summaryrefslogtreecommitdiff
path: root/common/draw_panel_gal.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'common/draw_panel_gal.cpp')
-rw-r--r--common/draw_panel_gal.cpp415
1 files changed, 415 insertions, 0 deletions
diff --git a/common/draw_panel_gal.cpp b/common/draw_panel_gal.cpp
new file mode 100644
index 0000000..2cd2e4c
--- /dev/null
+++ b/common/draw_panel_gal.cpp
@@ -0,0 +1,415 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2013-2015 CERN
+ * @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
+ * @author Maciej Suminski <maciej.suminski@cern.ch>
+ *
+ * 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 <wx/wx.h>
+#include <wx/frame.h>
+#include <wx/window.h>
+#include <wx/event.h>
+#include <wx/colour.h>
+#include <wx/filename.h>
+#include <confirm.h>
+
+#include <class_draw_panel_gal.h>
+#include <view/view.h>
+#include <view/wx_view_controls.h>
+#include <pcb_painter.h>
+
+#include <gal/graphics_abstraction_layer.h>
+#include <gal/opengl/opengl_gal.h>
+#include <gal/cairo/cairo_gal.h>
+
+#include <tool/tool_dispatcher.h>
+#include <tool/tool_manager.h>
+
+#include <boost/foreach.hpp>
+
+#ifdef __WXDEBUG__
+#include <profile.h>
+#endif /* __WXDEBUG__ */
+
+EDA_DRAW_PANEL_GAL::EDA_DRAW_PANEL_GAL( wxWindow* aParentWindow, wxWindowID aWindowId,
+ const wxPoint& aPosition, const wxSize& aSize,
+ GAL_TYPE aGalType ) :
+ wxScrolledCanvas( aParentWindow, aWindowId, aPosition, aSize )
+{
+ m_parent = aParentWindow;
+ m_gal = NULL;
+ m_backend = GAL_TYPE_NONE;
+ m_view = NULL;
+ m_painter = NULL;
+ m_eventDispatcher = NULL;
+ m_lostFocus = false;
+
+ SetLayoutDirection( wxLayout_LeftToRight );
+
+ SwitchBackend( aGalType );
+ SetBackgroundStyle( wxBG_STYLE_CUSTOM );
+
+// Scrollbars broken in GAL on OSX
+#ifdef __WXMAC__
+ ShowScrollbars( wxSHOW_SB_NEVER, wxSHOW_SB_NEVER );
+#else
+ ShowScrollbars( wxSHOW_SB_ALWAYS, wxSHOW_SB_ALWAYS );
+#endif
+ EnableScrolling( false, false ); // otherwise Zoom Auto disables GAL canvas
+
+ m_painter = new KIGFX::PCB_PAINTER( m_gal );
+
+ m_view = new KIGFX::VIEW( true );
+ m_view->SetPainter( m_painter );
+ m_view->SetGAL( m_gal );
+
+ Connect( wxEVT_SIZE, wxSizeEventHandler( EDA_DRAW_PANEL_GAL::onSize ), NULL, this );
+ Connect( wxEVT_ENTER_WINDOW, wxEventHandler( EDA_DRAW_PANEL_GAL::onEnter ), NULL, this );
+ Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( EDA_DRAW_PANEL_GAL::onLostFocus ), NULL, this );
+
+ const wxEventType events[] =
+ {
+ wxEVT_LEFT_UP, wxEVT_LEFT_DOWN, wxEVT_LEFT_DCLICK,
+ wxEVT_RIGHT_UP, wxEVT_RIGHT_DOWN, wxEVT_RIGHT_DCLICK,
+ wxEVT_MIDDLE_UP, wxEVT_MIDDLE_DOWN, wxEVT_MIDDLE_DCLICK,
+ wxEVT_MOTION, wxEVT_MOUSEWHEEL, wxEVT_CHAR,
+#ifdef USE_OSX_MAGNIFY_EVENT
+ wxEVT_MAGNIFY,
+#endif
+ KIGFX::WX_VIEW_CONTROLS::EVT_REFRESH_MOUSE
+ };
+
+ BOOST_FOREACH( wxEventType eventType, events )
+ {
+ Connect( eventType, wxEventHandler( EDA_DRAW_PANEL_GAL::onEvent ),
+ NULL, m_eventDispatcher );
+ }
+
+ // View controls is the first in the event handler chain, so the Tool Framework operates
+ // on updated viewport data.
+ m_viewControls = new KIGFX::WX_VIEW_CONTROLS( m_view, this );
+
+ // Set up timer that prevents too frequent redraw commands
+ m_refreshTimer.SetOwner( this );
+ m_pendingRefresh = false;
+ m_drawing = false;
+ m_drawingEnabled = false;
+ Connect( wxEVT_TIMER, wxTimerEventHandler( EDA_DRAW_PANEL_GAL::onRefreshTimer ), NULL, this );
+}
+
+
+EDA_DRAW_PANEL_GAL::~EDA_DRAW_PANEL_GAL()
+{
+ delete m_painter;
+ delete m_viewControls;
+ delete m_view;
+ delete m_gal;
+}
+
+
+void EDA_DRAW_PANEL_GAL::SetFocus()
+{
+// Windows has a strange manner on bringing up and activating windows
+// containing a GAL canvas just after moving the mouse cursor into its area.
+// Feel free to uncomment or extend the following #ifdef if you experience
+// similar problems on your platform.
+#ifdef __WINDOWS__
+ if( !GetParent()->IsDescendant( wxWindow::FindFocus() ) )
+ return;
+#endif
+
+ wxScrolledCanvas::SetFocus();
+ m_lostFocus = false;
+}
+
+
+void EDA_DRAW_PANEL_GAL::onPaint( wxPaintEvent& WXUNUSED( aEvent ) )
+{
+ m_pendingRefresh = false;
+
+ if( m_drawing )
+ return;
+
+ m_drawing = true;
+
+// Scrollbars broken in GAL on OSX
+#ifndef __WXMAC__
+ m_viewControls->UpdateScrollbars();
+#endif
+
+ m_view->UpdateItems();
+ m_gal->BeginDrawing();
+ m_gal->ClearScreen( m_painter->GetSettings()->GetBackgroundColor() );
+
+ KIGFX::COLOR4D gridColor = static_cast<KIGFX::PCB_RENDER_SETTINGS*> (m_painter->GetSettings())->GetLayerColor( ITEM_GAL_LAYER ( GRID_VISIBLE ) );
+ m_gal->SetGridColor ( gridColor );
+
+ if( m_view->IsDirty() )
+ {
+ m_view->ClearTargets();
+
+ // Grid has to be redrawn only when the NONCACHED target is redrawn
+ if( m_view->IsTargetDirty( KIGFX::TARGET_NONCACHED ) )
+ m_gal->DrawGrid();
+
+ m_view->Redraw();
+ }
+
+ m_gal->DrawCursor( m_viewControls->GetCursorPosition() );
+ m_gal->EndDrawing();
+
+ m_lastRefresh = wxGetLocalTimeMillis();
+ m_drawing = false;
+}
+
+
+void EDA_DRAW_PANEL_GAL::onSize( wxSizeEvent& aEvent )
+{
+ m_gal->ResizeScreen( aEvent.GetSize().x, aEvent.GetSize().y );
+ m_view->MarkTargetDirty( KIGFX::TARGET_CACHED );
+ m_view->MarkTargetDirty( KIGFX::TARGET_NONCACHED );
+}
+
+
+void EDA_DRAW_PANEL_GAL::Refresh( bool aEraseBackground, const wxRect* aRect )
+{
+ if( m_pendingRefresh )
+ return;
+
+ m_pendingRefresh = true;
+
+#ifdef __WXMAC__
+ // Timers on OS X may have a high latency (seen up to 500ms and more) which
+ // makes repaints jerky. No negative impact seen without throttling, so just
+ // do an unconditional refresh for OS X.
+ ForceRefresh();
+#else
+ wxLongLong t = wxGetLocalTimeMillis();
+ wxLongLong delta = t - m_lastRefresh;
+
+ if( delta >= MinRefreshPeriod )
+ {
+ ForceRefresh();
+ }
+ else
+ {
+ // One shot timer
+ m_refreshTimer.Start( ( MinRefreshPeriod - delta ).ToLong(), true );
+ }
+#endif
+}
+
+
+void EDA_DRAW_PANEL_GAL::ForceRefresh()
+{
+ wxPaintEvent redrawEvent;
+ wxPostEvent( this, redrawEvent );
+}
+
+
+void EDA_DRAW_PANEL_GAL::SetEventDispatcher( TOOL_DISPATCHER* aEventDispatcher )
+{
+ m_eventDispatcher = aEventDispatcher;
+ const wxEventType eventTypes[] = { wxEVT_TOOL };
+
+ if( m_eventDispatcher )
+ {
+ BOOST_FOREACH( wxEventType type, eventTypes )
+ {
+ m_parent->Connect( type, wxCommandEventHandler( TOOL_DISPATCHER::DispatchWxCommand ),
+ NULL, m_eventDispatcher );
+ }
+ }
+ else
+ {
+ BOOST_FOREACH( wxEventType type, eventTypes )
+ {
+ // While loop is used to be sure that all event handlers are removed.
+ while( m_parent->Disconnect( type,
+ wxCommandEventHandler( TOOL_DISPATCHER::DispatchWxCommand ),
+ NULL, m_eventDispatcher ) );
+ }
+ }
+}
+
+
+void EDA_DRAW_PANEL_GAL::StartDrawing()
+{
+ // Start querying GAL if it is ready
+ m_refreshTimer.StartOnce( 100 );
+}
+
+
+void EDA_DRAW_PANEL_GAL::StopDrawing()
+{
+ m_drawingEnabled = false;
+ Disconnect( wxEVT_PAINT, wxPaintEventHandler( EDA_DRAW_PANEL_GAL::onPaint ), NULL, this );
+ m_pendingRefresh = false;
+ m_drawing = true;
+ m_refreshTimer.Stop();
+}
+
+
+void EDA_DRAW_PANEL_GAL::SetHighContrastLayer( LAYER_ID aLayer )
+{
+ // Set display settings for high contrast mode
+ KIGFX::RENDER_SETTINGS* rSettings = m_view->GetPainter()->GetSettings();
+
+ SetTopLayer( aLayer );
+
+ rSettings->ClearActiveLayers();
+ rSettings->SetActiveLayer( aLayer );
+
+ m_view->UpdateAllLayersColor();
+}
+
+
+void EDA_DRAW_PANEL_GAL::SetTopLayer( LAYER_ID aLayer )
+{
+ m_view->ClearTopLayers();
+ m_view->SetTopLayer( aLayer );
+ m_view->UpdateAllLayersOrder();
+}
+
+
+double EDA_DRAW_PANEL_GAL::GetLegacyZoom() const
+{
+ double zoomFactor = m_gal->GetWorldScale() / m_gal->GetZoomFactor();
+ return ( 1.0 / ( zoomFactor * m_view->GetScale() ) );
+}
+
+
+bool EDA_DRAW_PANEL_GAL::SwitchBackend( GAL_TYPE aGalType )
+{
+ // Do not do anything if the currently used GAL is correct
+ if( aGalType == m_backend && m_gal != NULL )
+ return true;
+
+ bool result = true; // assume everything will be fine
+
+ // Prevent refreshing canvas during backend switch
+ StopDrawing();
+
+ KIGFX::GAL* new_gal = NULL;
+
+ try
+ {
+ switch( aGalType )
+ {
+ case GAL_TYPE_OPENGL:
+ new_gal = new KIGFX::OPENGL_GAL( this, this, this );
+ break;
+
+ case GAL_TYPE_CAIRO:
+ new_gal = new KIGFX::CAIRO_GAL( this, this, this );
+ break;
+
+ default:
+ assert( false );
+ // warn about unhandled GAL canvas type, but continue with the fallback option
+
+ case GAL_TYPE_NONE:
+ // KIGFX::GAL is a stub - it actually does cannot display anything,
+ // but prevents code relying on GAL canvas existence from crashing
+ new_gal = new KIGFX::GAL();
+ break;
+ }
+ }
+ catch( std::runtime_error& err )
+ {
+ new_gal = new KIGFX::GAL();
+ aGalType = GAL_TYPE_NONE;
+ DisplayError( m_parent, wxString( err.what() ) );
+ result = false;
+ }
+
+ assert( new_gal );
+ delete m_gal;
+ m_gal = new_gal;
+
+ wxSize size = GetClientSize();
+ m_gal->ResizeScreen( size.GetX(), size.GetY() );
+
+ if( m_painter )
+ m_painter->SetGAL( m_gal );
+
+ if( m_view )
+ m_view->SetGAL( m_gal );
+
+ m_backend = aGalType;
+
+ return result;
+}
+
+
+void EDA_DRAW_PANEL_GAL::onEvent( wxEvent& aEvent )
+{
+ if( m_lostFocus )
+ SetFocus();
+
+ if( !m_eventDispatcher )
+ aEvent.Skip();
+ else
+ m_eventDispatcher->DispatchWxEvent( aEvent );
+
+ Refresh();
+}
+
+
+void EDA_DRAW_PANEL_GAL::onEnter( wxEvent& aEvent )
+{
+ // Getting focus is necessary in order to receive key events properly
+ SetFocus();
+
+ aEvent.Skip();
+}
+
+
+void EDA_DRAW_PANEL_GAL::onLostFocus( wxFocusEvent& aEvent )
+{
+ m_lostFocus = true;
+
+ aEvent.Skip();
+}
+
+
+void EDA_DRAW_PANEL_GAL::onRefreshTimer( wxTimerEvent& aEvent )
+{
+ if( !m_drawingEnabled )
+ {
+ if( m_gal->IsInitialized() )
+ {
+ m_drawing = false;
+ m_pendingRefresh = true;
+ Connect( wxEVT_PAINT, wxPaintEventHandler( EDA_DRAW_PANEL_GAL::onPaint ), NULL, this );
+ m_drawingEnabled = true;
+ }
+ else
+ {
+ // Try again soon
+ m_refreshTimer.Start( 100, true );
+ return;
+ }
+ }
+
+ wxPaintEvent redrawEvent;
+ wxPostEvent( this, redrawEvent );
+}