Add GAL microwave tools
This adds the microwave tools to GAL in Pcbnew as a new tool: MICROWAVE_TOOL. Some new preview items are introduced to support this: * TWO_POINT_GEOM_MANAGER for managing construction of two-point geometries, such as that used to construct the inductor. * CENTRELINE_RECT_ITEM a preview item to draw ractangle with a given aspect ratio along a line (specified using a TWO_POINT_GEOM_MANAGER) PCB_TOOL gets a generic event loops which should be useful for more than just microwave tools: * doInteractiveItemPlacement() - handles event loops that wait for a click, create an item on click and then allow moving it around/flipping, etc. Fixes: lp:1531323 * https://bugs.launchpad.net/kicad/+bug/1531323
This commit is contained in:
parent
3994e9a268
commit
ce731f8b62
|
@ -187,6 +187,7 @@ set( COMMON_PAGE_LAYOUT_SRCS
|
||||||
set( COMMON_PREVIEW_ITEMS_SRCS
|
set( COMMON_PREVIEW_ITEMS_SRCS
|
||||||
preview_items/arc_assistant.cpp
|
preview_items/arc_assistant.cpp
|
||||||
preview_items/arc_geom_manager.cpp
|
preview_items/arc_geom_manager.cpp
|
||||||
|
preview_items/centreline_rect_item.cpp
|
||||||
preview_items/preview_utils.cpp
|
preview_items/preview_utils.cpp
|
||||||
preview_items/ruler_item.cpp
|
preview_items/ruler_item.cpp
|
||||||
preview_items/simple_overlay_item.cpp
|
preview_items/simple_overlay_item.cpp
|
||||||
|
|
|
@ -0,0 +1,100 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Kicad Developers, see change_log.txt for contributors.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, you may find one here:
|
||||||
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
||||||
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
||||||
|
* or you may write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <preview_items/centreline_rect_item.h>
|
||||||
|
|
||||||
|
#include <preview_items/two_point_geom_manager.h>
|
||||||
|
|
||||||
|
#include <gal/graphics_abstraction_layer.h>
|
||||||
|
|
||||||
|
|
||||||
|
using namespace KIGFX::PREVIEW;
|
||||||
|
|
||||||
|
static SHAPE_POLY_SET getRectangleAlongCentreLine(
|
||||||
|
const VECTOR2D& aClStart, const VECTOR2D& aClEnd, double aAspect )
|
||||||
|
{
|
||||||
|
SHAPE_POLY_SET poly;
|
||||||
|
poly.NewOutline();
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The point layout of the rectangle goes like this,
|
||||||
|
* but start/end don't have to be horz/vert
|
||||||
|
*
|
||||||
|
* 0 ---------------- 1 -----
|
||||||
|
* | | ^
|
||||||
|
* s--------cl------->e |cl|/aspect
|
||||||
|
* | | v
|
||||||
|
* 3----------------- 2 -----
|
||||||
|
*/
|
||||||
|
|
||||||
|
// vector down the centre line of the rectangle
|
||||||
|
const VECTOR2D cl = aClEnd - aClStart;
|
||||||
|
|
||||||
|
// the "side" of the rectangle is the centre line rotated by 90 deg
|
||||||
|
// and scaled by the aspect ratio
|
||||||
|
const VECTOR2D side = cl.Rotate( M_PI / 2.0 ) * aAspect;
|
||||||
|
|
||||||
|
VECTOR2D pt = aClStart + ( side / 2.0 );
|
||||||
|
poly.Append( pt );
|
||||||
|
|
||||||
|
pt += cl;
|
||||||
|
poly.Append( pt );
|
||||||
|
|
||||||
|
pt -= side;
|
||||||
|
poly.Append( pt );
|
||||||
|
|
||||||
|
pt -= cl;
|
||||||
|
poly.Append( pt );
|
||||||
|
|
||||||
|
return poly;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
CENTRELINE_RECT_ITEM::CENTRELINE_RECT_ITEM(
|
||||||
|
const TWO_POINT_GEOMETRY_MANAGER& aGeomMgr,
|
||||||
|
double aAspect ):
|
||||||
|
m_geomMgr( aGeomMgr ),
|
||||||
|
m_aspect( aAspect )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SHAPE_POLY_SET CENTRELINE_RECT_ITEM::getOutline() const
|
||||||
|
{
|
||||||
|
return getRectangleAlongCentreLine( m_geomMgr.GetOrigin(),
|
||||||
|
m_geomMgr.GetEnd(),
|
||||||
|
m_aspect );
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const BOX2I CENTRELINE_RECT_ITEM::ViewBBox() const
|
||||||
|
{
|
||||||
|
return getOutline().BBox();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void CENTRELINE_RECT_ITEM::drawPreviewShape( KIGFX::GAL& aGal ) const
|
||||||
|
{
|
||||||
|
aGal.DrawLine( m_geomMgr.GetOrigin(), m_geomMgr.GetEnd() );
|
||||||
|
aGal.DrawPolygon( getOutline() );
|
||||||
|
}
|
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 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
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PREVIEW_ITEMS_CENTERLINE_RECT_ITEM_H
|
||||||
|
#define PREVIEW_ITEMS_CENTERLINE_RECT_ITEM_H
|
||||||
|
|
||||||
|
#include <preview_items/simple_overlay_item.h>
|
||||||
|
|
||||||
|
#include <geometry/shape_poly_set.h>
|
||||||
|
|
||||||
|
#include <math/vector2d.h>
|
||||||
|
|
||||||
|
namespace KIGFX
|
||||||
|
{
|
||||||
|
class GAL;
|
||||||
|
|
||||||
|
namespace PREVIEW
|
||||||
|
{
|
||||||
|
class TWO_POINT_GEOMETRY_MANAGER;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class CENTRELINE_RECT_ITEM
|
||||||
|
*
|
||||||
|
* Represents an area drawn by drawing a rectangle of a given aspect
|
||||||
|
* along a vector, with the midpoiunt of one side on the start point
|
||||||
|
* and the mid point of the opposite side on the end.
|
||||||
|
*
|
||||||
|
* The centre line does not have to horizontal or vertical, it
|
||||||
|
* can be at any angle.
|
||||||
|
*/
|
||||||
|
class CENTRELINE_RECT_ITEM : public SIMPLE_OVERLAY_ITEM
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
CENTRELINE_RECT_ITEM( const TWO_POINT_GEOMETRY_MANAGER& aGeomMgr,
|
||||||
|
double aAspect );
|
||||||
|
|
||||||
|
///> Gets the bounding box of the rectangle
|
||||||
|
virtual const BOX2I ViewBBox() const override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
///> Get the rectangular outline
|
||||||
|
SHAPE_POLY_SET getOutline() const;
|
||||||
|
|
||||||
|
///> Draw rectangle and centre line onto GAL
|
||||||
|
void drawPreviewShape( KIGFX::GAL& aGal ) const override;
|
||||||
|
|
||||||
|
const TWO_POINT_GEOMETRY_MANAGER& m_geomMgr;
|
||||||
|
|
||||||
|
///> The aspect ratio of the rectangle to draw
|
||||||
|
double m_aspect;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // PREVIEW
|
||||||
|
} // KIGFX
|
||||||
|
|
||||||
|
#endif // PREVIEW_ITEMS_CENTERLINE_RECT_ITEM_H
|
|
@ -0,0 +1,97 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 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
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef PREVIEW_ITEMS_TWO_POINT_GEOMETRY_MANAGER_H
|
||||||
|
#define PREVIEW_ITEMS_TWO_POINT_GEOMETRY_MANAGER_H
|
||||||
|
|
||||||
|
#include <math/vector2d.h>
|
||||||
|
#include <common.h>
|
||||||
|
|
||||||
|
namespace KIGFX
|
||||||
|
{
|
||||||
|
class GAL;
|
||||||
|
|
||||||
|
namespace PREVIEW
|
||||||
|
{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class TWO_POINT_GEOMETRY_MANAGER
|
||||||
|
*
|
||||||
|
* Represents a very simple geometry manager for items that have
|
||||||
|
* a start and end point.
|
||||||
|
*/
|
||||||
|
class TWO_POINT_GEOMETRY_MANAGER
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
|
||||||
|
///> Set the origin of the ruler (the fixed end)
|
||||||
|
void SetOrigin( const VECTOR2I& aOrigin )
|
||||||
|
{
|
||||||
|
m_origin = aOrigin;
|
||||||
|
}
|
||||||
|
|
||||||
|
VECTOR2I GetOrigin() const
|
||||||
|
{
|
||||||
|
return m_origin;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the current end of the rectangle (the end that moves
|
||||||
|
* with the cursor.
|
||||||
|
*/
|
||||||
|
void SetEnd( const VECTOR2I& aEnd )
|
||||||
|
{
|
||||||
|
if( m_angleSnap )
|
||||||
|
{
|
||||||
|
const auto vec = aEnd - m_origin;
|
||||||
|
const auto len = vec.EuclideanNorm();
|
||||||
|
const auto angle = KiROUND( vec.Angle() / M_PI_4 ) * M_PI_4;
|
||||||
|
|
||||||
|
m_end = m_origin + VECTOR2I( len, 0 ).Rotate( angle );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_end = aEnd;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
VECTOR2I GetEnd() const
|
||||||
|
{
|
||||||
|
return m_end;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetAngleSnap( bool aSnap )
|
||||||
|
{
|
||||||
|
m_angleSnap = aSnap;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
VECTOR2I m_origin, m_end;
|
||||||
|
bool m_angleSnap = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // PREVIEW
|
||||||
|
} // KIGFX
|
||||||
|
|
||||||
|
#endif // PREVIEW_ITEMS_TWO_POINT_GEOMETRY_MANAGER_H
|
|
@ -294,6 +294,7 @@ set( PCBNEW_CLASS_SRCS
|
||||||
class_footprint_wizard.cpp
|
class_footprint_wizard.cpp
|
||||||
class_action_plugin.cpp
|
class_action_plugin.cpp
|
||||||
|
|
||||||
|
tools/pcb_tool.cpp
|
||||||
tools/selection_tool.cpp
|
tools/selection_tool.cpp
|
||||||
tools/pcb_selection_conditions.cpp
|
tools/pcb_selection_conditions.cpp
|
||||||
tools/pcb_bright_box.cpp
|
tools/pcb_bright_box.cpp
|
||||||
|
@ -304,6 +305,7 @@ set( PCBNEW_CLASS_SRCS
|
||||||
tools/edit_tool.cpp
|
tools/edit_tool.cpp
|
||||||
tools/pcbnew_control.cpp
|
tools/pcbnew_control.cpp
|
||||||
tools/pcb_editor_control.cpp
|
tools/pcb_editor_control.cpp
|
||||||
|
tools/microwave_tool.cpp
|
||||||
tools/module_editor_tools.cpp
|
tools/module_editor_tools.cpp
|
||||||
tools/placement_tool.cpp
|
tools/placement_tool.cpp
|
||||||
tools/pcb_actions.cpp
|
tools/pcb_actions.cpp
|
||||||
|
|
|
@ -307,10 +307,10 @@ MODULE* MWAVE::CreateMicrowaveInductor( INDUCTOR_PATTERN& inductorPattern,
|
||||||
|
|
||||||
// Enter the desired length.
|
// Enter the desired length.
|
||||||
msg = StringFromValue( g_UserUnit, inductorPattern.m_length );
|
msg = StringFromValue( g_UserUnit, inductorPattern.m_length );
|
||||||
wxTextEntryDialog dlg( NULL, wxEmptyString, _( "Length of Trace:" ), msg );
|
wxTextEntryDialog dlg( nullptr, wxEmptyString, _( "Length of Trace:" ), msg );
|
||||||
|
|
||||||
if( dlg.ShowModal() != wxID_OK )
|
if( dlg.ShowModal() != wxID_OK )
|
||||||
return NULL; // canceled by user
|
return nullptr; // canceled by user
|
||||||
|
|
||||||
msg = dlg.GetValue();
|
msg = dlg.GetValue();
|
||||||
inductorPattern.m_length = ValueFromString( g_UserUnit, msg );
|
inductorPattern.m_length = ValueFromString( g_UserUnit, msg );
|
||||||
|
@ -319,7 +319,7 @@ MODULE* MWAVE::CreateMicrowaveInductor( INDUCTOR_PATTERN& inductorPattern,
|
||||||
if( inductorPattern.m_length < min_len )
|
if( inductorPattern.m_length < min_len )
|
||||||
{
|
{
|
||||||
aErrorMessage = _( "Requested length < minimum length" );
|
aErrorMessage = _( "Requested length < minimum length" );
|
||||||
return NULL;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate the elements.
|
// Calculate the elements.
|
||||||
|
@ -331,16 +331,16 @@ MODULE* MWAVE::CreateMicrowaveInductor( INDUCTOR_PATTERN& inductorPattern,
|
||||||
if( !ll )
|
if( !ll )
|
||||||
{
|
{
|
||||||
aErrorMessage = _( "Requested length too large" );
|
aErrorMessage = _( "Requested length too large" );
|
||||||
return NULL;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate footprint. the value is also used as footprint name.
|
// Generate footprint. the value is also used as footprint name.
|
||||||
msg.Empty();
|
msg = "L";
|
||||||
wxTextEntryDialog cmpdlg( NULL, wxEmptyString, _( "Component Value:" ), msg );
|
wxTextEntryDialog cmpdlg( nullptr, wxEmptyString, _( "Component Value:" ), msg );
|
||||||
cmpdlg.SetTextValidator( FILE_NAME_CHAR_VALIDATOR( &msg ) );
|
cmpdlg.SetTextValidator( FILE_NAME_CHAR_VALIDATOR( &msg ) );
|
||||||
|
|
||||||
if( ( cmpdlg.ShowModal() != wxID_OK ) || msg.IsEmpty() )
|
if( ( cmpdlg.ShowModal() != wxID_OK ) || msg.IsEmpty() )
|
||||||
return NULL; // Aborted by user
|
return nullptr; // Aborted by user
|
||||||
|
|
||||||
MODULE* module = aPcbFrame->CreateNewModule( msg );
|
MODULE* module = aPcbFrame->CreateNewModule( msg );
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,385 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Kicad Developers, see change_log.txt for contributors.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, you may find one here:
|
||||||
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
||||||
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
||||||
|
* or you may write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "microwave_tool.h"
|
||||||
|
|
||||||
|
#include <gal/graphics_abstraction_layer.h>
|
||||||
|
#include <class_draw_panel_gal.h>
|
||||||
|
#include <view/view_controls.h>
|
||||||
|
#include <view/view.h>
|
||||||
|
#include <tool/tool_manager.h>
|
||||||
|
#include <board_commit.h>
|
||||||
|
#include <confirm.h>
|
||||||
|
#include <preview_items/two_point_geom_manager.h>
|
||||||
|
#include <preview_items/centreline_rect_item.h>
|
||||||
|
|
||||||
|
// For frame ToolID values
|
||||||
|
#include <pcbnew_id.h>
|
||||||
|
|
||||||
|
// For action icons
|
||||||
|
#include <bitmaps.h>
|
||||||
|
|
||||||
|
#include <class_board_item.h>
|
||||||
|
#include <class_module.h>
|
||||||
|
|
||||||
|
#include <microwave/microwave_inductor.h>
|
||||||
|
|
||||||
|
#include "pcb_actions.h"
|
||||||
|
#include "selection_tool.h"
|
||||||
|
#include "tool_event_utils.h"
|
||||||
|
|
||||||
|
|
||||||
|
///> Type of items that are "simple" - just get placed on
|
||||||
|
///> the board directly, without a graphical interactive setup stage
|
||||||
|
enum MWAVE_TOOL_SIMPLE_ID
|
||||||
|
{
|
||||||
|
GAP,
|
||||||
|
STUB,
|
||||||
|
STUB_ARC,
|
||||||
|
FUNCTION_SHAPE,
|
||||||
|
};
|
||||||
|
|
||||||
|
TOOL_ACTION PCB_ACTIONS::microwaveCreateGap(
|
||||||
|
"pcbnew.MicrowaveTool.createGap",
|
||||||
|
AS_GLOBAL, 0,
|
||||||
|
_( "Add Gap" ), _( "Create gap of specified length for microwave applications" ),
|
||||||
|
mw_add_gap_xpm, AF_ACTIVATE, (void*) MWAVE_TOOL_SIMPLE_ID::GAP );
|
||||||
|
|
||||||
|
TOOL_ACTION PCB_ACTIONS::microwaveCreateStub(
|
||||||
|
"pcbnew.MicrowaveTool.createStub",
|
||||||
|
AS_GLOBAL, 0,
|
||||||
|
_( "Add Stub" ), _( "Create stub of specified length for microwave applications" ),
|
||||||
|
mw_add_stub_xpm, AF_ACTIVATE, (void*) MWAVE_TOOL_SIMPLE_ID::STUB );
|
||||||
|
|
||||||
|
TOOL_ACTION PCB_ACTIONS::microwaveCreateStubArc(
|
||||||
|
"pcbnew.MicrowaveTool.createStubArc",
|
||||||
|
AS_GLOBAL, 0,
|
||||||
|
_( "Add Arc Stub" ), _( "Create stub (arc) of specified length for microwave applications" ),
|
||||||
|
mw_add_stub_arc_xpm, AF_ACTIVATE, (void*) MWAVE_TOOL_SIMPLE_ID::STUB_ARC );
|
||||||
|
|
||||||
|
TOOL_ACTION PCB_ACTIONS::microwaveCreateFunctionShape(
|
||||||
|
"pcbnew.MicrowaveTool.createFunctionShape",
|
||||||
|
AS_GLOBAL, 0,
|
||||||
|
_( "Add Polynomial Shape" ), _( "Create polynomial shape for microwave applications" ),
|
||||||
|
mw_add_gap_xpm, AF_ACTIVATE, (void*) MWAVE_TOOL_SIMPLE_ID::FUNCTION_SHAPE );
|
||||||
|
|
||||||
|
TOOL_ACTION PCB_ACTIONS::microwaveCreateLine(
|
||||||
|
"pcbnew.MicrowaveTool.createLine",
|
||||||
|
AS_GLOBAL, 0,
|
||||||
|
_( "Add Microwave Line" ), _( "Create line of specified length for microwave applications" ),
|
||||||
|
mw_add_line_xpm, AF_ACTIVATE );
|
||||||
|
|
||||||
|
|
||||||
|
MICROWAVE_TOOL::MICROWAVE_TOOL() :
|
||||||
|
PCB_TOOL( "pcbnew.MicrowaveTool" ),
|
||||||
|
m_menu( *this )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
MICROWAVE_TOOL::~MICROWAVE_TOOL()
|
||||||
|
{}
|
||||||
|
|
||||||
|
|
||||||
|
void MICROWAVE_TOOL::Reset( RESET_REASON aReason )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool MICROWAVE_TOOL::Init()
|
||||||
|
{
|
||||||
|
auto activeToolFunctor = [ this ] ( const SELECTION& aSel ) {
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
auto& ctxMenu = m_menu.GetMenu();
|
||||||
|
|
||||||
|
// cancel current goes in main context menu at the top if present
|
||||||
|
ctxMenu.AddItem( ACTIONS::cancelInteractive, activeToolFunctor, 1000 );
|
||||||
|
ctxMenu.AddSeparator( activeToolFunctor, 1000 );
|
||||||
|
|
||||||
|
m_menu.AddStandardSubMenus( *getEditFrame<PCB_BASE_FRAME>() );
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
struct MICROWAVE_TOOL_INFO
|
||||||
|
{
|
||||||
|
using MOD_CREATOR = std::function<std::unique_ptr<MODULE>()>;
|
||||||
|
|
||||||
|
wxString name;
|
||||||
|
int toolId;
|
||||||
|
MOD_CREATOR creatorFunc;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
MICROWAVE_TOOL_INFO getMicrowaveItemCreator( PCB_EDIT_FRAME& frame, int aParam )
|
||||||
|
{
|
||||||
|
MICROWAVE_TOOL_INFO info;
|
||||||
|
|
||||||
|
switch( aParam )
|
||||||
|
{
|
||||||
|
case MWAVE_TOOL_SIMPLE_ID::GAP:
|
||||||
|
info.name = _( "Add Gap" );
|
||||||
|
info.toolId = ID_PCB_MUWAVE_TOOL_GAP_CMD;
|
||||||
|
info.creatorFunc = [&frame] () {
|
||||||
|
return std::unique_ptr<MODULE>( frame.Create_MuWaveComponent( 0 ) );
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
|
||||||
|
case MWAVE_TOOL_SIMPLE_ID::STUB:
|
||||||
|
info.name = _( "Add Stub" );
|
||||||
|
info.toolId = ID_PCB_MUWAVE_TOOL_STUB_CMD;
|
||||||
|
info.creatorFunc = [&frame] () {
|
||||||
|
return std::unique_ptr<MODULE>( frame.Create_MuWaveComponent( 1 ) );
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
|
||||||
|
case MWAVE_TOOL_SIMPLE_ID::STUB_ARC:
|
||||||
|
info.name = _( "Add Stub (Arc)" );
|
||||||
|
info.toolId = ID_PCB_MUWAVE_TOOL_STUB_ARC_CMD;
|
||||||
|
info.creatorFunc = [&frame] () {
|
||||||
|
return std::unique_ptr<MODULE>( frame.Create_MuWaveComponent( 2 ) );
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
|
||||||
|
case MWAVE_TOOL_SIMPLE_ID::FUNCTION_SHAPE:
|
||||||
|
info.name = _( "Add Polynomial Shape" );
|
||||||
|
info.toolId = ID_PCB_MUWAVE_TOOL_FUNCTION_SHAPE_CMD;
|
||||||
|
info.creatorFunc = [&frame] () {
|
||||||
|
return std::unique_ptr<MODULE>( frame.Create_MuWavePolygonShape() );
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
};
|
||||||
|
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int MICROWAVE_TOOL::addMicrowaveFootprint( const TOOL_EVENT& aEvent )
|
||||||
|
{
|
||||||
|
auto& frame = *getEditFrame<PCB_EDIT_FRAME>();
|
||||||
|
|
||||||
|
const int param = aEvent.Parameter<intptr_t>();
|
||||||
|
|
||||||
|
MICROWAVE_TOOL_INFO info = getMicrowaveItemCreator( frame, param );
|
||||||
|
|
||||||
|
// failed to find suitable item info - shouldn't be possible
|
||||||
|
// if all the id's are handled
|
||||||
|
if( !info.name )
|
||||||
|
{
|
||||||
|
wxASSERT_MSG( 0, "Failed to find suitable microwave tool info" );
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
frame.SetToolID( info.toolId, wxCURSOR_PENCIL, info.name );
|
||||||
|
|
||||||
|
ITEM_CREATOR moduleCreator = [this, &info] ( const TOOL_EVENT& aAddingEvent ) {
|
||||||
|
auto module = info.creatorFunc();
|
||||||
|
|
||||||
|
// Module has been added in the legacy backend,
|
||||||
|
// so we have to remove it before committing the change
|
||||||
|
// @todo LEGACY
|
||||||
|
if( module )
|
||||||
|
{
|
||||||
|
board()->Remove( module.get() );
|
||||||
|
}
|
||||||
|
|
||||||
|
return module;
|
||||||
|
};
|
||||||
|
|
||||||
|
doInteractiveItemPlacement( moduleCreator, _( "Place microwave feature" ) );
|
||||||
|
|
||||||
|
frame.SetToolID( ID_NO_TOOL_SELECTED, wxCURSOR_DEFAULT, wxEmptyString );
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void MICROWAVE_TOOL::createInductorBetween( const VECTOR2I& aStart, const VECTOR2I& aEnd )
|
||||||
|
{
|
||||||
|
auto& frame = *getEditFrame<PCB_EDIT_FRAME>();
|
||||||
|
|
||||||
|
MWAVE::INDUCTOR_PATTERN pattern;
|
||||||
|
|
||||||
|
pattern.m_Width = board()->GetDesignSettings().GetCurrentTrackWidth();
|
||||||
|
|
||||||
|
pattern.m_Start = { aStart.x, aStart.y };
|
||||||
|
pattern.m_End = { aEnd.x, aEnd.y };
|
||||||
|
|
||||||
|
wxString errorMessage;
|
||||||
|
|
||||||
|
auto inductorModule = std::unique_ptr<MODULE>(
|
||||||
|
CreateMicrowaveInductor( pattern, &frame, errorMessage )
|
||||||
|
);
|
||||||
|
|
||||||
|
if( inductorModule )
|
||||||
|
{
|
||||||
|
// legacy mode tools add to the board
|
||||||
|
// so remove it and add it back with the commit object
|
||||||
|
// this has to happen, even if we don't end up storing the module
|
||||||
|
// @todo LEGACY
|
||||||
|
board()->Remove( inductorModule.get() );
|
||||||
|
}
|
||||||
|
|
||||||
|
// on any error, report if we can
|
||||||
|
if ( !inductorModule || !errorMessage.IsEmpty() )
|
||||||
|
{
|
||||||
|
if ( !errorMessage.IsEmpty() )
|
||||||
|
{
|
||||||
|
DisplayError( &frame, errorMessage );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// at this point, we can save the module
|
||||||
|
frame.SetCurItem( inductorModule.get() );
|
||||||
|
|
||||||
|
BOARD_COMMIT commit( this );
|
||||||
|
commit.Add( inductorModule.release() );
|
||||||
|
commit.Push( _("Add microwave inductor" ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static const COLOR4D inductorAreaFill( 0.3, 0.3, 0.5, 0.3 );
|
||||||
|
static const COLOR4D inductorAreaStroke( 0.4, 1.0, 1.0, 1.0 );
|
||||||
|
static const double inductorAreaStrokeWidth = 1.0;
|
||||||
|
|
||||||
|
///> Aspect of the preview rectangle - this is hardcoded in the
|
||||||
|
///> microwave backend for now
|
||||||
|
static const double inductorAreaAspect = 0.5;
|
||||||
|
|
||||||
|
|
||||||
|
int MICROWAVE_TOOL::drawMicrowaveInductor( const TOOL_EVENT& aEvent )
|
||||||
|
{
|
||||||
|
using namespace KIGFX::PREVIEW;
|
||||||
|
|
||||||
|
KIGFX::VIEW& view = *getView();
|
||||||
|
KIGFX::VIEW_CONTROLS& controls = *getViewControls();
|
||||||
|
auto& frame = *getEditFrame<PCB_EDIT_FRAME>();
|
||||||
|
|
||||||
|
frame.SetToolID( ID_PCB_MUWAVE_TOOL_SELF_CMD, wxCURSOR_PENCIL, _( "Add Microwave Inductor" ) );
|
||||||
|
|
||||||
|
Activate();
|
||||||
|
|
||||||
|
TWO_POINT_GEOMETRY_MANAGER tpGeomMgr;
|
||||||
|
|
||||||
|
CENTRELINE_RECT_ITEM previewRect( tpGeomMgr, inductorAreaAspect );
|
||||||
|
|
||||||
|
previewRect.SetFillColor( inductorAreaFill );
|
||||||
|
previewRect.SetStrokeColor( inductorAreaStroke );
|
||||||
|
previewRect.SetLineWidth( inductorAreaStrokeWidth );
|
||||||
|
|
||||||
|
bool originSet = false;
|
||||||
|
|
||||||
|
controls.ShowCursor( true );
|
||||||
|
controls.SetSnapping( true );
|
||||||
|
|
||||||
|
view.Add( &previewRect );
|
||||||
|
|
||||||
|
while( auto evt = Wait() )
|
||||||
|
{
|
||||||
|
VECTOR2I cursorPos = controls.GetCursorPosition();
|
||||||
|
|
||||||
|
if( TOOL_EVT_UTILS::IsCancelInteractive( *evt ) )
|
||||||
|
{
|
||||||
|
// overriding action, or we're cancelling without
|
||||||
|
// an in-progress preview area
|
||||||
|
if ( evt->IsActivate() || !originSet )
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// had an in-progress area, so start again but don't
|
||||||
|
// cancel the tool
|
||||||
|
originSet = false;
|
||||||
|
view.SetVisible( &previewRect, false );
|
||||||
|
view.Update( &previewRect, KIGFX::GEOMETRY );
|
||||||
|
}
|
||||||
|
|
||||||
|
// A click or drag starts
|
||||||
|
else if( !originSet &&
|
||||||
|
( evt->IsClick( BUT_LEFT ) || evt->IsDrag( BUT_LEFT ) ) )
|
||||||
|
{
|
||||||
|
tpGeomMgr.SetOrigin( cursorPos );
|
||||||
|
tpGeomMgr.SetEnd( cursorPos );
|
||||||
|
|
||||||
|
originSet = true;
|
||||||
|
|
||||||
|
controls.CaptureCursor( true );
|
||||||
|
controls.SetAutoPan( true );
|
||||||
|
}
|
||||||
|
|
||||||
|
// another click after origin set is the end
|
||||||
|
// left up is also the end, as you'll only get that after a drag
|
||||||
|
else if( originSet &&
|
||||||
|
( evt->IsClick( BUT_LEFT ) || evt->IsMouseUp( BUT_LEFT ) ) )
|
||||||
|
{
|
||||||
|
// second click, we're done:
|
||||||
|
// delegate to the point-to-point inductor creator function
|
||||||
|
createInductorBetween( tpGeomMgr.GetOrigin(), tpGeomMgr.GetEnd() );
|
||||||
|
|
||||||
|
// start again if needed
|
||||||
|
originSet = false;
|
||||||
|
|
||||||
|
view.SetVisible( &previewRect, false );
|
||||||
|
view.Update( &previewRect, KIGFX::GEOMETRY );
|
||||||
|
}
|
||||||
|
|
||||||
|
// any move or drag once the origin was set updates
|
||||||
|
// the end point
|
||||||
|
else if( originSet &&
|
||||||
|
( evt->IsMotion() || evt->IsDrag( BUT_LEFT ) ) )
|
||||||
|
{
|
||||||
|
tpGeomMgr.SetAngleSnap( evt->Modifier( MD_CTRL ) );
|
||||||
|
tpGeomMgr.SetEnd( cursorPos );
|
||||||
|
|
||||||
|
view.SetVisible( &previewRect, true );
|
||||||
|
view.Update( &previewRect, KIGFX::GEOMETRY );
|
||||||
|
}
|
||||||
|
|
||||||
|
else if( evt->IsClick( BUT_RIGHT ) )
|
||||||
|
{
|
||||||
|
m_menu.ShowContextMenu();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
view.Remove( &previewRect );
|
||||||
|
|
||||||
|
frame.SetToolID( ID_NO_TOOL_SELECTED, wxCURSOR_DEFAULT, wxEmptyString );
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void MICROWAVE_TOOL::SetTransitions()
|
||||||
|
{
|
||||||
|
Go( &MICROWAVE_TOOL::addMicrowaveFootprint, PCB_ACTIONS::microwaveCreateGap.MakeEvent() );
|
||||||
|
Go( &MICROWAVE_TOOL::addMicrowaveFootprint, PCB_ACTIONS::microwaveCreateStub.MakeEvent() );
|
||||||
|
Go( &MICROWAVE_TOOL::addMicrowaveFootprint, PCB_ACTIONS::microwaveCreateStubArc.MakeEvent() );
|
||||||
|
Go( &MICROWAVE_TOOL::addMicrowaveFootprint, PCB_ACTIONS::microwaveCreateFunctionShape.MakeEvent() );
|
||||||
|
|
||||||
|
Go( &MICROWAVE_TOOL::drawMicrowaveInductor, PCB_ACTIONS::microwaveCreateLine.MakeEvent() );
|
||||||
|
}
|
|
@ -0,0 +1,68 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Kicad Developers, see change_log.txt for contributors.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, you may find one here:
|
||||||
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
||||||
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
||||||
|
* or you may write to the Free Software Foundation, Inc.,
|
||||||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef TOOLS_MICROWAVE_TOOL_H
|
||||||
|
#define TOOLS_MICROWAVE_TOOL_H
|
||||||
|
|
||||||
|
#include <tools/pcb_tool.h>
|
||||||
|
|
||||||
|
#include <tool/tool_menu.h>
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class MICROWAVE_TOOL
|
||||||
|
*
|
||||||
|
* Tool responsible for adding microwave features to PCBs
|
||||||
|
*/
|
||||||
|
class MICROWAVE_TOOL : public PCB_TOOL
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
MICROWAVE_TOOL();
|
||||||
|
~MICROWAVE_TOOL();
|
||||||
|
|
||||||
|
///> React to model/view changes
|
||||||
|
void Reset( RESET_REASON aReason ) override;
|
||||||
|
|
||||||
|
///> Basic initalization
|
||||||
|
bool Init() override;
|
||||||
|
|
||||||
|
///> Bind handlers to corresponding TOOL_ACTIONs
|
||||||
|
void SetTransitions() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
///> Main interactive tool
|
||||||
|
int addMicrowaveFootprint( const TOOL_EVENT& aEvent );
|
||||||
|
|
||||||
|
///> Create an inductor between the two points
|
||||||
|
void createInductorBetween( const VECTOR2I& aStart, const VECTOR2I& aEnd );
|
||||||
|
|
||||||
|
///> Draw a microwave inductor interactively
|
||||||
|
int drawMicrowaveInductor( const TOOL_EVENT& aEvent );
|
||||||
|
|
||||||
|
/// Menu model displayed by the tool.
|
||||||
|
TOOL_MENU m_menu;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#endif // TOOLS_MICROWAVE_TOOL_H
|
|
@ -163,6 +163,21 @@ boost::optional<TOOL_EVENT> PCB_ACTIONS::TranslateLegacyId( int aId )
|
||||||
|
|
||||||
case ID_PCB_SHOW_1_RATSNEST_BUTT:
|
case ID_PCB_SHOW_1_RATSNEST_BUTT:
|
||||||
return PCB_ACTIONS::toBeDone.MakeEvent();
|
return PCB_ACTIONS::toBeDone.MakeEvent();
|
||||||
|
|
||||||
|
case ID_PCB_MUWAVE_TOOL_GAP_CMD:
|
||||||
|
return PCB_ACTIONS::microwaveCreateGap.MakeEvent();
|
||||||
|
|
||||||
|
case ID_PCB_MUWAVE_TOOL_STUB_CMD:
|
||||||
|
return PCB_ACTIONS::microwaveCreateStub.MakeEvent();
|
||||||
|
|
||||||
|
case ID_PCB_MUWAVE_TOOL_STUB_ARC_CMD:
|
||||||
|
return PCB_ACTIONS::microwaveCreateStubArc.MakeEvent();
|
||||||
|
|
||||||
|
case ID_PCB_MUWAVE_TOOL_FUNCTION_SHAPE_CMD:
|
||||||
|
return PCB_ACTIONS::microwaveCreateFunctionShape.MakeEvent();
|
||||||
|
|
||||||
|
case ID_PCB_MUWAVE_TOOL_SELF_CMD:
|
||||||
|
return PCB_ACTIONS::microwaveCreateLine.MakeEvent();
|
||||||
}
|
}
|
||||||
|
|
||||||
return boost::optional<TOOL_EVENT>();
|
return boost::optional<TOOL_EVENT>();
|
||||||
|
|
|
@ -294,6 +294,17 @@ public:
|
||||||
/// Copy the current pad's settings to other pads in the module or on the board
|
/// Copy the current pad's settings to other pads in the module or on the board
|
||||||
static TOOL_ACTION pushPadSettings;
|
static TOOL_ACTION pushPadSettings;
|
||||||
|
|
||||||
|
// Microwave tools
|
||||||
|
static TOOL_ACTION microwaveCreateGap;
|
||||||
|
|
||||||
|
static TOOL_ACTION microwaveCreateStub;
|
||||||
|
|
||||||
|
static TOOL_ACTION microwaveCreateStubArc;
|
||||||
|
|
||||||
|
static TOOL_ACTION microwaveCreateFunctionShape;
|
||||||
|
|
||||||
|
static TOOL_ACTION microwaveCreateLine;
|
||||||
|
|
||||||
/// Cursor control with keyboard
|
/// Cursor control with keyboard
|
||||||
static TOOL_ACTION cursorUp;
|
static TOOL_ACTION cursorUp;
|
||||||
static TOOL_ACTION cursorDown;
|
static TOOL_ACTION cursorDown;
|
||||||
|
|
|
@ -0,0 +1,169 @@
|
||||||
|
/*
|
||||||
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 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
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "pcb_tool.h"
|
||||||
|
|
||||||
|
#include <view/view_controls.h>
|
||||||
|
#include <view/view.h>
|
||||||
|
#include <tool/tool_manager.h>
|
||||||
|
#include <board_commit.h>
|
||||||
|
|
||||||
|
#include <class_module.h>
|
||||||
|
|
||||||
|
#include "selection_tool.h"
|
||||||
|
#include "pcb_actions.h"
|
||||||
|
#include "tool_event_utils.h"
|
||||||
|
|
||||||
|
|
||||||
|
void PCB_TOOL::doInteractiveItemPlacement( ITEM_CREATOR aItemCreator,
|
||||||
|
const wxString& aCommitMessage )
|
||||||
|
{
|
||||||
|
using namespace std::placeholders;
|
||||||
|
|
||||||
|
KIGFX::VIEW& view = *getView();
|
||||||
|
KIGFX::VIEW_CONTROLS& controls = *getViewControls();
|
||||||
|
auto& frame = *getEditFrame<PCB_EDIT_FRAME>();
|
||||||
|
|
||||||
|
std::unique_ptr<BOARD_ITEM> newItem;
|
||||||
|
|
||||||
|
Activate();
|
||||||
|
|
||||||
|
BOARD_COMMIT commit( &frame );
|
||||||
|
|
||||||
|
GetManager()->RunAction( PCB_ACTIONS::selectionClear, true );
|
||||||
|
|
||||||
|
// do not capture or auto-pan until we start placing an item
|
||||||
|
controls.ShowCursor( true );
|
||||||
|
controls.SetSnapping( true );
|
||||||
|
|
||||||
|
// Add a VIEW_GROUP that serves as a preview for the new item
|
||||||
|
SELECTION preview;
|
||||||
|
view.Add( &preview );
|
||||||
|
|
||||||
|
// Main loop: keep receiving events
|
||||||
|
while( OPT_TOOL_EVENT evt = Wait() )
|
||||||
|
{
|
||||||
|
VECTOR2I cursorPos = controls.GetCursorPosition();
|
||||||
|
|
||||||
|
if( TOOL_EVT_UTILS::IsCancelInteractive( *evt ) )
|
||||||
|
{
|
||||||
|
if( newItem )
|
||||||
|
{
|
||||||
|
// Delete the old item and have another try
|
||||||
|
newItem = nullptr;
|
||||||
|
|
||||||
|
preview.Clear();
|
||||||
|
|
||||||
|
controls.SetAutoPan( false );
|
||||||
|
controls.CaptureCursor( false );
|
||||||
|
controls.ShowCursor( true );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( evt->IsActivate() ) // now finish unconditionally
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
else if( evt->IsClick( BUT_LEFT ) )
|
||||||
|
{
|
||||||
|
if( !newItem )
|
||||||
|
{
|
||||||
|
// create the item if possible
|
||||||
|
newItem = aItemCreator( *evt );
|
||||||
|
|
||||||
|
// no item created, so wait for another click
|
||||||
|
if( !newItem )
|
||||||
|
continue;
|
||||||
|
|
||||||
|
controls.CaptureCursor( true );
|
||||||
|
controls.SetAutoPan( true );
|
||||||
|
|
||||||
|
newItem->SetPosition( wxPoint( cursorPos.x, cursorPos.y ) );
|
||||||
|
|
||||||
|
preview.Add( newItem.get() );
|
||||||
|
|
||||||
|
if( newItem->Type() == PCB_MODULE_T )
|
||||||
|
{
|
||||||
|
auto module = dyn_cast<MODULE*>( newItem.get() );
|
||||||
|
|
||||||
|
// modules have more drawable parts
|
||||||
|
module->RunOnChildren( std::bind( &KIGFX::VIEW_GROUP::Add, &preview, _1 ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
newItem->ClearFlags();
|
||||||
|
preview.Remove( newItem.get() );
|
||||||
|
|
||||||
|
if( newItem->Type() == PCB_MODULE_T )
|
||||||
|
{
|
||||||
|
auto module = dyn_cast<MODULE*>( newItem.get() );
|
||||||
|
module->RunOnChildren( std::bind( &KIGFX::VIEW_GROUP::Remove, &preview, _1 ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
commit.Add( newItem.release() );
|
||||||
|
commit.Push( aCommitMessage );
|
||||||
|
|
||||||
|
controls.CaptureCursor( false );
|
||||||
|
controls.SetAutoPan( false );
|
||||||
|
controls.ShowCursor( true );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if( newItem && evt->Category() == TC_COMMAND )
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Handle any events that can affect the item as we move
|
||||||
|
* it around, eg rotate and flip
|
||||||
|
*/
|
||||||
|
|
||||||
|
if( TOOL_EVT_UTILS::IsRotateToolEvt( *evt ) )
|
||||||
|
{
|
||||||
|
const auto rotationAngle = TOOL_EVT_UTILS::GetEventRotationAngle(
|
||||||
|
frame, *evt );
|
||||||
|
newItem->Rotate( newItem->GetPosition(), rotationAngle );
|
||||||
|
view.Update( &preview );
|
||||||
|
}
|
||||||
|
else if( evt->IsAction( &PCB_ACTIONS::flip ) )
|
||||||
|
{
|
||||||
|
newItem->Flip( newItem->GetPosition() );
|
||||||
|
view.Update( &preview );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if( newItem && evt->IsMotion() )
|
||||||
|
{
|
||||||
|
// track the cursor
|
||||||
|
newItem->SetPosition( wxPoint( cursorPos.x, cursorPos.y ) );
|
||||||
|
|
||||||
|
// Show a preview of the item
|
||||||
|
view.Update( &preview );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
view.Remove( &preview );
|
||||||
|
}
|
|
@ -33,6 +33,9 @@
|
||||||
#include <class_board.h>
|
#include <class_board.h>
|
||||||
#include <view/view_group.h>
|
#include <view/view_group.h>
|
||||||
|
|
||||||
|
#include <functional>
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class PCB_TOOL
|
* Class PCB_TOOL
|
||||||
*
|
*
|
||||||
|
@ -78,6 +81,30 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callable that returns a new board item.
|
||||||
|
*
|
||||||
|
* The event that triggered it is provided, so you can check modifier
|
||||||
|
* keys, position, etc, if required
|
||||||
|
*/
|
||||||
|
using ITEM_CREATOR = std::function< std::unique_ptr< BOARD_ITEM >( const TOOL_EVENT& aEvt ) >;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper function for performing a common interactive idiom:
|
||||||
|
* wait for a left click, place an item there (perhaps with a
|
||||||
|
* dialog or other user interaction), then have it move with
|
||||||
|
* the mouse and respond to rotate/flip, etc
|
||||||
|
*
|
||||||
|
* More complex interactive processes are not supported here, you
|
||||||
|
* should implement a customised event loop for those.
|
||||||
|
*
|
||||||
|
* @param aItemCreator the callable that will attempt to create the item
|
||||||
|
* @param aCommitMessage the message used on a successful commit
|
||||||
|
*/
|
||||||
|
void doInteractiveItemPlacement( ITEM_CREATOR aItemCreator,
|
||||||
|
const wxString& aCommitMessage );
|
||||||
|
|
||||||
KIGFX::VIEW* view() const { return getView(); }
|
KIGFX::VIEW* view() const { return getView(); }
|
||||||
PCB_EDIT_FRAME* frame() const { return getEditFrame<PCB_EDIT_FRAME>(); }
|
PCB_EDIT_FRAME* frame() const { return getEditFrame<PCB_EDIT_FRAME>(); }
|
||||||
BOARD* board() const { return getModel<BOARD>(); }
|
BOARD* board() const { return getModel<BOARD>(); }
|
||||||
|
|
|
@ -38,6 +38,7 @@
|
||||||
#include <tools/pcb_editor_control.h>
|
#include <tools/pcb_editor_control.h>
|
||||||
#include <tools/placement_tool.h>
|
#include <tools/placement_tool.h>
|
||||||
#include <tools/pad_tool.h>
|
#include <tools/pad_tool.h>
|
||||||
|
#include <tools/microwave_tool.h>
|
||||||
#include <tools/pcb_actions.h>
|
#include <tools/pcb_actions.h>
|
||||||
|
|
||||||
#include <router/router_tool.h>
|
#include <router/router_tool.h>
|
||||||
|
@ -58,4 +59,5 @@ void PCB_ACTIONS::RegisterAllTools( TOOL_MANAGER* aToolManager )
|
||||||
aToolManager->RegisterTool( new PCBNEW_CONTROL );
|
aToolManager->RegisterTool( new PCBNEW_CONTROL );
|
||||||
aToolManager->RegisterTool( new PCB_EDITOR_CONTROL );
|
aToolManager->RegisterTool( new PCB_EDITOR_CONTROL );
|
||||||
aToolManager->RegisterTool( new PLACEMENT_TOOL );
|
aToolManager->RegisterTool( new PLACEMENT_TOOL );
|
||||||
|
aToolManager->RegisterTool( new MICROWAVE_TOOL );
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue