Break out tools' common menu functions and ownership
Add a class TOOL_MENU, which provides a management class for a CONDITIONAL_MENU and a set of CONTEXT_MENUs. The aim of this is to provide a central place where all TOOL_INTERACTIVEs can get a "basic" context menu that either they or other tools can register new items and sub-menus against. This means that "top-level" tools no longer need to manage the lifetimes of any CONTEXT_MENUs that they add, and can also delegate simple menu display functions.
This commit is contained in:
parent
3d7a509046
commit
d7db84e282
|
@ -294,6 +294,7 @@ set( PCBNEW_CLASS_SRCS
|
||||||
tools/picker_tool.cpp
|
tools/picker_tool.cpp
|
||||||
tools/zoom_tool.cpp
|
tools/zoom_tool.cpp
|
||||||
tools/tools_common.cpp
|
tools/tools_common.cpp
|
||||||
|
tools/tool_menu.cpp
|
||||||
|
|
||||||
tools/grid_menu.cpp
|
tools/grid_menu.cpp
|
||||||
tools/zoom_menu.cpp
|
tools/zoom_menu.cpp
|
||||||
|
|
|
@ -43,6 +43,7 @@
|
||||||
#include <router/direction.h>
|
#include <router/direction.h>
|
||||||
#include <ratsnest_data.h>
|
#include <ratsnest_data.h>
|
||||||
#include <board_commit.h>
|
#include <board_commit.h>
|
||||||
|
#include <scoped_set_reset.h>
|
||||||
|
|
||||||
#include <class_board.h>
|
#include <class_board.h>
|
||||||
#include <class_edge_mod.h>
|
#include <class_edge_mod.h>
|
||||||
|
@ -51,11 +52,6 @@
|
||||||
#include <class_zone.h>
|
#include <class_zone.h>
|
||||||
#include <class_module.h>
|
#include <class_module.h>
|
||||||
|
|
||||||
#include <scoped_set_reset.h>
|
|
||||||
|
|
||||||
#include "zoom_menu.h"
|
|
||||||
#include "grid_menu.h"
|
|
||||||
|
|
||||||
#include <tools/selection_tool.h>
|
#include <tools/selection_tool.h>
|
||||||
|
|
||||||
|
|
||||||
|
@ -66,8 +62,7 @@ DRAWING_TOOL::DRAWING_TOOL() :
|
||||||
m_view( nullptr ), m_controls( nullptr ),
|
m_view( nullptr ), m_controls( nullptr ),
|
||||||
m_board( nullptr ), m_frame( nullptr ), m_mode( MODE::NONE ),
|
m_board( nullptr ), m_frame( nullptr ), m_mode( MODE::NONE ),
|
||||||
m_lineWidth( 1 ),
|
m_lineWidth( 1 ),
|
||||||
m_menu( this ), m_contextMenu( nullptr ),
|
m_menu( *this )
|
||||||
m_gridMenu( nullptr), m_zoomMenu( nullptr)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -80,19 +75,7 @@ DRAWING_TOOL::~DRAWING_TOOL()
|
||||||
bool DRAWING_TOOL::Init()
|
bool DRAWING_TOOL::Init()
|
||||||
{
|
{
|
||||||
// Drawing type-specific options will be added by the PCB control tool
|
// Drawing type-specific options will be added by the PCB control tool
|
||||||
|
m_menu.AddStandardSubMenus( *getEditFrame<PCB_BASE_FRAME>() );
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomCenter, SELECTION_CONDITIONS::ShowAlways, 1000 );
|
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomIn, SELECTION_CONDITIONS::ShowAlways, 1000 );
|
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomOut , SELECTION_CONDITIONS::ShowAlways, 1000 );
|
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomFitScreen , SELECTION_CONDITIONS::ShowAlways, 1000 );
|
|
||||||
|
|
||||||
PCB_BASE_FRAME* frame = getEditFrame<PCB_BASE_FRAME>();
|
|
||||||
|
|
||||||
m_zoomMenu = new ZOOM_MENU( frame );
|
|
||||||
m_menu.AddMenu( m_zoomMenu, _( "Zoom" ), false, SELECTION_CONDITIONS::ShowAlways, 1000 );
|
|
||||||
|
|
||||||
m_gridMenu = new GRID_MENU( frame );
|
|
||||||
m_menu.AddMenu( m_gridMenu, _( "Grid" ), false, SELECTION_CONDITIONS::ShowAlways, 1000 );
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -268,7 +251,7 @@ int DRAWING_TOOL::PlaceText( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
else if ( evt->IsClick( BUT_RIGHT ) )
|
else if ( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
else if( evt->IsClick( BUT_LEFT ) )
|
else if( evt->IsClick( BUT_LEFT ) )
|
||||||
|
@ -442,7 +425,7 @@ int DRAWING_TOOL::DrawDimension( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
else if ( evt->IsClick( BUT_RIGHT ) )
|
else if ( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
else if( evt->IsClick( BUT_LEFT ) )
|
else if( evt->IsClick( BUT_LEFT ) )
|
||||||
|
@ -658,7 +641,7 @@ int DRAWING_TOOL::PlaceDXF( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
else if ( evt->IsClick( BUT_RIGHT ) )
|
else if ( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
else if( evt->IsClick( BUT_LEFT ) )
|
else if( evt->IsClick( BUT_LEFT ) )
|
||||||
|
@ -790,7 +773,7 @@ int DRAWING_TOOL::SetAnchor( const TOOL_EVENT& aEvent )
|
||||||
}
|
}
|
||||||
else if ( evt->IsClick( BUT_RIGHT ) )
|
else if ( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
else if( evt->IsCancel() || evt->IsActivate() )
|
else if( evt->IsCancel() || evt->IsActivate() )
|
||||||
break;
|
break;
|
||||||
|
@ -807,17 +790,6 @@ int DRAWING_TOOL::SetAnchor( const TOOL_EVENT& aEvent )
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void DRAWING_TOOL::showContextMenu()
|
|
||||||
{
|
|
||||||
// Dummy selection - the drawing tool doesn't depend on a selection
|
|
||||||
SELECTION aSelection;
|
|
||||||
m_contextMenu = m_menu.Generate( aSelection );
|
|
||||||
|
|
||||||
if( m_contextMenu->GetMenuItemCount() > 0 )
|
|
||||||
SetContextMenu( m_contextMenu, CMENU_NOW );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool DRAWING_TOOL::drawSegment( int aShape, DRAWSEGMENT*& aGraphic,
|
bool DRAWING_TOOL::drawSegment( int aShape, DRAWSEGMENT*& aGraphic,
|
||||||
boost::optional<VECTOR2D> aStartingPoint )
|
boost::optional<VECTOR2D> aStartingPoint )
|
||||||
{
|
{
|
||||||
|
@ -898,7 +870,7 @@ bool DRAWING_TOOL::drawSegment( int aShape, DRAWSEGMENT*& aGraphic,
|
||||||
}
|
}
|
||||||
else if( evt->IsClick( BUT_RIGHT ) )
|
else if( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
|
else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
|
||||||
{
|
{
|
||||||
|
@ -1047,7 +1019,7 @@ bool DRAWING_TOOL::drawArc( DRAWSEGMENT*& aGraphic )
|
||||||
}
|
}
|
||||||
else if( evt->IsClick( BUT_RIGHT ) )
|
else if( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
else if( evt->IsClick( BUT_LEFT ) )
|
else if( evt->IsClick( BUT_LEFT ) )
|
||||||
{
|
{
|
||||||
|
@ -1260,7 +1232,7 @@ int DRAWING_TOOL::drawZone( bool aKeepout )
|
||||||
}
|
}
|
||||||
else if( evt->IsClick( BUT_RIGHT ) )
|
else if( evt->IsClick( BUT_RIGHT ) )
|
||||||
{
|
{
|
||||||
showContextMenu();
|
m_menu.ShowContextMenu();
|
||||||
}
|
}
|
||||||
else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
|
else if( evt->IsClick( BUT_LEFT ) || evt->IsDblClick( BUT_LEFT ) )
|
||||||
{
|
{
|
||||||
|
|
|
@ -28,7 +28,7 @@
|
||||||
#include <tools/pcb_tool.h>
|
#include <tools/pcb_tool.h>
|
||||||
#include <boost/optional.hpp>
|
#include <boost/optional.hpp>
|
||||||
|
|
||||||
#include "conditional_menu.h"
|
#include "tool_menu.h"
|
||||||
|
|
||||||
namespace KIGFX
|
namespace KIGFX
|
||||||
{
|
{
|
||||||
|
@ -38,9 +38,6 @@ namespace KIGFX
|
||||||
class BOARD;
|
class BOARD;
|
||||||
class PCB_BASE_EDIT_FRAME;
|
class PCB_BASE_EDIT_FRAME;
|
||||||
class DRAWSEGMENT;
|
class DRAWSEGMENT;
|
||||||
class CONTEXT_MENU;
|
|
||||||
class GRID_MENU;
|
|
||||||
class ZOOM_MENU;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class DRAWING_TOOL
|
* Class DRAWING_TOOL
|
||||||
|
@ -63,7 +60,7 @@ public:
|
||||||
///> Get the DRAWING_TOOL top-level context menu
|
///> Get the DRAWING_TOOL top-level context menu
|
||||||
inline CONDITIONAL_MENU& GetMenu()
|
inline CONDITIONAL_MENU& GetMenu()
|
||||||
{
|
{
|
||||||
return m_menu;
|
return m_menu.GetMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
///> The possible drawing modes of DRAWING_TOOL
|
///> The possible drawing modes of DRAWING_TOOL
|
||||||
|
@ -208,13 +205,8 @@ private:
|
||||||
/// Stores the current line width for multisegment drawing.
|
/// Stores the current line width for multisegment drawing.
|
||||||
unsigned int m_lineWidth;
|
unsigned int m_lineWidth;
|
||||||
|
|
||||||
/// Menu displayed by the tool.
|
/// Menu model displayed by the tool.
|
||||||
CONDITIONAL_MENU m_menu;
|
TOOL_MENU m_menu;
|
||||||
|
|
||||||
/// Pointers to context menus
|
|
||||||
CONTEXT_MENU* m_contextMenu;
|
|
||||||
GRID_MENU* m_gridMenu;
|
|
||||||
ZOOM_MENU* m_zoomMenu;
|
|
||||||
|
|
||||||
// How does line width change after one -/+ key press.
|
// How does line width change after one -/+ key press.
|
||||||
static const int WIDTH_STEP;
|
static const int WIDTH_STEP;
|
||||||
|
|
|
@ -50,8 +50,6 @@ using namespace std::placeholders;
|
||||||
|
|
||||||
#include "selection_tool.h"
|
#include "selection_tool.h"
|
||||||
#include "selection_area.h"
|
#include "selection_area.h"
|
||||||
#include "zoom_menu.h"
|
|
||||||
#include "grid_menu.h"
|
|
||||||
#include "bright_box.h"
|
#include "bright_box.h"
|
||||||
#include "common_actions.h"
|
#include "common_actions.h"
|
||||||
|
|
||||||
|
@ -70,8 +68,7 @@ public:
|
||||||
SELECTION_TOOL::SELECTION_TOOL() :
|
SELECTION_TOOL::SELECTION_TOOL() :
|
||||||
PCB_TOOL( "pcbnew.InteractiveSelection" ),
|
PCB_TOOL( "pcbnew.InteractiveSelection" ),
|
||||||
m_frame( NULL ), m_additive( false ), m_multiple( false ),
|
m_frame( NULL ), m_additive( false ), m_multiple( false ),
|
||||||
m_locked( true ), m_menu( this ), m_contextMenu( NULL ), m_selectMenu( NULL ),
|
m_locked( true ), m_menu( *this )
|
||||||
m_zoomMenu( NULL ), m_gridMenu( NULL )
|
|
||||||
{
|
{
|
||||||
// Do not leave uninitialized members:
|
// Do not leave uninitialized members:
|
||||||
m_preliminary = false;
|
m_preliminary = false;
|
||||||
|
@ -81,11 +78,6 @@ SELECTION_TOOL::SELECTION_TOOL() :
|
||||||
SELECTION_TOOL::~SELECTION_TOOL()
|
SELECTION_TOOL::~SELECTION_TOOL()
|
||||||
{
|
{
|
||||||
getView()->Remove( &m_selection );
|
getView()->Remove( &m_selection );
|
||||||
|
|
||||||
delete m_contextMenu;
|
|
||||||
delete m_selectMenu;
|
|
||||||
delete m_zoomMenu;
|
|
||||||
delete m_gridMenu;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -96,25 +88,17 @@ bool SELECTION_TOOL::Init()
|
||||||
auto showSelectMenuFunctor = ( S_C::OnlyType( PCB_VIA_T ) || S_C::OnlyType( PCB_TRACE_T ) )
|
auto showSelectMenuFunctor = ( S_C::OnlyType( PCB_VIA_T ) || S_C::OnlyType( PCB_TRACE_T ) )
|
||||||
&& S_C::Count( 1 );
|
&& S_C::Count( 1 );
|
||||||
|
|
||||||
m_selectMenu = new SELECT_MENU;
|
auto selectMenu = std::make_shared<SELECT_MENU>();
|
||||||
m_selectMenu->SetTool( this );
|
selectMenu->SetTool( this );
|
||||||
|
m_menu.AddSubMenu( selectMenu );
|
||||||
|
|
||||||
m_menu.AddMenu( m_selectMenu, _( "Select..." ), false, showSelectMenuFunctor );
|
auto& menu = m_menu.GetMenu();
|
||||||
|
|
||||||
|
menu.AddMenu( selectMenu.get(), _( "Select..." ), false, showSelectMenuFunctor );
|
||||||
// only show separator if there is a Select menu to show above it
|
// only show separator if there is a Select menu to show above it
|
||||||
GetMenu().AddSeparator( showSelectMenuFunctor, 1000 );
|
menu.AddSeparator( showSelectMenuFunctor, 1000 );
|
||||||
|
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomCenter, S_C::ShowAlways, 1000 );
|
m_menu.AddStandardSubMenus( *getEditFrame<PCB_BASE_FRAME>() );
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomIn, S_C::ShowAlways, 1000 );
|
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomOut , S_C::ShowAlways, 1000 );
|
|
||||||
m_menu.AddItem( COMMON_ACTIONS::zoomFitScreen , S_C::ShowAlways, 1000 );
|
|
||||||
|
|
||||||
PCB_BASE_FRAME* frame = getEditFrame<PCB_BASE_FRAME>();
|
|
||||||
|
|
||||||
m_zoomMenu = new ZOOM_MENU( frame );
|
|
||||||
m_menu.AddMenu( m_zoomMenu, _( "Zoom" ), false, S_C::ShowAlways, 1000 );
|
|
||||||
|
|
||||||
m_gridMenu = new GRID_MENU( frame );
|
|
||||||
m_menu.AddMenu( m_gridMenu, _( "Grid" ), false, S_C::ShowAlways, 1000 );
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -177,11 +161,7 @@ int SELECTION_TOOL::Main( const TOOL_EVENT& aEvent )
|
||||||
if( emptySelection )
|
if( emptySelection )
|
||||||
selectPoint( evt->Position() );
|
selectPoint( evt->Position() );
|
||||||
|
|
||||||
delete m_contextMenu;
|
m_menu.ShowContextMenu( m_selection );
|
||||||
m_contextMenu = m_menu.Generate( m_selection );
|
|
||||||
|
|
||||||
if( m_contextMenu->GetMenuItemCount() > 0 )
|
|
||||||
SetContextMenu( m_contextMenu, CMENU_NOW );
|
|
||||||
|
|
||||||
m_preliminary = emptySelection;
|
m_preliminary = emptySelection;
|
||||||
}
|
}
|
||||||
|
@ -288,11 +268,7 @@ int SELECTION_TOOL::Main( const TOOL_EVENT& aEvent )
|
||||||
if( m_preliminary )
|
if( m_preliminary )
|
||||||
clearSelection();
|
clearSelection();
|
||||||
|
|
||||||
if( evt->Parameter<CONTEXT_MENU*>() == m_contextMenu )
|
m_menu.CloseContextMenu( evt );
|
||||||
{
|
|
||||||
delete m_contextMenu;
|
|
||||||
m_contextMenu = NULL;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,15 +33,12 @@
|
||||||
#include <tool/context_menu.h>
|
#include <tool/context_menu.h>
|
||||||
|
|
||||||
#include "selection_conditions.h"
|
#include "selection_conditions.h"
|
||||||
#include "conditional_menu.h"
|
#include "tool_menu.h"
|
||||||
|
|
||||||
class PCB_BASE_FRAME;
|
class PCB_BASE_FRAME;
|
||||||
class SELECTION_AREA;
|
class SELECTION_AREA;
|
||||||
class BOARD_ITEM;
|
class BOARD_ITEM;
|
||||||
class GENERAL_COLLECTOR;
|
class GENERAL_COLLECTOR;
|
||||||
class SELECT_MENU;
|
|
||||||
class ZOOM_MENU;
|
|
||||||
class GRID_MENU;
|
|
||||||
|
|
||||||
namespace KIGFX
|
namespace KIGFX
|
||||||
{
|
{
|
||||||
|
@ -198,7 +195,7 @@ public:
|
||||||
|
|
||||||
inline CONDITIONAL_MENU& GetMenu()
|
inline CONDITIONAL_MENU& GetMenu()
|
||||||
{
|
{
|
||||||
return m_menu;
|
return m_menu.GetMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
///> Checks if the user has agreed to modify locked items for the given selection.
|
///> Checks if the user has agreed to modify locked items for the given selection.
|
||||||
|
@ -392,14 +389,8 @@ private:
|
||||||
/// Determines if the selection is preliminary or final.
|
/// Determines if the selection is preliminary or final.
|
||||||
bool m_preliminary;
|
bool m_preliminary;
|
||||||
|
|
||||||
/// Menu displayed by the tool.
|
/// Menu model displayed by the tool.
|
||||||
CONDITIONAL_MENU m_menu;
|
TOOL_MENU m_menu;
|
||||||
|
|
||||||
/// Pointers to context menus
|
|
||||||
CONTEXT_MENU* m_contextMenu;
|
|
||||||
SELECT_MENU* m_selectMenu;
|
|
||||||
ZOOM_MENU* m_zoomMenu;
|
|
||||||
GRID_MENU* m_gridMenu;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -0,0 +1,104 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 KiCad Developers, see CHANGELOG.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
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "tool_menu.h"
|
||||||
|
|
||||||
|
#include <tool/context_menu.h>
|
||||||
|
|
||||||
|
#include "common_actions.h"
|
||||||
|
#include "zoom_menu.h"
|
||||||
|
#include "grid_menu.h"
|
||||||
|
#include "selection_tool.h" // For SELECTION
|
||||||
|
|
||||||
|
|
||||||
|
TOOL_MENU::TOOL_MENU( TOOL_INTERACTIVE& aTool ) :
|
||||||
|
m_menu( &aTool ),
|
||||||
|
m_tool( aTool )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
TOOL_MENU::~TOOL_MENU()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
CONDITIONAL_MENU& TOOL_MENU::GetMenu()
|
||||||
|
{
|
||||||
|
return m_menu;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void TOOL_MENU::AddSubMenu( std::shared_ptr<CONTEXT_MENU> aSubMenu )
|
||||||
|
{
|
||||||
|
// store a copy of the menu (keeps a reference)
|
||||||
|
m_subMenus.push_back( std::move( aSubMenu ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void TOOL_MENU::ShowContextMenu( SELECTION& aSelection )
|
||||||
|
{
|
||||||
|
m_contextMenu = std::unique_ptr<CONTEXT_MENU>(
|
||||||
|
m_menu.Generate( aSelection ) );
|
||||||
|
|
||||||
|
if( m_contextMenu->GetMenuItemCount() > 0 )
|
||||||
|
{
|
||||||
|
m_tool.SetContextMenu( m_contextMenu.get(), CMENU_NOW );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void TOOL_MENU::ShowContextMenu()
|
||||||
|
{
|
||||||
|
SELECTION dummySelection;
|
||||||
|
|
||||||
|
ShowContextMenu( dummySelection );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void TOOL_MENU::CloseContextMenu( OPT_TOOL_EVENT& evt )
|
||||||
|
{
|
||||||
|
// m_contextMenu can be null here, that's OK
|
||||||
|
if( evt->Parameter<CONTEXT_MENU*>() == m_contextMenu.get() )
|
||||||
|
{
|
||||||
|
m_contextMenu = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// This makes the factory functions a bit less verbose
|
||||||
|
using S_C = SELECTION_CONDITIONS;
|
||||||
|
|
||||||
|
void TOOL_MENU::AddStandardSubMenus( EDA_DRAW_FRAME& aFrame )
|
||||||
|
{
|
||||||
|
m_menu.AddItem( COMMON_ACTIONS::zoomCenter, S_C::ShowAlways, 1000 );
|
||||||
|
m_menu.AddItem( COMMON_ACTIONS::zoomIn, S_C::ShowAlways, 1000 );
|
||||||
|
m_menu.AddItem( COMMON_ACTIONS::zoomOut, S_C::ShowAlways, 1000 );
|
||||||
|
m_menu.AddItem( COMMON_ACTIONS::zoomFitScreen, S_C::ShowAlways, 1000 );
|
||||||
|
|
||||||
|
m_menu.AddMenu( createOwnSubMenu<ZOOM_MENU>( &aFrame ).get(),
|
||||||
|
_( "Zoom" ), false, S_C::ShowAlways, 1000 );
|
||||||
|
|
||||||
|
m_menu.AddMenu( createOwnSubMenu<GRID_MENU>( &aFrame ).get(),
|
||||||
|
_( "Grid" ), false, S_C::ShowAlways, 1000 );
|
||||||
|
}
|
|
@ -0,0 +1,173 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 KiCad Developers, see CHANGELOG.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
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef TOOLS_TOOL_MENU__H_
|
||||||
|
#define TOOLS_TOOL_MENU__H_
|
||||||
|
|
||||||
|
#include "conditional_menu.h"
|
||||||
|
#include "pcb_tool.h"
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
class CONTEXT_MENU;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class TOOL_MENU
|
||||||
|
*
|
||||||
|
* Manages a CONDITIONAL_MENU and some number of
|
||||||
|
* CONTEXT_MENUs as sub-menus
|
||||||
|
*
|
||||||
|
* Each "top-level" interactive tool can have one of these,
|
||||||
|
* and other tools can contribute CONTEXT_MENUS to it.
|
||||||
|
*
|
||||||
|
* There are also helper functions for adding common sets of
|
||||||
|
* menu items, for example zoom and grid controls.
|
||||||
|
*/
|
||||||
|
class TOOL_MENU
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function TOOL_MENU
|
||||||
|
*
|
||||||
|
* Construct a new TOOL_MENU for a specific tool. This menu
|
||||||
|
* will be empty - it's up to the caller to add the relevant
|
||||||
|
* items. This can be done directy, using the reference returned
|
||||||
|
* by TOOL_MENU::GetMenu(), or the helpers for common command sets
|
||||||
|
* can be used, or a combination of the two.
|
||||||
|
*/
|
||||||
|
TOOL_MENU( TOOL_INTERACTIVE& aTool );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Destructor, also destructs any submenus created with
|
||||||
|
* TOOL_MENU::CreateSubMenu().
|
||||||
|
*/
|
||||||
|
~TOOL_MENU();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function GetMenu
|
||||||
|
*
|
||||||
|
* @return reference to the CONDITIONAL_MENU model, which can be
|
||||||
|
* used tby tools to add their own commands to the menu.
|
||||||
|
*/
|
||||||
|
CONDITIONAL_MENU& GetMenu();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function CreateSubMenu
|
||||||
|
*
|
||||||
|
* Store a submenu of this menu model. This can be shared with
|
||||||
|
* other menu models.
|
||||||
|
*
|
||||||
|
* It is the callers responsibility to add the submenu to
|
||||||
|
* m_menu (via GetMenu() ) in the right way, as well
|
||||||
|
* as to set the tool with SetTool(), since it's not a given
|
||||||
|
* that the menu's tool is the tool that directly owns this
|
||||||
|
* TOOL_MENU
|
||||||
|
*
|
||||||
|
* @param aSubMenu: a sub menu to add
|
||||||
|
*/
|
||||||
|
void AddSubMenu( std::shared_ptr<CONTEXT_MENU> aSubMenu );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function ShowContextMenu
|
||||||
|
*
|
||||||
|
* Helper function to set and immediately show a CONTEXT_MENU
|
||||||
|
* based on the internal CONDITIONAL_MENU in concert with
|
||||||
|
* the given SELECTION
|
||||||
|
*
|
||||||
|
* You don't have to use this function, if the caller has a
|
||||||
|
* different way to show the menu, it can create one from
|
||||||
|
* the reference returned by TOOL_MENU::GetMenu(), but it will
|
||||||
|
* have to be managed externally to this class.
|
||||||
|
*/
|
||||||
|
void ShowContextMenu( SELECTION& aSelection );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function ShowContextMenu
|
||||||
|
*
|
||||||
|
* Helper function to show a context menu without any selection
|
||||||
|
* for tools that can't make selections.
|
||||||
|
*/
|
||||||
|
void ShowContextMenu();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function CloseContextMenu
|
||||||
|
*
|
||||||
|
* Helper function to close a menu previously opened with
|
||||||
|
* ShowContextMenu(), if a suitable event is received
|
||||||
|
*/
|
||||||
|
void CloseContextMenu( OPT_TOOL_EVENT& evt );
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function CreateBasicMenu
|
||||||
|
*
|
||||||
|
* Construct a "basic" menu for a tool, containing only items
|
||||||
|
* that apply to all tools (e.g. zoom and grid)
|
||||||
|
*/
|
||||||
|
void AddStandardSubMenus( EDA_DRAW_FRAME& aFrame );
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Helper function for factories to abe able to easily add
|
||||||
|
* their own new sub menus. This sets the tool to the TOOL_MENUs
|
||||||
|
* owner and adds to the store.
|
||||||
|
*
|
||||||
|
* Note, this won't share the menu between multiple invocations
|
||||||
|
* of the factory. But if different top-level tools are using the
|
||||||
|
* same factory, which one would be used for SetTool()?
|
||||||
|
*/
|
||||||
|
template <typename T, typename ... Args>
|
||||||
|
std::shared_ptr<T> createOwnSubMenu( Args&& ... args )
|
||||||
|
{
|
||||||
|
auto subMenuPtr = std::make_shared<T>( args ... );
|
||||||
|
|
||||||
|
subMenuPtr->SetTool( &m_tool );
|
||||||
|
AddSubMenu( subMenuPtr );
|
||||||
|
|
||||||
|
return subMenuPtr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The conditional model of the menu displayed by the tool
|
||||||
|
*/
|
||||||
|
CONDITIONAL_MENU m_menu;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The actual menu displayed by the tool
|
||||||
|
*/
|
||||||
|
std::unique_ptr<CONTEXT_MENU> m_contextMenu;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The tool that owns this menu
|
||||||
|
*/
|
||||||
|
TOOL_INTERACTIVE& m_tool;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Lifetime-managing container of submenus
|
||||||
|
*/
|
||||||
|
std::vector<std::shared_ptr<CONTEXT_MENU> > m_subMenus;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // TOOLS_TOOL_MENU__H_
|
Loading…
Reference in New Issue