2013-08-02 14:46:53 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2013 CERN
|
|
|
|
* @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __TOOL_EVENT_H
|
|
|
|
#define __TOOL_EVENT_H
|
|
|
|
|
|
|
|
#include <cstdio>
|
|
|
|
#include <deque>
|
|
|
|
|
|
|
|
#include <math/vector2d.h>
|
|
|
|
|
|
|
|
#include <boost/optional.hpp>
|
|
|
|
|
|
|
|
class TOOL_MANAGER;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Internal (GUI-independent) event definitions.
|
|
|
|
* Enums are mostly self-explanatory.
|
|
|
|
*/
|
2013-08-07 09:20:12 +00:00
|
|
|
enum TOOL_EventCategory
|
|
|
|
{
|
2013-08-21 15:37:27 +00:00
|
|
|
TC_None = 0x00,
|
|
|
|
TC_Mouse = 0x01,
|
|
|
|
TC_Keyboard = 0x02,
|
|
|
|
TC_Command = 0x04,
|
|
|
|
TC_Message = 0x08,
|
|
|
|
TC_View = 0x10,
|
|
|
|
TC_Any = 0xffffffff
|
2013-08-02 14:46:53 +00:00
|
|
|
};
|
|
|
|
|
2013-08-07 09:20:12 +00:00
|
|
|
enum TOOL_Actions
|
|
|
|
{
|
2013-08-21 15:37:27 +00:00
|
|
|
// UI input events
|
|
|
|
TA_None = 0x0000,
|
|
|
|
TA_MouseClick = 0x0001,
|
|
|
|
TA_MouseUp = 0x0002,
|
|
|
|
TA_MouseDown = 0x0004,
|
|
|
|
TA_MouseDrag = 0x0008,
|
|
|
|
TA_MouseMotion = 0x0010,
|
|
|
|
TA_MouseWheel = 0x0020,
|
|
|
|
TA_Mouse = 0x003f,
|
|
|
|
TA_KeyUp = 0x0040,
|
|
|
|
TA_KeyDown = 0x0080,
|
|
|
|
TA_Keyboard = TA_KeyUp | TA_KeyDown,
|
|
|
|
|
|
|
|
// View related events
|
|
|
|
TA_ViewRefresh = 0x0100,
|
|
|
|
TA_ViewZoom = 0x0200,
|
|
|
|
TA_ViewPan = 0x0400,
|
|
|
|
TA_ViewDirty = 0x0800,
|
|
|
|
TA_ChangeLayer = 0x1000,
|
2013-08-02 14:46:53 +00:00
|
|
|
|
|
|
|
// Tool cancel event. Issued automagically when the user hits escape or selects End Tool from the context menu.
|
2013-08-21 15:37:27 +00:00
|
|
|
TA_CancelTool = 0x2000,
|
2013-08-02 14:46:53 +00:00
|
|
|
|
|
|
|
// Tool activation event. Issued by the GUI upon pressing a button/menu selection.
|
|
|
|
TA_ActivateTool = 0x4000,
|
|
|
|
|
|
|
|
// Context menu update. Issued whenever context menu is open and the user hovers the mouse over one of choices.
|
|
|
|
// Used in dynamic highligting in disambiguation menu
|
|
|
|
TA_ContextMenuUpdate = 0x8000,
|
|
|
|
|
|
|
|
// Context menu choice. Sent if the user picked something from the context menu or closed it without selecting anything.
|
|
|
|
TA_ContextMenuChoice = 0x10000,
|
|
|
|
TA_Any = 0xffffffff
|
|
|
|
};
|
|
|
|
|
2013-08-07 09:20:12 +00:00
|
|
|
enum TOOL_MouseButtons
|
|
|
|
{
|
2013-08-21 15:37:27 +00:00
|
|
|
MB_None = 0x0,
|
|
|
|
MB_Left = 0x1,
|
|
|
|
MB_Right = 0x2,
|
|
|
|
MB_Middle = 0x4,
|
|
|
|
MB_ButtonMask = MB_Left | MB_Right | MB_Middle,
|
|
|
|
MB_Any = 0xffffffff
|
2013-08-02 14:46:53 +00:00
|
|
|
};
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
enum TOOL_Modifiers
|
|
|
|
{
|
|
|
|
MD_ModShift = 0x1000,
|
|
|
|
MD_ModCtrl = 0x2000,
|
|
|
|
MD_ModAlt = 0x4000,
|
|
|
|
MD_ModifierMask = MD_ModShift | MD_ModCtrl | MD_ModAlt,
|
|
|
|
};
|
2013-08-02 14:46:53 +00:00
|
|
|
|
|
|
|
// Defines when a context menu is opened.
|
2013-09-02 12:21:12 +00:00
|
|
|
enum CONTEXT_MENU_TRIGGER
|
2013-08-07 09:20:12 +00:00
|
|
|
{
|
2013-08-21 15:37:27 +00:00
|
|
|
CMENU_BUTTON = 0, // On the right button
|
|
|
|
CMENU_NOW, // Right now (after TOOL_INTERACTIVE::SetContextMenu)
|
|
|
|
CMENU_OFF // Never
|
2013-08-02 14:46:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class TOOL_EVENT
|
|
|
|
*
|
|
|
|
* Generic, UI-independent tool event.
|
|
|
|
*/
|
|
|
|
class TOOL_EVENT
|
|
|
|
{
|
2013-08-07 09:20:12 +00:00
|
|
|
public:
|
|
|
|
const std::string Format() const;
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
TOOL_EVENT( TOOL_EventCategory aCategory = TC_None, TOOL_Actions aAction = TA_None ) :
|
2013-08-07 09:20:12 +00:00
|
|
|
m_category( aCategory ),
|
|
|
|
m_actions( aAction ),
|
2013-08-21 15:37:27 +00:00
|
|
|
m_mouseButtons( 0 ),
|
|
|
|
m_keyCode( 0 ),
|
|
|
|
m_modifiers( 0 ) {}
|
2013-08-07 09:20:12 +00:00
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
TOOL_EVENT( TOOL_EventCategory aCategory, TOOL_Actions aAction, int aExtraParam ) :
|
2013-08-07 09:20:12 +00:00
|
|
|
m_category( aCategory ),
|
|
|
|
m_actions( aAction )
|
|
|
|
{
|
|
|
|
if( aCategory == TC_Mouse )
|
2013-08-21 15:37:27 +00:00
|
|
|
{
|
|
|
|
m_mouseButtons = aExtraParam & MB_ButtonMask;
|
|
|
|
}
|
|
|
|
else if( aCategory == TC_Keyboard )
|
|
|
|
{
|
|
|
|
m_keyCode = aExtraParam & ~MD_ModifierMask; // Filter out modifiers
|
|
|
|
}
|
2013-08-07 09:20:12 +00:00
|
|
|
else if ( aCategory == TC_Command )
|
2013-08-21 15:37:27 +00:00
|
|
|
{
|
2013-08-07 09:20:12 +00:00
|
|
|
m_commandId = aExtraParam;
|
2013-08-21 15:37:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if( aCategory & ( TC_Mouse | TC_Keyboard ) )
|
|
|
|
{
|
|
|
|
m_modifiers = aExtraParam & MD_ModifierMask;
|
|
|
|
}
|
2013-08-07 09:20:12 +00:00
|
|
|
}
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
TOOL_EVENT( TOOL_EventCategory aCategory, TOOL_Actions aAction,
|
|
|
|
const std::string& aExtraParam ) :
|
2013-08-07 09:20:12 +00:00
|
|
|
m_category( aCategory ),
|
|
|
|
m_actions( aAction ),
|
|
|
|
m_mouseButtons( 0 )
|
|
|
|
{
|
|
|
|
if( aCategory == TC_Command )
|
|
|
|
m_commandStr = aExtraParam;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TOOL_EventCategory Category() const
|
|
|
|
{
|
|
|
|
return m_category;
|
|
|
|
}
|
|
|
|
|
|
|
|
TOOL_Actions Action() const
|
|
|
|
{
|
|
|
|
return m_actions;
|
|
|
|
}
|
|
|
|
|
|
|
|
const VECTOR2D Delta() const
|
|
|
|
{
|
|
|
|
return m_mouseDelta;
|
|
|
|
}
|
|
|
|
|
|
|
|
const VECTOR2D& Position() const
|
|
|
|
{
|
|
|
|
return m_mousePos;
|
|
|
|
}
|
|
|
|
|
|
|
|
const VECTOR2D& DragOrigin() const
|
|
|
|
{
|
|
|
|
return m_mouseDragOrigin;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Buttons() const
|
|
|
|
{
|
|
|
|
return m_mouseButtons;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsClick( int aButtonMask = MB_Any ) const
|
|
|
|
{
|
|
|
|
return ( m_actions == TA_MouseClick )
|
|
|
|
&& ( ( m_mouseButtons & aButtonMask ) == aButtonMask );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsDrag( int aButtonMask = MB_Any ) const
|
|
|
|
{
|
|
|
|
return ( m_actions == TA_MouseDrag ) && ( ( m_mouseButtons & aButtonMask ) == aButtonMask );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsMouseUp( int aButtonMask = MB_Any ) const
|
|
|
|
{
|
|
|
|
return ( m_actions == TA_MouseUp ) && ( ( m_mouseButtons & aButtonMask ) == aButtonMask );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsMotion() const
|
|
|
|
{
|
|
|
|
return ( m_actions == TA_MouseMotion );
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsCancel() const
|
|
|
|
{
|
|
|
|
return m_actions == TA_CancelTool;
|
|
|
|
}
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
bool Modifier( int aMask = MD_ModifierMask ) const
|
|
|
|
{
|
|
|
|
return ( m_modifiers & aMask );
|
|
|
|
}
|
|
|
|
|
|
|
|
int KeyCode() const
|
2013-08-07 09:20:12 +00:00
|
|
|
{
|
2013-08-21 15:37:27 +00:00
|
|
|
return m_keyCode;
|
2013-08-07 09:20:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Ignore();
|
|
|
|
|
|
|
|
void SetMouseDragOrigin( const VECTOR2D &aP )
|
|
|
|
{
|
|
|
|
m_mouseDragOrigin = aP;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetMousePosition( const VECTOR2D& aP )
|
|
|
|
{
|
|
|
|
m_mousePos = aP;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetMouseDelta( const VECTOR2D& aP )
|
|
|
|
{
|
|
|
|
m_mouseDelta = aP;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Matches( const TOOL_EVENT& aEvent ) const
|
|
|
|
{
|
2013-08-19 07:47:36 +00:00
|
|
|
if( !( m_category & aEvent.m_category ) )
|
2013-08-07 09:20:12 +00:00
|
|
|
return false;
|
|
|
|
|
2013-08-19 07:47:36 +00:00
|
|
|
if( !( m_actions & aEvent.m_actions ) )
|
2013-08-07 09:20:12 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if( m_category == TC_Command )
|
|
|
|
{
|
|
|
|
if( m_commandStr && aEvent.m_commandStr )
|
|
|
|
return ( *m_commandStr == *aEvent.m_commandStr );
|
|
|
|
if( m_commandId && aEvent.m_commandId )
|
|
|
|
return ( *m_commandId == *aEvent.m_commandId );
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
boost::optional<int> GetCommandId()
|
|
|
|
{
|
|
|
|
return m_commandId;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
friend class TOOL_MANAGER;
|
|
|
|
|
|
|
|
TOOL_EventCategory m_category;
|
|
|
|
TOOL_Actions m_actions;
|
|
|
|
|
|
|
|
VECTOR2D m_mouseDelta;
|
|
|
|
VECTOR2D m_mousePos;
|
|
|
|
VECTOR2D m_mouseDragOrigin;
|
|
|
|
|
|
|
|
int m_mouseButtons;
|
2013-08-21 15:37:27 +00:00
|
|
|
int m_keyCode;
|
|
|
|
int m_modifiers;
|
2013-08-07 09:20:12 +00:00
|
|
|
boost::optional<int> m_commandId;
|
|
|
|
boost::optional<std::string> m_commandStr;
|
2013-08-02 14:46:53 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef boost::optional<TOOL_EVENT> OPT_TOOL_EVENT;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class TOOL_EVENT_LIST
|
|
|
|
*
|
|
|
|
* A list of TOOL_EVENTs, with overloaded || operators allowing for
|
|
|
|
* concatenating TOOL_EVENTs with little code.
|
|
|
|
*/
|
|
|
|
class TOOL_EVENT_LIST {
|
2013-08-07 09:20:12 +00:00
|
|
|
public:
|
|
|
|
typedef TOOL_EVENT value_type;
|
|
|
|
typedef std::deque<TOOL_EVENT>::iterator iterator;
|
|
|
|
typedef std::deque<TOOL_EVENT>::const_iterator const_iterator;
|
|
|
|
|
|
|
|
TOOL_EVENT_LIST() {};
|
|
|
|
TOOL_EVENT_LIST( const TOOL_EVENT& aSingleEvent )
|
|
|
|
{
|
|
|
|
m_events.push_back(aSingleEvent);
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string Format() const;
|
|
|
|
|
|
|
|
boost::optional<const TOOL_EVENT&> Matches( const TOOL_EVENT &b ) const
|
|
|
|
{
|
|
|
|
for( const_iterator i = m_events.begin(); i != m_events.end(); ++i )
|
|
|
|
if ( i->Matches( b ) )
|
|
|
|
return *i;
|
|
|
|
return boost::optional<const TOOL_EVENT&>();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Add( const TOOL_EVENT& aEvent )
|
|
|
|
{
|
|
|
|
m_events.push_back( aEvent );
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator begin()
|
|
|
|
{
|
|
|
|
return m_events.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator end()
|
|
|
|
{
|
|
|
|
return m_events.end();
|
|
|
|
}
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
const_iterator cbegin() const
|
2013-08-07 09:20:12 +00:00
|
|
|
{
|
|
|
|
return m_events.begin();
|
|
|
|
}
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
const_iterator cend() const
|
2013-08-07 09:20:12 +00:00
|
|
|
{
|
|
|
|
return m_events.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
int size() const
|
|
|
|
{
|
|
|
|
return m_events.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
m_events.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
TOOL_EVENT_LIST& operator=( const TOOL_EVENT_LIST& b )
|
|
|
|
{
|
|
|
|
m_events.clear();
|
|
|
|
|
|
|
|
for( std::deque<TOOL_EVENT>::const_iterator i = b.m_events.begin();
|
|
|
|
i != b.m_events.end(); ++i )
|
|
|
|
{
|
|
|
|
m_events.push_back(*i);
|
|
|
|
}
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
TOOL_EVENT_LIST& operator=( const TOOL_EVENT& b )
|
|
|
|
{
|
|
|
|
m_events.clear();
|
|
|
|
m_events.push_back( b );
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
TOOL_EVENT_LIST& operator||( const TOOL_EVENT& b )
|
|
|
|
{
|
|
|
|
Add( b );
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
TOOL_EVENT_LIST& operator||( const TOOL_EVENT_LIST& b )
|
|
|
|
{
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::deque<TOOL_EVENT> m_events;
|
2013-08-02 14:46:53 +00:00
|
|
|
};
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
inline const TOOL_EVENT_LIST operator||( const TOOL_EVENT& a, const TOOL_EVENT& b )
|
2013-08-02 14:46:53 +00:00
|
|
|
{
|
|
|
|
TOOL_EVENT_LIST l;
|
|
|
|
|
2013-08-07 09:20:12 +00:00
|
|
|
l.Add( a );
|
|
|
|
l.Add( b );
|
2013-08-02 14:46:53 +00:00
|
|
|
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
2013-08-21 15:37:27 +00:00
|
|
|
inline const TOOL_EVENT_LIST operator||( const TOOL_EVENT& a, const TOOL_EVENT_LIST& b )
|
2013-08-02 14:46:53 +00:00
|
|
|
{
|
2013-08-07 09:20:12 +00:00
|
|
|
TOOL_EVENT_LIST l( b );
|
2013-08-02 14:46:53 +00:00
|
|
|
|
2013-08-07 09:20:12 +00:00
|
|
|
l.Add( a );
|
2013-08-02 14:46:53 +00:00
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|