summaryrefslogtreecommitdiffstats
path: root/kparts/partmanager.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kparts/partmanager.cpp')
-rw-r--r--kparts/partmanager.cpp601
1 files changed, 601 insertions, 0 deletions
diff --git a/kparts/partmanager.cpp b/kparts/partmanager.cpp
new file mode 100644
index 000000000..2925b601e
--- /dev/null
+++ b/kparts/partmanager.cpp
@@ -0,0 +1,601 @@
+// -*- mode: c++; c-basic-offset: 2 -*-
+/* This file is part of the KDE project
+ Copyright (C) 1999 Simon Hausmann <hausmann@kde.org>
+ (C) 1999 David Faure <faure@kde.org>
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 of the License, or (at your option) any later version.
+
+ This library 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
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+*/
+
+#include "partmanager.h"
+#include <kparts/event.h>
+#include <kparts/part.h>
+#include <kglobal.h>
+#include <kdebug.h>
+
+#include <qapplication.h>
+
+//#define DEBUG_PARTMANAGER
+
+using namespace KParts;
+
+template class QPtrList<Part>;
+
+namespace KParts {
+
+class PartManagerPrivate
+{
+public:
+ PartManagerPrivate()
+ {
+ m_activeWidget = 0;
+ m_activePart = 0;
+ m_selectedPart = 0;
+ m_selectedWidget = 0;
+ m_bAllowNestedParts = false;
+ m_bIgnoreScrollBars = false;
+ m_activationButtonMask = Qt::LeftButton | Qt::MidButton | Qt::RightButton;
+ m_reason = PartManager::NoReason;
+ }
+ ~PartManagerPrivate()
+ {
+ }
+ void setReason( QEvent* ev ) {
+ switch( ev->type() ) {
+ case QEvent::MouseButtonPress:
+ case QEvent::MouseButtonDblClick: {
+ QMouseEvent* mev = static_cast<QMouseEvent *>( ev );
+ m_reason = mev->button() == Qt::LeftButton
+ ? PartManager::ReasonLeftClick
+ : ( mev->button() == Qt::MidButton
+ ? PartManager::ReasonMidClick
+ : PartManager::ReasonRightClick );
+ break;
+ }
+ case QEvent::FocusIn:
+ m_reason = static_cast<QFocusEvent *>( ev )->reason();
+ break;
+ default:
+ kdWarning(1000) << "PartManagerPrivate::setReason got unexpected ev type " << ev->type() << endl;
+ break;
+ }
+ }
+
+ Part * m_activePart;
+ QWidget *m_activeWidget;
+
+ QPtrList<Part> m_parts;
+
+ PartManager::SelectionPolicy m_policy;
+
+ Part *m_selectedPart;
+ QWidget *m_selectedWidget;
+
+ QPtrList<QWidget> m_managedTopLevelWidgets;
+ short int m_activationButtonMask;
+ bool m_bIgnoreScrollBars;
+ bool m_bAllowNestedParts;
+ int m_reason;
+};
+
+}
+
+PartManager::PartManager( QWidget * parent, const char * name )
+ : QObject( parent, name )
+{
+ d = new PartManagerPrivate;
+
+ qApp->installEventFilter( this );
+
+ d->m_policy = Direct;
+
+ addManagedTopLevelWidget( parent );
+}
+
+PartManager::PartManager( QWidget *topLevel, QObject *parent, const char *name )
+ : QObject( parent, name )
+{
+ d = new PartManagerPrivate;
+
+ qApp->installEventFilter( this );
+
+ d->m_policy = Direct;
+
+ addManagedTopLevelWidget( topLevel );
+}
+
+PartManager::~PartManager()
+{
+ for ( QPtrListIterator<QWidget> it( d->m_managedTopLevelWidgets );
+ it.current(); ++it )
+ disconnect( it.current(), SIGNAL( destroyed() ),
+ this, SLOT( slotManagedTopLevelWidgetDestroyed() ) );
+
+ for ( QPtrListIterator<Part> it( d->m_parts ); it.current(); ++it )
+ {
+ it.current()->setManager( 0 );
+ }
+
+ // core dumps ... setActivePart( 0L );
+ qApp->removeEventFilter( this );
+ delete d;
+}
+
+void PartManager::setSelectionPolicy( SelectionPolicy policy )
+{
+ d->m_policy = policy;
+}
+
+PartManager::SelectionPolicy PartManager::selectionPolicy() const
+{
+ return d->m_policy;
+}
+
+void PartManager::setAllowNestedParts( bool allow )
+{
+ d->m_bAllowNestedParts = allow;
+}
+
+bool PartManager::allowNestedParts() const
+{
+ return d->m_bAllowNestedParts;
+}
+
+void PartManager::setIgnoreScrollBars( bool ignore )
+{
+ d->m_bIgnoreScrollBars = ignore;
+}
+
+bool PartManager::ignoreScrollBars() const
+{
+ return d->m_bIgnoreScrollBars;
+}
+
+void PartManager::setActivationButtonMask( short int buttonMask )
+{
+ d->m_activationButtonMask = buttonMask;
+}
+
+short int PartManager::activationButtonMask() const
+{
+ return d->m_activationButtonMask;
+}
+
+bool PartManager::eventFilter( QObject *obj, QEvent *ev )
+{
+
+ if ( ev->type() != QEvent::MouseButtonPress &&
+ ev->type() != QEvent::MouseButtonDblClick &&
+ ev->type() != QEvent::FocusIn )
+ return false;
+
+ if ( !obj->isWidgetType() )
+ return false;
+
+ QWidget *w = static_cast<QWidget *>( obj );
+
+ if ( ( w->testWFlags( WType_Dialog ) && w->isModal() ) ||
+ w->testWFlags( WType_Popup ) || w->testWFlags( WStyle_Tool ) )
+ return false;
+
+ QMouseEvent* mev = 0L;
+ if ( ev->type() == QEvent::MouseButtonPress || ev->type() == QEvent::MouseButtonDblClick )
+ {
+ mev = static_cast<QMouseEvent *>( ev );
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << "PartManager::eventFilter button: " << mev->button() << " " << "d->m_activationButtonMask=" << d->m_activationButtonMask << endl;
+#endif
+ if ( ( mev->button() & d->m_activationButtonMask ) == 0 )
+ return false; // ignore this button
+ }
+
+ Part * part;
+ while ( w )
+ {
+ QPoint pos;
+
+ if ( !d->m_managedTopLevelWidgets.containsRef( w->topLevelWidget() ) )
+ return false;
+
+ if ( d->m_bIgnoreScrollBars && w->inherits( "QScrollBar" ) )
+ return false;
+
+ if ( mev ) // mouse press or mouse double-click event
+ {
+ pos = mev->globalPos();
+ part = findPartFromWidget( w, pos );
+ } else
+ part = findPartFromWidget( w );
+
+#ifdef DEBUG_PARTMANAGER
+ QCString evType = ( ev->type() == QEvent::MouseButtonPress ) ? "MouseButtonPress"
+ : ( ev->type() == QEvent::MouseButtonDblClick ) ? "MouseButtonDblClick"
+ : ( ev->type() == QEvent::FocusIn ) ? "FocusIn" : "OTHER! ERROR!";
+#endif
+ if ( part ) // We found a part whose widget is w
+ {
+ if ( d->m_policy == PartManager::TriState )
+ {
+ if ( ev->type() == QEvent::MouseButtonDblClick )
+ {
+ if ( part == d->m_activePart && w == d->m_activeWidget )
+ return false;
+
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << "PartManager::eventFilter dblclick -> setActivePart" << part << endl;
+#endif
+ d->setReason( ev );
+ setActivePart( part, w );
+ d->m_reason = NoReason;
+ return true;
+ }
+
+ if ( ( d->m_selectedWidget != w || d->m_selectedPart != part ) &&
+ ( d->m_activeWidget != w || d->m_activePart != part ) )
+ {
+ if ( part->isSelectable() )
+ setSelectedPart( part, w );
+ else {
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << "Part " << part << " (non-selectable) made active because " << w->className() << " got event" << " " << evType << endl;
+#endif
+ d->setReason( ev );
+ setActivePart( part, w );
+ d->m_reason = NoReason;
+ }
+ return true;
+ }
+ else if ( d->m_selectedWidget == w && d->m_selectedPart == part )
+ {
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << "Part " << part << " made active (from selected) because " << w->className() << " got event" << " " << evType << endl;
+#endif
+ d->setReason( ev );
+ setActivePart( part, w );
+ d->m_reason = NoReason;
+ return true;
+ }
+ else if ( d->m_activeWidget == w && d->m_activePart == part )
+ {
+ setSelectedPart( 0L );
+ return false;
+ }
+
+ return false;
+ }
+ else if ( part != d->m_activePart )
+ {
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << "Part " << part << " made active because " << w->className() << " got event" << " " << evType << endl;
+#endif
+ d->setReason( ev );
+ setActivePart( part, w );
+ d->m_reason = NoReason;
+ }
+
+ return false;
+ }
+
+ w = w->parentWidget();
+
+ if ( w && ( ( w->testWFlags( WType_Dialog ) && w->isModal() ) ||
+ w->testWFlags( WType_Popup ) || w->testWFlags( WStyle_Tool ) ) )
+ {
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << QString("No part made active although %1/%2 got event - loop aborted").arg(obj->name()).arg(obj->className()) << endl;
+#endif
+ return false;
+ }
+
+ }
+
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << QString("No part made active although %1/%2 got event").arg(obj->name()).arg(obj->className()) << endl;
+#endif
+ return false;
+}
+
+Part * PartManager::findPartFromWidget( QWidget * widget, const QPoint &pos )
+{
+ QPtrListIterator<Part> it ( d->m_parts );
+ for ( ; it.current() ; ++it )
+ {
+ Part *part = it.current()->hitTest( widget, pos );
+ if ( part && d->m_parts.findRef( part ) != -1 )
+ return part;
+ }
+ return 0L;
+}
+
+Part * PartManager::findPartFromWidget( QWidget * widget )
+{
+ QPtrListIterator<Part> it ( d->m_parts );
+ for ( ; it.current() ; ++it )
+ {
+ if ( widget == it.current()->widget() )
+ return it.current();
+ }
+ return 0L;
+}
+
+void PartManager::addPart( Part *part, bool setActive )
+{
+ if ( d->m_parts.findRef( part ) != -1 ) // don't add parts more than once :)
+ {
+#ifdef DEBUG_PARTMANAGER
+ kdWarning(1000) << k_funcinfo << part << " already added" << kdBacktrace(5) << endl;
+#endif
+ return;
+ }
+
+ d->m_parts.append( part );
+
+ part->setManager( this );
+
+ if ( setActive )
+ {
+ setActivePart( part );
+ if ( part->widget() )
+ part->widget()->setFocus();
+ }
+
+ // Prevent focus problems
+ if ( part->widget() && part->widget()->focusPolicy() == QWidget::NoFocus )
+ {
+ kdWarning(1000) << "Part '" << part->name() << "' has a widget " << part->widget()->name() << " with a focus policy of NoFocus. It should have at least a ClickFocus policy, for part activation to work well." << endl;
+ }
+ if ( part->widget() && part->widget()->focusPolicy() == QWidget::TabFocus )
+ {
+ kdWarning(1000) << "Part '" << part->name() << "' has a widget " << part->widget()->name() << " with a focus policy of TabFocus. It should have at least a ClickFocus policy, for part activation to work well." << endl;
+ }
+
+ if ( setActive && part->widget() )
+ part->widget()->show();
+ emit partAdded( part );
+}
+
+void PartManager::removePart( Part *part )
+{
+ if ( d->m_parts.findRef( part ) == -1 )
+ {
+ kdFatal(1000) << QString("Can't remove part %1, not in KPartManager's list.").arg(part->name()) << endl;
+ return;
+ }
+
+ //Warning. The part could be already deleted
+ //kdDebug(1000) << QString("Part %1 removed").arg(part->name()) << endl;
+ int nb = d->m_parts.count();
+ bool ok = d->m_parts.removeRef( part );
+ Q_ASSERT( ok );
+ Q_ASSERT( (int)d->m_parts.count() == nb-1 );
+ part->setManager(0);
+
+ emit partRemoved( part );
+
+ if ( part == d->m_activePart )
+ setActivePart( 0 );
+ if ( part == d->m_selectedPart )
+ setSelectedPart( 0 );
+}
+
+void PartManager::replacePart( Part * oldPart, Part * newPart, bool setActive )
+{
+ //kdDebug(1000) << "replacePart " << oldPart->name() << "-> " << newPart->name() << " setActive=" << setActive << endl;
+ // This methods does exactly removePart + addPart but without calling setActivePart(0) in between
+ if ( d->m_parts.findRef( oldPart ) == -1 )
+ {
+ kdFatal(1000) << QString("Can't remove part %1, not in KPartManager's list.").arg(oldPart->name()) << endl;
+ return;
+ }
+
+ d->m_parts.removeRef( oldPart );
+ oldPart->setManager(0);
+
+ emit partRemoved( oldPart );
+
+ addPart( newPart, setActive );
+}
+
+void PartManager::setActivePart( Part *part, QWidget *widget )
+{
+ if ( part && d->m_parts.findRef( part ) == -1 )
+ {
+ kdWarning( 1000 ) << "PartManager::setActivePart : trying to activate a non-registered part! " << part->name() << endl;
+ return; // don't allow someone call setActivePart with a part we don't know about
+ }
+
+ //check whether nested parts are disallowed and activate the top parent part then, by traversing the
+ //tree recursively (Simon)
+ if ( part && !d->m_bAllowNestedParts )
+ {
+ QObject *parentPart = part->parent(); // ### this relies on people using KParts::Factory!
+ if ( parentPart && parentPart->inherits( "KParts::Part" ) )
+ {
+ KParts::Part *parPart = static_cast<KParts::Part *>( parentPart );
+ setActivePart( parPart, parPart->widget() );
+ return;
+ }
+ }
+
+#ifdef DEBUG_PARTMANAGER
+ kdDebug(1000) << "PartManager::setActivePart d->m_activePart=" << d->m_activePart << "<->part=" << part
+ << " d->m_activeWidget=" << d->m_activeWidget << "<->widget=" << widget << endl;
+#endif
+
+ // don't activate twice
+ if ( d->m_activePart && part && d->m_activePart == part &&
+ (!widget || d->m_activeWidget == widget) )
+ return;
+
+ KParts::Part *oldActivePart = d->m_activePart;
+ QWidget *oldActiveWidget = d->m_activeWidget;
+
+ setSelectedPart( 0L );
+
+ d->m_activePart = part;
+ d->m_activeWidget = widget;
+
+ if ( oldActivePart )
+ {
+ KParts::Part *savedActivePart = part;
+ QWidget *savedActiveWidget = widget;
+
+ PartActivateEvent ev( false, oldActivePart, oldActiveWidget );
+ QApplication::sendEvent( oldActivePart, &ev );
+ if ( oldActiveWidget )
+ {
+ disconnect( oldActiveWidget, SIGNAL( destroyed() ),
+ this, SLOT( slotWidgetDestroyed() ) );
+ QApplication::sendEvent( oldActiveWidget, &ev );
+ }
+
+ d->m_activePart = savedActivePart;
+ d->m_activeWidget = savedActiveWidget;
+ }
+
+ if ( d->m_activePart )
+ {
+ if ( !widget )
+ d->m_activeWidget = part->widget();
+
+ PartActivateEvent ev( true, d->m_activePart, d->m_activeWidget );
+ QApplication::sendEvent( d->m_activePart, &ev );
+ if ( d->m_activeWidget )
+ {
+ connect( d->m_activeWidget, SIGNAL( destroyed() ),
+ this, SLOT( slotWidgetDestroyed() ) );
+ QApplication::sendEvent( d->m_activeWidget, &ev );
+ }
+ }
+ // Set the new active instance in KGlobal
+ setActiveInstance( d->m_activePart ? d->m_activePart->instance() : 0L );
+
+ kdDebug(1000) << this << " emitting activePartChanged " << d->m_activePart << endl;
+ emit activePartChanged( d->m_activePart );
+}
+
+void PartManager::setActiveInstance( KInstance * instance )
+{
+ // It's a separate method to allow redefining this behavior
+ KGlobal::_activeInstance = instance;
+}
+
+Part *PartManager::activePart() const
+{
+ return d->m_activePart;
+}
+
+QWidget *PartManager::activeWidget() const
+{
+ return d->m_activeWidget;
+}
+
+void PartManager::setSelectedPart( Part *part, QWidget *widget )
+{
+ if ( part == d->m_selectedPart && widget == d->m_selectedWidget )
+ return;
+
+ Part *oldPart = d->m_selectedPart;
+ QWidget *oldWidget = d->m_selectedWidget;
+
+ d->m_selectedPart = part;
+ d->m_selectedWidget = widget;
+
+ if ( part && !widget )
+ d->m_selectedWidget = part->widget();
+
+ if ( oldPart )
+ {
+ PartSelectEvent ev( false, oldPart, oldWidget );
+ QApplication::sendEvent( oldPart, &ev );
+ QApplication::sendEvent( oldWidget, &ev );
+ }
+
+ if ( d->m_selectedPart )
+ {
+ PartSelectEvent ev( true, d->m_selectedPart, d->m_selectedWidget );
+ QApplication::sendEvent( d->m_selectedPart, &ev );
+ QApplication::sendEvent( d->m_selectedWidget, &ev );
+ }
+}
+
+Part *PartManager::selectedPart() const
+{
+ return d->m_selectedPart;
+}
+
+QWidget *PartManager::selectedWidget() const
+{
+ return d->m_selectedWidget;
+}
+
+void PartManager::slotObjectDestroyed()
+{
+ kdDebug(1000) << "KPartManager::slotObjectDestroyed()" << endl;
+ removePart( const_cast<Part *>( static_cast<const Part *>( sender() ) ) );
+}
+
+void PartManager::slotWidgetDestroyed()
+{
+ kdDebug(1000) << "KPartsManager::slotWidgetDestroyed()" << endl;
+ if ( static_cast<const QWidget *>( sender() ) == d->m_activeWidget )
+ setActivePart( 0L ); //do not remove the part because if the part's widget dies, then the
+ //part will delete itself anyway, invoking removePart() in its destructor
+}
+
+const QPtrList<Part> *PartManager::parts() const
+{
+ return &d->m_parts;
+}
+
+void PartManager::addManagedTopLevelWidget( const QWidget *topLevel )
+{
+ if ( !topLevel->isTopLevel() )
+ return;
+
+ if ( d->m_managedTopLevelWidgets.containsRef( topLevel ) )
+ return;
+
+ d->m_managedTopLevelWidgets.append( topLevel );
+ connect( topLevel, SIGNAL( destroyed() ),
+ this, SLOT( slotManagedTopLevelWidgetDestroyed() ) );
+}
+
+void PartManager::removeManagedTopLevelWidget( const QWidget *topLevel )
+{
+ if ( !topLevel->isTopLevel() )
+ return;
+
+ if ( d->m_managedTopLevelWidgets.findRef( topLevel ) == -1 )
+ return;
+
+ d->m_managedTopLevelWidgets.remove();
+}
+
+void PartManager::slotManagedTopLevelWidgetDestroyed()
+{
+ const QWidget *widget = static_cast<const QWidget *>( sender() );
+ removeManagedTopLevelWidget( widget );
+}
+
+int PartManager::reason() const
+{
+ return d->m_reason;
+}
+
+void PartManager::virtual_hook( int, void* )
+{ /*BASE::virtual_hook( id, data );*/ }
+
+#include "partmanager.moc"