forked from qt-creator/qt-creator
		
	When opening a help url with anchor for context help, the help viewer would loose focus, breaking behavior like pressing F1 twice to go to help mode. Task-number: QTCREATORBUG-9093 Change-Id: Ib6b6ecb4f6bec63f545c1f4ff973d878547f3ee1 Reviewed-by: Eike Ziller <eike.ziller@digia.com>
		
			
				
	
	
		
			217 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			217 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /****************************************************************************
 | |
| **
 | |
| ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
 | |
| ** Contact: http://www.qt-project.org/legal
 | |
| **
 | |
| ** This file is part of Qt Creator.
 | |
| **
 | |
| ** Commercial License Usage
 | |
| ** Licensees holding valid commercial Qt licenses may use this file in
 | |
| ** accordance with the commercial license agreement provided with the
 | |
| ** Software or, alternatively, in accordance with the terms contained in
 | |
| ** a written agreement between you and Digia.  For licensing terms and
 | |
| ** conditions see http://qt.digia.com/licensing.  For further information
 | |
| ** use the contact form at http://qt.digia.com/contact-us.
 | |
| **
 | |
| ** GNU Lesser General Public License Usage
 | |
| ** Alternatively, this file may be used under the terms of the GNU Lesser
 | |
| ** General Public License version 2.1 as published by the Free Software
 | |
| ** Foundation and appearing in the file LICENSE.LGPL included in the
 | |
| ** packaging of this file.  Please review the following information to
 | |
| ** ensure the GNU Lesser General Public License version 2.1 requirements
 | |
| ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | |
| **
 | |
| ** In addition, as a special exception, Digia gives you certain additional
 | |
| ** rights.  These rights are described in the Digia Qt LGPL Exception
 | |
| ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
 | |
| **
 | |
| ****************************************************************************/
 | |
| 
 | |
| #include "rightpane.h"
 | |
| 
 | |
| #include <coreplugin/modemanager.h>
 | |
| 
 | |
| #include <QSettings>
 | |
| 
 | |
| #include <QVBoxLayout>
 | |
| #include <QSplitter>
 | |
| #include <QResizeEvent>
 | |
| 
 | |
| 
 | |
| using namespace Core;
 | |
| using namespace Core::Internal;
 | |
| 
 | |
| RightPanePlaceHolder *RightPanePlaceHolder::m_current = 0;
 | |
| 
 | |
| RightPanePlaceHolder* RightPanePlaceHolder::current()
 | |
| {
 | |
|     return m_current;
 | |
| }
 | |
| 
 | |
| RightPanePlaceHolder::RightPanePlaceHolder(Core::IMode *mode, QWidget *parent)
 | |
|     :QWidget(parent), m_mode(mode)
 | |
| {
 | |
|     setLayout(new QVBoxLayout);
 | |
|     layout()->setMargin(0);
 | |
|     connect(Core::ModeManager::instance(), SIGNAL(currentModeChanged(Core::IMode*)),
 | |
|             this, SLOT(currentModeChanged(Core::IMode*)));
 | |
| }
 | |
| 
 | |
| RightPanePlaceHolder::~RightPanePlaceHolder()
 | |
| {
 | |
|     if (m_current == this) {
 | |
|         RightPaneWidget::instance()->setParent(0);
 | |
|         RightPaneWidget::instance()->hide();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void RightPanePlaceHolder::applyStoredSize(int width)
 | |
| {
 | |
|     if (width) {
 | |
|         QSplitter *splitter = qobject_cast<QSplitter *>(parentWidget());
 | |
|         if (splitter) {
 | |
|             // A splitter we need to resize the splitter sizes
 | |
|             QList<int> sizes = splitter->sizes();
 | |
|             int index = splitter->indexOf(this);
 | |
|             int diff = width - sizes.at(index);
 | |
|             int adjust = sizes.count() > 1 ? (diff / (sizes.count() - 1)) : 0;
 | |
|             for (int i = 0; i < sizes.count(); ++i) {
 | |
|                 if (i != index)
 | |
|                     sizes[i] -= adjust;
 | |
|             }
 | |
|             sizes[index]= width;
 | |
|             splitter->setSizes(sizes);
 | |
|         } else {
 | |
|             QSize s = size();
 | |
|             s.setWidth(width);
 | |
|             resize(s);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // This function does work even though the order in which
 | |
| // the placeHolder get the signal is undefined.
 | |
| // It does ensure that after all PlaceHolders got the signal
 | |
| // m_current points to the current PlaceHolder, or zero if there
 | |
| // is no PlaceHolder in this mode
 | |
| // And that the parent of the RightPaneWidget gets the correct parent
 | |
| void RightPanePlaceHolder::currentModeChanged(Core::IMode *mode)
 | |
| {
 | |
|     if (m_current == this) {
 | |
|         m_current = 0;
 | |
|         RightPaneWidget::instance()->setParent(0);
 | |
|         RightPaneWidget::instance()->hide();
 | |
|     }
 | |
|     if (m_mode == mode) {
 | |
|         m_current = this;
 | |
| 
 | |
|         int width = RightPaneWidget::instance()->storedWidth();
 | |
| 
 | |
|         layout()->addWidget(RightPaneWidget::instance());
 | |
|         RightPaneWidget::instance()->show();
 | |
| 
 | |
|         applyStoredSize(width);
 | |
|         setVisible(RightPaneWidget::instance()->isShown());
 | |
|     }
 | |
| }
 | |
| 
 | |
| /////
 | |
| // RightPaneWidget
 | |
| /////
 | |
| 
 | |
| 
 | |
| RightPaneWidget *RightPaneWidget::m_instance = 0;
 | |
| 
 | |
| RightPaneWidget::RightPaneWidget()
 | |
|     : m_shown(true), m_width(0)
 | |
| {
 | |
|     m_instance = this;
 | |
| 
 | |
|     QVBoxLayout *layout = new QVBoxLayout;
 | |
|     layout->setMargin(0);
 | |
|     setLayout(layout);
 | |
| }
 | |
| 
 | |
| RightPaneWidget::~RightPaneWidget()
 | |
| {
 | |
|     clearWidget();
 | |
|     m_instance = 0;
 | |
| }
 | |
| 
 | |
| RightPaneWidget *RightPaneWidget::instance()
 | |
| {
 | |
|     return m_instance;
 | |
| }
 | |
| 
 | |
| void RightPaneWidget::setWidget(QWidget *widget)
 | |
| {
 | |
|     if (widget == m_widget)
 | |
|         return;
 | |
|     clearWidget();
 | |
|     m_widget = widget;
 | |
|     if (m_widget) {
 | |
|         m_widget->setParent(this);
 | |
|         layout()->addWidget(m_widget);
 | |
|         setFocusProxy(m_widget);
 | |
|         m_widget->show();
 | |
|     }
 | |
| }
 | |
| 
 | |
| int RightPaneWidget::storedWidth()
 | |
| {
 | |
|     return m_width;
 | |
| }
 | |
| 
 | |
| void RightPaneWidget::resizeEvent(QResizeEvent *re)
 | |
| {
 | |
|     if (m_width && re->size().width())
 | |
|         m_width = re->size().width();
 | |
|     QWidget::resizeEvent(re);
 | |
| }
 | |
| 
 | |
| void RightPaneWidget::saveSettings(QSettings *settings)
 | |
| {
 | |
|     settings->setValue(QLatin1String("RightPane/Visible"), isShown());
 | |
|     settings->setValue(QLatin1String("RightPane/Width"), m_width);
 | |
| }
 | |
| 
 | |
| void RightPaneWidget::readSettings(QSettings *settings)
 | |
| {
 | |
|     if (settings->contains(QLatin1String("RightPane/Visible")))
 | |
|         setShown(settings->value(QLatin1String("RightPane/Visible")).toBool());
 | |
|     else
 | |
|         setShown(false);
 | |
| 
 | |
|     if (settings->contains(QLatin1String("RightPane/Width"))) {
 | |
|         m_width = settings->value(QLatin1String("RightPane/Width")).toInt();
 | |
|         if (!m_width)
 | |
|             m_width = 500;
 | |
|     } else {
 | |
|         m_width = 500; //pixel
 | |
|     }
 | |
|     // Apply
 | |
|     if (RightPanePlaceHolder::m_current)
 | |
|         RightPanePlaceHolder::m_current->applyStoredSize(m_width);
 | |
| }
 | |
| 
 | |
| void RightPaneWidget::setShown(bool b)
 | |
| {
 | |
|     if (RightPanePlaceHolder::m_current)
 | |
|         RightPanePlaceHolder::m_current->setVisible(b);
 | |
|     m_shown = b;
 | |
| }
 | |
| 
 | |
| bool RightPaneWidget::isShown()
 | |
| {
 | |
|     return m_shown;
 | |
| }
 | |
| 
 | |
| void RightPaneWidget::clearWidget()
 | |
| {
 | |
|     if (m_widget) {
 | |
|         m_widget->hide();
 | |
|         m_widget->setParent(0);
 | |
|         m_widget = 0;
 | |
|     }
 | |
| }
 |