2012-01-09 17:24:01 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
|
2019-04-11 01:23:34 +00:00
|
|
|
* Copyright (C) 2015-2019 KiCad Developers, see change_log.txt for contributors.
|
2012-01-09 17:24:01 +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
|
|
|
|
*/
|
|
|
|
|
2012-04-13 18:51:24 +00:00
|
|
|
#include <base_units.h>
|
Modular-Kicad milestone B), major portions:
*) Rework the set language support, simplify it by using KIWAY. Now any major
frame with a "change language" menu can change the language for all KIWAY_PLAYERs
in the whole KIWAY. Multiple KIWAYs are not supported yet.
*) Simplify "modal wxFrame" support, and add that support exclusively to
KIWAY_PLAYER where it is inherited by all derivatives. The function
KIWAY_PLAYER::ShowModal() is in the vtable and so is cross module capable.
*) Remove the requirements and assumptions that the wxFrame hierarchy always
had PCB_EDIT_FRAME and SCH_EDIT_FRAME as immediate parents of their viewers
and editors. This is no longer the case, nor required.
*) Use KIWAY::Player() everywhere to make KIWAY_PLAYERs, this registers the
KIWAY_PLAYER within the KIWAY and makes it very easy to find an open frame
quickly. It also gives control to the KIWAY as to frame hierarchical
relationships.
*) Change single_top to use the KIWAY for loading a KIFACE and instantiating
the single KIWAY_PLAYER, see bullet immediately above.
*) Add KIWAY::OnKiwayEnd() and call it from PGM_BASE at program termination, this
gives the KIFACEs a chance to save their final configuration dope to disk.
*) Add dedicated FRAME_T's for the modal frames, so m_Ident can be tested and
these modal frames are distinctly different than their non-modal equivalents.
KIWAY_PLAYER::IsModal() is !not! a valid test during the wxFrame's constructor,
so this is another important reason for having a dedicated FRAME_T for each
modal wxFrame.
On balance, more lines were deleted than were added to achieve all this.
2014-05-03 17:40:19 +00:00
|
|
|
#include <kiway.h>
|
2018-08-03 12:18:26 +00:00
|
|
|
#include <sch_draw_panel.h>
|
|
|
|
#include <sch_view.h>
|
|
|
|
#include <sch_painter.h>
|
|
|
|
#include <gal/graphics_abstraction_layer.h>
|
2017-10-06 18:07:43 +00:00
|
|
|
#include <confirm.h>
|
2018-09-05 22:17:22 +00:00
|
|
|
#include <preview_items/selection_area.h>
|
2017-10-06 18:07:43 +00:00
|
|
|
#include <class_library.h>
|
2017-08-12 12:09:39 +00:00
|
|
|
#include <eeschema_id.h>
|
2018-01-30 10:49:51 +00:00
|
|
|
#include <lib_edit_frame.h>
|
2017-08-12 12:09:39 +00:00
|
|
|
#include <viewlib_frame.h>
|
|
|
|
#include <sch_base_frame.h>
|
|
|
|
#include <symbol_lib_table.h>
|
2018-05-14 17:34:18 +00:00
|
|
|
#include <dialog_configure_paths.h>
|
2019-04-13 17:41:11 +00:00
|
|
|
#include <tool/tool_manager.h>
|
|
|
|
#include <tool/tool_dispatcher.h>
|
|
|
|
#include <tools/sch_actions.h>
|
2018-05-14 17:34:18 +00:00
|
|
|
|
2018-07-20 17:46:56 +00:00
|
|
|
#include "dialogs/panel_sym_lib_table.h"
|
2017-08-12 12:09:39 +00:00
|
|
|
|
|
|
|
|
2015-04-22 11:39:00 +00:00
|
|
|
|
2017-10-06 18:07:43 +00:00
|
|
|
LIB_ALIAS* SchGetLibAlias( const LIB_ID& aLibId, SYMBOL_LIB_TABLE* aLibTable, PART_LIB* aCacheLib,
|
|
|
|
wxWindow* aParent, bool aShowErrorMsg )
|
|
|
|
{
|
2017-10-22 00:48:25 +00:00
|
|
|
// wxCHECK_MSG( aLibId.IsValid(), NULL, "LIB_ID is not valid." );
|
2017-10-06 18:07:43 +00:00
|
|
|
wxCHECK_MSG( aLibTable, NULL, "Invalid symbol library table." );
|
|
|
|
|
|
|
|
LIB_ALIAS* alias = NULL;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
alias = aLibTable->LoadSymbol( aLibId );
|
|
|
|
|
|
|
|
if( !alias && aCacheLib )
|
2017-12-07 23:01:03 +00:00
|
|
|
alias = aCacheLib->FindAlias( aLibId );
|
2017-10-06 18:07:43 +00:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
|
|
|
if( aShowErrorMsg )
|
|
|
|
{
|
2019-04-17 19:09:48 +00:00
|
|
|
wxString msg = wxString::Format( _( "Error loading symbol '%s' from library '%s'." ),
|
|
|
|
aLibId.GetLibItemName().wx_str(),
|
|
|
|
aLibId.GetLibNickname().wx_str() );
|
2017-10-06 18:07:43 +00:00
|
|
|
DisplayErrorMessage( aParent, msg, ioe.What() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return alias;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LIB_PART* SchGetLibPart( const LIB_ID& aLibId, SYMBOL_LIB_TABLE* aLibTable, PART_LIB* aCacheLib,
|
|
|
|
wxWindow* aParent, bool aShowErrorMsg )
|
|
|
|
{
|
|
|
|
LIB_ALIAS* alias = SchGetLibAlias( aLibId, aLibTable, aCacheLib, aParent, aShowErrorMsg );
|
|
|
|
|
|
|
|
return ( alias ) ? alias->GetPart() : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-08 20:57:56 +00:00
|
|
|
// Static members:
|
2015-04-22 11:39:00 +00:00
|
|
|
|
2019-04-17 19:09:48 +00:00
|
|
|
SCH_BASE_FRAME::SCH_BASE_FRAME( KIWAY* aKiway, wxWindow* aParent, FRAME_T aWindowType,
|
|
|
|
const wxString& aTitle, const wxPoint& aPosition,
|
|
|
|
const wxSize& aSize, long aStyle, const wxString& aFrameName ) :
|
|
|
|
EDA_DRAW_FRAME( aKiway, aParent, aWindowType, aTitle, aPosition, aSize, aStyle, aFrameName )
|
2012-02-19 19:53:11 +00:00
|
|
|
{
|
2018-09-10 13:37:28 +00:00
|
|
|
createCanvas();
|
|
|
|
|
2015-04-22 11:39:00 +00:00
|
|
|
m_zoomLevelCoeff = 11.0; // Adjusted to roughly displays zoom level = 1
|
2015-01-06 07:38:58 +00:00
|
|
|
// when the screen shows a 1:1 image
|
|
|
|
// obviously depends on the monitor,
|
|
|
|
// but this is an acceptable value
|
2015-04-22 11:39:00 +00:00
|
|
|
m_repeatStep = wxPoint( DEFAULT_REPEAT_OFFSET_X, DEFAULT_REPEAT_OFFSET_Y );
|
|
|
|
m_repeatDeltaLabel = DEFAULT_REPEAT_LABEL_INC;
|
2012-02-19 19:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-12 12:09:39 +00:00
|
|
|
SCH_BASE_FRAME::~SCH_BASE_FRAME()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-04-13 17:41:11 +00:00
|
|
|
void SCH_BASE_FRAME::setupTools()
|
|
|
|
{
|
|
|
|
// Create the manager and dispatcher & route draw panel events to the dispatcher
|
|
|
|
m_toolManager = new TOOL_MANAGER;
|
|
|
|
m_toolManager->SetEnvironment( GetScreen(), GetCanvas()->GetView(),
|
|
|
|
GetCanvas()->GetViewControls(), this );
|
|
|
|
m_actions = new SCH_ACTIONS();
|
|
|
|
m_toolDispatcher = new TOOL_DISPATCHER( m_toolManager, m_actions );
|
|
|
|
|
|
|
|
// Register tools
|
|
|
|
m_actions->RegisterAllTools( m_toolManager );
|
|
|
|
m_toolManager->InitTools();
|
2019-04-23 09:56:57 +00:00
|
|
|
|
|
|
|
// Run the selection tool, it is supposed to be always active
|
2019-04-23 17:40:29 +00:00
|
|
|
m_toolManager->InvokeTool( "eeschema.InteractiveSelection" );
|
2019-04-23 09:56:57 +00:00
|
|
|
|
2019-04-13 17:41:11 +00:00
|
|
|
GetCanvas()->SetEventDispatcher( m_toolDispatcher );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-10 13:37:28 +00:00
|
|
|
void SCH_BASE_FRAME::OnUpdateSwitchCanvas( wxUpdateUIEvent& aEvent )
|
|
|
|
{
|
|
|
|
wxMenuBar* menuBar = GetMenuBar();
|
|
|
|
EDA_DRAW_PANEL_GAL* gal_canvas = GetGalCanvas();
|
|
|
|
EDA_DRAW_PANEL_GAL::GAL_TYPE canvasType = gal_canvas->GetBackend();
|
|
|
|
|
|
|
|
struct { int menuId; int galType; } menuList[] =
|
|
|
|
{
|
|
|
|
{ ID_MENU_CANVAS_OPENGL, EDA_DRAW_PANEL_GAL::GAL_TYPE_OPENGL },
|
|
|
|
{ ID_MENU_CANVAS_CAIRO, EDA_DRAW_PANEL_GAL::GAL_TYPE_CAIRO },
|
|
|
|
};
|
|
|
|
|
|
|
|
for( auto ii: menuList )
|
|
|
|
{
|
|
|
|
wxMenuItem* item = menuBar->FindItem( ii.menuId );
|
|
|
|
if( ii.galType == canvasType )
|
|
|
|
item->Check( true );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::OnSwitchCanvas( wxCommandEvent& aEvent )
|
|
|
|
{
|
|
|
|
auto new_type = EDA_DRAW_PANEL_GAL::GAL_TYPE_OPENGL;
|
|
|
|
|
|
|
|
if( aEvent.GetId() == ID_MENU_CANVAS_CAIRO )
|
|
|
|
new_type = EDA_DRAW_PANEL_GAL::GAL_TYPE_CAIRO;
|
|
|
|
|
|
|
|
if( m_canvasType == new_type )
|
|
|
|
return;
|
|
|
|
|
|
|
|
GetGalCanvas()->SwitchBackend( new_type );
|
2018-09-14 08:15:10 +00:00
|
|
|
m_canvasType = new_type;
|
2018-09-10 13:37:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-19 19:53:11 +00:00
|
|
|
void SCH_BASE_FRAME::OnOpenLibraryViewer( wxCommandEvent& event )
|
|
|
|
{
|
Modular-Kicad milestone B), major portions:
*) Rework the set language support, simplify it by using KIWAY. Now any major
frame with a "change language" menu can change the language for all KIWAY_PLAYERs
in the whole KIWAY. Multiple KIWAYs are not supported yet.
*) Simplify "modal wxFrame" support, and add that support exclusively to
KIWAY_PLAYER where it is inherited by all derivatives. The function
KIWAY_PLAYER::ShowModal() is in the vtable and so is cross module capable.
*) Remove the requirements and assumptions that the wxFrame hierarchy always
had PCB_EDIT_FRAME and SCH_EDIT_FRAME as immediate parents of their viewers
and editors. This is no longer the case, nor required.
*) Use KIWAY::Player() everywhere to make KIWAY_PLAYERs, this registers the
KIWAY_PLAYER within the KIWAY and makes it very easy to find an open frame
quickly. It also gives control to the KIWAY as to frame hierarchical
relationships.
*) Change single_top to use the KIWAY for loading a KIFACE and instantiating
the single KIWAY_PLAYER, see bullet immediately above.
*) Add KIWAY::OnKiwayEnd() and call it from PGM_BASE at program termination, this
gives the KIFACEs a chance to save their final configuration dope to disk.
*) Add dedicated FRAME_T's for the modal frames, so m_Ident can be tested and
these modal frames are distinctly different than their non-modal equivalents.
KIWAY_PLAYER::IsModal() is !not! a valid test during the wxFrame's constructor,
so this is another important reason for having a dedicated FRAME_T for each
modal wxFrame.
On balance, more lines were deleted than were added to achieve all this.
2014-05-03 17:40:19 +00:00
|
|
|
LIB_VIEW_FRAME* viewlibFrame = (LIB_VIEW_FRAME*) Kiway().Player( FRAME_SCH_VIEWER, true );
|
2012-09-12 09:53:11 +00:00
|
|
|
|
2015-08-25 14:16:15 +00:00
|
|
|
viewlibFrame->PushPreferences( m_canvas );
|
|
|
|
|
2015-07-16 19:29:40 +00:00
|
|
|
// On Windows, Raise() does not bring the window on screen, when iconized
|
|
|
|
if( viewlibFrame->IsIconized() )
|
|
|
|
viewlibFrame->Iconize( false );
|
|
|
|
|
Modular-Kicad milestone B), major portions:
*) Rework the set language support, simplify it by using KIWAY. Now any major
frame with a "change language" menu can change the language for all KIWAY_PLAYERs
in the whole KIWAY. Multiple KIWAYs are not supported yet.
*) Simplify "modal wxFrame" support, and add that support exclusively to
KIWAY_PLAYER where it is inherited by all derivatives. The function
KIWAY_PLAYER::ShowModal() is in the vtable and so is cross module capable.
*) Remove the requirements and assumptions that the wxFrame hierarchy always
had PCB_EDIT_FRAME and SCH_EDIT_FRAME as immediate parents of their viewers
and editors. This is no longer the case, nor required.
*) Use KIWAY::Player() everywhere to make KIWAY_PLAYERs, this registers the
KIWAY_PLAYER within the KIWAY and makes it very easy to find an open frame
quickly. It also gives control to the KIWAY as to frame hierarchical
relationships.
*) Change single_top to use the KIWAY for loading a KIFACE and instantiating
the single KIWAY_PLAYER, see bullet immediately above.
*) Add KIWAY::OnKiwayEnd() and call it from PGM_BASE at program termination, this
gives the KIFACEs a chance to save their final configuration dope to disk.
*) Add dedicated FRAME_T's for the modal frames, so m_Ident can be tested and
these modal frames are distinctly different than their non-modal equivalents.
KIWAY_PLAYER::IsModal() is !not! a valid test during the wxFrame's constructor,
so this is another important reason for having a dedicated FRAME_T for each
modal wxFrame.
On balance, more lines were deleted than were added to achieve all this.
2014-05-03 17:40:19 +00:00
|
|
|
viewlibFrame->Show( true );
|
|
|
|
viewlibFrame->Raise();
|
2012-02-19 19:53:11 +00:00
|
|
|
}
|
2012-01-09 17:24:01 +00:00
|
|
|
|
2017-08-12 12:09:39 +00:00
|
|
|
|
2012-01-09 17:24:01 +00:00
|
|
|
SCH_SCREEN* SCH_BASE_FRAME::GetScreen() const
|
|
|
|
{
|
|
|
|
return (SCH_SCREEN*) EDA_DRAW_FRAME::GetScreen();
|
|
|
|
}
|
|
|
|
|
2017-10-06 18:07:43 +00:00
|
|
|
|
2019-04-17 19:09:48 +00:00
|
|
|
void SCH_BASE_FRAME::SetScreen( BASE_SCREEN* aScreen )
|
|
|
|
{
|
|
|
|
EDA_DRAW_FRAME::SetScreen( aScreen );
|
|
|
|
|
|
|
|
if( m_toolManager )
|
|
|
|
{
|
|
|
|
m_toolManager->SetEnvironment( GetScreen(), GetCanvas()->GetView(),
|
|
|
|
GetCanvas()->GetViewControls(), this );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-06 07:38:58 +00:00
|
|
|
const wxString SCH_BASE_FRAME::GetZoomLevelIndicator() const
|
|
|
|
{
|
|
|
|
return EDA_DRAW_FRAME::GetZoomLevelIndicator();
|
|
|
|
}
|
2012-01-09 17:24:01 +00:00
|
|
|
|
2017-10-06 18:07:43 +00:00
|
|
|
|
2012-01-09 17:24:01 +00:00
|
|
|
void SCH_BASE_FRAME::SetPageSettings( const PAGE_INFO& aPageSettings )
|
|
|
|
{
|
|
|
|
GetScreen()->SetPageSettings( aPageSettings );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const PAGE_INFO& SCH_BASE_FRAME::GetPageSettings () const
|
|
|
|
{
|
|
|
|
return GetScreen()->GetPageSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const wxSize SCH_BASE_FRAME::GetPageSizeIU() const
|
|
|
|
{
|
|
|
|
// GetSizeIU is compile time dependent:
|
|
|
|
return GetScreen()->GetPageSettings().GetSizeIU();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-03 05:15:23 +00:00
|
|
|
const wxPoint& SCH_BASE_FRAME::GetAuxOrigin() const
|
2012-01-09 17:24:01 +00:00
|
|
|
{
|
|
|
|
wxASSERT( GetScreen() );
|
2013-08-03 05:15:23 +00:00
|
|
|
return GetScreen()->GetAuxOrigin();
|
2012-01-09 17:24:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-03 05:15:23 +00:00
|
|
|
void SCH_BASE_FRAME::SetAuxOrigin( const wxPoint& aPosition )
|
2012-01-09 17:24:01 +00:00
|
|
|
{
|
|
|
|
wxASSERT( GetScreen() );
|
2013-08-03 05:15:23 +00:00
|
|
|
GetScreen()->SetAuxOrigin( aPosition );
|
2012-01-09 17:24:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const TITLE_BLOCK& SCH_BASE_FRAME::GetTitleBlock() const
|
|
|
|
{
|
|
|
|
wxASSERT( GetScreen() );
|
|
|
|
return GetScreen()->GetTitleBlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::SetTitleBlock( const TITLE_BLOCK& aTitleBlock )
|
|
|
|
{
|
|
|
|
wxASSERT( GetScreen() );
|
|
|
|
GetScreen()->SetTitleBlock( aTitleBlock );
|
|
|
|
}
|
2012-04-13 18:51:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::UpdateStatusBar()
|
|
|
|
{
|
|
|
|
wxString line;
|
|
|
|
BASE_SCREEN* screen = GetScreen();
|
|
|
|
|
|
|
|
if( !screen )
|
|
|
|
return;
|
|
|
|
|
|
|
|
EDA_DRAW_FRAME::UpdateStatusBar();
|
|
|
|
|
|
|
|
// Display absolute coordinates:
|
2018-02-02 15:56:09 +00:00
|
|
|
double dXpos = To_User_Unit( GetUserUnits(), GetCrossHairPosition().x );
|
|
|
|
double dYpos = To_User_Unit( GetUserUnits(), GetCrossHairPosition().y );
|
2012-04-13 18:51:24 +00:00
|
|
|
|
2018-02-02 15:56:09 +00:00
|
|
|
if ( GetUserUnits() == MILLIMETRES )
|
2012-04-13 18:51:24 +00:00
|
|
|
{
|
|
|
|
dXpos = RoundTo0( dXpos, 100.0 );
|
|
|
|
dYpos = RoundTo0( dYpos, 100.0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
wxString absformatter;
|
|
|
|
wxString locformatter;
|
|
|
|
|
2018-02-02 15:56:09 +00:00
|
|
|
switch( GetUserUnits() )
|
2012-04-13 18:51:24 +00:00
|
|
|
{
|
|
|
|
case INCHES:
|
2019-02-14 09:55:57 +00:00
|
|
|
absformatter = "X %.3f Y %.3f";
|
|
|
|
locformatter = "dx %.3f dy %.3f dist %.3f";
|
2012-04-13 18:51:24 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MILLIMETRES:
|
2019-02-14 09:55:57 +00:00
|
|
|
absformatter = "X %.2f Y %.2f";
|
|
|
|
locformatter = "dx %.2f dy %.2f dist %.2f";
|
2012-04-13 18:51:24 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case UNSCALED_UNITS:
|
2019-02-14 09:55:57 +00:00
|
|
|
absformatter = "X %f Y %f";
|
|
|
|
locformatter = "dx %f dy %f dist %f";
|
2012-04-13 18:51:24 +00:00
|
|
|
break;
|
2015-02-19 15:03:45 +00:00
|
|
|
|
|
|
|
case DEGREES:
|
|
|
|
wxASSERT( false );
|
|
|
|
break;
|
2012-04-13 18:51:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
line.Printf( absformatter, dXpos, dYpos );
|
|
|
|
SetStatusText( line, 2 );
|
|
|
|
|
|
|
|
// Display relative coordinates:
|
2019-02-14 09:55:57 +00:00
|
|
|
double dx = (double)GetCrossHairPosition().x - (double)screen->m_O_Curseur.x;
|
|
|
|
double dy = (double)GetCrossHairPosition().y - (double)screen->m_O_Curseur.y;
|
2013-08-03 05:15:23 +00:00
|
|
|
|
2018-02-02 15:56:09 +00:00
|
|
|
dXpos = To_User_Unit( GetUserUnits(), dx );
|
|
|
|
dYpos = To_User_Unit( GetUserUnits(), dy );
|
2012-04-13 18:51:24 +00:00
|
|
|
|
2018-02-02 15:56:09 +00:00
|
|
|
if( GetUserUnits() == MILLIMETRES )
|
2012-04-13 18:51:24 +00:00
|
|
|
{
|
|
|
|
dXpos = RoundTo0( dXpos, 100.0 );
|
|
|
|
dYpos = RoundTo0( dYpos, 100.0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
// We already decided the formatter above
|
2013-05-01 17:32:36 +00:00
|
|
|
line.Printf( locformatter, dXpos, dYpos, hypot( dXpos, dYpos ) );
|
2012-04-13 18:51:24 +00:00
|
|
|
SetStatusText( line, 3 );
|
|
|
|
|
|
|
|
// refresh units display
|
|
|
|
DisplayUnitsMsg();
|
|
|
|
}
|
2017-08-12 12:09:39 +00:00
|
|
|
|
|
|
|
|
2018-05-03 13:52:43 +00:00
|
|
|
void SCH_BASE_FRAME::OnConfigurePaths( wxCommandEvent& aEvent )
|
|
|
|
{
|
2018-05-14 17:34:18 +00:00
|
|
|
DIALOG_CONFIGURE_PATHS dlg( this, nullptr );
|
|
|
|
dlg.ShowModal();
|
2018-05-03 13:52:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-12 12:09:39 +00:00
|
|
|
void SCH_BASE_FRAME::OnEditSymbolLibTable( wxCommandEvent& aEvent )
|
|
|
|
{
|
2018-08-06 18:33:28 +00:00
|
|
|
InvokeSchEditSymbolLibTable( &Kiway(), this );
|
2017-08-12 12:09:39 +00:00
|
|
|
}
|
2017-10-06 18:07:43 +00:00
|
|
|
|
|
|
|
|
2018-08-06 18:33:28 +00:00
|
|
|
LIB_ALIAS* SCH_BASE_FRAME::GetLibAlias( const LIB_ID& aLibId, bool aUseCacheLib, bool aShowError )
|
2018-07-20 17:46:56 +00:00
|
|
|
{
|
2017-10-06 18:07:43 +00:00
|
|
|
PART_LIB* cache = ( aUseCacheLib ) ? Prj().SchLibs()->GetCacheLibrary() : NULL;
|
|
|
|
|
2018-08-06 18:33:28 +00:00
|
|
|
return SchGetLibAlias( aLibId, Prj().SchSymbolLibTable(), cache, this, aShowError );
|
2017-10-06 18:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LIB_PART* SCH_BASE_FRAME::GetLibPart( const LIB_ID& aLibId, bool aUseCacheLib, bool aShowErrorMsg )
|
|
|
|
{
|
|
|
|
PART_LIB* cache = ( aUseCacheLib ) ? Prj().SchLibs()->GetCacheLibrary() : NULL;
|
|
|
|
|
|
|
|
return SchGetLibPart( aLibId, Prj().SchSymbolLibTable(), cache, this, aShowErrorMsg );
|
|
|
|
}
|
2017-11-13 10:53:19 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool SCH_BASE_FRAME::saveSymbolLibTables( bool aGlobal, bool aProject )
|
|
|
|
{
|
2018-08-06 18:33:28 +00:00
|
|
|
wxString msg;
|
2017-11-13 10:53:19 +00:00
|
|
|
bool success = true;
|
|
|
|
|
|
|
|
if( aGlobal )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2018-08-06 18:33:28 +00:00
|
|
|
SYMBOL_LIB_TABLE::GetGlobalLibTable().Save( SYMBOL_LIB_TABLE::GetGlobalTableFileName() );
|
2017-11-13 10:53:19 +00:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
|
|
|
success = false;
|
2019-04-17 19:09:48 +00:00
|
|
|
msg.Printf( _( "Error saving global symbol library table:\n%s" ), ioe.What() );
|
2017-11-13 10:53:19 +00:00
|
|
|
wxMessageBox( msg, _( "File Save Error" ), wxOK | wxICON_ERROR );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( aProject && !Prj().GetProjectName().IsEmpty() )
|
|
|
|
{
|
|
|
|
wxFileName fn( Prj().GetProjectPath(), SYMBOL_LIB_TABLE::GetSymbolLibTableFileName() );
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
Prj().SchSymbolLibTable()->Save( fn.GetFullPath() );
|
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
|
|
|
success = false;
|
2019-04-17 19:09:48 +00:00
|
|
|
msg.Printf( _( "Error saving project-specific symbol library table:\n%s" ), ioe.What() );
|
2017-11-13 10:53:19 +00:00
|
|
|
wxMessageBox( msg, _( "File Save Error" ), wxOK | wxICON_ERROR );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
2018-08-03 12:18:26 +00:00
|
|
|
|
2018-09-08 20:57:56 +00:00
|
|
|
|
|
|
|
// Set the zoom level to show the contents of the view.
|
2018-08-03 12:18:26 +00:00
|
|
|
void SCH_BASE_FRAME::Zoom_Automatique( bool aWarpPointer )
|
|
|
|
{
|
2018-09-03 13:58:47 +00:00
|
|
|
EDA_DRAW_PANEL_GAL* galCanvas = GetGalCanvas();
|
|
|
|
KIGFX::VIEW* view = GetGalCanvas()->GetView();
|
2018-08-03 12:18:26 +00:00
|
|
|
|
|
|
|
BOX2I bBox = GetDocumentExtents();
|
|
|
|
|
|
|
|
VECTOR2D scrollbarSize = VECTOR2D( galCanvas->GetSize() - galCanvas->GetClientSize() );
|
|
|
|
VECTOR2D screenSize = view->ToWorld( galCanvas->GetClientSize(), false );
|
|
|
|
|
|
|
|
if( bBox.GetWidth() == 0 || bBox.GetHeight() == 0 )
|
|
|
|
bBox = galCanvas->GetDefaultViewBBox();
|
|
|
|
|
|
|
|
VECTOR2D vsize = bBox.GetSize();
|
|
|
|
double scale = view->GetScale() / std::max( fabs( vsize.x / screenSize.x ),
|
|
|
|
fabs( vsize.y / screenSize.y ) );
|
|
|
|
|
|
|
|
// Reserve a 10% margin around component bounding box.
|
|
|
|
double margin_scale_factor = 1.1;
|
|
|
|
|
|
|
|
// Leave 20% for library editors & viewers
|
|
|
|
if( IsType( FRAME_PCB_MODULE_VIEWER ) || IsType( FRAME_PCB_MODULE_VIEWER_MODAL )
|
|
|
|
|| IsType( FRAME_SCH_VIEWER ) || IsType( FRAME_SCH_VIEWER_MODAL )
|
|
|
|
|| IsType( FRAME_SCH_LIB_EDITOR ) || IsType( FRAME_PCB_MODULE_EDITOR ) )
|
|
|
|
{
|
|
|
|
margin_scale_factor = 1.2;
|
|
|
|
}
|
|
|
|
|
2018-09-03 13:58:47 +00:00
|
|
|
scale /= margin_scale_factor;
|
|
|
|
|
|
|
|
GetScreen()->SetScalingFactor( 1 / scale );
|
|
|
|
|
|
|
|
view->SetScale( scale );
|
2018-08-03 12:18:26 +00:00
|
|
|
view->SetCenter( bBox.Centre() );
|
|
|
|
|
|
|
|
// Take scrollbars into account
|
|
|
|
VECTOR2D worldScrollbarSize = view->ToWorld( scrollbarSize, false );
|
|
|
|
view->SetCenter( view->GetCenter() + worldScrollbarSize / 2.0 );
|
|
|
|
galCanvas->Refresh();
|
|
|
|
}
|
|
|
|
|
2018-09-08 20:57:56 +00:00
|
|
|
|
|
|
|
// Set the zoom level to show the area of aRect
|
|
|
|
void SCH_BASE_FRAME::Window_Zoom( EDA_RECT& aRect )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2018-09-03 13:58:47 +00:00
|
|
|
KIGFX::VIEW* view = GetGalCanvas()->GetView();
|
2018-09-08 20:57:56 +00:00
|
|
|
BOX2I selectionBox ( aRect.GetPosition(), aRect.GetSize() );
|
2018-08-03 12:18:26 +00:00
|
|
|
|
|
|
|
VECTOR2D screenSize = view->ToWorld( GetGalCanvas()->GetClientSize(), false );
|
|
|
|
|
|
|
|
if( selectionBox.GetWidth() == 0 || selectionBox.GetHeight() == 0 )
|
|
|
|
return;
|
|
|
|
|
|
|
|
VECTOR2D vsize = selectionBox.GetSize();
|
|
|
|
double scale;
|
2019-04-17 19:09:48 +00:00
|
|
|
double ratio = std::max( fabs( vsize.x / screenSize.x ), fabs( vsize.y / screenSize.y ) );
|
2018-08-03 12:18:26 +00:00
|
|
|
|
|
|
|
scale = view->GetScale() / ratio;
|
|
|
|
|
2018-09-03 13:58:47 +00:00
|
|
|
GetScreen()->SetScalingFactor( 1 / scale );
|
|
|
|
|
2018-08-03 12:18:26 +00:00
|
|
|
view->SetScale( scale );
|
|
|
|
view->SetCenter( selectionBox.Centre() );
|
|
|
|
GetGalCanvas()->Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::RedrawScreen( const wxPoint& aCenterPoint, bool aWarpPointer )
|
|
|
|
{
|
2018-09-03 13:58:47 +00:00
|
|
|
KIGFX::GAL* gal = GetCanvas()->GetGAL();
|
|
|
|
|
|
|
|
double selectedZoom = GetScreen()->GetZoom();
|
|
|
|
double zoomFactor = gal->GetWorldScale() / gal->GetZoomFactor();
|
|
|
|
double scale = 1.0 / ( zoomFactor * selectedZoom );
|
|
|
|
|
|
|
|
if( aCenterPoint != wxPoint( 0, 0 ) )
|
|
|
|
GetCanvas()->GetView()->SetScale( scale, aCenterPoint );
|
|
|
|
else
|
|
|
|
GetCanvas()->GetView()->SetScale( scale );
|
|
|
|
|
2019-03-11 21:32:05 +00:00
|
|
|
GetCanvas()->GetView()->SetCenter( aCenterPoint );
|
|
|
|
|
2018-09-03 13:58:47 +00:00
|
|
|
if( aWarpPointer )
|
2018-09-14 08:15:10 +00:00
|
|
|
GetCanvas()->GetViewControls()->CenterOnCursor();
|
2018-09-03 13:58:47 +00:00
|
|
|
|
|
|
|
GetCanvas()->Refresh();
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::RedrawScreen2( const wxPoint& posBefore )
|
|
|
|
{
|
2018-09-03 13:58:47 +00:00
|
|
|
KIGFX::GAL* gal = GetCanvas()->GetGAL();
|
|
|
|
|
|
|
|
double selectedZoom = GetScreen()->GetZoom();
|
|
|
|
double zoomFactor = gal->GetWorldScale() / gal->GetZoomFactor();
|
|
|
|
double scale = 1.0 / ( zoomFactor * selectedZoom );
|
|
|
|
|
|
|
|
GetCanvas()->GetView()->SetScale( scale );
|
|
|
|
|
2018-08-03 12:18:26 +00:00
|
|
|
GetGalCanvas()->Refresh();
|
|
|
|
}
|
|
|
|
|
2018-09-05 22:17:22 +00:00
|
|
|
|
2018-09-08 20:57:56 +00:00
|
|
|
void SCH_BASE_FRAME::CenterScreen( const wxPoint& aCenterPoint, bool aWarpPointer )
|
|
|
|
{
|
|
|
|
GetCanvas()->GetView()->SetCenter( aCenterPoint );
|
|
|
|
|
|
|
|
if( aWarpPointer )
|
2018-11-12 22:09:03 +00:00
|
|
|
GetCanvas()->GetViewControls()->WarpCursor( aCenterPoint, true );
|
2018-09-08 20:57:56 +00:00
|
|
|
|
|
|
|
GetGalCanvas()->Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-05 22:17:22 +00:00
|
|
|
void SCH_BASE_FRAME::HardRedraw()
|
|
|
|
{
|
2019-04-11 01:23:34 +00:00
|
|
|
GetGalCanvas()->GetView()->UpdateAllItems( KIGFX::ALL );
|
|
|
|
GetGalCanvas()->ForceRefresh();
|
2018-09-05 22:17:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-02 20:19:22 +00:00
|
|
|
SCH_DRAW_PANEL* SCH_BASE_FRAME::GetCanvas() const
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
|
|
|
return static_cast<SCH_DRAW_PANEL*>( GetGalCanvas() );
|
2018-09-02 20:19:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
KIGFX::SCH_RENDER_SETTINGS* SCH_BASE_FRAME::GetRenderSettings()
|
|
|
|
{
|
|
|
|
KIGFX::PAINTER* painter = GetGalCanvas()->GetView()->GetPainter();
|
|
|
|
return static_cast<KIGFX::SCH_RENDER_SETTINGS*>( painter->GetSettings() );
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SCH_BASE_FRAME::HandleBlockBegin( wxDC* aDC, EDA_KEY aKey, const wxPoint& aPosition,
|
2018-10-30 10:48:23 +00:00
|
|
|
int aExplicitCommand )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
|
|
|
BLOCK_SELECTOR* block = &GetScreen()->m_BlockLocate;
|
|
|
|
|
|
|
|
if( ( block->GetCommand() != BLOCK_IDLE ) || ( block->GetState() != STATE_NO_BLOCK ) )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if( aExplicitCommand == 0 )
|
|
|
|
block->SetCommand( (BLOCK_COMMAND_T) BlockCommand( aKey ) );
|
|
|
|
else
|
|
|
|
block->SetCommand( (BLOCK_COMMAND_T) aExplicitCommand );
|
|
|
|
|
|
|
|
if( block->GetCommand() == 0 )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch( block->GetCommand() )
|
|
|
|
{
|
|
|
|
case BLOCK_IDLE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLOCK_MOVE: // Move
|
|
|
|
case BLOCK_DRAG: // Drag (block defined)
|
|
|
|
case BLOCK_DRAG_ITEM: // Drag from a drag item command
|
|
|
|
case BLOCK_DUPLICATE: // Duplicate
|
|
|
|
case BLOCK_DUPLICATE_AND_INCREMENT: // Duplicate and increment relevant references
|
|
|
|
case BLOCK_DELETE: // Delete
|
|
|
|
case BLOCK_COPY: // Copy
|
|
|
|
case BLOCK_FLIP: // Flip
|
|
|
|
case BLOCK_ZOOM: // Window Zoom
|
|
|
|
case BLOCK_MIRROR_X:
|
|
|
|
case BLOCK_MIRROR_Y: // mirror
|
|
|
|
case BLOCK_PRESELECT_MOVE: // Move with preselection list
|
|
|
|
block->InitData( m_canvas, aPosition );
|
|
|
|
GetCanvas()->GetView()->ShowSelectionArea();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLOCK_PASTE:
|
2018-09-05 22:17:22 +00:00
|
|
|
{
|
2018-08-03 12:18:26 +00:00
|
|
|
block->InitData( m_canvas, aPosition );
|
|
|
|
InitBlockPasteInfos();
|
|
|
|
|
2018-10-30 10:48:23 +00:00
|
|
|
wxRect bounds( 0, 0, 0, 0 );
|
|
|
|
|
|
|
|
for( unsigned i = 0; i < block->GetCount(); ++i )
|
|
|
|
bounds.Union( block->GetItem( i )->GetBoundingBox() );
|
|
|
|
|
|
|
|
block->SetOrigin( bounds.GetPosition() );
|
|
|
|
block->SetSize( bounds.GetSize() );
|
|
|
|
block->SetLastCursorPosition( wxPoint( 0, 0 ) );
|
2018-09-05 22:17:22 +00:00
|
|
|
|
2018-08-03 12:18:26 +00:00
|
|
|
if( block->GetCount() == 0 ) // No data to paste
|
|
|
|
{
|
2018-09-05 22:17:22 +00:00
|
|
|
DisplayError( this, _( "Nothing to paste" ), 20 );
|
2018-08-03 12:18:26 +00:00
|
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_IDLE );
|
|
|
|
m_canvas->SetMouseCaptureCallback( NULL );
|
|
|
|
block->SetState( STATE_NO_BLOCK );
|
|
|
|
block->SetMessageBlock( this );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( !m_canvas->IsMouseCaptured() )
|
|
|
|
{
|
|
|
|
block->ClearItemsList();
|
2018-09-05 22:17:22 +00:00
|
|
|
wxFAIL_MSG( "SCH_BASE_FRAME::HandleBlockBegin() error: m_mouseCaptureCallback NULL" );
|
2018-08-03 12:18:26 +00:00
|
|
|
block->SetState( STATE_NO_BLOCK );
|
|
|
|
block->SetMessageBlock( this );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
block->SetState( STATE_BLOCK_MOVE );
|
2018-09-05 22:17:22 +00:00
|
|
|
block->SetFlags( IS_MOVED );
|
2018-08-03 12:18:26 +00:00
|
|
|
m_canvas->CallMouseCapture( aDC, aPosition, false );
|
2018-09-06 21:42:51 +00:00
|
|
|
m_canvas->Refresh();
|
2018-09-05 22:17:22 +00:00
|
|
|
}
|
2018-08-03 12:18:26 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2018-09-05 22:17:22 +00:00
|
|
|
wxFAIL_MSG( wxString::Format( "SCH_BASE_FRAME::HandleBlockBegin() unknown command: %s",
|
|
|
|
block->GetCommand() ) );
|
2018-08-03 12:18:26 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
block->SetMessageBlock( this );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-09-10 13:37:28 +00:00
|
|
|
void SCH_BASE_FRAME::createCanvas()
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2018-09-15 07:00:13 +00:00
|
|
|
m_canvasType = LoadCanvasTypeSetting();
|
2018-09-10 13:37:28 +00:00
|
|
|
|
|
|
|
// Allows only a CAIRO or OPENGL canvas:
|
2019-04-17 19:09:48 +00:00
|
|
|
if( m_canvasType != EDA_DRAW_PANEL_GAL::GAL_TYPE_OPENGL
|
|
|
|
&& m_canvasType != EDA_DRAW_PANEL_GAL::GAL_TYPE_CAIRO )
|
|
|
|
{
|
2018-09-15 07:00:13 +00:00
|
|
|
m_canvasType = EDA_DRAW_PANEL_GAL::GAL_TYPE_OPENGL;
|
2019-04-17 19:09:48 +00:00
|
|
|
}
|
2018-09-10 13:37:28 +00:00
|
|
|
|
2018-08-28 22:32:36 +00:00
|
|
|
m_canvas = new SCH_DRAW_PANEL( this, wxID_ANY, wxPoint( 0, 0 ), m_FrameSize,
|
2018-09-15 07:00:13 +00:00
|
|
|
GetGalDisplayOptions(), m_canvasType );
|
2018-08-03 12:18:26 +00:00
|
|
|
|
|
|
|
m_useSingleCanvasPane = true;
|
|
|
|
|
|
|
|
SetGalCanvas( static_cast<SCH_DRAW_PANEL*> (m_canvas) );
|
|
|
|
UseGalCanvas( true );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-10 10:58:50 +00:00
|
|
|
void SCH_BASE_FRAME::RefreshItem( SCH_ITEM* aItem, bool isAddOrDelete )
|
|
|
|
{
|
|
|
|
EDA_ITEM* parent = aItem->GetParent();
|
|
|
|
|
|
|
|
if( aItem->Type() == SCH_SHEET_PIN_T )
|
|
|
|
{
|
|
|
|
// Sheet pins aren't in the view. Refresh their parent.
|
|
|
|
if( parent )
|
|
|
|
GetCanvas()->GetView()->Update( parent );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( !isAddOrDelete )
|
|
|
|
GetCanvas()->GetView()->Update( aItem );
|
|
|
|
|
|
|
|
// Component children are drawn from their parents. Mark them for re-paint.
|
|
|
|
if( parent && parent->Type() == SCH_COMPONENT_T )
|
|
|
|
GetCanvas()->GetView()->Update( parent, KIGFX::REPAINT );
|
|
|
|
}
|
|
|
|
|
|
|
|
GetCanvas()->Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-03-11 21:32:05 +00:00
|
|
|
void SCH_BASE_FRAME::AddToScreen( SCH_ITEM* aItem, SCH_SCREEN* aScreen )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2019-03-11 21:32:05 +00:00
|
|
|
auto screen = aScreen;
|
|
|
|
|
|
|
|
if( aScreen == nullptr )
|
|
|
|
screen = GetScreen();
|
|
|
|
|
|
|
|
screen->Append( aItem );
|
|
|
|
|
|
|
|
if( screen == GetScreen() )
|
|
|
|
{
|
|
|
|
GetCanvas()->GetView()->Add( aItem );
|
|
|
|
RefreshItem( aItem, true ); // handle any additional parent semantics
|
|
|
|
}
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
2018-09-08 18:07:41 +00:00
|
|
|
|
2019-03-11 21:32:05 +00:00
|
|
|
void SCH_BASE_FRAME::RemoveFromScreen( SCH_ITEM* aItem, SCH_SCREEN* aScreen )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2019-03-11 21:32:05 +00:00
|
|
|
auto screen = aScreen;
|
|
|
|
|
|
|
|
if( aScreen == nullptr )
|
|
|
|
screen = GetScreen();
|
|
|
|
|
|
|
|
if( screen == GetScreen() )
|
|
|
|
GetCanvas()->GetView()->Remove( aItem );
|
|
|
|
|
|
|
|
screen->Remove( aItem );
|
|
|
|
|
|
|
|
if( screen == GetScreen() )
|
|
|
|
RefreshItem( aItem, true ); // handle any additional parent semantics
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
2018-09-08 18:07:41 +00:00
|
|
|
|
2018-08-03 12:18:26 +00:00
|
|
|
void SCH_BASE_FRAME::SyncView()
|
|
|
|
{
|
|
|
|
auto screen = GetScreen();
|
|
|
|
auto gal = GetGalCanvas()->GetGAL();
|
|
|
|
|
|
|
|
auto gs = screen->GetGridSize();
|
|
|
|
gal->SetGridSize( VECTOR2D( gs.x, gs.y ));
|
|
|
|
GetGalCanvas()->GetView()->UpdateAllItems( KIGFX::ALL );
|
|
|
|
}
|