/* This file is part of the KDevelop project Copyright 2002 Falk Brettschneider Copyright 2003 John Firebaugh Copyright 2006 Adam Treat Copyright 2006, 2007 Alexander Dymo 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 "mainwindow_p.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "core.h" #include "partdocument.h" #include "partcontroller.h" #include "uicontroller.h" #include "statusbar.h" #include "mainwindow.h" #include "textdocument.h" #include "sessioncontroller.h" #include #include #include #include #include namespace KDevelop { bool MainWindowPrivate::s_quitRequested = false; MainWindowPrivate::MainWindowPrivate(MainWindow *mainWindow) : m_mainWindow(mainWindow), m_statusBar(0), lastXMLGUIClientView(0), m_changingActiveView(false) { } void MainWindowPrivate::setupGui() { m_statusBar = new KDevelop::StatusBar(m_mainWindow); setupStatusBar(); } void MainWindowPrivate::setupStatusBar() { QWidget *location = m_mainWindow->statusBarLocation(); if (m_statusBar) location->layout()->addWidget(m_statusBar); } void MainWindowPrivate::addPlugin( IPlugin *plugin ) { kDebug() << "add plugin" << plugin << plugin->componentData().componentName(); Q_ASSERT( plugin ); //The direct plugin client can only be added to the first mainwindow if(m_mainWindow == Core::self()->uiControllerInternal()->mainWindows()[0]) m_mainWindow->guiFactory()->addClient( plugin ); Q_ASSERT(!m_pluginCustomClients.contains(plugin)); KXMLGUIClient* ownClient = plugin->createGUIForMainWindow(m_mainWindow); if(ownClient) { m_pluginCustomClients[plugin] = ownClient; connect(plugin, SIGNAL(destroyed(QObject*)), SLOT(pluginDestroyed(QObject*))); m_mainWindow->guiFactory()->addClient(ownClient); } } void MainWindowPrivate::pluginDestroyed(QObject* pluginObj) { IPlugin* plugin = static_cast(pluginObj); Q_ASSERT(m_pluginCustomClients.contains(plugin)); m_mainWindow->guiFactory()->removeClient( m_pluginCustomClients[plugin] ); delete m_pluginCustomClients[plugin]; m_pluginCustomClients.remove(plugin); } MainWindowPrivate::~MainWindowPrivate() { foreach(KXMLGUIClient* client, m_pluginCustomClients.values()) delete client; } void MainWindowPrivate::removePlugin( IPlugin *plugin ) { Q_ASSERT( plugin ); if(m_pluginCustomClients.contains(plugin)) { m_mainWindow->guiFactory()->removeClient( m_pluginCustomClients[plugin] ); delete m_pluginCustomClients[plugin]; m_pluginCustomClients.remove(plugin); disconnect(plugin, SIGNAL(destroyed(QObject*)), this, SLOT(pluginDestroyed(QObject*))); } m_mainWindow->guiFactory()->removeClient( plugin ); } void MainWindowPrivate::activePartChanged(KParts::Part *part) { if ( Core::self()->uiController()->activeMainWindow() == m_mainWindow) m_mainWindow->createGUI(part); } void MainWindowPrivate::changeActiveView(Sublime::View *view) { //disable updates on a window to avoid toolbar flickering on xmlgui client change Sublime::HoldUpdates s(m_mainWindow); mergeView(view); if(!view) return; IDocument *doc = dynamic_cast(view->document()); if (doc) { doc->activate(view, m_mainWindow); } else { //activated view is not a part document so we need to remove active part gui ///@todo adymo: only this window needs to remove GUI // KParts::Part *activePart = Core::self()->partController()->activePart(); // if (activePart) // guiFactory()->removeClient(activePart); } } void MainWindowPrivate::mergeView(Sublime::View* view) { PushPositiveValue block(m_changingActiveView, true); // If the previous view was KXMLGUIClient, remove its actions // In the case that that view was removed, lastActiveView // will auto-reset, and xmlguifactory will disconnect that // client, I think. if (lastXMLGUIClientView) { kDebug() << "clearing last XML GUI client" << lastXMLGUIClientView; m_mainWindow->guiFactory()->removeClient(dynamic_cast(lastXMLGUIClientView)); disconnect (lastXMLGUIClientView, SIGNAL(destroyed(QObject*)), this, 0); lastXMLGUIClientView = NULL; } if (!view) return; QWidget* viewWidget = view->widget(); Q_ASSERT(viewWidget); kDebug() << "changing active view to" << view << "doc" << view->document() << "mw" << m_mainWindow; // If the new view is KXMLGUIClient, add it. if (KXMLGUIClient* c = dynamic_cast(viewWidget)) { kDebug() << "setting new XMLGUI client" << viewWidget; lastXMLGUIClientView = viewWidget; m_mainWindow->guiFactory()->addClient(c); connect(viewWidget, SIGNAL(destroyed(QObject*)), this, SLOT(xmlguiclientDestroyed(QObject*))); } } void MainWindowPrivate::xmlguiclientDestroyed(QObject* obj) { /* We're informed the QWidget for the active view that is also KXMLGUIclient is dying. KXMLGUIFactory will not like deleted clients, really. Unfortunately, there's nothing we can do at this point. For example, KateView derives from QWidget and KXMLGUIClient. The destroyed() signal is emitted by ~QWidget. At this point, event attempt to cross-cast to KXMLGUIClient is undefined behaviour. We hope to catch view deletion a bit later, but if we fail, we better report it now, rather than get a weird crash a bit later. */ Q_ASSERT(obj == lastXMLGUIClientView); Q_ASSERT(false && "xmlgui clients management is messed up"); Q_UNUSED(obj); } void MainWindowPrivate::setupActions() { connect(Core::self()->sessionController(), SIGNAL(quitSession()), SLOT(quitAll())); KAction *action; QString app = qApp->applicationName(); QString text = i18nc( "%1 = application name", "Configure %1", app ); action = KStandardAction::preferences( this, SLOT(settingsDialog()), actionCollection()); action->setToolTip( text ); action->setWhatsThis( i18n( "Lets you customize %1.", app ) ); action = actionCollection()->addAction( "show_editorconfig", this, SLOT(showEditorConfig()) ); action->setIcon( KIcon("preferences-other") ); action->setText( i18n("Configure Editor...")); action->setWhatsThis( i18nc("@info:whatsthis", "Configure various aspects of this editor.") ); action = KStandardAction::configureNotifications(this, SLOT(configureNotifications()), actionCollection()); action->setText( i18n("Configure Notifications...") ); action->setToolTip( i18nc("@info:tooltip", "Configure notifications") ); action->setWhatsThis( i18nc( "@info:whatsthis", "Shows a dialog that lets you configure notifications." ) ); action = actionCollection()->addAction( "about_platform", this, SLOT(showAboutPlatform()) ); action->setText( i18n("About KDevelop Platform") ); action->setStatusTip( i18n("Show Information about KDevelop Platform") ); action->setWhatsThis( i18nc( "@info:whatsthis", "Shows a dialog with information about KDevelop Platform." ) ); action = actionCollection()->addAction( "loaded_plugins", this, SLOT(showLoadedPlugins()) ); action->setText( i18n("Loaded Plugins") ); action->setStatusTip( i18n("Show a list of all loaded plugins") ); action->setWhatsThis( i18nc( "@info:whatsthis", "Shows a dialog with information about all loaded plugins." ) ); action = actionCollection()->addAction( "view_next_window" ); action->setText( i18n( "&Next Window" ) ); connect( action, SIGNAL(triggered(bool)), SLOT(gotoNextWindow()) ); action->setShortcut( Qt::ALT + Qt::SHIFT + Qt::Key_Right ); action->setToolTip( i18nc( "@info:tooltip", "Next window" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Switches to the next window." ) ); action->setIcon(KIcon("go-next")); action = actionCollection()->addAction( "view_previous_window" ); action->setText( i18n( "&Previous Window" ) ); connect( action, SIGNAL(triggered(bool)), SLOT(gotoPreviousWindow()) ); action->setShortcut( Qt::ALT + Qt::SHIFT + Qt::Key_Left ); action->setToolTip( i18nc( "@info:tooltip", "Previous window" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Switches to the previous window." ) ); action->setIcon(KIcon("go-previous")); action = actionCollection()->addAction( "split_horizontal" ); action->setIcon(KIcon( "view-split-top-bottom" )); action->setText( i18n( "Split View &Top/Bottom" ) ); action->setShortcut( Qt::CTRL + Qt::SHIFT + Qt::Key_T ); connect( action, SIGNAL(triggered(bool)), SLOT(splitHorizontal()) ); action->setToolTip( i18nc( "@info:tooltip", "Split horizontal" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Splits the current view horizontally." ) ); action = actionCollection()->addAction( "split_vertical" ); action->setIcon(KIcon( "view-split-left-right" )); action->setText( i18n( "Split View &Left/Right" ) ); action->setShortcut( Qt::CTRL + Qt::SHIFT + Qt::Key_L ); connect( action, SIGNAL(triggered(bool)), SLOT(splitVertical()) ); action->setToolTip( i18nc( "@info:tooltip", "Split vertical" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Splits the current view vertically." ) ); action = actionCollection()->addAction( "view_next_split" ); action->setText( i18n( "&Next Split View" ) ); connect( action, SIGNAL(triggered(bool)), SLOT(gotoNextSplit()) ); action->setShortcut( Qt::CTRL + Qt::SHIFT + Qt::Key_N ); action->setToolTip( i18nc( "@info:tooltip", "Next split view" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Switches to the next split view." ) ); action->setIcon(KIcon("go-next")); action = actionCollection()->addAction( "view_previous_split" ); action->setText( i18n( "&Previous Split View" ) ); connect( action, SIGNAL(triggered(bool)), SLOT(gotoPreviousSplit()) ); action->setShortcut( Qt::CTRL + Qt::SHIFT + Qt::Key_P ); action->setToolTip( i18nc( "@info:tooltip", "Previous split view" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Switches to the previous split view." ) ); action->setIcon(KIcon("go-previous")); action = KStandardAction::fullScreen( this, SLOT(toggleFullScreen(bool)), m_mainWindow, actionCollection() ); action = actionCollection()->addAction( "file_new" ); action->setIcon(KIcon("document-new")); action->setShortcut( Qt::CTRL + Qt::Key_N ); action->setText( i18n( "&New" ) ); action->setIconText( i18nc( "Shorter Text for 'New File' shown in the toolbar", "New") ); connect( action, SIGNAL(triggered(bool)), SLOT(fileNew()) ); action->setToolTip( i18nc( "@info:tooltip", "New file" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Creates an empty file." ) ); action = actionCollection()->addAction( "add_toolview" ); action->setIcon(KIcon("window-new")); action->setShortcut( Qt::CTRL + Qt::SHIFT + Qt::Key_V ); action->setText( i18n( "&Add Tool View..." ) ); connect( action, SIGNAL(triggered(bool)), SLOT(viewAddNewToolView()) ); action->setToolTip( i18nc( "@info:tooltip", "Add tool view" ) ); action->setWhatsThis( i18nc( "@info:whatsthis", "Adds a new tool view to this window." ) ); } void MainWindowPrivate::toggleArea(bool b) { if (!b) return; KAction *action = qobject_cast(sender()); if (!action) return; m_mainWindow->controller()->showArea(action->data().toString(), m_mainWindow); } KActionCollection * MainWindowPrivate::actionCollection() { return m_mainWindow->actionCollection(); } bool MainWindowPrivate::applicationQuitRequested() const { return s_quitRequested; } void MainWindowPrivate::registerStatus(QObject* status) { m_statusBar->registerStatus(status); } void MainWindowPrivate::showErrorMessage(QString message, int timeout) { m_statusBar->showErrorMessage(message, timeout); } void MainWindowPrivate::tabContextMenuRequested(Sublime::View* view, KMenu* menu) { m_tabView = view; QAction* action; action = menu->addAction(KIcon("view-split-top-bottom"), i18n("Split View Top/Bottom")); connect(action, SIGNAL(triggered(bool)), this, SLOT(contextMenuSplitHorizontal())); action = menu->addAction(KIcon("view-split-left-right"), i18n("Split View Left/Right")); connect(action, SIGNAL(triggered(bool)), this, SLOT(contextMenuSplitVertical())); menu->addSeparator(); action = menu->addAction(KIcon("document-new"), i18n("New File")); connect(action, SIGNAL(triggered(bool)), this, SLOT(contextMenuFileNew())); if ( TextDocument* doc = dynamic_cast(view->document()) ) { action = menu->addAction(KIcon("view-refresh"), i18n("Reload")); connect(action, SIGNAL(triggered(bool)), doc, SLOT(reload())); action = menu->addAction(KIcon("view-refresh"), i18n("Reload All")); connect(action, SIGNAL(triggered(bool)), this, SLOT(reloadAll())); } } void MainWindowPrivate::tabToolTipRequested(Sublime::View* view, Sublime::Container* container, int tab) { if (m_tabTooltip.second) { if (m_tabTooltip.first == view) { // tooltip already shown, don't do anything. prevents flicker when moving mouse over same tab return; } else { m_tabTooltip.second.data()->close(); } } DUChainReadLocker lock; Sublime::UrlDocument* urlDoc = dynamic_cast(view->document()); if (urlDoc) { TopDUContext* top = DUChainUtils::standardContextForUrl(urlDoc->url()); if (top) { if ( QWidget* navigationWidget = top->createNavigationWidget() ) { NavigationToolTip* tooltip = new KDevelop::NavigationToolTip(m_mainWindow, QCursor::pos() + QPoint(20, 20), navigationWidget); tooltip->resize(navigationWidget->sizeHint() + QSize(10, 10)); tooltip->addExtendRect(container->tabRect(tab)); m_tabTooltip.first = view; m_tabTooltip.second = tooltip; ActiveToolTip::showToolTip(m_tabTooltip.second.data()); } } } } void MainWindowPrivate::dockBarContextMenuRequested(Qt::DockWidgetArea area, const QPoint& position) { KMenu menu; menu.addTitle(KIcon("window-new"), i18n("Add Tool View")); QMap factories = Core::self()->uiControllerInternal()->factoryDocuments(); QMap actionToFactory; if ( !factories.isEmpty() ) { // sorted actions QMap actionMap; for (QMap::const_iterator it = factories.constBegin(); it != factories.constEnd(); ++it) { QAction* action = new QAction(it.value()->statusIcon(), it.value()->title(), &menu); action->setIcon(it.value()->statusIcon()); if (!it.key()->allowMultiple() && Core::self()->uiControllerInternal()->toolViewPresent(it.value(), m_mainWindow->area())) { action->setDisabled(true); } actionToFactory.insert(action, it.key()); actionMap[action->text()] = action; } menu.addActions(actionMap.values()); } QAction* triggered = menu.exec(position); if ( !triggered ) { return; } Core::self()->uiControllerInternal()->addToolViewToDockArea( actionToFactory[triggered], area ); } bool MainWindowPrivate::changingActiveView() const { return m_changingActiveView; } } #include "mainwindow_actions.cpp" #include "moc_mainwindow_p.cpp"