2019-05-08 18:56:03 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2019-08-14 08:28:07 +00:00
|
|
|
* Copyright (C) 2019 CERN
|
2019-05-08 18:56:03 +00:00
|
|
|
* Copyright (C) 2019 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 <functional>
|
|
|
|
using namespace std::placeholders;
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
#include "ee_point_editor.h"
|
2019-05-08 18:56:03 +00:00
|
|
|
#include <tool/tool_manager.h>
|
|
|
|
#include <view/view_controls.h>
|
|
|
|
#include <geometry/seg.h>
|
2019-05-10 17:19:48 +00:00
|
|
|
#include <tools/ee_actions.h>
|
|
|
|
#include <tools/ee_selection_tool.h>
|
2019-05-08 18:56:03 +00:00
|
|
|
#include <bitmaps.h>
|
|
|
|
#include <sch_edit_frame.h>
|
|
|
|
#include <sch_line.h>
|
2019-06-19 10:31:21 +00:00
|
|
|
#include <sch_bitmap.h>
|
2019-05-08 18:56:03 +00:00
|
|
|
#include <sch_sheet.h>
|
|
|
|
#include <lib_edit_frame.h>
|
|
|
|
#include <lib_arc.h>
|
|
|
|
#include <lib_circle.h>
|
|
|
|
#include <lib_rectangle.h>
|
|
|
|
#include <lib_polyline.h>
|
|
|
|
|
|
|
|
|
|
|
|
// Few constants to avoid using bare numbers for point indices
|
|
|
|
enum ARC_POINTS
|
|
|
|
{
|
|
|
|
ARC_CENTER, ARC_START, ARC_END
|
|
|
|
};
|
|
|
|
|
|
|
|
enum CIRCLE_POINTS
|
|
|
|
{
|
|
|
|
CIRC_CENTER, CIRC_END
|
|
|
|
};
|
|
|
|
|
|
|
|
enum RECTANGLE_POINTS
|
|
|
|
{
|
|
|
|
RECT_TOPLEFT, RECT_TOPRIGHT, RECT_BOTLEFT, RECT_BOTRIGHT
|
|
|
|
};
|
|
|
|
|
|
|
|
enum LINE_POINTS
|
|
|
|
{
|
|
|
|
LINE_START, LINE_END
|
|
|
|
};
|
|
|
|
|
|
|
|
class EDIT_POINTS_FACTORY
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
static std::shared_ptr<EDIT_POINTS> Make( EDA_ITEM* aItem, SCH_BASE_FRAME* frame )
|
|
|
|
{
|
|
|
|
std::shared_ptr<EDIT_POINTS> points = std::make_shared<EDIT_POINTS>( aItem );
|
|
|
|
|
|
|
|
if( !aItem )
|
|
|
|
return points;
|
|
|
|
|
|
|
|
// Generate list of edit points based on the item type
|
|
|
|
switch( aItem->Type() )
|
|
|
|
{
|
|
|
|
case LIB_ARC_T:
|
|
|
|
{
|
|
|
|
LIB_ARC* arc = (LIB_ARC*) aItem;
|
|
|
|
|
|
|
|
points->AddPoint( mapCoords( arc->GetPosition() ) );
|
|
|
|
points->AddPoint( mapCoords( arc->GetStart() ) );
|
|
|
|
points->AddPoint( mapCoords( arc->GetEnd() ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LIB_CIRCLE_T:
|
|
|
|
{
|
|
|
|
LIB_CIRCLE* circle = (LIB_CIRCLE*) aItem;
|
|
|
|
|
|
|
|
points->AddPoint( mapCoords( circle->GetPosition() ) );
|
|
|
|
points->AddPoint( mapCoords( circle->GetEnd() ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LIB_POLYLINE_T:
|
|
|
|
{
|
|
|
|
LIB_POLYLINE* lines = (LIB_POLYLINE*) aItem;
|
|
|
|
const std::vector<wxPoint>& pts = lines->GetPolyPoints();
|
|
|
|
|
|
|
|
for( wxPoint pt : pts )
|
|
|
|
points->AddPoint( mapCoords( pt ) );
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case LIB_RECTANGLE_T:
|
|
|
|
{
|
|
|
|
LIB_RECTANGLE* rect = (LIB_RECTANGLE*) aItem;
|
2019-07-26 08:40:40 +00:00
|
|
|
// point editor works only with rectangles having width and height > 0
|
|
|
|
// Some symbols can have rectangles with width or height < 0
|
|
|
|
// So normalize the size:
|
|
|
|
BOX2I dummy;
|
|
|
|
dummy.SetOrigin( mapCoords( rect->GetPosition() ) );
|
|
|
|
dummy.SetEnd( mapCoords( rect->GetEnd() ) );
|
|
|
|
dummy.Normalize();
|
|
|
|
VECTOR2I topLeft = dummy.GetPosition();
|
|
|
|
VECTOR2I botRight = dummy.GetEnd();
|
|
|
|
|
|
|
|
points->AddPoint( topLeft );
|
|
|
|
points->AddPoint( VECTOR2I( botRight.x, topLeft.y ) );
|
|
|
|
points->AddPoint( VECTOR2I( topLeft.x, botRight.y ) );
|
|
|
|
points->AddPoint( botRight );
|
2019-05-08 18:56:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SCH_SHEET_T:
|
|
|
|
{
|
|
|
|
SCH_SHEET* sheet = (SCH_SHEET*) aItem;
|
|
|
|
wxPoint topLeft = sheet->GetPosition();
|
|
|
|
wxPoint botRight = sheet->GetPosition() + sheet->GetSize();
|
|
|
|
|
|
|
|
points->AddPoint( (wxPoint) topLeft );
|
|
|
|
points->AddPoint( wxPoint( botRight.x, topLeft.y ) );
|
|
|
|
points->AddPoint( wxPoint( topLeft.x, botRight.y ) );
|
|
|
|
points->AddPoint( (wxPoint) botRight );
|
|
|
|
break;
|
|
|
|
}
|
2019-06-19 10:31:21 +00:00
|
|
|
case SCH_BITMAP_T:
|
|
|
|
{
|
|
|
|
SCH_BITMAP* bitmap = (SCH_BITMAP*) aItem;
|
|
|
|
wxPoint topLeft = bitmap->GetPosition() - bitmap->GetSize() / 2;
|
|
|
|
wxPoint botRight = bitmap->GetPosition() + bitmap->GetSize() / 2;
|
|
|
|
|
|
|
|
points->AddPoint( (wxPoint) topLeft );
|
|
|
|
points->AddPoint( wxPoint( botRight.x, topLeft.y ) );
|
|
|
|
points->AddPoint( wxPoint( topLeft.x, botRight.y ) );
|
|
|
|
points->AddPoint( (wxPoint) botRight );
|
|
|
|
break;
|
|
|
|
}
|
2019-05-08 18:56:03 +00:00
|
|
|
case SCH_LINE_T:
|
|
|
|
{
|
|
|
|
SCH_LINE* line = (SCH_LINE*) aItem;
|
|
|
|
SCH_LINE* connectedStart = nullptr;
|
|
|
|
SCH_LINE* connectedEnd = nullptr;
|
|
|
|
|
2019-06-25 23:39:58 +00:00
|
|
|
for( auto test : frame->GetScreen()->Items().OfType( SCH_LINE_T ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-06-25 23:39:58 +00:00
|
|
|
if( test->GetLayer() != LAYER_NOTES )
|
2019-05-08 18:56:03 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if( test == aItem )
|
|
|
|
continue;
|
|
|
|
|
2019-06-25 23:39:58 +00:00
|
|
|
auto testLine = static_cast<SCH_LINE*>( test );
|
2019-05-08 18:56:03 +00:00
|
|
|
testLine->ClearFlags( STARTPOINT | ENDPOINT );
|
|
|
|
|
|
|
|
if( testLine->GetStartPoint() == line->GetStartPoint() )
|
|
|
|
{
|
|
|
|
connectedStart = testLine;
|
|
|
|
testLine->SetFlags( STARTPOINT );
|
|
|
|
}
|
|
|
|
else if( testLine->GetEndPoint() == line->GetStartPoint() )
|
|
|
|
{
|
|
|
|
connectedStart = testLine;
|
|
|
|
testLine->SetFlags( ENDPOINT );
|
|
|
|
}
|
|
|
|
else if( testLine->GetStartPoint() == line->GetEndPoint() )
|
|
|
|
{
|
|
|
|
connectedEnd = testLine;
|
|
|
|
testLine->SetFlags( STARTPOINT );
|
|
|
|
}
|
|
|
|
else if( testLine->GetEndPoint() == line->GetEndPoint() )
|
|
|
|
{
|
|
|
|
connectedEnd = testLine;
|
|
|
|
testLine->SetFlags( ENDPOINT );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-25 23:39:58 +00:00
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
points->AddPoint( line->GetStartPoint(), connectedStart );
|
|
|
|
points->AddPoint( line->GetEndPoint(), connectedEnd );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
points.reset();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return points;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
EDIT_POINTS_FACTORY() {};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
EE_POINT_EDITOR::EE_POINT_EDITOR() :
|
2019-05-12 11:49:58 +00:00
|
|
|
EE_TOOL_BASE<SCH_BASE_FRAME>( "eeschema.PointEditor" ),
|
2019-05-08 18:56:03 +00:00
|
|
|
m_editedPoint( nullptr )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::Reset( RESET_REASON aReason )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-05-12 11:49:58 +00:00
|
|
|
EE_TOOL_BASE::Reset( aReason );
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
m_editPoints.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
bool EE_POINT_EDITOR::Init()
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-05-12 11:49:58 +00:00
|
|
|
EE_TOOL_BASE::Init();
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
auto& menu = m_selectionTool->GetToolMenu().GetMenu();
|
2019-05-10 17:19:48 +00:00
|
|
|
menu.AddItem( EE_ACTIONS::pointEditorAddCorner,
|
|
|
|
std::bind( &EE_POINT_EDITOR::addCornerCondition, this, _1 ) );
|
|
|
|
menu.AddItem( EE_ACTIONS::pointEditorRemoveCorner,
|
|
|
|
std::bind( &EE_POINT_EDITOR::removeCornerCondition, this, _1 ) );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::updateEditedPoint( const TOOL_EVENT& aEvent )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
EDIT_POINT* point = m_editedPoint;
|
|
|
|
|
|
|
|
if( aEvent.IsMotion() )
|
|
|
|
{
|
|
|
|
point = m_editPoints->FindPoint( aEvent.Position(), getView() );
|
|
|
|
}
|
|
|
|
else if( aEvent.IsDrag( BUT_LEFT ) )
|
|
|
|
{
|
|
|
|
point = m_editPoints->FindPoint( aEvent.DragOrigin(), getView() );
|
|
|
|
}
|
2019-06-27 21:33:48 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
point = m_editPoints->FindPoint( getViewControls()->GetCursorPosition(), getView() );
|
|
|
|
}
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
if( m_editedPoint != point )
|
|
|
|
setEditedPoint( point );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
int EE_POINT_EDITOR::Main( const TOOL_EVENT& aEvent )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-06-19 10:31:21 +00:00
|
|
|
static KICAD_T supportedTypes[] = {
|
|
|
|
LIB_ARC_T,
|
|
|
|
LIB_CIRCLE_T,
|
|
|
|
LIB_POLYLINE_T,
|
|
|
|
LIB_RECTANGLE_T,
|
|
|
|
SCH_SHEET_T,
|
|
|
|
SCH_LINE_LOCATE_GRAPHIC_LINE_T,
|
|
|
|
SCH_BITMAP_T,
|
|
|
|
EOT
|
|
|
|
};
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
if( !m_selectionTool )
|
|
|
|
return 0;
|
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
const EE_SELECTION& selection = m_selectionTool->GetSelection();
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-06-19 10:31:21 +00:00
|
|
|
if( selection.Size() != 1 || !selection.Front()->IsType( supportedTypes ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
return 0;
|
|
|
|
|
2019-05-24 11:39:30 +00:00
|
|
|
// Wait till drawing tool is done
|
|
|
|
if( selection.Front()->IsNew() )
|
|
|
|
return 0;
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
Activate();
|
|
|
|
|
|
|
|
KIGFX::VIEW_CONTROLS* controls = getViewControls();
|
|
|
|
KIGFX::VIEW* view = getView();
|
|
|
|
EDA_ITEM* item = (EDA_ITEM*) selection.Front();
|
|
|
|
|
|
|
|
controls->ShowCursor( true );
|
|
|
|
|
|
|
|
m_editPoints = EDIT_POINTS_FACTORY::Make( item, m_frame );
|
|
|
|
view->Add( m_editPoints.get() );
|
|
|
|
setEditedPoint( nullptr );
|
2019-06-27 21:33:48 +00:00
|
|
|
updateEditedPoint( aEvent );
|
2019-05-08 18:56:03 +00:00
|
|
|
bool inDrag = false;
|
|
|
|
bool modified = false;
|
|
|
|
|
|
|
|
// Main loop: keep receiving events
|
2019-06-17 13:43:22 +00:00
|
|
|
while( TOOL_EVENT* evt = Wait() )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-07-01 21:01:33 +00:00
|
|
|
if( !m_editPoints || evt->IsSelectionEvent() )
|
2019-05-08 18:56:03 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
if ( !inDrag )
|
|
|
|
updateEditedPoint( *evt );
|
|
|
|
|
|
|
|
if( evt->IsDrag( BUT_LEFT ) && m_editedPoint )
|
|
|
|
{
|
|
|
|
if( !inDrag )
|
|
|
|
{
|
|
|
|
saveItemsToUndo();
|
|
|
|
controls->ForceCursorPosition( false );
|
|
|
|
inDrag = true;
|
|
|
|
modified = true;
|
|
|
|
}
|
|
|
|
|
2019-06-29 18:56:07 +00:00
|
|
|
bool snap = !evt->Modifier( MD_ALT );
|
|
|
|
|
|
|
|
if( item->Type() == LIB_ARC_T && getEditedPointIndex() == ARC_CENTER )
|
|
|
|
snap = false;
|
|
|
|
|
|
|
|
m_editedPoint->SetPosition( controls->GetCursorPosition( snap ) );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
updateItem();
|
|
|
|
updatePoints();
|
|
|
|
}
|
|
|
|
|
2019-06-17 13:43:22 +00:00
|
|
|
else if( inDrag && evt->IsMouseUp( BUT_LEFT ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
controls->SetAutoPan( false );
|
|
|
|
inDrag = false;
|
|
|
|
}
|
|
|
|
|
2019-07-01 21:01:33 +00:00
|
|
|
else if( evt->IsCancelInteractive() || evt->IsActivate() )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( inDrag ) // Restore the last change
|
|
|
|
{
|
|
|
|
rollbackFromUndo();
|
2019-06-18 17:56:40 +00:00
|
|
|
inDrag = false;
|
2019-05-08 18:56:03 +00:00
|
|
|
modified = false;
|
2019-07-16 00:09:10 +00:00
|
|
|
break;
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
2019-07-01 21:01:33 +00:00
|
|
|
else if( evt->IsCancelInteractive() )
|
|
|
|
break;
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-07-01 21:01:33 +00:00
|
|
|
if( evt->IsActivate() && !evt->IsMoveTool() )
|
|
|
|
break;
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else
|
2019-06-16 11:06:49 +00:00
|
|
|
evt->SetPassEvent();
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
controls->SetAutoPan( inDrag );
|
|
|
|
controls->CaptureCursor( inDrag );
|
|
|
|
}
|
|
|
|
|
|
|
|
controls->SetAutoPan( false );
|
|
|
|
controls->CaptureCursor( false );
|
|
|
|
|
|
|
|
if( m_editPoints )
|
|
|
|
{
|
|
|
|
view->Remove( m_editPoints.get() );
|
|
|
|
|
|
|
|
if( modified )
|
|
|
|
m_frame->OnModify();
|
|
|
|
|
|
|
|
m_editPoints.reset();
|
|
|
|
m_frame->GetCanvas()->Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void pinEditedCorner( int editedPointIndex, int minWidth, int minHeight, VECTOR2I& topLeft,
|
|
|
|
VECTOR2I& topRight, VECTOR2I& botLeft, VECTOR2I& botRight )
|
|
|
|
{
|
|
|
|
switch( editedPointIndex )
|
|
|
|
{
|
|
|
|
case RECT_TOPLEFT:
|
|
|
|
// pin edited point within opposite corner
|
|
|
|
topLeft.x = std::min( topLeft.x, botRight.x - minWidth );
|
|
|
|
topLeft.y = std::min( topLeft.y, botRight.y - minHeight );
|
|
|
|
|
|
|
|
// push edited point edges to adjacent corners
|
|
|
|
topRight.y = topLeft.y;
|
|
|
|
botLeft.x = topLeft.x;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RECT_TOPRIGHT:
|
|
|
|
// pin edited point within opposite corner
|
|
|
|
topRight.x = std::max( topRight.x, botLeft.x + minWidth );
|
|
|
|
topRight.y = std::min( topRight.y, botLeft.y - minHeight );
|
|
|
|
|
|
|
|
// push edited point edges to adjacent corners
|
|
|
|
topLeft.y = topRight.y;
|
|
|
|
botRight.x = topRight.x;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RECT_BOTLEFT:
|
|
|
|
// pin edited point within opposite corner
|
|
|
|
botLeft.x = std::min( botLeft.x, topRight.x - minWidth );
|
|
|
|
botLeft.y = std::max( botLeft.y, topRight.y + minHeight );
|
|
|
|
|
|
|
|
// push edited point edges to adjacent corners
|
|
|
|
botRight.y = botLeft.y;
|
|
|
|
topLeft.x = botLeft.x;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RECT_BOTRIGHT:
|
|
|
|
// pin edited point within opposite corner
|
|
|
|
botRight.x = std::max( botRight.x, topLeft.x + minWidth );
|
|
|
|
botRight.y = std::max( botRight.y, topLeft.y + minHeight );
|
|
|
|
|
|
|
|
// push edited point edges to adjacent corners
|
|
|
|
botLeft.y = botRight.y;
|
|
|
|
topRight.x = botRight.x;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::updateItem() const
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
EDA_ITEM* item = m_editPoints->GetParent();
|
|
|
|
|
|
|
|
if( !item )
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch( item->Type() )
|
|
|
|
{
|
|
|
|
case LIB_ARC_T:
|
|
|
|
{
|
|
|
|
LIB_ARC* arc = (LIB_ARC*) item;
|
2019-06-29 13:39:46 +00:00
|
|
|
int i = getEditedPointIndex();
|
|
|
|
|
|
|
|
if( i == ARC_CENTER )
|
|
|
|
{
|
|
|
|
arc->SetEditState( 4 );
|
|
|
|
arc->CalcEdit( mapCoords( m_editPoints->Point( ARC_CENTER ).GetPosition() ) );
|
|
|
|
}
|
|
|
|
else if( i == ARC_START )
|
|
|
|
{
|
|
|
|
arc->SetEditState( 2 );
|
|
|
|
arc->CalcEdit( mapCoords( m_editPoints->Point( ARC_START ).GetPosition() ) );
|
|
|
|
}
|
|
|
|
else if( i == ARC_END )
|
|
|
|
{
|
|
|
|
arc->SetEditState( 3 );
|
|
|
|
arc->CalcEdit( mapCoords( m_editPoints->Point( ARC_END ).GetPosition() ) );
|
|
|
|
}
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LIB_CIRCLE_T:
|
|
|
|
{
|
|
|
|
LIB_CIRCLE* circle = (LIB_CIRCLE*) item;
|
|
|
|
|
|
|
|
circle->SetPosition( mapCoords( m_editPoints->Point( CIRC_CENTER ).GetPosition() ) );
|
|
|
|
circle->SetEnd( mapCoords( m_editPoints->Point( CIRC_END ).GetPosition() ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LIB_POLYLINE_T:
|
|
|
|
{
|
|
|
|
LIB_POLYLINE* lines = (LIB_POLYLINE*) item;
|
|
|
|
|
|
|
|
lines->ClearPoints();
|
|
|
|
|
2019-05-11 09:12:39 +00:00
|
|
|
for( unsigned i = 0; i < m_editPoints->PointsSize(); ++i )
|
2019-05-08 18:56:03 +00:00
|
|
|
lines->AddPoint( mapCoords( m_editPoints->Point( i ).GetPosition() ) );
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LIB_RECTANGLE_T:
|
|
|
|
{
|
|
|
|
VECTOR2I topLeft = m_editPoints->Point( RECT_TOPLEFT ).GetPosition();
|
|
|
|
VECTOR2I topRight = m_editPoints->Point( RECT_TOPRIGHT ).GetPosition();
|
|
|
|
VECTOR2I botLeft = m_editPoints->Point( RECT_BOTLEFT ).GetPosition();
|
|
|
|
VECTOR2I botRight = m_editPoints->Point( RECT_BOTRIGHT ).GetPosition();
|
|
|
|
|
|
|
|
pinEditedCorner( getEditedPointIndex(), Mils2iu( 1 ), Mils2iu( 1 ),
|
|
|
|
topLeft, topRight, botLeft, botRight );
|
|
|
|
|
2019-07-26 08:40:40 +00:00
|
|
|
LIB_RECTANGLE* rect = (LIB_RECTANGLE*) item;
|
2019-05-13 16:14:50 +00:00
|
|
|
rect->SetPosition( mapCoords( topLeft ) );
|
|
|
|
rect->SetEnd( mapCoords( botRight ) );
|
2019-05-08 18:56:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-06-19 10:31:21 +00:00
|
|
|
case SCH_BITMAP_T:
|
|
|
|
{
|
|
|
|
SCH_BITMAP* bitmap = (SCH_BITMAP*) item;
|
|
|
|
VECTOR2I topLeft = m_editPoints->Point( RECT_TOPLEFT ).GetPosition();
|
|
|
|
VECTOR2I topRight = m_editPoints->Point( RECT_TOPRIGHT ).GetPosition();
|
|
|
|
VECTOR2I botLeft = m_editPoints->Point( RECT_BOTLEFT ).GetPosition();
|
|
|
|
VECTOR2I botRight = m_editPoints->Point( RECT_BOTRIGHT ).GetPosition();
|
|
|
|
|
|
|
|
pinEditedCorner( getEditedPointIndex(), Mils2iu( 50 ), Mils2iu( 50 ),
|
|
|
|
topLeft, topRight, botLeft, botRight );
|
|
|
|
|
|
|
|
double oldWidth = bitmap->GetSize().x;
|
|
|
|
double newWidth = topRight.x - topLeft.x;
|
|
|
|
double widthRatio = newWidth / oldWidth;
|
|
|
|
|
|
|
|
double oldHeight = bitmap->GetSize().y;
|
|
|
|
double newHeight = botLeft.y - topLeft.y;
|
|
|
|
double heightRatio = newHeight / oldHeight;
|
|
|
|
|
|
|
|
bitmap->SetImageScale( bitmap->GetImageScale() * std::min( widthRatio, heightRatio ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
case SCH_SHEET_T:
|
|
|
|
{
|
|
|
|
SCH_SHEET* sheet = (SCH_SHEET*) item;
|
|
|
|
VECTOR2I topLeft = m_editPoints->Point( RECT_TOPLEFT ).GetPosition();
|
|
|
|
VECTOR2I topRight = m_editPoints->Point( RECT_TOPRIGHT ).GetPosition();
|
|
|
|
VECTOR2I botLeft = m_editPoints->Point( RECT_BOTLEFT ).GetPosition();
|
|
|
|
VECTOR2I botRight = m_editPoints->Point( RECT_BOTRIGHT ).GetPosition();
|
|
|
|
|
|
|
|
pinEditedCorner( getEditedPointIndex(), sheet->GetMinWidth(), sheet->GetMinHeight(),
|
|
|
|
topLeft, topRight, botLeft, botRight );
|
|
|
|
|
|
|
|
sheet->SetPosition( (wxPoint) topLeft );
|
|
|
|
sheet->SetSize( wxSize( botRight.x - topLeft.x, botRight.y - topLeft.y ) );
|
2019-05-16 21:27:03 +00:00
|
|
|
|
|
|
|
// Keep sheet pins attached to edges:
|
|
|
|
for( SCH_SHEET_PIN& pin : sheet->GetPins() )
|
|
|
|
{
|
|
|
|
wxPoint pos = pin.GetPosition();
|
|
|
|
|
|
|
|
switch( pin.GetEdge() )
|
|
|
|
{
|
2019-05-17 16:45:27 +00:00
|
|
|
case SHEET_LEFT_SIDE: pos.x = topLeft.x; break;
|
|
|
|
case SHEET_RIGHT_SIDE: pos.x = topRight.x; break;
|
|
|
|
case SHEET_TOP_SIDE: pos.y = topLeft.y; break;
|
|
|
|
case SHEET_BOTTOM_SIDE: pos.y = botLeft.y; break;
|
|
|
|
case SHEET_UNDEFINED_SIDE: break;
|
2019-05-16 21:27:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pin.SetPosition( pos );
|
|
|
|
}
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SCH_LINE_T:
|
|
|
|
{
|
|
|
|
SCH_LINE* line = (SCH_LINE*) item;
|
|
|
|
|
|
|
|
line->SetStartPoint( (wxPoint) m_editPoints->Point( LINE_START ).GetPosition() );
|
|
|
|
line->SetEndPoint( (wxPoint) m_editPoints->Point( LINE_END ).GetPosition() );
|
|
|
|
|
|
|
|
SCH_LINE* connection = (SCH_LINE*) ( m_editPoints->Point( LINE_START ).GetConnection() );
|
|
|
|
|
|
|
|
if( connection )
|
|
|
|
{
|
2019-08-27 12:12:34 +00:00
|
|
|
if( connection->HasFlag( STARTPOINT ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
connection->SetStartPoint( line->GetPosition() );
|
2019-08-27 12:12:34 +00:00
|
|
|
else if( connection->HasFlag( ENDPOINT ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
connection->SetEndPoint( line->GetPosition() );
|
|
|
|
|
|
|
|
getView()->Update( connection, KIGFX::GEOMETRY );
|
|
|
|
}
|
|
|
|
|
|
|
|
connection = (SCH_LINE*) ( m_editPoints->Point( LINE_END ).GetConnection() );
|
|
|
|
|
|
|
|
if( connection )
|
|
|
|
{
|
2019-08-27 12:12:34 +00:00
|
|
|
if( connection->HasFlag( STARTPOINT ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
connection->SetStartPoint( line->GetEndPoint() );
|
2019-08-27 12:12:34 +00:00
|
|
|
else if( connection->HasFlag( ENDPOINT ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
connection->SetEndPoint( line->GetEndPoint() );
|
|
|
|
|
|
|
|
getView()->Update( connection, KIGFX::GEOMETRY );
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-12 11:49:58 +00:00
|
|
|
updateView( item );
|
2019-05-08 18:56:03 +00:00
|
|
|
m_frame->SetMsgPanel( item );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::updatePoints()
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( !m_editPoints )
|
|
|
|
return;
|
|
|
|
|
|
|
|
EDA_ITEM* item = m_editPoints->GetParent();
|
|
|
|
|
|
|
|
if( !item )
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch( item->Type() )
|
|
|
|
{
|
|
|
|
case LIB_ARC_T:
|
|
|
|
{
|
|
|
|
LIB_ARC* arc = (LIB_ARC*) item;
|
|
|
|
|
|
|
|
m_editPoints->Point( ARC_CENTER ).SetPosition( mapCoords( arc->GetPosition() ) );
|
|
|
|
m_editPoints->Point( ARC_START ).SetPosition( mapCoords( arc->GetStart() ) );
|
|
|
|
m_editPoints->Point( ARC_END ).SetPosition( mapCoords( arc->GetEnd() ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LIB_CIRCLE_T:
|
|
|
|
{
|
|
|
|
LIB_CIRCLE* circle = (LIB_CIRCLE*) item;
|
|
|
|
|
|
|
|
m_editPoints->Point( CIRC_CENTER ).SetPosition( mapCoords( circle->GetPosition() ) );
|
|
|
|
m_editPoints->Point( CIRC_END ).SetPosition( mapCoords( circle->GetEnd() ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LIB_POLYLINE_T:
|
|
|
|
{
|
|
|
|
LIB_POLYLINE* lines = (LIB_POLYLINE*) item;
|
|
|
|
const std::vector<wxPoint>& pts = lines->GetPolyPoints();
|
|
|
|
|
|
|
|
if( m_editPoints->PointsSize() != (unsigned) pts.size() )
|
|
|
|
{
|
|
|
|
getView()->Remove( m_editPoints.get() );
|
|
|
|
m_editedPoint = nullptr;
|
|
|
|
m_editPoints = EDIT_POINTS_FACTORY::Make( item, m_frame );
|
|
|
|
getView()->Add(m_editPoints.get() );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( unsigned i = 0; i < pts.size(); i++ )
|
|
|
|
m_editPoints->Point( i ).SetPosition( mapCoords( pts[i] ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LIB_RECTANGLE_T:
|
|
|
|
{
|
|
|
|
LIB_RECTANGLE* rect = (LIB_RECTANGLE*) item;
|
2019-07-26 08:40:40 +00:00
|
|
|
// point editor works only with rectangles having width and height > 0
|
|
|
|
// Some symbols can have rectangles with width or height < 0
|
|
|
|
// So normalize the size:
|
|
|
|
BOX2I dummy;
|
|
|
|
dummy.SetOrigin( mapCoords( rect->GetPosition() ) );
|
|
|
|
dummy.SetEnd( mapCoords( rect->GetEnd() ) );
|
|
|
|
dummy.Normalize();
|
|
|
|
VECTOR2I topLeft = dummy.GetPosition();
|
|
|
|
VECTOR2I botRight = dummy.GetEnd();
|
|
|
|
|
|
|
|
m_editPoints->Point( RECT_TOPLEFT ).SetPosition( topLeft );
|
|
|
|
m_editPoints->Point( RECT_TOPRIGHT ).SetPosition( VECTOR2I( botRight.x, topLeft.y ) );
|
|
|
|
m_editPoints->Point( RECT_BOTLEFT ).SetPosition( VECTOR2I( topLeft.x, botRight.y ) );
|
|
|
|
m_editPoints->Point( RECT_BOTRIGHT ).SetPosition( botRight );
|
2019-05-08 18:56:03 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-06-19 10:31:21 +00:00
|
|
|
case SCH_BITMAP_T:
|
|
|
|
{
|
|
|
|
SCH_BITMAP* bitmap = (SCH_BITMAP*) item;
|
|
|
|
wxPoint topLeft = bitmap->GetPosition() - bitmap->GetSize() / 2;
|
|
|
|
wxPoint botRight = bitmap->GetPosition() + bitmap->GetSize() / 2;
|
|
|
|
|
|
|
|
m_editPoints->Point( RECT_TOPLEFT ).SetPosition( topLeft );
|
|
|
|
m_editPoints->Point( RECT_TOPRIGHT ).SetPosition( botRight.x, topLeft.y );
|
|
|
|
m_editPoints->Point( RECT_BOTLEFT ).SetPosition( topLeft.x, botRight.y );
|
|
|
|
m_editPoints->Point( RECT_BOTRIGHT ).SetPosition( botRight );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
case SCH_SHEET_T:
|
|
|
|
{
|
|
|
|
SCH_SHEET* sheet = (SCH_SHEET*) item;
|
|
|
|
wxPoint topLeft = sheet->GetPosition();
|
|
|
|
wxPoint botRight = sheet->GetPosition() + sheet->GetSize();
|
|
|
|
|
|
|
|
m_editPoints->Point( RECT_TOPLEFT ).SetPosition( topLeft );
|
|
|
|
m_editPoints->Point( RECT_TOPRIGHT ).SetPosition( botRight.x, topLeft.y );
|
|
|
|
m_editPoints->Point( RECT_BOTLEFT ).SetPosition( topLeft.x, botRight.y );
|
|
|
|
m_editPoints->Point( RECT_BOTRIGHT ).SetPosition( botRight );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case SCH_LINE_T:
|
|
|
|
{
|
|
|
|
SCH_LINE* line = (SCH_LINE*) item;
|
|
|
|
|
|
|
|
m_editPoints->Point( LINE_START ).SetPosition( line->GetStartPoint() );
|
|
|
|
m_editPoints->Point( LINE_END ).SetPosition( line->GetEndPoint() );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
getView()->Update( m_editPoints.get() );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::setEditedPoint( EDIT_POINT* aPoint )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
KIGFX::VIEW_CONTROLS* controls = getViewControls();
|
|
|
|
|
|
|
|
if( aPoint )
|
|
|
|
{
|
2019-06-29 13:39:46 +00:00
|
|
|
m_frame->GetCanvas()->SetCurrentCursor( wxCURSOR_ARROW );
|
2019-05-08 18:56:03 +00:00
|
|
|
controls->ForceCursorPosition( true, aPoint->GetPosition() );
|
|
|
|
controls->ShowCursor( true );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-06-29 13:39:46 +00:00
|
|
|
if( m_frame->ToolStackIsEmpty() )
|
|
|
|
controls->ShowCursor( false );
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
controls->ForceCursorPosition( false );
|
|
|
|
}
|
|
|
|
|
|
|
|
m_editedPoint = aPoint;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
bool EE_POINT_EDITOR::removeCornerCondition( const SELECTION& )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( !m_editPoints || !m_editedPoint )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
LIB_POLYLINE* polyLine = dynamic_cast<LIB_POLYLINE*>( m_editPoints->GetParent() );
|
|
|
|
|
|
|
|
if( !polyLine || polyLine->GetCornerCount() < 3 )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const std::vector<wxPoint>& pts = polyLine->GetPolyPoints();
|
|
|
|
|
2019-05-11 09:12:39 +00:00
|
|
|
for( unsigned i = 0; i < polyLine->GetCornerCount(); ++i )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( pts[i] == mapCoords( m_editedPoint->GetPosition() ) )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
bool EE_POINT_EDITOR::addCornerCondition( const SELECTION& )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( !m_editPoints || !m_editedPoint )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
LIB_POLYLINE* polyLine = dynamic_cast<LIB_POLYLINE*>( m_editPoints->GetParent() );
|
|
|
|
|
|
|
|
if( !polyLine )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
VECTOR2I cursorPos = getViewControls()->GetCursorPosition();
|
|
|
|
double threshold = getView()->ToWorld( EDIT_POINT::POINT_SIZE );
|
|
|
|
|
|
|
|
return polyLine->HitTest( mapCoords( cursorPos ), (int) threshold );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
int EE_POINT_EDITOR::addCorner( const TOOL_EVENT& aEvent )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( !m_editPoints )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
LIB_POLYLINE* polyLine = dynamic_cast<LIB_POLYLINE*>( m_editPoints->GetParent() );
|
|
|
|
|
|
|
|
if( !polyLine )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
VECTOR2I cursorPos = getViewControls()->GetCursorPosition( !aEvent.Modifier( MD_ALT ) );
|
|
|
|
polyLine->AddCorner( mapCoords( cursorPos ) );
|
|
|
|
|
2019-05-12 11:49:58 +00:00
|
|
|
updateView( polyLine );
|
2019-05-08 18:56:03 +00:00
|
|
|
updatePoints();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
int EE_POINT_EDITOR::removeCorner( const TOOL_EVENT& aEvent )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( !m_editPoints || !m_editedPoint )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
LIB_POLYLINE* polyLine = dynamic_cast<LIB_POLYLINE*>( m_editPoints->GetParent() );
|
|
|
|
|
|
|
|
if( !polyLine || polyLine->GetCornerCount() < 3 )
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
polyLine->RemoveCorner( getEditedPointIndex() );
|
|
|
|
|
2019-05-12 11:49:58 +00:00
|
|
|
updateView( polyLine );
|
2019-05-08 18:56:03 +00:00
|
|
|
updatePoints();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
int EE_POINT_EDITOR::modifiedSelection( const TOOL_EVENT& aEvent )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
updatePoints();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::saveItemsToUndo()
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( m_isLibEdit )
|
|
|
|
{
|
2019-05-12 11:49:58 +00:00
|
|
|
saveCopyInUndoList( m_editPoints->GetParent()->GetParent(), UR_LIBEDIT );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-05-12 11:49:58 +00:00
|
|
|
saveCopyInUndoList( (SCH_ITEM*) m_editPoints->GetParent(), UR_CHANGED );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
if( m_editPoints->GetParent()->Type() == SCH_LINE_T )
|
|
|
|
{
|
|
|
|
EDA_ITEM* connection = m_editPoints->Point( LINE_START ).GetConnection();
|
|
|
|
|
|
|
|
if( connection )
|
2019-05-12 11:49:58 +00:00
|
|
|
saveCopyInUndoList( (SCH_ITEM*) connection, UR_CHANGED, true );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
connection = m_editPoints->Point( LINE_END ).GetConnection();
|
|
|
|
|
|
|
|
if( connection )
|
2019-05-12 11:49:58 +00:00
|
|
|
saveCopyInUndoList( (SCH_ITEM*) connection, UR_CHANGED, true );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::rollbackFromUndo()
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( m_isLibEdit )
|
|
|
|
static_cast<LIB_EDIT_FRAME*>( m_frame )->RollbackPartFromUndo();
|
|
|
|
else
|
|
|
|
static_cast<SCH_EDIT_FRAME*>( m_frame )->RollbackSchematicFromUndo();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
void EE_POINT_EDITOR::setTransitions()
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-05-24 11:39:30 +00:00
|
|
|
Go( &EE_POINT_EDITOR::Main, EVENTS::SelectedEvent );
|
|
|
|
Go( &EE_POINT_EDITOR::Main, ACTIONS::activatePointEditor.MakeEvent() );
|
2019-05-10 17:19:48 +00:00
|
|
|
Go( &EE_POINT_EDITOR::addCorner, EE_ACTIONS::pointEditorAddCorner.MakeEvent() );
|
|
|
|
Go( &EE_POINT_EDITOR::removeCorner, EE_ACTIONS::pointEditorRemoveCorner.MakeEvent() );
|
|
|
|
Go( &EE_POINT_EDITOR::modifiedSelection, EVENTS::SelectedItemsModified );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|