2015-02-12 03:22:24 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2014 John Beard, john.j.beard@gmail.com
|
2020-04-03 20:47:02 +00:00
|
|
|
* Copyright (C) 2018-2020 KiCad Developers, see AUTHORS.txt for contributors.
|
2015-02-12 03:22:24 +00:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
2019-02-23 13:58:57 +00:00
|
|
|
#include <dialogs/dialog_move_exact.h>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <math/util.h> // for KiROUND
|
2019-02-23 13:58:57 +00:00
|
|
|
#include <widgets/tab_traversal.h>
|
|
|
|
#include <pcb_edit_frame.h>
|
2020-10-14 03:37:48 +00:00
|
|
|
#include <trigo.h>
|
2015-02-12 03:22:24 +00:00
|
|
|
|
|
|
|
// initialise statics
|
|
|
|
DIALOG_MOVE_EXACT::MOVE_EXACT_OPTIONS DIALOG_MOVE_EXACT::m_options;
|
|
|
|
|
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
DIALOG_MOVE_EXACT::DIALOG_MOVE_EXACT( PCB_BASE_FRAME *aParent, wxPoint& aTranslate,
|
2019-06-20 13:28:51 +00:00
|
|
|
double& aRotate, ROTATION_ANCHOR& aAnchor,
|
|
|
|
const EDA_RECT& aBbox ) :
|
2015-02-12 03:22:24 +00:00
|
|
|
DIALOG_MOVE_EXACT_BASE( aParent ),
|
2018-06-12 07:36:35 +00:00
|
|
|
m_translation( aTranslate ),
|
|
|
|
m_rotation( aRotate ),
|
|
|
|
m_rotationAnchor( aAnchor ),
|
2019-06-20 13:28:51 +00:00
|
|
|
m_bbox( aBbox ),
|
2018-06-12 07:36:35 +00:00
|
|
|
m_moveX( aParent, m_xLabel, m_xEntry, m_xUnit ),
|
|
|
|
m_moveY( aParent, m_yLabel, m_yEntry, m_yUnit ),
|
2020-02-29 02:21:18 +00:00
|
|
|
m_rotate( aParent, m_rotLabel, m_rotEntry, m_rotUnit ),
|
|
|
|
m_stateX( 0.0 ),
|
|
|
|
m_stateY( 0.0 ),
|
|
|
|
m_stateRadius( 0.0 ),
|
|
|
|
m_stateTheta( 0.0 )
|
2015-02-12 03:22:24 +00:00
|
|
|
{
|
2019-08-26 17:36:35 +00:00
|
|
|
// We can't set the tab order through wxWidgets due to shortcomings in their mnemonics
|
|
|
|
// implementation on MSW
|
|
|
|
m_tabOrder = {
|
|
|
|
m_xEntry,
|
|
|
|
m_yEntry,
|
|
|
|
m_rotEntry,
|
|
|
|
m_anchorOptions,
|
|
|
|
m_stdButtonsOK,
|
|
|
|
m_stdButtonsCancel
|
|
|
|
};
|
2015-02-12 03:22:24 +00:00
|
|
|
|
2020-07-06 14:41:29 +00:00
|
|
|
// Configure display origin transforms
|
|
|
|
m_moveX.SetCoordType( ORIGIN_TRANSFORMS::REL_X_COORD );
|
|
|
|
m_moveY.SetCoordType( ORIGIN_TRANSFORMS::REL_Y_COORD );
|
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
updateDialogControls( m_options.polarCoords );
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
m_menuIDs.push_back( aAnchor );
|
|
|
|
m_menuIDs.push_back( ROTATE_AROUND_USER_ORIGIN );
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2019-09-05 22:00:47 +00:00
|
|
|
if( aParent->IsType( FRAME_PCB_EDITOR ) )
|
2018-06-12 07:36:35 +00:00
|
|
|
m_menuIDs.push_back( ROTATE_AROUND_AUX_ORIGIN );
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
buildRotationAnchorMenu();
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
// and set up the entries according to the saved options
|
|
|
|
m_polarCoords->SetValue( m_options.polarCoords );
|
2020-04-03 20:47:02 +00:00
|
|
|
m_moveX.SetValue( m_options.entry1 );
|
|
|
|
m_moveY.SetValue( m_options.entry2 );
|
2017-04-22 20:07:29 +00:00
|
|
|
|
2019-12-20 14:11:39 +00:00
|
|
|
m_rotate.SetUnits( EDA_UNITS::DEGREES );
|
2018-06-12 07:36:35 +00:00
|
|
|
m_rotate.SetValue( m_options.entryRotation );
|
|
|
|
m_anchorOptions->SetSelection( std::min( m_options.entryAnchorSelection, m_menuIDs.size() ) );
|
2015-02-12 03:22:24 +00:00
|
|
|
|
2015-02-22 14:43:44 +00:00
|
|
|
m_stdButtonsOK->SetDefault();
|
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
FinishDialogSettings();
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
void DIALOG_MOVE_EXACT::buildRotationAnchorMenu()
|
2015-02-12 03:22:24 +00:00
|
|
|
{
|
2018-06-12 07:36:35 +00:00
|
|
|
wxArrayString menuItems;
|
|
|
|
|
|
|
|
for( auto anchorID : m_menuIDs )
|
|
|
|
{
|
|
|
|
switch( anchorID )
|
|
|
|
{
|
|
|
|
case ROTATE_AROUND_ITEM_ANCHOR:
|
|
|
|
menuItems.push_back( _( "Rotate around item anchor" ) );
|
|
|
|
break;
|
|
|
|
case ROTATE_AROUND_SEL_CENTER:
|
|
|
|
menuItems.push_back( _( "Rotate around selection center" ) );
|
|
|
|
break;
|
|
|
|
case ROTATE_AROUND_USER_ORIGIN:
|
2018-12-03 22:11:23 +00:00
|
|
|
menuItems.push_back( _( "Rotate around local coordinates origin" ) );
|
2018-06-12 07:36:35 +00:00
|
|
|
break;
|
|
|
|
case ROTATE_AROUND_AUX_ORIGIN:
|
|
|
|
menuItems.push_back( _( "Rotate around drill/place origin" ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_anchorOptions->Set( menuItems );
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DIALOG_MOVE_EXACT::ToPolarDeg( double x, double y, double& r, double& q )
|
|
|
|
{
|
|
|
|
// convert to polar coordinates
|
2018-06-12 07:36:35 +00:00
|
|
|
r = hypot( x, y );
|
2015-02-12 03:22:24 +00:00
|
|
|
|
|
|
|
q = ( r != 0) ? RAD2DEG( atan2( y, x ) ) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-02-29 02:21:18 +00:00
|
|
|
bool DIALOG_MOVE_EXACT::GetTranslationInIU( wxRealPoint& val, bool polar )
|
2015-02-12 03:22:24 +00:00
|
|
|
{
|
|
|
|
if( polar )
|
|
|
|
{
|
2020-02-29 02:21:18 +00:00
|
|
|
const double r = m_moveX.GetDoubleValue();
|
|
|
|
const double q = m_moveY.GetDoubleValue();
|
2015-02-12 03:22:24 +00:00
|
|
|
|
|
|
|
val.x = r * cos( DEG2RAD( q / 10.0 ) );
|
|
|
|
val.y = r * sin( DEG2RAD( q / 10.0 ) );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// direct read
|
2020-02-29 02:21:18 +00:00
|
|
|
val.x = m_moveX.GetDoubleValue();
|
|
|
|
val.y = m_moveY.GetDoubleValue();
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// no validation to do here, but in future, you could return false here
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DIALOG_MOVE_EXACT::OnPolarChanged( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
bool newPolar = m_polarCoords->IsChecked();
|
2020-02-29 02:21:18 +00:00
|
|
|
double moveX = m_moveX.GetDoubleValue();
|
|
|
|
double moveY = m_moveY.GetDoubleValue();
|
2018-06-12 07:36:35 +00:00
|
|
|
updateDialogControls( newPolar );
|
|
|
|
|
2015-02-12 03:22:24 +00:00
|
|
|
if( newPolar )
|
|
|
|
{
|
2020-02-29 02:21:18 +00:00
|
|
|
if( moveX != m_stateX || moveY != m_stateY )
|
|
|
|
{
|
|
|
|
m_stateX = moveX;
|
|
|
|
m_stateY = moveY;
|
|
|
|
ToPolarDeg( m_stateX, m_stateY, m_stateRadius, m_stateTheta );
|
|
|
|
m_stateTheta *= 10.0;
|
|
|
|
|
|
|
|
m_moveX.SetDoubleValue( m_stateRadius );
|
|
|
|
m_stateRadius = m_moveX.GetDoubleValue();
|
|
|
|
m_moveY.SetDoubleValue( m_stateTheta );
|
|
|
|
m_stateTheta = m_moveY.GetDoubleValue();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_moveX.SetDoubleValue( m_stateRadius );
|
|
|
|
m_moveY.SetDoubleValue( m_stateTheta );
|
|
|
|
}
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-02-29 02:21:18 +00:00
|
|
|
if( moveX != m_stateRadius || moveY != m_stateTheta )
|
|
|
|
{
|
|
|
|
m_stateRadius = moveX;
|
|
|
|
m_stateTheta = moveY;
|
|
|
|
m_stateX = m_stateRadius * cos( DEG2RAD( m_stateTheta / 10.0 ) );
|
|
|
|
m_stateY = m_stateRadius * sin( DEG2RAD( m_stateTheta / 10.0 ) );
|
|
|
|
|
|
|
|
m_moveX.SetDoubleValue( m_stateX );
|
|
|
|
m_stateX = m_moveX.GetDoubleValue();
|
|
|
|
m_moveY.SetDoubleValue( m_stateY );
|
|
|
|
m_stateY = m_moveY.GetDoubleValue();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_moveX.SetDoubleValue( m_stateX );
|
|
|
|
m_moveY.SetDoubleValue( m_stateY );
|
|
|
|
}
|
2015-02-22 14:43:44 +00:00
|
|
|
}
|
2017-04-22 20:07:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-06-12 07:36:35 +00:00
|
|
|
void DIALOG_MOVE_EXACT::updateDialogControls( bool aPolar )
|
2015-02-22 14:43:44 +00:00
|
|
|
{
|
|
|
|
if( aPolar )
|
|
|
|
{
|
2018-06-12 07:36:35 +00:00
|
|
|
m_moveX.SetLabel( _( "Distance:" ) ); // Polar radius
|
|
|
|
m_moveY.SetLabel( _( "Angle:" ) ); // Polar theta or angle
|
2019-12-20 14:11:39 +00:00
|
|
|
m_moveY.SetUnits( EDA_UNITS::DEGREES );
|
2015-02-22 14:43:44 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-06-12 07:36:35 +00:00
|
|
|
m_moveX.SetLabel( _( "Move X:" ) );
|
|
|
|
m_moveY.SetLabel( _( "Move Y:" ) );
|
|
|
|
m_moveY.SetUnits( GetUserUnits() );
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
2018-06-12 07:36:35 +00:00
|
|
|
|
|
|
|
Layout();
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DIALOG_MOVE_EXACT::OnClear( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
wxObject* obj = event.GetEventObject();
|
|
|
|
|
|
|
|
if( obj == m_clearX )
|
|
|
|
{
|
2018-06-12 07:36:35 +00:00
|
|
|
m_moveX.SetValue( 0 );
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
else if( obj == m_clearY )
|
|
|
|
{
|
2018-06-12 07:36:35 +00:00
|
|
|
m_moveY.SetValue( 0 );
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
else if( obj == m_clearRot )
|
|
|
|
{
|
2018-06-12 07:36:35 +00:00
|
|
|
m_rotate.SetValue( 0 );
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
2018-02-08 09:51:05 +00:00
|
|
|
// Keep m_stdButtonsOK focused to allow enter key actiavte the OK button
|
|
|
|
m_stdButtonsOK->SetFocus();
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-08 09:51:05 +00:00
|
|
|
bool DIALOG_MOVE_EXACT::TransferDataFromWindow()
|
2015-02-12 03:22:24 +00:00
|
|
|
{
|
|
|
|
// for the output, we only deliver a Cartesian vector
|
2020-02-29 02:21:18 +00:00
|
|
|
wxRealPoint translation;
|
|
|
|
bool ok = GetTranslationInIU( translation, m_polarCoords->IsChecked() );
|
|
|
|
m_translation.x = KiROUND(translation.x);
|
|
|
|
m_translation.y = KiROUND(translation.y);
|
2020-10-30 16:59:04 +00:00
|
|
|
m_rotation = m_rotate.GetDoubleValue();
|
2018-06-12 07:36:35 +00:00
|
|
|
m_rotationAnchor = m_menuIDs[ m_anchorOptions->GetSelection() ];
|
2015-02-12 03:22:24 +00:00
|
|
|
|
2015-07-11 23:11:34 +00:00
|
|
|
if( ok )
|
2015-02-12 03:22:24 +00:00
|
|
|
{
|
|
|
|
// save the settings
|
|
|
|
m_options.polarCoords = m_polarCoords->GetValue();
|
2020-04-03 20:47:02 +00:00
|
|
|
m_options.entry1 = m_moveX.GetOriginalText();
|
|
|
|
m_options.entry2 = m_moveY.GetOriginalText();
|
|
|
|
m_options.entryRotation = m_rotate.GetOriginalText();
|
2018-06-12 07:36:35 +00:00
|
|
|
m_options.entryAnchorSelection = (size_t) std::max( m_anchorOptions->GetSelection(), 0 );
|
2018-02-08 09:51:05 +00:00
|
|
|
return true;
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
2018-02-08 09:51:05 +00:00
|
|
|
|
|
|
|
return false;
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DIALOG_MOVE_EXACT::OnTextFocusLost( wxFocusEvent& event )
|
|
|
|
{
|
|
|
|
wxTextCtrl* obj = static_cast<wxTextCtrl*>( event.GetEventObject() );
|
|
|
|
|
|
|
|
if( obj->GetValue().IsEmpty() )
|
2015-07-11 23:11:34 +00:00
|
|
|
obj->SetValue( "0" );
|
2015-02-20 19:04:32 +00:00
|
|
|
|
|
|
|
event.Skip();
|
2015-02-12 03:22:24 +00:00
|
|
|
}
|
2019-06-20 13:28:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
void DIALOG_MOVE_EXACT::OnTextChanged( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
|
2020-02-29 02:21:18 +00:00
|
|
|
double delta_x = m_moveX.GetDoubleValue();
|
|
|
|
double delta_y = m_moveY.GetDoubleValue();
|
2019-06-20 13:46:41 +00:00
|
|
|
double max_border = std::numeric_limits<int>::max() * 0.7071;
|
2019-06-20 13:28:51 +00:00
|
|
|
|
|
|
|
if( m_bbox.GetLeft() + delta_x < -max_border ||
|
|
|
|
m_bbox.GetRight() + delta_x > max_border ||
|
|
|
|
m_bbox.GetTop() + delta_y < -max_border ||
|
|
|
|
m_bbox.GetBottom() + delta_y > max_border )
|
|
|
|
{
|
|
|
|
const wxString invalid_length = _( "Invalid movement values. Movement would place selection "
|
|
|
|
"outside of the maximum board area." );
|
|
|
|
|
|
|
|
m_xEntry->SetToolTip( invalid_length );
|
|
|
|
m_xEntry->SetForegroundColour( *wxRED );
|
|
|
|
m_yEntry->SetToolTip( invalid_length );
|
|
|
|
m_yEntry->SetForegroundColour( *wxRED );
|
|
|
|
m_stdButtons->GetAffirmativeButton()->Disable();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_xEntry->SetToolTip( "" );
|
|
|
|
m_xEntry->SetForegroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOWTEXT ) );
|
|
|
|
m_yEntry->SetToolTip( "" );
|
|
|
|
m_yEntry->SetForegroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOWTEXT ) );
|
|
|
|
m_stdButtons->GetAffirmativeButton()->Enable();
|
|
|
|
event.Skip();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|