You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							514 lines
						
					
					
						
							16 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							514 lines
						
					
					
						
							16 KiB
						
					
					
				| /* | |
|  * This program source code file is part of KiCad, a free EDA CAD application. | |
|  * | |
|  * Copyright (C) 2004 Jean-Pierre Charras, jaen-pierre.charras@gipsa-lab.inpg.com | |
|  * Copyright (C) 2008-2016 Wayne Stambaugh <stambaughw@verizon.net> | |
|  * Copyright (C) 2004-2016 KiCad Developers, see change_log.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 | |
|  */ | |
| 
 | |
| /** | |
|  * @file eeschema/find.cpp | |
|  * @brief Functions for searching for a schematic item. | |
|  */ | |
| 
 | |
| /* | |
|  *  Search a text (text, value, reference) within a component or | |
|  *  search a component in libraries, a marker ..., | |
|  *  in current sheet or whole the project | |
|  */ | |
| #include <fctsys.h> | |
| #include <pgm_base.h> | |
| #include <class_drawpanel.h> | |
| #include <confirm.h> | |
| #include <kicad_string.h> | |
| #include <gestfich.h> | |
| #include <schframe.h> | |
| #include <base_units.h> | |
|  | |
| #include <general.h> | |
| #include <class_library.h> | |
| #include <lib_pin.h> | |
| #include <sch_marker.h> | |
| #include <sch_component.h> | |
| #include <sch_sheet.h> | |
| #include <sch_sheet_path.h> | |
|  | |
| #include <kicad_device_context.h> | |
|  | |
| #include <dialogs/dialog_schematic_find.h> | |
|  | |
| 
 | |
| void SCH_EDIT_FRAME::OnFindDrcMarker( wxFindDialogEvent& event ) | |
| { | |
|     static SCH_MARKER* lastMarker = NULL; | |
| 
 | |
|     wxString           msg; | |
|     SCH_SHEET_LIST     schematic( g_RootSheet ); | |
|     SCH_SHEET_PATH*    sheetFoundIn = NULL; | |
|     bool               wrap = ( event.GetFlags() & FR_SEARCH_WRAP ) != 0; | |
|     bool               warpCursor = ( ( event.GetId() == wxEVT_COMMAND_FIND_CLOSE ) || | |
|                                       !( event.GetFlags() & FR_NO_WARP_CURSOR ) ); | |
| 
 | |
|     if( event.GetFlags() & FR_CURRENT_SHEET_ONLY ) | |
|     { | |
|         sheetFoundIn = m_CurrentSheet; | |
|         lastMarker = (SCH_MARKER*) m_CurrentSheet->FindNextItem( SCH_MARKER_T, lastMarker, wrap ); | |
|     } | |
|     else | |
|     { | |
|         lastMarker = (SCH_MARKER*) schematic.FindNextItem( SCH_MARKER_T, &sheetFoundIn, | |
|                                                            lastMarker, wrap ); | |
|     } | |
| 
 | |
|     if( lastMarker != NULL ) | |
|     { | |
|         if( *sheetFoundIn != *m_CurrentSheet ) | |
|         { | |
|             sheetFoundIn->LastScreen()->SetZoom( GetScreen()->GetZoom() ); | |
|             *m_CurrentSheet = *sheetFoundIn; | |
|             m_CurrentSheet->UpdateAllScreenReferences(); | |
|         } | |
| 
 | |
|         SetCrossHairPosition( lastMarker->GetPosition() ); | |
| 
 | |
|         RedrawScreen( lastMarker->GetPosition(), warpCursor ); | |
| 
 | |
|         wxString path = sheetFoundIn->Path(); | |
|         wxString units = GetAbbreviatedUnitsLabel(); | |
|         double x = To_User_Unit( g_UserUnit, (double) lastMarker->GetPosition().x ); | |
|         double y = To_User_Unit( g_UserUnit, (double) lastMarker->GetPosition().y ); | |
|         msg.Printf( _( "Design rule check marker found in sheet %s at %0.3f%s, %0.3f%s" ), | |
|                     GetChars( path ), x, GetChars( units ), y, GetChars( units) ); | |
|         SetStatusText( msg ); | |
|     } | |
|     else | |
|     { | |
|         SetStatusText( _( "No more markers were found." ) ); | |
|     } | |
| } | |
| 
 | |
| 
 | |
| SCH_ITEM* SCH_EDIT_FRAME::FindComponentAndItem( const wxString& aReference, | |
|                                                 bool            aSearchHierarchy, | |
|                                                 SCH_SEARCH_T    aSearchType, | |
|                                                 const wxString& aSearchText, | |
|                                                 bool            aWarpMouse ) | |
| { | |
|     SCH_SHEET_PATH* sheet = NULL; | |
|     SCH_SHEET_PATH* sheetWithComponentFound = NULL; | |
|     SCH_ITEM*       item = NULL; | |
|     SCH_COMPONENT*  Component = NULL; | |
|     wxPoint         pos; | |
|     bool            centerAndRedraw = false; | |
|     bool            notFound = true; | |
|     LIB_PIN*        pin; | |
|     SCH_SHEET_LIST  sheetList( g_RootSheet ); | |
| 
 | |
|     if( !aSearchHierarchy ) | |
|         sheetList.push_back( *m_CurrentSheet ); | |
|     else | |
|         sheetList.BuildSheetList( g_RootSheet ); | |
| 
 | |
|     for( SCH_SHEET_PATHS_ITER it = sheetList.begin(); it != sheetList.end(); ++it ) | |
|     { | |
|         sheet = &(*it); | |
|         item = (*it).LastDrawList(); | |
| 
 | |
|         for( ; ( item != NULL ) && ( notFound == true ); item = item->Next() ) | |
|         { | |
|             if( item->Type() != SCH_COMPONENT_T ) | |
|                 continue; | |
| 
 | |
|             SCH_COMPONENT* pSch = (SCH_COMPONENT*) item; | |
| 
 | |
|             if( aReference.CmpNoCase( pSch->GetRef( sheet ) ) == 0 ) | |
|             { | |
|                 Component = pSch; | |
|                 sheetWithComponentFound = sheet; | |
| 
 | |
|                 switch( aSearchType ) | |
|                 { | |
|                 default: | |
|                 case FIND_COMPONENT_ONLY:    // Find component only | |
|                     notFound = false; | |
|                     pos = pSch->GetPosition(); | |
|                     break; | |
| 
 | |
|                 case FIND_PIN:               // find a pin | |
|                     pos = pSch->GetPosition();  // temporary: will be changed if the pin is found. | |
|                     pin = pSch->GetPin( aSearchText ); | |
| 
 | |
|                     if( pin == NULL ) | |
|                         break; | |
| 
 | |
|                     notFound = false; | |
|                     pos += pin->GetPosition(); | |
|                     break; | |
| 
 | |
|                 case FIND_REFERENCE:         // find reference | |
|                     notFound = false; | |
|                     pos = pSch->GetField( REFERENCE )->GetPosition(); | |
|                     break; | |
| 
 | |
|                 case FIND_VALUE:             // find value | |
|                     pos = pSch->GetPosition(); | |
| 
 | |
|                     if( aSearchText.CmpNoCase( pSch->GetField( VALUE )->GetShownText() ) != 0 ) | |
|                         break; | |
| 
 | |
|                     notFound = false; | |
|                     pos = pSch->GetField( VALUE )->GetPosition(); | |
|                     break; | |
|                 } | |
|             } | |
|         } | |
| 
 | |
|         if( notFound == false ) | |
|             break; | |
|     } | |
| 
 | |
|     if( Component ) | |
|     { | |
|         sheet = sheetWithComponentFound; | |
| 
 | |
|         if( *sheet != *m_CurrentSheet ) | |
|         { | |
|             sheet->LastScreen()->SetZoom( GetScreen()->GetZoom() ); | |
|             *m_CurrentSheet = *sheet; | |
|             m_CurrentSheet->UpdateAllScreenReferences(); | |
|             centerAndRedraw = true; | |
|         } | |
| 
 | |
|         wxPoint delta; | |
|         pos  -= Component->GetPosition(); | |
|         delta = Component->GetTransform().TransformCoordinate( pos ); | |
|         pos   = delta + Component->GetPosition(); | |
| 
 | |
| 
 | |
|         /* There may be need to reframe the drawing */ | |
|         if( ! m_canvas->IsPointOnDisplay( pos ) ) | |
|         { | |
|             centerAndRedraw = true; | |
|         } | |
| 
 | |
|         if( centerAndRedraw ) | |
|         { | |
|             SetCrossHairPosition( pos ); | |
|             RedrawScreen( pos, aWarpMouse ); | |
|         } | |
| 
 | |
|         else | |
|         { | |
|             INSTALL_UNBUFFERED_DC( dc, m_canvas ); | |
| 
 | |
|             m_canvas->CrossHairOff( &dc ); | |
| 
 | |
|             if( aWarpMouse ) | |
|                 m_canvas->MoveCursor( pos ); | |
| 
 | |
|             SetCrossHairPosition( pos ); | |
| 
 | |
|             m_canvas->CrossHairOn( &dc ); | |
|         } | |
|     } | |
| 
 | |
| 
 | |
|     /* Print diag */ | |
|     wxString msg_item; | |
|     wxString msg; | |
| 
 | |
|     switch( aSearchType ) | |
|     { | |
|     default: | |
|     case FIND_COMPONENT_ONLY:      // Find component only | |
|         msg_item = _( "component" ); | |
|         break; | |
| 
 | |
|     case FIND_PIN:                 // find a pin | |
|         msg_item.Printf( _( "pin %s" ), GetChars( aSearchText ) ); | |
|         break; | |
| 
 | |
|     case FIND_REFERENCE:           // find reference | |
|         msg_item.Printf( _( "reference %s" ), GetChars( aSearchText ) ); | |
|         break; | |
| 
 | |
|     case FIND_VALUE:               // find value | |
|         msg_item.Printf( _( "value %s" ), GetChars( aSearchText ) ); | |
|         break; | |
| 
 | |
|     case FIND_FIELD:               // find field. todo | |
|         msg_item.Printf( _( "field %s" ), GetChars( aSearchText ) ); | |
|         break; | |
|     } | |
| 
 | |
|     if( Component ) | |
|     { | |
|         if( !notFound ) | |
|         { | |
|             msg.Printf( _( "%s %s found" ), | |
|                         GetChars( aReference ), GetChars( msg_item ) ); | |
|         } | |
|         else | |
|         { | |
|             msg.Printf( _( "%s found but %s not found" ), | |
|                         GetChars( aReference ), GetChars( msg_item ) ); | |
|         } | |
|     } | |
|     else | |
|     { | |
|         msg.Printf( _( "Component %s not found" ), | |
|                     GetChars( aReference ) ); | |
|     } | |
| 
 | |
|     SetStatusText( msg ); | |
| 
 | |
|     return item; | |
| } | |
| 
 | |
| 
 | |
| bool SCH_EDIT_FRAME::IsSearchCacheObsolete( const SCH_FIND_REPLACE_DATA& aSearchCriteria ) | |
| { | |
|     PART_LIBS*  libs = Prj().SchLibs(); | |
|     int         mod_hash = libs->GetModifyHash(); | |
| 
 | |
|     // the cache is obsolete whenever any library changes. | |
|     if( mod_hash != m_foundItems.GetLibHash() ) | |
|     { | |
|         m_foundItems.SetForceSearch(); | |
|         m_foundItems.SetLibHash( mod_hash ); | |
|         return true; | |
|     } | |
|     else if( m_foundItems.IsSearchRequired( aSearchCriteria ) ) | |
|         return true; | |
|     else | |
|         return false; | |
| } | |
| 
 | |
| 
 | |
| void SCH_EDIT_FRAME::OnFindSchematicItem( wxFindDialogEvent& aEvent ) | |
| { | |
|     SCH_FIND_REPLACE_DATA   searchCriteria; | |
|     SCH_FIND_COLLECTOR_DATA data; | |
| 
 | |
|     searchCriteria.SetFlags( aEvent.GetFlags() ); | |
|     searchCriteria.SetFindString( aEvent.GetFindString() ); | |
|     searchCriteria.SetReplaceString( aEvent.GetReplaceString() ); | |
| 
 | |
|     if( aEvent.GetEventType() == wxEVT_COMMAND_FIND_CLOSE ) | |
|     { | |
|         if( m_foundItems.GetCount() == 0 ) | |
|             return; | |
| 
 | |
|         // Refresh the search cache in case something has changed.  This prevents any stale | |
|         // pointers from crashing Eeschema when the wxEVT_FIND_CLOSE event is handled. | |
|         if( IsSearchCacheObsolete( searchCriteria ) ) | |
|         { | |
|             if( aEvent.GetFlags() & FR_CURRENT_SHEET_ONLY && g_RootSheet->CountSheets() > 1 ) | |
|             { | |
|                 m_foundItems.Collect( searchCriteria, m_CurrentSheet ); | |
|             } | |
|             else | |
|             { | |
|                 m_foundItems.Collect( searchCriteria ); | |
|             } | |
|         } | |
|     } | |
|     else if( IsSearchCacheObsolete( searchCriteria ) ) | |
|     { | |
|         if( aEvent.GetFlags() & FR_CURRENT_SHEET_ONLY && g_RootSheet->CountSheets() > 1 ) | |
|         { | |
|             m_foundItems.Collect( searchCriteria, m_CurrentSheet ); | |
|         } | |
|         else | |
|         { | |
|             m_foundItems.Collect( searchCriteria ); | |
|         } | |
|     } | |
|     else | |
|     { | |
|         EDA_ITEM* currentItem = m_foundItems.GetItem( data ); | |
| 
 | |
|         if( currentItem != NULL ) | |
|             currentItem->SetForceVisible( false ); | |
| 
 | |
|         m_foundItems.UpdateIndex(); | |
|     } | |
| 
 | |
|     updateFindReplaceView( aEvent ); | |
| } | |
| 
 | |
| 
 | |
| void SCH_EDIT_FRAME::OnFindReplace( wxFindDialogEvent& aEvent ) | |
| { | |
|     static int              nextFoundIndex = 0; | |
|     SCH_ITEM*               item; | |
|     SCH_SHEET_PATH*         sheet; | |
|     SCH_SHEET_LIST          schematic( g_RootSheet ); | |
|     SCH_FIND_COLLECTOR_DATA data; | |
|     SCH_FIND_REPLACE_DATA   searchCriteria; | |
| 
 | |
|     searchCriteria.SetFlags( aEvent.GetFlags() ); | |
|     searchCriteria.SetFindString( aEvent.GetFindString() ); | |
|     searchCriteria.SetReplaceString( aEvent.GetReplaceString() ); | |
|     m_foundItems.SetReplaceString( aEvent.GetReplaceString() ); | |
| 
 | |
|     if( IsSearchCacheObsolete( searchCriteria ) ) | |
|     { | |
|         if( aEvent.GetFlags() & FR_CURRENT_SHEET_ONLY && g_RootSheet->CountSheets() > 1 ) | |
|         { | |
|             m_foundItems.Collect( searchCriteria, m_CurrentSheet ); | |
|         } | |
|         else | |
|         { | |
|             m_foundItems.Collect( searchCriteria ); | |
|         } | |
| 
 | |
|         // Restore the next found index on cache refresh.  Prevents single replace events | |
|         // from starting back at the beginning of the cache. | |
|         m_foundItems.SetFoundIndex( nextFoundIndex ); | |
|     } | |
| 
 | |
|     if( aEvent.GetEventType() == wxEVT_COMMAND_FIND_REPLACE_ALL ) | |
|     { | |
|         while( ( item = (SCH_ITEM*) m_foundItems.GetItem( data ) ) != NULL ) | |
|         { | |
|             SCH_ITEM* undoItem = data.GetParent(); | |
| 
 | |
|             // Don't save child items in undo list. | |
|             if( undoItem == NULL ) | |
|                 undoItem = item; | |
| 
 | |
|             SetUndoItem( undoItem ); | |
| 
 | |
|             sheet = schematic.GetSheetByPath( data.GetSheetPath() ); | |
| 
 | |
|             wxCHECK_RET( sheet != NULL, wxT( "Could not find sheet path " ) + data.GetSheetPath() ); | |
| 
 | |
|             if( m_foundItems.ReplaceItem( sheet ) ) | |
|             { | |
|                 OnModify(); | |
|                 SaveUndoItemInUndoList( undoItem ); | |
|                 updateFindReplaceView( aEvent ); | |
|             } | |
| 
 | |
|             m_foundItems.IncrementIndex(); | |
| 
 | |
|             if( m_foundItems.PassedEnd() ) | |
|                 break; | |
|         } | |
|     } | |
|     else | |
|     { | |
|         item = (SCH_ITEM*) m_foundItems.GetItem( data ); | |
| 
 | |
|         wxCHECK_RET( item != NULL, wxT( "Invalid replace item in find collector list." ) ); | |
| 
 | |
|         SCH_ITEM* undoItem = data.GetParent(); | |
| 
 | |
|         if( undoItem == NULL ) | |
|             undoItem = item; | |
| 
 | |
|         SetUndoItem( undoItem ); | |
| 
 | |
|         sheet = schematic.GetSheetByPath( data.GetSheetPath() ); | |
| 
 | |
|         wxCHECK_RET( sheet != NULL, wxT( "Could not find sheet path " ) + data.GetSheetPath() ); | |
| 
 | |
|         if( m_foundItems.ReplaceItem( sheet ) ) | |
|         { | |
|             OnModify(); | |
|             SaveUndoItemInUndoList( undoItem ); | |
|             updateFindReplaceView( aEvent ); | |
|         } | |
| 
 | |
|         m_foundItems.IncrementIndex(); | |
|         nextFoundIndex = m_foundItems.GetFoundIndex(); | |
|     } | |
| 
 | |
|     // End the replace if we are at the end if the list.  This prevents an infinite loop if | |
|     // wrap search is selected and all of the items have been replaced with a value that | |
|     // still satisfies the search criteria. | |
|     if( m_foundItems.PassedEnd() ) | |
|         aEvent.SetFlags( aEvent.GetFlags() & ~FR_REPLACE_ITEM_FOUND ); | |
| } | |
| 
 | |
| 
 | |
| void SCH_EDIT_FRAME::updateFindReplaceView( wxFindDialogEvent& aEvent ) | |
| { | |
|     wxString                msg; | |
|     SCH_SHEET_LIST          schematic( g_RootSheet ); | |
|     SCH_FIND_COLLECTOR_DATA data; | |
|     SCH_FIND_REPLACE_DATA   searchCriteria; | |
|     bool                    warpCursor = !( aEvent.GetFlags() & FR_NO_WARP_CURSOR ); | |
| 
 | |
|     searchCriteria.SetFlags( aEvent.GetFlags() ); | |
|     searchCriteria.SetFindString( aEvent.GetFindString() ); | |
|     searchCriteria.SetReplaceString( aEvent.GetReplaceString() ); | |
| 
 | |
|     if( m_foundItems.GetItem( data ) != NULL ) | |
|     { | |
|         wxLogTrace( traceFindReplace, wxT( "Found " ) + m_foundItems.GetText() ); | |
| 
 | |
|         SCH_SHEET_PATH* sheet = schematic.GetSheetByPath( data.GetSheetPath() ); | |
| 
 | |
|         wxCHECK_RET( sheet != NULL, wxT( "Could not find sheet path " ) + | |
|                      data.GetSheetPath() ); | |
| 
 | |
|         SCH_ITEM* item = (SCH_ITEM*)m_foundItems.GetItem( data ); | |
| 
 | |
|         // Make the item temporarily visible just in case it's hide flag is set.  This | |
|         // has no effect on objects that don't support hiding.  If this is a close find | |
|         // dialog event, clear the temporary visibility flag. | |
|         if( item ) | |
|         { | |
|             if( aEvent.GetEventType() == wxEVT_COMMAND_FIND_CLOSE ) | |
|                 item->SetForceVisible( false ); | |
|             else if( item->Type() == SCH_FIELD_T && !( (SCH_FIELD*) item )->IsVisible() ) | |
|                 item->SetForceVisible( true ); | |
|         } | |
| 
 | |
|         if( sheet->PathHumanReadable() != m_CurrentSheet->PathHumanReadable() ) | |
|         { | |
|             sheet->LastScreen()->SetZoom( GetScreen()->GetZoom() ); | |
|             *m_CurrentSheet = *sheet; | |
|             m_CurrentSheet->UpdateAllScreenReferences(); | |
|             SetScreen( sheet->LastScreen() ); | |
|         } | |
| 
 | |
|         // careful here | |
|         SetCrossHairPosition( data.GetPosition() ); | |
| 
 | |
|         RedrawScreen( data.GetPosition(), warpCursor ); | |
| 
 | |
|         msg = m_foundItems.GetText(); | |
| 
 | |
|         if( aEvent.GetFlags() & FR_SEARCH_REPLACE ) | |
|             aEvent.SetFlags( aEvent.GetFlags() | FR_REPLACE_ITEM_FOUND ); | |
|     } | |
|     else | |
|     { | |
|         if( aEvent.GetFlags() & FR_SEARCH_REPLACE ) | |
|             aEvent.SetFlags( aEvent.GetFlags() & ~FR_REPLACE_ITEM_FOUND ); | |
| 
 | |
|         msg.Printf( _( "No item found matching %s." ), GetChars( aEvent.GetFindString() ) ); | |
|     } | |
| 
 | |
|     SetStatusText( msg ); | |
| }
 |