1020 lines
30 KiB
C++
1020 lines
30 KiB
C++
/**
|
|
* @file eeschema/hotkeys.cpp
|
|
*/
|
|
|
|
#include "fctsys.h"
|
|
#include "eeschema_id.h"
|
|
#include "hotkeys.h"
|
|
#include "wxEeschemaStruct.h"
|
|
|
|
#include "general.h"
|
|
#include "protos.h"
|
|
#include "libeditframe.h"
|
|
#include "class_libentry.h"
|
|
#include "sch_junction.h"
|
|
#include "sch_line.h"
|
|
#include "sch_component.h"
|
|
#include "sch_sheet.h"
|
|
|
|
#include "dialogs/dialog_schematic_find.h"
|
|
|
|
|
|
/* How to add a new hotkey:
|
|
* add a new id in the enum hotkey_id_command like MY_NEW_ID_FUNCTION (see
|
|
* hotkeys.h).
|
|
* add a new EDA_HOTKEY entry like:
|
|
* static EDA_HOTKEY HkMyNewEntry(wxT("Command Label"), MY_NEW_ID_FUNCTION,
|
|
* default key value);
|
|
* wxT("Command Label") is the name used in hotkey list display, and the
|
|
* identifier in the hotkey list file
|
|
* MY_NEW_ID_FUNCTION is an equivalent id function used in the switch in
|
|
* OnHotKey() function.
|
|
* default key value is the default hotkey for this command. Can be overridden
|
|
* by the user hotkey list file
|
|
* add the HkMyNewEntry pointer in the s_Schematic_Hotkey_List list or the
|
|
* s_LibEdit_Hotkey_List list or s_Common_Hotkey_List if the same command is
|
|
* added both in Eeschema and libedit)
|
|
* Add the new code in the switch in OnHotKey() function.
|
|
* when the variable itemInEdit is true, an item is currently edited.
|
|
* This can be useful if the new function cannot be executed while an item is
|
|
* currently being edited
|
|
* ( For example, one cannot start a new wire when a component is moving.)
|
|
*
|
|
* Note: If an hotkey is a special key be sure the corresponding wxWidget
|
|
* keycode (WXK_XXXX) is handled in the hotkey_name_descr
|
|
* s_Hotkey_Name_List list (see hotkeys_basic.cpp) and see this list
|
|
* for some ascii keys (space ...)
|
|
*
|
|
* Key modifier are: GR_KB_CTRL GR_KB_ALT
|
|
*/
|
|
|
|
|
|
/* local variables */
|
|
/* Hotkey list: */
|
|
|
|
/**
|
|
* Common commands
|
|
*/
|
|
|
|
/* Fit on Screen */
|
|
#if !defined( __WXMAC__ )
|
|
static EDA_HOTKEY HkZoomAuto( wxT( "Fit on Screen" ), HK_ZOOM_AUTO, WXK_HOME );
|
|
#else
|
|
static EDA_HOTKEY HkZoomAuto( wxT( "Zoom Auto" ), HK_ZOOM_AUTO, GR_KB_CTRL + '0' );
|
|
#endif
|
|
|
|
static EDA_HOTKEY HkZoomCenter( wxT( "Zoom Center" ), HK_ZOOM_CENTER, WXK_F4 );
|
|
|
|
/* Refresh Screen */
|
|
#if !defined( __WXMAC__ )
|
|
static EDA_HOTKEY HkZoomRedraw( wxT( "Zoom Redraw" ), HK_ZOOM_REDRAW, WXK_F3 );
|
|
#else
|
|
static EDA_HOTKEY HkZoomRedraw( wxT( "Zoom Redraw" ), HK_ZOOM_REDRAW, GR_KB_CTRL + 'R' );
|
|
#endif
|
|
|
|
/* Zoom In */
|
|
#if !defined( __WXMAC__ )
|
|
static EDA_HOTKEY HkZoomIn( wxT( "Zoom In" ), HK_ZOOM_IN, WXK_F1 );
|
|
#else
|
|
static EDA_HOTKEY HkZoomIn( wxT( "Zoom In" ), HK_ZOOM_IN, GR_KB_CTRL + '+' );
|
|
#endif
|
|
|
|
/* Zoom Out */
|
|
#if !defined( __WXMAC__ )
|
|
static EDA_HOTKEY HkZoomOut( wxT( "Zoom Out" ), HK_ZOOM_OUT, WXK_F2 );
|
|
#else
|
|
static EDA_HOTKEY HkZoomOut( wxT( "Zoom Out" ), HK_ZOOM_OUT, GR_KB_CTRL + '-' );
|
|
#endif
|
|
|
|
static EDA_HOTKEY HkHelp( wxT( "Help (this window)" ), HK_HELP, '?' );
|
|
static EDA_HOTKEY HkResetLocalCoord( wxT( "Reset Local Coordinates" ),
|
|
HK_RESET_LOCAL_COORD, ' ' );
|
|
|
|
/* Undo */
|
|
static EDA_HOTKEY HkUndo( wxT( "Undo" ), HK_UNDO, GR_KB_CTRL + 'Z', (int) wxID_UNDO );
|
|
|
|
/* Redo */
|
|
#if !defined( __WXMAC__ )
|
|
static EDA_HOTKEY HkRedo( wxT( "Redo" ), HK_REDO, GR_KB_CTRL + 'Y', (int) wxID_REDO );
|
|
#else
|
|
static EDA_HOTKEY HkRedo( wxT( "Redo" ), HK_REDO, GR_KB_SHIFT + GR_KB_CTRL + 'Z',
|
|
(int) wxID_REDO );
|
|
#endif
|
|
|
|
// Schematic editor
|
|
static EDA_HOTKEY HkAddLabel( wxT( "add Label" ), HK_ADD_LABEL, 'L' );
|
|
static EDA_HOTKEY HkAddHierarchicalLabel( wxT( "Add Hierarchical Label" ), HK_ADD_HLABEL, 'H' );
|
|
static EDA_HOTKEY HkAddGlobalLabel( wxT( "Add Global Label" ), HK_ADD_GLABEL, GR_KB_CTRL + 'L' );
|
|
static EDA_HOTKEY HkAddJunction( wxT( "Add Junction" ), HK_ADD_JUNCTION, 'J' );
|
|
static EDA_HOTKEY HkBeginWire( wxT( "Draw Wire" ), HK_BEGIN_WIRE, 'W' );
|
|
static EDA_HOTKEY HkBeginBus( wxT( "Draw Bus" ), HK_BEGIN_BUS, 'B' );
|
|
static EDA_HOTKEY HkAddComponent( wxT( "Add Component" ), HK_ADD_NEW_COMPONENT, 'A' );
|
|
static EDA_HOTKEY HkAddPower( wxT( "Add Power" ), HK_ADD_NEW_POWER, 'P' );
|
|
static EDA_HOTKEY HkAddNoConn( wxT( "Add NoConnected Flag" ), HK_ADD_NOCONN_FLAG, 'Q' );
|
|
static EDA_HOTKEY HkAddHierSheet( wxT( "Add Sheet" ), HK_ADD_HIER_SHEET, 'S' );
|
|
static EDA_HOTKEY HkAddBusEntry( wxT( "Add Bus Entry" ), HK_ADD_BUS_ENTRY, '/' );
|
|
static EDA_HOTKEY HkAddWireEntry( wxT( "Add Wire Entry" ), HK_ADD_WIRE_ENTRY, 'Z' );
|
|
static EDA_HOTKEY HkAddGraphicPolyLine( wxT( "Add Graphic PolyLine" ), HK_ADD_GRAPHIC_POLYLINE, 'I' );
|
|
static EDA_HOTKEY HkAddGraphicText( wxT( "Add Graphic Text" ), HK_ADD_GRAPHIC_TEXT, 'T' );
|
|
static EDA_HOTKEY HkMirrorYComponent( wxT( "Mirror Y Component" ), HK_MIRROR_Y_COMPONENT, 'Y' );
|
|
static EDA_HOTKEY HkMirrorXComponent( wxT( "Mirror X Component" ), HK_MIRROR_X_COMPONENT, 'X' );
|
|
static EDA_HOTKEY HkOrientNormalComponent( wxT( "Orient Normal Component" ),
|
|
HK_ORIENT_NORMAL_COMPONENT, 'N' );
|
|
static EDA_HOTKEY HkRotate( wxT( "Rotate Item" ), HK_ROTATE, 'R' );
|
|
static EDA_HOTKEY HkEdit( wxT( "Edit Schematic Item" ), HK_EDIT, 'E' );
|
|
static EDA_HOTKEY HkEditComponentValue( wxT( "Edit Component Value" ),
|
|
HK_EDIT_COMPONENT_VALUE, 'V',
|
|
ID_POPUP_SCH_EDIT_VALUE_CMP );
|
|
static EDA_HOTKEY HkEditComponentFootprint( wxT( "Edit Component Footprint" ),
|
|
HK_EDIT_COMPONENT_FOOTPRINT, 'F',
|
|
ID_POPUP_SCH_EDIT_FOOTPRINT_CMP );
|
|
static EDA_HOTKEY HkMove( wxT( "Move Schematic Item" ),
|
|
HK_MOVE_COMPONENT_OR_ITEM, 'M',
|
|
ID_POPUP_SCH_MOVE_ITEM );
|
|
|
|
static EDA_HOTKEY HkCopyComponentOrText( wxT( "Copy Component or Label" ),
|
|
HK_COPY_COMPONENT_OR_LABEL, 'C',
|
|
ID_POPUP_SCH_COPY_ITEM );
|
|
|
|
static EDA_HOTKEY HkDrag( wxT( "Drag Schematic Item" ), HK_DRAG, 'G',
|
|
ID_POPUP_SCH_DRAG_CMP_REQUEST );
|
|
static EDA_HOTKEY HkMove2Drag( wxT( "Move Block -> Drag Block" ),
|
|
HK_MOVEBLOCK_TO_DRAGBLOCK, '\t' );
|
|
static EDA_HOTKEY HkInsert( wxT( "Repeat Last Item" ), HK_REPEAT_LAST, WXK_INSERT );
|
|
static EDA_HOTKEY HkDelete( wxT( "Delete Item" ), HK_DELETE, WXK_DELETE );
|
|
|
|
static EDA_HOTKEY HkFindItem( wxT( "Find Item" ), HK_FIND_ITEM, 'F' + GR_KB_CTRL );
|
|
static EDA_HOTKEY HkFindNextItem( wxT( "Find Next Item" ), HK_FIND_NEXT_ITEM, WXK_F5 );
|
|
static EDA_HOTKEY HkFindNextDrcMarker( wxT( "Find Next DRC Marker" ), HK_FIND_NEXT_DRC_MARKER,
|
|
WXK_F5 + GR_KB_SHIFT );
|
|
|
|
// Special keys for library editor:
|
|
static EDA_HOTKEY HkCreatePin( wxT( "Create Pin" ), HK_LIBEDIT_CREATE_PIN, 'P' );
|
|
static EDA_HOTKEY HkInsertPin( wxT( "Repeat Pin" ), HK_REPEAT_LAST, WXK_INSERT );
|
|
static EDA_HOTKEY HkMoveLibItem( wxT( "Move Library Item" ), HK_LIBEDIT_MOVE_GRAPHIC_ITEM, 'M' );
|
|
|
|
|
|
// List of common hotkey descriptors
|
|
EDA_HOTKEY* s_Common_Hotkey_List[] =
|
|
{
|
|
&HkHelp,
|
|
&HkZoomIn,
|
|
&HkZoomOut,
|
|
&HkZoomRedraw,
|
|
&HkZoomCenter,
|
|
&HkZoomAuto,
|
|
&HkResetLocalCoord,
|
|
&HkUndo,
|
|
&HkRedo,
|
|
NULL
|
|
};
|
|
|
|
// List of hotkey descriptors for schematic
|
|
EDA_HOTKEY* s_Schematic_Hotkey_List[] =
|
|
{
|
|
&HkFindItem,
|
|
&HkFindNextItem,
|
|
&HkFindNextDrcMarker,
|
|
&HkDelete,
|
|
&HkInsert,
|
|
&HkMove2Drag,
|
|
&HkMove,
|
|
&HkCopyComponentOrText,
|
|
&HkDrag,
|
|
&HkAddComponent,
|
|
&HkAddPower,
|
|
&HkRotate,
|
|
&HkMirrorXComponent,
|
|
&HkMirrorYComponent,
|
|
&HkOrientNormalComponent,
|
|
&HkEdit,
|
|
&HkEditComponentValue,
|
|
&HkEditComponentFootprint,
|
|
&HkBeginWire,
|
|
&HkBeginBus,
|
|
&HkAddLabel,
|
|
&HkAddHierarchicalLabel,
|
|
&HkAddGlobalLabel,
|
|
&HkAddJunction,
|
|
&HkAddNoConn,
|
|
&HkAddHierSheet,
|
|
&HkAddWireEntry,
|
|
&HkAddBusEntry,
|
|
&HkAddGraphicPolyLine,
|
|
&HkAddGraphicText,
|
|
NULL
|
|
};
|
|
|
|
// List of hotkey descriptors for library editor
|
|
EDA_HOTKEY* s_LibEdit_Hotkey_List[] =
|
|
{
|
|
&HkCreatePin,
|
|
&HkInsertPin,
|
|
&HkEdit,
|
|
&HkMoveLibItem,
|
|
&HkDelete,
|
|
&HkRotate,
|
|
&HkDrag,
|
|
NULL
|
|
};
|
|
|
|
// list of sections and corresponding hotkey list for Eeschema (used to create
|
|
// an hotkey config file)
|
|
struct EDA_HOTKEY_CONFIG s_Eeschema_Hokeys_Descr[] =
|
|
{
|
|
{ &g_CommonSectionTag, s_Common_Hotkey_List, L"Common keys" },
|
|
{ &g_SchematicSectionTag, s_Schematic_Hotkey_List, L"Schematic editor keys" },
|
|
{ &g_LibEditSectionTag, s_LibEdit_Hotkey_List, L"library editor keys" },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
// list of sections and corresponding hotkey list for the schematic editor
|
|
// (used to list current hotkeys)
|
|
struct EDA_HOTKEY_CONFIG s_Schematic_Hokeys_Descr[] =
|
|
{
|
|
{ &g_CommonSectionTag, s_Common_Hotkey_List, NULL },
|
|
{ &g_SchematicSectionTag, s_Schematic_Hotkey_List, NULL },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
// list of sections and corresponding hotkey list for the component editor
|
|
// (used to list current hotkeys)
|
|
struct EDA_HOTKEY_CONFIG s_Libedit_Hokeys_Descr[] =
|
|
{
|
|
{ &g_CommonSectionTag, s_Common_Hotkey_List, NULL },
|
|
{ &g_LibEditSectionTag, s_LibEdit_Hotkey_List, NULL },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
// list of sections and corresponding hotkey list for the component browser
|
|
// (used to list current hotkeys)
|
|
struct EDA_HOTKEY_CONFIG s_Viewlib_Hokeys_Descr[] =
|
|
{
|
|
{ &g_CommonSectionTag, s_Common_Hotkey_List, NULL },
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
/*
|
|
* Hot keys. Some commands are relative to the item under the mouse cursor
|
|
* Commands are case insensitive
|
|
*/
|
|
void SCH_EDIT_FRAME::OnHotKey( wxDC* aDC, int aHotKey, const wxPoint& aPosition, EDA_ITEM* aItem )
|
|
{
|
|
if( aHotKey == 0 )
|
|
return;
|
|
|
|
wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED );
|
|
|
|
cmd.SetEventObject( this );
|
|
|
|
SCH_SCREEN* screen = GetScreen();
|
|
|
|
// itemInEdit == false means no item currently edited. We can ask for editing a new item
|
|
bool itemInEdit = screen->GetCurItem() && screen->GetCurItem()->GetFlags();
|
|
|
|
// notBusy == true means no item currently edited and no other command in progress
|
|
// We can change active tool and ask for editing a new item
|
|
bool notBusy = (!itemInEdit) && (screen->m_BlockLocate.m_State == STATE_NO_BLOCK);
|
|
|
|
/* Convert lower to upper case (the usual toupper function has problem
|
|
* with non ascii codes like function keys */
|
|
if( (aHotKey >= 'a') && (aHotKey <= 'z') )
|
|
aHotKey += 'A' - 'a';
|
|
|
|
// Search command from key :
|
|
EDA_HOTKEY* HK_Descr = GetDescriptorFromHotkey( aHotKey, s_Common_Hotkey_List );
|
|
|
|
if( HK_Descr == NULL )
|
|
HK_Descr = GetDescriptorFromHotkey( aHotKey, s_Schematic_Hotkey_List );
|
|
|
|
if( HK_Descr == NULL )
|
|
return;
|
|
|
|
switch( HK_Descr->m_Idcommand )
|
|
{
|
|
default:
|
|
case HK_NOT_FOUND:
|
|
return;
|
|
|
|
case HK_HELP: // Display Current hotkey list
|
|
DisplayHotkeyList( this, s_Schematic_Hokeys_Descr );
|
|
break;
|
|
|
|
case HK_RESET_LOCAL_COORD: /* Reset the relative coord */
|
|
GetScreen()->m_O_Curseur = GetScreen()->GetCrossHairPosition();
|
|
break;
|
|
|
|
case HK_ZOOM_IN:
|
|
cmd.SetId( ID_POPUP_ZOOM_IN );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_OUT:
|
|
cmd.SetId( ID_POPUP_ZOOM_OUT );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_REDRAW:
|
|
cmd.SetId( ID_ZOOM_REDRAW );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_CENTER:
|
|
cmd.SetId( ID_POPUP_ZOOM_CENTER );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_AUTO:
|
|
cmd.SetId( ID_ZOOM_PAGE );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_UNDO:
|
|
case HK_REDO:
|
|
if( notBusy )
|
|
{
|
|
wxCommandEvent event( wxEVT_COMMAND_TOOL_CLICKED, HK_Descr->m_IdMenuEvent );
|
|
wxPostEvent( this, event );
|
|
}
|
|
break;
|
|
|
|
case HK_MOVEBLOCK_TO_DRAGBLOCK: // Switch to drag mode, when block moving
|
|
HandleBlockEndByPopUp( BLOCK_DRAG, aDC );
|
|
break;
|
|
|
|
case HK_DELETE:
|
|
if( notBusy )
|
|
DeleteItemAtCrossHair( aDC );
|
|
|
|
break;
|
|
|
|
case HK_REPEAT_LAST:
|
|
if( notBusy && m_itemToRepeat && ( m_itemToRepeat->GetFlags() == 0 ) )
|
|
RepeatDrawItem( aDC );
|
|
break;
|
|
|
|
case HK_FIND_ITEM:
|
|
if( notBusy )
|
|
{
|
|
wxCommandEvent evt;
|
|
evt.SetId( ID_FIND_ITEMS );
|
|
Process_Special_Functions( evt );
|
|
}
|
|
break;
|
|
|
|
case HK_FIND_NEXT_ITEM:
|
|
if( notBusy )
|
|
{
|
|
wxFindDialogEvent event( wxEVT_COMMAND_FIND, GetId() );
|
|
event.SetEventObject( this );
|
|
event.SetFlags( m_findReplaceData->GetFlags() );
|
|
event.SetFindString( m_findReplaceData->GetFindString() );
|
|
GetEventHandler()->ProcessEvent( event );
|
|
}
|
|
break;
|
|
|
|
case HK_FIND_NEXT_DRC_MARKER:
|
|
if( notBusy )
|
|
{
|
|
wxFindDialogEvent event( EVT_COMMAND_FIND_DRC_MARKER, GetId() );
|
|
event.SetEventObject( this );
|
|
event.SetFlags( m_findReplaceData->GetFlags() );
|
|
event.SetFindString( m_findReplaceData->GetFindString() );
|
|
GetEventHandler()->ProcessEvent( event );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_NEW_COMPONENT: // Add component
|
|
if( !itemInEdit )
|
|
{
|
|
if( GetToolId() != ID_SCH_PLACE_COMPONENT )
|
|
SetToolID( ID_SCH_PLACE_COMPONENT, wxCURSOR_PENCIL, _( "Add Component" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_NEW_POWER: // Add power component
|
|
if( !itemInEdit )
|
|
{
|
|
if( GetToolId() != ID_PLACE_POWER_BUTT )
|
|
SetToolID( ID_PLACE_POWER_BUTT, wxCURSOR_PENCIL, _( "Add Power" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_LABEL:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_LABEL_BUTT )
|
|
SetToolID( ID_LABEL_BUTT, wxCURSOR_PENCIL, _( "Add Label" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_HLABEL:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_HIERLABEL_BUTT )
|
|
SetToolID( ID_HIERLABEL_BUTT, wxCURSOR_PENCIL, _( "Add Hierarchical Label" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_GLABEL:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_GLABEL_BUTT )
|
|
SetToolID( ID_GLABEL_BUTT, wxCURSOR_PENCIL, _( "Add Global Label" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_JUNCTION:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_JUNCTION_BUTT )
|
|
SetToolID( ID_JUNCTION_BUTT, wxCURSOR_PENCIL, _( "Add Junction" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_WIRE_ENTRY:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_WIRETOBUS_ENTRY_BUTT )
|
|
SetToolID( ID_WIRETOBUS_ENTRY_BUTT, wxCURSOR_PENCIL, _( "Add Wire to Bus Entry" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_BUS_ENTRY:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_BUSTOBUS_ENTRY_BUTT )
|
|
SetToolID( ID_BUSTOBUS_ENTRY_BUTT, wxCURSOR_PENCIL, _( "Add Bus to Bus Entry" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_HIER_SHEET:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_SHEET_SYMBOL_BUTT )
|
|
SetToolID( ID_SHEET_SYMBOL_BUTT, wxCURSOR_PENCIL, _( "Add Sheet" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_GRAPHIC_TEXT:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_TEXT_COMMENT_BUTT )
|
|
SetToolID( ID_TEXT_COMMENT_BUTT, wxCURSOR_PENCIL, _( "Add Text" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_GRAPHIC_POLYLINE:
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_LINE_COMMENT_BUTT )
|
|
SetToolID( ID_LINE_COMMENT_BUTT, wxCURSOR_PENCIL, _( "Add Lines" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_BEGIN_BUS:
|
|
// An item can be selected. If not a Bus, a begin command is not possible
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_BUS_BUTT )
|
|
SetToolID( ID_BUS_BUTT, wxCURSOR_PENCIL, _( "Add Bus" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
break;
|
|
}
|
|
|
|
if( aItem && aItem->IsNew() && ( GetToolId() == ID_BUS_BUTT ) )
|
|
{
|
|
if( aItem->Type() == SCH_LINE_T )
|
|
{
|
|
SCH_LINE* segment = (SCH_LINE*) aItem;
|
|
|
|
if( segment->GetLayer() != LAYER_BUS )
|
|
break;
|
|
|
|
// Bus in progress:
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case HK_BEGIN_WIRE:
|
|
// An item can be selected. If not a wire, a begin command is not possible
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_WIRE_BUTT )
|
|
SetToolID( ID_WIRE_BUTT, wxCURSOR_PENCIL, _( "Add Wire" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
break;
|
|
}
|
|
|
|
if( aItem && aItem->IsNew() && ( GetToolId() == ID_WIRE_BUTT ) )
|
|
{
|
|
if( aItem->Type() == SCH_LINE_T )
|
|
{
|
|
SCH_LINE* segment = (SCH_LINE*) aItem;
|
|
|
|
if( segment->GetLayer() != LAYER_WIRE )
|
|
break;
|
|
|
|
// Wire in progress:
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case HK_ADD_NOCONN_FLAG: // Add a no connected flag
|
|
if( notBusy )
|
|
{
|
|
if( GetToolId() != ID_NOCONN_BUTT )
|
|
SetToolID( ID_NOCONN_BUTT, wxCURSOR_PENCIL, _( "Add \"No Connect\" Flag" ) );
|
|
|
|
OnLeftClick( aDC, aPosition );
|
|
}
|
|
break;
|
|
|
|
case HK_ROTATE: // Component or other schematic item rotation
|
|
if ( screen->m_BlockLocate.m_State != STATE_NO_BLOCK )//allows bloc operation on hotkey
|
|
{
|
|
HandleBlockEndByPopUp(BLOCK_ROTATE, aDC );
|
|
break;
|
|
}
|
|
|
|
if( aItem == NULL )
|
|
{
|
|
// Find the schematic object to rotate under the cursor
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::RotatableItems );
|
|
|
|
if( aItem == NULL )
|
|
break;
|
|
}
|
|
|
|
switch( aItem->Type() )
|
|
{
|
|
case SCH_SHEET_T: //TODO allow sheet rotate on hotkey
|
|
//wxPostEvent( this, eventRotateSheet );
|
|
break;
|
|
|
|
case SCH_COMPONENT_T:
|
|
cmd.SetId( ID_POPUP_SCH_ROTATE_CMP_COUNTERCLOCKWISE );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_TEXT_T:
|
|
case SCH_LABEL_T:
|
|
case SCH_GLOBAL_LABEL_T:
|
|
case SCH_HIERARCHICAL_LABEL_T:
|
|
cmd.SetId( ID_POPUP_SCH_ROTATE_TEXT );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_FIELD_T:
|
|
cmd.SetId( ID_POPUP_SCH_ROTATE_FIELD );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_BITMAP_T:
|
|
cmd.SetId( ID_POPUP_SCH_ROTATE_IMAGE );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
break;
|
|
|
|
case HK_MIRROR_Y_COMPONENT: // Mirror Y (Component)
|
|
if( screen->m_BlockLocate.m_State != STATE_NO_BLOCK )
|
|
{
|
|
HandleBlockEndByPopUp( BLOCK_MIRROR_Y, aDC );
|
|
break;
|
|
}
|
|
|
|
if( aItem == NULL )
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::ComponentsOnly );
|
|
|
|
if( aItem )
|
|
{
|
|
screen->SetCurItem( (SCH_ITEM*) aItem );
|
|
cmd.SetId( ID_POPUP_SCH_MIRROR_Y_CMP );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
}
|
|
break;
|
|
|
|
case HK_MIRROR_X_COMPONENT: // Mirror X (Component)
|
|
if( screen->m_BlockLocate.m_State != STATE_NO_BLOCK ) //allows bloc operation on hotkey
|
|
{
|
|
HandleBlockEndByPopUp( BLOCK_MIRROR_X, aDC );
|
|
break;
|
|
}
|
|
|
|
if( aItem == NULL )
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::ComponentsOnly );
|
|
|
|
if( aItem )
|
|
{
|
|
screen->SetCurItem( (SCH_ITEM*) aItem );
|
|
cmd.SetId( ID_POPUP_SCH_MIRROR_X_CMP );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
}
|
|
break;
|
|
|
|
case HK_ORIENT_NORMAL_COMPONENT: // Orient 0, no mirror (Component)
|
|
if( aItem == NULL )
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::ComponentsOnly );
|
|
|
|
if( aItem )
|
|
{
|
|
screen->SetCurItem( (SCH_ITEM*) aItem );
|
|
cmd.SetId( ID_POPUP_SCH_ORIENT_NORMAL_CMP );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
}
|
|
|
|
break;
|
|
|
|
case HK_COPY_COMPONENT_OR_LABEL: // Duplicate component or text/label
|
|
if( itemInEdit )
|
|
break;
|
|
|
|
if( aItem == NULL )
|
|
{
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::MovableItems );
|
|
|
|
if( aItem == NULL )
|
|
break;
|
|
}
|
|
|
|
cmd.SetId( HK_Descr->m_IdMenuEvent );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case HK_DRAG: // Start drag
|
|
if( itemInEdit )
|
|
break;
|
|
|
|
if( aItem == NULL )
|
|
{
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::DraggableItems,
|
|
HK_Descr->m_Idcommand );
|
|
|
|
if( aItem == NULL )
|
|
break;
|
|
}
|
|
|
|
if( aItem->GetFlags() == 0 )
|
|
{
|
|
switch( aItem->Type() )
|
|
{
|
|
// select the correct event for moving an schematic object
|
|
// and add it to the event queue
|
|
case SCH_COMPONENT_T:
|
|
case SCH_GLOBAL_LABEL_T:
|
|
case SCH_HIERARCHICAL_LABEL_T:
|
|
case SCH_SHEET_T:
|
|
cmd.SetId( HK_Descr->m_IdMenuEvent );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_BUS_ENTRY_T:
|
|
case SCH_LINE_T:
|
|
case SCH_JUNCTION_T:
|
|
if( ((SCH_ITEM*) aItem )->GetLayer() != LAYER_BUS )
|
|
{
|
|
cmd.SetId( ID_POPUP_SCH_DRAG_WIRE_REQUEST );
|
|
wxPostEvent( this, cmd );
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
;
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case HK_MOVE_COMPONENT_OR_ITEM: // Start move component or other schematic item
|
|
if( itemInEdit )
|
|
break;
|
|
|
|
if( aItem == NULL )
|
|
{
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::MovableItems,
|
|
HK_Descr->m_Idcommand );
|
|
|
|
if( aItem == NULL )
|
|
break;
|
|
}
|
|
|
|
if( aItem->GetFlags() == 0 )
|
|
{
|
|
switch( aItem->Type() )
|
|
{
|
|
// select the correct event for moving an schematic object
|
|
// and add it to the event queue
|
|
case SCH_SHEET_T:
|
|
case SCH_COMPONENT_T:
|
|
case SCH_LABEL_T:
|
|
case SCH_GLOBAL_LABEL_T:
|
|
case SCH_HIERARCHICAL_LABEL_T:
|
|
case SCH_TEXT_T:
|
|
case SCH_FIELD_T:
|
|
case SCH_BUS_ENTRY_T:
|
|
case SCH_BITMAP_T:
|
|
cmd.SetId( HK_Descr->m_IdMenuEvent );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_SHEET_PIN_T:
|
|
cmd.SetId( ID_POPUP_SCH_MOVE_SHEET_PIN );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_LINE_T:
|
|
/**
|
|
* @todo Determine why moving bus lines are not handled here.
|
|
*/
|
|
if( ((SCH_ITEM*) aItem )->GetLayer() != LAYER_BUS )
|
|
{
|
|
cmd.SetId( ID_POPUP_SCH_MOVE_ITEM );
|
|
wxPostEvent( this, cmd );
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
;
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case HK_EDIT:
|
|
|
|
if( itemInEdit )
|
|
break;
|
|
|
|
if( aItem == NULL )
|
|
{
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::EditableItems );
|
|
|
|
if( aItem == NULL )
|
|
break;
|
|
}
|
|
|
|
switch( aItem->Type() )
|
|
{
|
|
case SCH_COMPONENT_T:
|
|
EditComponent( (SCH_COMPONENT*) aItem );
|
|
break;
|
|
|
|
case SCH_SHEET_T:
|
|
cmd.SetId( ID_POPUP_SCH_EDIT_SHEET );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_SHEET_PIN_T:
|
|
cmd.SetId( ID_POPUP_SCH_EDIT_SHEET_PIN );
|
|
wxPostEvent( this, cmd );
|
|
break;
|
|
|
|
case SCH_TEXT_T:
|
|
case SCH_LABEL_T:
|
|
case SCH_GLOBAL_LABEL_T:
|
|
case SCH_HIERARCHICAL_LABEL_T:
|
|
EditSchematicText( (SCH_TEXT*) aItem );
|
|
break;
|
|
|
|
case SCH_FIELD_T:
|
|
EditComponentFieldText( (SCH_FIELD*) aItem, aDC );
|
|
break;
|
|
|
|
case SCH_BITMAP_T:
|
|
EditImage( (SCH_BITMAP*) aItem );
|
|
break;
|
|
default:
|
|
;
|
|
}
|
|
|
|
break;
|
|
|
|
case HK_EDIT_COMPONENT_VALUE:
|
|
case HK_EDIT_COMPONENT_FOOTPRINT:
|
|
|
|
if( itemInEdit )
|
|
break;
|
|
|
|
if( aItem == NULL )
|
|
aItem = LocateAndShowItem( aPosition, SCH_COLLECTOR::ComponentsOnly );
|
|
|
|
if( aItem )
|
|
{
|
|
cmd.SetId( HK_Descr->m_IdMenuEvent );
|
|
wxPostEvent( this, cmd );
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Hot keys for the component editor. Some commands are relatives to the item
|
|
* under the mouse cursor
|
|
* Commands are case insensitive
|
|
*/
|
|
void LIB_EDIT_FRAME::OnHotKey( wxDC* aDC, int aHotKey, const wxPoint& aPosition, EDA_ITEM* aItem )
|
|
{
|
|
if( aHotKey == 0 )
|
|
return;
|
|
|
|
wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED );
|
|
|
|
cmd.SetEventObject( this );
|
|
|
|
bool itemInEdit = m_drawItem && m_drawItem->GetFlags();
|
|
|
|
/* Convert lower to upper case (the usual toupper function has problem
|
|
* with non ascii codes like function keys */
|
|
if( (aHotKey >= 'a') && (aHotKey <= 'z') )
|
|
aHotKey += 'A' - 'a';
|
|
|
|
EDA_HOTKEY* HK_Descr = GetDescriptorFromHotkey( aHotKey, s_Common_Hotkey_List );
|
|
|
|
if( HK_Descr == NULL )
|
|
HK_Descr = GetDescriptorFromHotkey( aHotKey, s_LibEdit_Hotkey_List );
|
|
|
|
if( HK_Descr == NULL )
|
|
return;
|
|
|
|
switch( HK_Descr->m_Idcommand )
|
|
{
|
|
default:
|
|
case HK_NOT_FOUND:
|
|
return;
|
|
|
|
case HK_HELP: // Display Current hotkey list
|
|
DisplayHotkeyList( this, s_Libedit_Hokeys_Descr );
|
|
break;
|
|
|
|
case HK_RESET_LOCAL_COORD: /* Reset the relative coord */
|
|
GetScreen()->m_O_Curseur = GetScreen()->GetCrossHairPosition();
|
|
break;
|
|
|
|
case HK_ZOOM_IN:
|
|
cmd.SetId( ID_POPUP_ZOOM_IN );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_OUT:
|
|
cmd.SetId( ID_POPUP_ZOOM_OUT );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_REDRAW:
|
|
cmd.SetId( ID_ZOOM_REDRAW );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_CENTER:
|
|
cmd.SetId( ID_POPUP_ZOOM_CENTER );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_ZOOM_AUTO:
|
|
cmd.SetId( ID_ZOOM_PAGE );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case HK_UNDO:
|
|
if( !itemInEdit )
|
|
{
|
|
cmd.SetId( wxID_UNDO );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
}
|
|
break;
|
|
|
|
case HK_REDO:
|
|
if( !itemInEdit )
|
|
{
|
|
cmd.SetId( wxID_REDO );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
}
|
|
break;
|
|
|
|
case HK_REPEAT_LAST:
|
|
if( m_lastDrawItem && (m_lastDrawItem->GetFlags() == 0)
|
|
&& ( m_lastDrawItem->Type() == LIB_PIN_T ) )
|
|
RepeatPinItem( aDC, (LIB_PIN*) m_lastDrawItem );
|
|
break;
|
|
|
|
case HK_EDIT:
|
|
if( ! itemInEdit )
|
|
m_drawItem = LocateItemUsingCursor( aPosition );
|
|
|
|
if( m_drawItem )
|
|
{
|
|
switch( m_drawItem->Type() )
|
|
{
|
|
case LIB_PIN_T:
|
|
cmd.SetId( ID_LIBEDIT_EDIT_PIN );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case LIB_ARC_T:
|
|
case LIB_CIRCLE_T:
|
|
case LIB_RECTANGLE_T:
|
|
case LIB_POLYLINE_T:
|
|
case LIB_TEXT_T:
|
|
cmd.SetId( ID_POPUP_LIBEDIT_BODY_EDIT_ITEM );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
case LIB_FIELD_T:
|
|
cmd.SetId( ID_POPUP_LIBEDIT_FIELD_EDIT_ITEM );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case HK_ROTATE:
|
|
if( ! itemInEdit )
|
|
m_drawItem = LocateItemUsingCursor( aPosition );
|
|
|
|
if( m_drawItem )
|
|
{
|
|
cmd.SetId( ID_LIBEDIT_ROTATE_ITEM );
|
|
GetEventHandler()->ProcessEvent( cmd );
|
|
}
|
|
break;
|
|
|
|
case HK_LIBEDIT_CREATE_PIN:
|
|
SetToolID( ID_LIBEDIT_PIN_BUTT, wxCURSOR_PENCIL, _( "Add Pin" ) );
|
|
OnLeftClick( aDC, aPosition );
|
|
break;
|
|
|
|
case HK_DELETE:
|
|
if( !itemInEdit )
|
|
m_drawItem = LocateItemUsingCursor( aPosition );
|
|
|
|
if( m_drawItem && !m_drawItem->InEditMode() )
|
|
{
|
|
cmd.SetId( ID_POPUP_LIBEDIT_DELETE_ITEM );
|
|
Process_Special_Functions( cmd );
|
|
}
|
|
break;
|
|
|
|
case HK_LIBEDIT_MOVE_GRAPHIC_ITEM:
|
|
if( !itemInEdit )
|
|
{
|
|
m_drawItem = LocateItemUsingCursor( aPosition );
|
|
|
|
if( m_drawItem )
|
|
{
|
|
cmd.SetId( ID_POPUP_LIBEDIT_MOVE_ITEM_REQUEST );
|
|
Process_Special_Functions( cmd );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case HK_DRAG:
|
|
m_drawItem = LocateItemUsingCursor( aPosition );
|
|
|
|
if( m_drawItem && !m_drawItem->InEditMode() )
|
|
{
|
|
cmd.SetId( ID_POPUP_LIBEDIT_MODIFY_ITEM );
|
|
Process_Special_Functions( cmd );
|
|
}
|
|
break;
|
|
}
|
|
}
|