diff options
Diffstat (limited to 'pcbnew/hotkeys_board_editor.cpp')
-rw-r--r-- | pcbnew/hotkeys_board_editor.cpp | 1205 |
1 files changed, 1205 insertions, 0 deletions
diff --git a/pcbnew/hotkeys_board_editor.cpp b/pcbnew/hotkeys_board_editor.cpp new file mode 100644 index 0000000..3dd119d --- /dev/null +++ b/pcbnew/hotkeys_board_editor.cpp @@ -0,0 +1,1205 @@ +/* + * This program source code file is part of KiCad, a free EDA CAD application. + * + * Copyright (C) 2009 Jean-Pierre Charras, jp.charras@wanadoo.fr + * Copyright (C) 1992-2016 KiCad Developers, see AUTHORS.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 hotkeys_board_editor.cpp + */ + +#include <fctsys.h> +#include <wxPcbStruct.h> +#include <class_drawpanel.h> +#include <confirm.h> + +#include <class_board.h> +#include <class_module.h> +#include <class_track.h> +#include <class_pcb_text.h> +#include <class_mire.h> +#include <class_drawsegment.h> + +#include <pcbnew.h> +#include <pcbnew_id.h> +#include <hotkeys.h> +#include <class_zone.h> +#include <tool/tool_manager.h> + +/* How to add a new hotkey: + * see hotkeys.cpp + */ + + +void PCB_EDIT_FRAME::RecordMacros(wxDC* aDC, int aNumber) +{ + wxASSERT( aNumber >= 0 && aNumber < 10 ); + wxString msg; + + if( m_RecordingMacros < 0 ) + { + m_RecordingMacros = aNumber; + m_Macros[aNumber].m_StartPosition = GetCrossHairPosition( false ); + m_Macros[aNumber].m_Record.clear(); + + msg.Printf( _( "Recording macro %d" ), aNumber ); + SetStatusText( msg ); + } + else + { + m_RecordingMacros = -1; + + msg.Printf( _( "Macro %d recorded" ), aNumber ); + SetStatusText( msg ); + } +} + + +void PCB_EDIT_FRAME::CallMacros( wxDC* aDC, const wxPoint& aPosition, int aNumber ) +{ + wxPoint tPosition; + + wxString msg; + + msg.Printf( _( "Call macro %d" ), aNumber ); + SetStatusText( msg ); + + wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED ); + cmd.SetEventObject( this ); + + tPosition = GetNearestGridPosition( aPosition ); + + m_canvas->CrossHairOff( aDC ); + SetMousePosition( tPosition ); + GeneralControl( aDC, tPosition ); + + for( std::list<MACROS_RECORD>::iterator i = m_Macros[aNumber].m_Record.begin(); + i != m_Macros[aNumber].m_Record.end(); ++i ) + { + wxPoint tmpPos = GetNearestGridPosition( tPosition + i->m_Position ); + + SetMousePosition( tmpPos ); + + GeneralControl( aDC, tmpPos, i->m_HotkeyCode ); + } + + cmd.SetId( ID_ZOOM_REDRAW ); + GetEventHandler()->ProcessEvent( cmd ); + + m_canvas->CrossHairOn( aDC ); +} + + +EDA_HOTKEY* PCB_EDIT_FRAME::GetHotKeyDescription( int aCommand ) const +{ + EDA_HOTKEY* HK_Descr = GetDescriptorFromCommand( aCommand, common_Hotkey_List ); + + if( HK_Descr == NULL ) + HK_Descr = GetDescriptorFromCommand( aCommand, board_edit_Hotkey_List ); + + return HK_Descr; +} + + +bool PCB_EDIT_FRAME::OnHotKey( wxDC* aDC, int aHotkeyCode, const wxPoint& aPosition, + EDA_ITEM* aItem ) +{ + if( aHotkeyCode == 0 ) + return false; + + bool itemCurrentlyEdited = GetCurItem() && GetCurItem()->GetFlags(); + MODULE* module = NULL; + int evt_type = 0; //Used to post a wxCommandEvent on demand + PCB_SCREEN* screen = GetScreen(); + DISPLAY_OPTIONS* displ_opts = (DISPLAY_OPTIONS*)GetDisplayOptions(); + + /* Convert lower to upper case + * (the usual toupper function has problem with non ascii codes like function keys + */ + if( (aHotkeyCode >= 'a') && (aHotkeyCode <= 'z') ) + aHotkeyCode += 'A' - 'a'; + + EDA_HOTKEY* HK_Descr = GetDescriptorFromHotkey( aHotkeyCode, common_Hotkey_List ); + + if( HK_Descr == NULL ) + HK_Descr = GetDescriptorFromHotkey( aHotkeyCode, board_edit_Hotkey_List ); + + if( HK_Descr == NULL ) + return false; + + int hk_id = HK_Descr->m_Idcommand; + + if( (m_RecordingMacros != -1) && + !( hk_id > HK_MACRO_ID_BEGIN && hk_id < HK_MACRO_ID_END) ) + { + MACROS_RECORD macros_record; + macros_record.m_HotkeyCode = aHotkeyCode; + macros_record.m_Idcommand = HK_Descr->m_Idcommand; + macros_record.m_Position = GetNearestGridPosition( aPosition ) - + m_Macros[m_RecordingMacros].m_StartPosition; + m_Macros[m_RecordingMacros].m_Record.push_back( macros_record ); + wxString msg; + msg.Printf( _( "Add key [%c] in macro %d" ), aHotkeyCode, m_RecordingMacros ); + SetStatusText( msg ); + } + + // Create a wxCommandEvent that will be posted in some hot keys functions + wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED ); + cmd.SetEventObject( this ); + + LAYER_NUM ll; + + switch( hk_id ) + { + default: + case HK_NOT_FOUND: + return false; + + case HK_LEFT_CLICK: + OnLeftClick( aDC, aPosition ); + break; + + case HK_LEFT_DCLICK: // Simulate a double left click: generate 2 events + OnLeftClick( aDC, aPosition ); + OnLeftDClick( aDC, aPosition ); + break; + + case HK_RECORD_MACROS_0: + case HK_RECORD_MACROS_1: + case HK_RECORD_MACROS_2: + case HK_RECORD_MACROS_3: + case HK_RECORD_MACROS_4: + case HK_RECORD_MACROS_5: + case HK_RECORD_MACROS_6: + case HK_RECORD_MACROS_7: + case HK_RECORD_MACROS_8: + case HK_RECORD_MACROS_9: + RecordMacros( aDC, hk_id - HK_RECORD_MACROS_0 ); + break; + + case HK_CALL_MACROS_0: + case HK_CALL_MACROS_1: + case HK_CALL_MACROS_2: + case HK_CALL_MACROS_3: + case HK_CALL_MACROS_4: + case HK_CALL_MACROS_5: + case HK_CALL_MACROS_6: + case HK_CALL_MACROS_7: + case HK_CALL_MACROS_8: + case HK_CALL_MACROS_9: + CallMacros( aDC, GetCrossHairPosition( false ), hk_id - HK_CALL_MACROS_0 ); + break; + + case HK_SWITCH_TRACK_WIDTH_TO_NEXT: + if( GetCanvas()->IsMouseCaptured() ) + GetCanvas()->CallMouseCapture( aDC, wxDefaultPosition, false ); + + if( GetDesignSettings().GetTrackWidthIndex() < GetDesignSettings().m_TrackWidthList.size() - 1 ) + GetDesignSettings().SetTrackWidthIndex( GetDesignSettings().GetTrackWidthIndex() + 1 ); + else + GetDesignSettings().SetTrackWidthIndex( 0 ); + + if( GetCanvas()->IsMouseCaptured() ) + GetCanvas()->CallMouseCapture( aDC, wxDefaultPosition, false ); + + break; + + case HK_SWITCH_TRACK_WIDTH_TO_PREVIOUS: + if( GetCanvas()->IsMouseCaptured() ) + GetCanvas()->CallMouseCapture( aDC, wxDefaultPosition, false ); + + if( GetDesignSettings().GetTrackWidthIndex() <= 0 ) + GetDesignSettings().SetTrackWidthIndex( GetDesignSettings().m_TrackWidthList.size() -1 ); + else + GetDesignSettings().SetTrackWidthIndex( GetDesignSettings().GetTrackWidthIndex() - 1 ); + + if( GetCanvas()->IsMouseCaptured() ) + GetCanvas()->CallMouseCapture( aDC, wxDefaultPosition, false ); + + break; + + case HK_SWITCH_GRID_TO_FASTGRID1: + SetFastGrid1(); + break; + + case HK_SWITCH_GRID_TO_FASTGRID2: + SetFastGrid2(); + break; + + case HK_SWITCH_GRID_TO_NEXT: + SetNextGrid(); + break; + + case HK_SWITCH_GRID_TO_PREVIOUS: + SetPrevGrid(); + break; + + case HK_SWITCH_LAYER_TO_PREVIOUS: + ll = GetActiveLayer(); + + if( !IsCopperLayer( ll ) ) + break; + + if( ll == F_Cu ) + ll = B_Cu; + else if( ll == B_Cu ) + ll = ToLAYER_ID( GetBoard()->GetCopperLayerCount() - 2 ); + else + ll = ll - 1; + + SwitchLayer( aDC, ToLAYER_ID( ll ) ); + break; + + case HK_SWITCH_LAYER_TO_NEXT: + ll = GetActiveLayer(); + + if( !IsCopperLayer( ll ) ) + break; + + if( ll == B_Cu ) + ll = F_Cu; + else if( ++ll >= GetBoard()->GetCopperLayerCount() - 1 ) + ll = B_Cu; + + SwitchLayer( aDC, ToLAYER_ID( ll ) ); + break; + + case HK_SWITCH_LAYER_TO_COMPONENT: + SwitchLayer( aDC, F_Cu ); + break; + + case HK_SWITCH_LAYER_TO_COPPER: + SwitchLayer( aDC, B_Cu ); + break; + + case HK_SWITCH_LAYER_TO_INNER1: + SwitchLayer( aDC, In1_Cu ); + break; + + case HK_SWITCH_LAYER_TO_INNER2: + SwitchLayer( aDC, In2_Cu ); + break; + + case HK_SWITCH_LAYER_TO_INNER3: + SwitchLayer( aDC, In3_Cu ); + break; + + case HK_SWITCH_LAYER_TO_INNER4: + SwitchLayer( aDC, In4_Cu ); + break; + + case HK_SWITCH_LAYER_TO_INNER5: + SwitchLayer( aDC, In5_Cu ); + break; + + case HK_SWITCH_LAYER_TO_INNER6: + SwitchLayer( aDC, In6_Cu ); + break; + + case HK_HELP: // Display Current hotkey list + DisplayHotkeyList( this, g_Board_Editor_Hokeys_Descr ); + break; + + case HK_ZOOM_IN: + evt_type = ID_POPUP_ZOOM_IN; + break; + + case HK_ZOOM_OUT: + evt_type = ID_POPUP_ZOOM_OUT; + break; + + case HK_ZOOM_REDRAW: + evt_type = ID_ZOOM_REDRAW; + break; + + case HK_ZOOM_AUTO: + evt_type = ID_ZOOM_PAGE; + break; + + case HK_ZOOM_CENTER: + evt_type = ID_POPUP_ZOOM_CENTER; + break; + + case HK_ADD_MODULE: + evt_type = ID_PCB_MODULE_BUTT; + break; + + case HK_UNDO: + case HK_REDO: + if( !itemCurrentlyEdited ) + { + wxCommandEvent event( wxEVT_COMMAND_TOOL_CLICKED, HK_Descr->m_IdMenuEvent ); + wxPostEvent( this, event ); + } + + break; + + case HK_RESET_LOCAL_COORD: // Set the relative coord + GetScreen()->m_O_Curseur = GetCrossHairPosition(); + break; + + case HK_SET_GRID_ORIGIN: + SetGridOrigin( GetCrossHairPosition() ); + OnModify(); // because grid origin is saved in board, show as modified + m_canvas->Refresh(); + break; + + case HK_RESET_GRID_ORIGIN: + SetGridOrigin( wxPoint( 0,0 ) ); + OnModify(); // because grid origin is saved in board, show as modified + m_canvas->Refresh(); + break; + + case HK_SWITCH_UNITS: + evt_type = (g_UserUnit == INCHES) ? + ID_TB_OPTIONS_SELECT_UNIT_MM : ID_TB_OPTIONS_SELECT_UNIT_INCH; + break; + + case HK_SWITCH_TRACK_DISPLAY_MODE: + displ_opts->m_DisplayPcbTrackFill = !displ_opts->m_DisplayPcbTrackFill; + m_canvas->Refresh(); + break; + + case HK_DELETE: + OnHotkeyDeleteItem( aDC ); + break; + + case HK_BACK_SPACE: + if( IsCopperLayer( GetActiveLayer() ) ) + { + if( !itemCurrentlyEdited ) + { + // no track is currently being edited - select a segment and remove it. + // @todo: possibly? pass the HK command code to PcbGeneralLocateAndDisplay() + // so it can restrict its search to specific item types. + BOARD_ITEM * item = PcbGeneralLocateAndDisplay(); + + // don't let backspace delete modules!! + if( item && item->IsTrack() ) + { + Delete_Segment( aDC, (TRACK*) item ); + SetCurItem( NULL ); + } + + OnModify(); + } + else if( GetCurItem()->IsTrack() ) + { + // then an element is being edited - remove the last segment. + // simple lines for debugger: + TRACK* track = (TRACK*) GetCurItem(); + track = Delete_Segment( aDC, track ); + SetCurItem( track ); + OnModify(); + } + } + + break; + + case HK_GET_AND_MOVE_FOOTPRINT: + if( !itemCurrentlyEdited ) + evt_type = ID_POPUP_PCB_GET_AND_MOVE_MODULE_REQUEST; + + break; + + case HK_FIND_ITEM: + if( !itemCurrentlyEdited ) + evt_type = ID_FIND_ITEMS; + + break; + + case HK_LOAD_BOARD: + if( !itemCurrentlyEdited ) + evt_type = ID_LOAD_FILE ; + + break; + + case HK_SAVE_BOARD: + if( !itemCurrentlyEdited ) + evt_type = ID_SAVE_BOARD; + + break; + + case HK_ADD_MICROVIA: // Place a micro via if a track is in progress + if( GetToolId() != ID_TRACK_BUTT ) + return true; + + if( !itemCurrentlyEdited ) // no track in progress: nothing to do + break; + + if( GetCurItem()->Type() != PCB_TRACE_T ) // Should not occur + return true; + + if( !GetCurItem()->IsNew() ) + return true; + + // place micro via and switch layer + if( IsMicroViaAcceptable() ) + evt_type = ID_POPUP_PCB_PLACE_MICROVIA; + + break; + + case HK_ADD_BLIND_BURIED_VIA: + case HK_ADD_THROUGH_VIA: // Switch to alternate layer and Place a via if a track is in progress + if( GetBoard()->GetDesignSettings().m_BlindBuriedViaAllowed && + hk_id == HK_ADD_BLIND_BURIED_VIA ) + GetBoard()->GetDesignSettings().m_CurrentViaType = VIA_BLIND_BURIED; + else + GetBoard()->GetDesignSettings().m_CurrentViaType = VIA_THROUGH; + + if( !itemCurrentlyEdited ) // no track in progress: switch layer only + { + Other_Layer_Route( NULL, aDC ); + if( displ_opts->m_ContrastModeDisplay ) + m_canvas->Refresh(); + break; + } + + if( GetToolId() != ID_TRACK_BUTT ) + return true; + + if( GetCurItem()->Type() != PCB_TRACE_T ) + return true; + + if( !GetCurItem()->IsNew() ) + return true; + + evt_type = hk_id == HK_ADD_BLIND_BURIED_VIA ? + ID_POPUP_PCB_PLACE_BLIND_BURIED_VIA : ID_POPUP_PCB_PLACE_THROUGH_VIA; + break; + + case HK_SEL_LAYER_AND_ADD_THROUGH_VIA: + case HK_SEL_LAYER_AND_ADD_BLIND_BURIED_VIA: + if( GetCurItem() == NULL || !GetCurItem()->IsNew() || + GetCurItem()->Type() != PCB_TRACE_T ) + break; + + evt_type = hk_id == HK_SEL_LAYER_AND_ADD_BLIND_BURIED_VIA ? + ID_POPUP_PCB_SELECT_CU_LAYER_AND_PLACE_BLIND_BURIED_VIA : + ID_POPUP_PCB_SELECT_CU_LAYER_AND_PLACE_THROUGH_VIA; + break; + + case HK_SWITCH_TRACK_POSTURE: + /* change the position of initial segment when creating new tracks + * switch from _/ to -\ . + */ + evt_type = ID_POPUP_PCB_SWITCH_TRACK_POSTURE ; + break; + + case HK_DRAG_TRACK_KEEP_SLOPE: + OnHotkeyMoveItem( HK_DRAG_TRACK_KEEP_SLOPE ); + break; + + case HK_PLACE_ITEM: + OnHotkeyPlaceItem( aDC ); + break; + + case HK_ADD_NEW_TRACK: // Start new track, if possible + OnHotkeyBeginRoute( aDC ); + break; + + case HK_EDIT_ITEM: // Edit board item + OnHotkeyEditItem( HK_EDIT_ITEM ); + break; + + case HK_EDIT_MODULE_WITH_MODEDIT: // Edit module with module editor + OnHotkeyEditItem( HK_EDIT_MODULE_WITH_MODEDIT ); + break; + + // Footprint edition: + case HK_LOCK_UNLOCK_FOOTPRINT: // toggle module "MODULE_is_LOCKED" status: + // get any module, locked or not locked and toggle its locked status + if( !itemCurrentlyEdited ) + { + wxPoint pos = RefPos( true ); + module = GetBoard()->GetFootprint( pos, screen->m_Active_Layer, true ); + } + else if( GetCurItem()->Type() == PCB_MODULE_T ) + { + module = (MODULE*) GetCurItem(); + } + + if( module ) + { + SetCurItem( module ); + module->SetLocked( !module->IsLocked() ); + OnModify(); + SetMsgPanel( module ); + } + break; + + case HK_DRAG_ITEM: // Start drag module or track segment + OnHotkeyMoveItem( HK_DRAG_ITEM ); + break; + + case HK_MOVE_ITEM: // Start move item + OnHotkeyMoveItem( HK_MOVE_ITEM ); + break; + + case HK_COPY_ITEM: + evt_type = OnHotkeyCopyItem(); + break; + + case HK_ROTATE_ITEM: // Rotation + OnHotkeyRotateItem( HK_ROTATE_ITEM ); + break; + + case HK_FLIP_ITEM: + OnHotkeyFlipItem( HK_FLIP_ITEM ); + break; + + case HK_MOVE_ITEM_EXACT: + case HK_DUPLICATE_ITEM: + case HK_DUPLICATE_ITEM_AND_INCREMENT: + case HK_CREATE_ARRAY: + OnHotkeyDuplicateOrArrayItem( HK_Descr->m_Idcommand ); + break; + + case HK_SWITCH_HIGHCONTRAST_MODE: // switch to high contrast mode and refresh the canvas + displ_opts->m_ContrastModeDisplay = !displ_opts->m_ContrastModeDisplay; + m_canvas->Refresh(); + break; + + case HK_CANVAS_CAIRO: + evt_type = ID_MENU_CANVAS_CAIRO; + break; + + case HK_CANVAS_OPENGL: + evt_type = ID_MENU_CANVAS_OPENGL; + break; + + case HK_CANVAS_DEFAULT: + evt_type = ID_MENU_CANVAS_DEFAULT; + break; + case HK_ZONE_FILL_OR_REFILL: + evt_type = ID_POPUP_PCB_FILL_ALL_ZONES; + break; + case HK_ZONE_REMOVE_FILLED: + evt_type = ID_POPUP_PCB_REMOVE_FILLED_AREAS_IN_ALL_ZONES; + break; + + } + + if( evt_type != 0 ) + { + wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED ); + evt.SetEventObject( this ); + evt.SetId( evt_type ); + GetEventHandler()->ProcessEvent( evt ); + } + + return true; +} + + +bool PCB_EDIT_FRAME::OnHotkeyDeleteItem( wxDC* aDC ) +{ + BOARD_ITEM* item = GetCurItem(); + bool ItemFree = (item == NULL) || (item->GetFlags() == 0); + + switch( GetToolId() ) + { + case ID_TRACK_BUTT: + if( !IsCopperLayer ( GetActiveLayer() ) ) + return false; + + if( ItemFree ) + { + item = PcbGeneralLocateAndDisplay(); + + if( item && !item->IsTrack() ) + return false; + + Delete_Track( aDC, (TRACK*) item ); + } + else if( item->IsTrack( ) ) + { + // simple lines for debugger: + TRACK* track = (TRACK*) item; + track = Delete_Segment( aDC, track ); + SetCurItem( track ); + OnModify(); + return true; + } + break; + + case ID_PCB_MODULE_BUTT: + if( ItemFree ) + { + wxPoint pos = RefPos( false ); + MODULE* module = GetBoard()->GetFootprint( pos, UNDEFINED_LAYER, false ); + + if( module == NULL || module->IsLocked() ) + return false; + + RemoveStruct( module, aDC ); + } + else + return false; + break; + + default: + if( ItemFree ) + { + item = PcbGeneralLocateAndDisplay(); + + // Shouldn't there be a check for locked tracks and vias here? + if( item == NULL || (item->Type() == PCB_MODULE_T && (MODULE*)item->IsLocked()) ) + return false; + + RemoveStruct( item, aDC ); + } + else + return false; + } + + OnModify(); + SetCurItem( NULL ); + return true; +} + + +bool PCB_EDIT_FRAME::OnHotkeyEditItem( int aIdCommand ) +{ + BOARD_ITEM* item = GetCurItem(); + bool itemCurrentlyEdited = item && item->GetFlags(); + + if( itemCurrentlyEdited ) + return false; + + item = PcbGeneralLocateAndDisplay(); + + if( item == NULL ) + return false; + + SetCurItem( item ); + + int evt_type = 0; //Used to post a wxCommandEvent on demand + + switch( item->Type() ) + { + case PCB_TRACE_T: + case PCB_VIA_T: + if( aIdCommand == HK_EDIT_ITEM ) + { + // Be sure the corresponding netclass is selected before edit: + SetCurrentNetClass( ( (BOARD_CONNECTED_ITEM*)item )->GetNetClassName() ); + evt_type = ID_POPUP_PCB_EDIT_TRACKSEG; + } + + break; + + case PCB_TEXT_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_TEXTEPCB; + + break; + + case PCB_MODULE_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_MODULE_PRMS; + else if( aIdCommand == HK_EDIT_MODULE_WITH_MODEDIT ) + evt_type = ID_POPUP_PCB_EDIT_MODULE_WITH_MODEDIT; + + break; + + case PCB_PAD_T: + // Until dec 2012 a EDIT_MODULE event is posted here to prevent pads + // from being edited by hotkeys. + // Process_Special_Functions takes care of finding the parent. + // After dec 2012 a EDIT_PAD event is posted, because there is no + // reason to not allow pad edit by hotkey + // (pad coordinates are no more modified by rounding, in nanometer version + // when using inches or mm in dialog) + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_PAD; + + break; + + case PCB_TARGET_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_MIRE; + + break; + + case PCB_DIMENSION_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_DIMENSION; + + break; + + case PCB_MODULE_TEXT_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_TEXTMODULE; + + break; + + case PCB_LINE_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_DRAWING; + + break; + + case PCB_ZONE_AREA_T: + if( aIdCommand == HK_EDIT_ITEM ) + evt_type = ID_POPUP_PCB_EDIT_ZONE_PARAMS; + + break; + + default: + break; + } + + if( evt_type != 0 ) + { + wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED ); + evt.SetEventObject( this ); + evt.SetId( evt_type ); + GetEventHandler()->ProcessEvent( evt ); + return true; + } + + return false; +} + + +int PCB_EDIT_FRAME::OnHotkeyCopyItem() +{ + BOARD_ITEM* item = GetCurItem(); + bool itemCurrentlyEdited = item && item->GetFlags(); + + if( itemCurrentlyEdited ) + return 0; + + item = PcbGeneralLocateAndDisplay(); + + if( item == NULL ) + return 0; + + SetCurItem( item ); + + int eventId = 0; + + switch( item->Type() ) + { + case PCB_TEXT_T: + eventId = ID_POPUP_PCB_COPY_TEXTEPCB; + break; + default: + eventId = 0; + break; + } + + return eventId; +} + + +bool PCB_EDIT_FRAME::OnHotkeyMoveItem( int aIdCommand ) +{ + BOARD_ITEM* item = GetCurItem(); + bool itemCurrentlyEdited = item && item->GetFlags(); + + if( itemCurrentlyEdited ) + return false; + + item = PcbGeneralLocateAndDisplay(); + + if( item == NULL ) + return false; + + SetCurItem( item ); + + int evt_type = 0; // Used to post a wxCommandEvent on demand + + switch( item->Type() ) + { + case PCB_TRACE_T: + case PCB_VIA_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_TRACK_NODE; + + if( aIdCommand == HK_DRAG_ITEM ) + evt_type = ID_POPUP_PCB_DRAG_TRACK_SEGMENT; + + if( aIdCommand == HK_DRAG_TRACK_KEEP_SLOPE ) + evt_type = ID_POPUP_PCB_DRAG_TRACK_SEGMENT_KEEP_SLOPE; + + break; + + case PCB_MODULE_T: + { + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_MODULE_REQUEST; + + if( aIdCommand == HK_DRAG_ITEM ) + evt_type = ID_POPUP_PCB_DRAG_MODULE_REQUEST; + } + break; + + case PCB_PAD_T: + // Post MODULE_REQUEST events here to prevent pads + // from being moved or dragged by hotkeys. + // Process_Special_Functions takes care of finding + // the parent. + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_MODULE_REQUEST; + + if( aIdCommand == HK_DRAG_ITEM ) + evt_type = ID_POPUP_PCB_DRAG_MODULE_REQUEST; + + break; + + case PCB_TEXT_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_TEXTEPCB_REQUEST; + + break; + + case PCB_TARGET_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_MIRE_REQUEST; + + break; + + case PCB_ZONE_AREA_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_ZONE_OUTLINES; + + if( aIdCommand == HK_DRAG_ITEM ) + evt_type = ID_POPUP_PCB_DRAG_ZONE_OUTLINE_SEGMENT; + + break; + + case PCB_MODULE_TEXT_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_TEXTMODULE_REQUEST; + + break; + + case PCB_LINE_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_DRAWING_REQUEST; + + break; + + case PCB_DIMENSION_T: + if( aIdCommand == HK_MOVE_ITEM ) + evt_type = ID_POPUP_PCB_MOVE_TEXT_DIMENSION_REQUEST; + break; + + default: + break; + } + + if( evt_type != 0 ) + { + wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED ); + evt.SetEventObject( this ); + evt.SetId( evt_type ); + GetEventHandler()->ProcessEvent( evt ); + return true; + } + + return false; +} + + +bool PCB_EDIT_FRAME::OnHotkeyPlaceItem( wxDC* aDC ) +{ + BOARD_ITEM* item = GetCurItem(); + bool no_tool = GetToolId() == ID_NO_TOOL_SELECTED; + bool itemCurrentlyEdited = item && item->GetFlags(); + + m_canvas->SetAutoPanRequest( false ); + + if( itemCurrentlyEdited ) + { + m_canvas->SetIgnoreMouseEvents( true ); + m_canvas->CrossHairOff( aDC ); + + switch( item->Type() ) + { + case PCB_TRACE_T: + case PCB_VIA_T: + if( item->IsDragging() ) + PlaceDraggedOrMovedTrackSegment( static_cast<TRACK*>( item ), aDC ); + + break; + + case PCB_TEXT_T: + Place_Texte_Pcb( static_cast<TEXTE_PCB*>( item ), aDC ); + break; + + case PCB_MODULE_TEXT_T: + PlaceTexteModule( static_cast<TEXTE_MODULE*>( item ), aDC ); + break; + + case PCB_PAD_T: + PlacePad( static_cast<D_PAD*>( item ), aDC ); + break; + + case PCB_MODULE_T: + PlaceModule( static_cast<MODULE*>( item ), aDC ); + break; + + case PCB_TARGET_T: + PlaceTarget( static_cast<PCB_TARGET*>( item ), aDC ); + break; + + case PCB_LINE_T: + if( no_tool ) // when no tools: existing item moving. + Place_DrawItem( static_cast<DRAWSEGMENT*>( item ), aDC ); + + break; + + default: + break; + } + + m_canvas->SetIgnoreMouseEvents( false ); + m_canvas->CrossHairOn( aDC ); + + return true; + } + + return false; +} + + +TRACK * PCB_EDIT_FRAME::OnHotkeyBeginRoute( wxDC* aDC ) +{ + if( !IsCopperLayer( GetActiveLayer() ) ) + return NULL; + + bool itemCurrentlyEdited = GetCurItem() && GetCurItem()->GetFlags(); + + // Ensure the track tool is active + if( GetToolId() != ID_TRACK_BUTT && !itemCurrentlyEdited ) + { + wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED ); + cmd.SetEventObject( this ); + cmd.SetId( ID_TRACK_BUTT ); + GetEventHandler()->ProcessEvent( cmd ); + } + + if( GetToolId() != ID_TRACK_BUTT ) + return NULL; + + TRACK* track = NULL; + + if( !itemCurrentlyEdited ) // no track in progress: + { + track = Begin_Route( NULL, aDC ); + SetCurItem( track ); + + if( track ) + m_canvas->SetAutoPanRequest( true ); + } + else if( GetCurItem()->IsNew() ) + { + track = Begin_Route( (TRACK*) GetCurItem(), aDC ); + + // SetCurItem() must not write to the msg panel + // because a track info is displayed while moving the mouse cursor + if( track ) // A new segment was created + SetCurItem( track, false ); + + m_canvas->SetAutoPanRequest( true ); + } + + return track; +} + + +bool PCB_EDIT_FRAME::OnHotkeyFlipItem( int aIdCommand ) +{ + BOARD_ITEM* item = GetCurItem(); + bool itemCurrentlyEdited = item && item->GetFlags(); + int evt_type = 0; // Used to post a wxCommandEvent on demand + + wxASSERT( aIdCommand == HK_FLIP_ITEM ); + + if( GetScreen()->m_BlockLocate.GetState() != STATE_NO_BLOCK ) + { + evt_type = ID_POPUP_FLIP_BLOCK; + } + else + { + if( !itemCurrentlyEdited ) + item = PcbGeneralLocateAndDisplay(); + + if( item == NULL ) + return false; + + SetCurItem( item ); + + switch( item->Type() ) + { + case PCB_MODULE_T: + evt_type = ID_POPUP_PCB_CHANGE_SIDE_MODULE; + break; + + case PCB_TEXT_T: + evt_type = ID_POPUP_PCB_FLIP_TEXTEPCB; + break; + + default: + break; + } + } + + if( evt_type != 0 ) + { + wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED ); + evt.SetEventObject( this ); + evt.SetId( evt_type ); + GetEventHandler()->ProcessEvent( evt ); + return true; + } + else + { + return false; + } +} + + +bool PCB_EDIT_FRAME::OnHotkeyRotateItem( int aIdCommand ) +{ + BOARD_ITEM* item = GetCurItem(); + bool itemCurrentlyEdited = item && item->GetFlags(); + int evt_type = 0; // Used to post a wxCommandEvent on demand + + wxASSERT( aIdCommand == HK_ROTATE_ITEM ); + + // Allows block rotate operation on hot key. + if( GetScreen()->m_BlockLocate.GetState() != STATE_NO_BLOCK ) + { + evt_type = ID_POPUP_ROTATE_BLOCK; + } + else + { + if( !itemCurrentlyEdited ) + item = PcbGeneralLocateAndDisplay(); + + if( item == NULL ) + return false; + + SetCurItem( item ); + + switch( item->Type() ) + { + case PCB_MODULE_T: + evt_type = ID_POPUP_PCB_ROTATE_MODULE_COUNTERCLOCKWISE; + break; + + case PCB_TEXT_T: + evt_type = ID_POPUP_PCB_ROTATE_TEXTEPCB; + break; + + case PCB_MODULE_TEXT_T: + evt_type = ID_POPUP_PCB_ROTATE_TEXTMODULE; + break; + + default: + break; + } + } + + if( evt_type != 0 ) + { + wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED ); + evt.SetEventObject( this ); + evt.SetId( evt_type ); + GetEventHandler()->ProcessEvent( evt ); + return true; + } + + return false; +} + + +bool PCB_EDIT_FRAME::OnHotkeyDuplicateOrArrayItem( int aIdCommand ) +{ + BOARD_ITEM* item = GetCurItem(); + bool itemCurrentlyEdited = item && item->GetFlags(); + + if( itemCurrentlyEdited ) + return false; + + item = PcbGeneralLocateAndDisplay(); + + if( item == NULL ) + return false; + + SetCurItem( item ); + + int evt_type = 0; // Used to post a wxCommandEvent on demand + + bool canDuplicate = true; + + switch( item->Type() ) + { + // Only handle items we know we can handle + case PCB_PAD_T: + canDuplicate = false; + // no break + case PCB_MODULE_T: + case PCB_LINE_T: + case PCB_TEXT_T: + case PCB_TRACE_T: + case PCB_ZONE_AREA_T: + case PCB_TARGET_T: + case PCB_DIMENSION_T: + switch( aIdCommand ) + { + case HK_CREATE_ARRAY: + if( canDuplicate ) + evt_type = ID_POPUP_PCB_CREATE_ARRAY; + break; + + case HK_DUPLICATE_ITEM_AND_INCREMENT: + if( canDuplicate ) + evt_type = ID_POPUP_PCB_DUPLICATE_ITEM_AND_INCREMENT; + break; + + case HK_DUPLICATE_ITEM: + if( canDuplicate ) + evt_type = ID_POPUP_PCB_DUPLICATE_ITEM; + break; + + case HK_MOVE_ITEM_EXACT: + evt_type = ID_POPUP_PCB_MOVE_EXACT; + break; + + default: + // We don't handle other commands here + break; + } + break; + + default: + wxASSERT_MSG( false, "Unhandled move, duplicate or array for " + "object type " + item->Type() ); + break; + } + + return PostCommandMenuEvent( evt_type ); +} |