Files
qt-creator/src/plugins/coreplugin/editortoolbar.cpp

479 lines
18 KiB
C++
Raw Normal View History

/****************************************************************************
**
** 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://www.qt.io/licensing. For further information
** use the contact form at http://www.qt.io/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 or version 3 as published by the Free
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
** following information to ensure the GNU Lesser General Public License
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
** 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
2010-12-17 16:01:08 +01:00
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
#include "editortoolbar.h"
#include <coreplugin/actionmanager/actionmanager.h>
#include <coreplugin/coreconstants.h>
#include <coreplugin/editormanager/documentmodel.h>
#include <coreplugin/editormanager/editormanager.h>
#include <coreplugin/editormanager/editormanager_p.h>
#include <coreplugin/editormanager/ieditor.h>
#include <coreplugin/fileiconprovider.h>
#include <coreplugin/icore.h>
#include <utils/fileutils.h>
#include <utils/hostosinfo.h>
#include <utils/qtcassert.h>
#include <QApplication>
#include <QComboBox>
#include <QDir>
#include <QDrag>
#include <QLabel>
#include <QMenu>
#include <QMimeData>
#include <QMouseEvent>
#include <QTimer>
#include <QToolButton>
#include <QVBoxLayout>
#include <QDebug>
enum {
debug = false
};
namespace Core {
struct EditorToolBarPrivate
{
2010-09-16 12:26:28 +02:00
explicit EditorToolBarPrivate(QWidget *parent, EditorToolBar *q);
QComboBox *m_editorList;
QToolButton *m_closeEditorButton;
2010-09-16 12:26:28 +02:00
QToolButton *m_lockButton;
QToolButton *m_dragHandle;
QMenu *m_dragHandleMenu;
EditorToolBar::MenuProvider m_menuProvider;
2010-09-16 12:26:28 +02:00
QAction *m_goBackAction;
QAction *m_goForwardAction;
QToolButton *m_backButton;
QToolButton *m_forwardButton;
QToolButton *m_splitButton;
QAction *m_horizontalSplitAction;
QAction *m_verticalSplitAction;
QAction *m_splitNewWindowAction;
QToolButton *m_closeSplitButton;
2010-09-16 12:26:28 +02:00
QWidget *m_activeToolBar;
QWidget *m_toolBarPlaceholder;
QWidget *m_defaultToolBar;
QPoint m_dragStartPosition;
2010-09-16 12:26:28 +02:00
bool m_isStandalone;
};
EditorToolBarPrivate::EditorToolBarPrivate(QWidget *parent, EditorToolBar *q) :
m_editorList(new QComboBox(q)),
m_closeEditorButton(new QToolButton(q)),
m_lockButton(new QToolButton(q)),
m_dragHandle(new QToolButton(q)),
2010-09-16 12:26:28 +02:00
m_goBackAction(new QAction(QIcon(QLatin1String(Constants::ICON_PREV)), EditorManager::tr("Go Back"), parent)),
m_goForwardAction(new QAction(QIcon(QLatin1String(Constants::ICON_NEXT)), EditorManager::tr("Go Forward"), parent)),
m_splitButton(new QToolButton(q)),
m_horizontalSplitAction(new QAction(QIcon(QLatin1String(Constants::ICON_SPLIT_HORIZONTAL)), EditorManager::tr("Split"), parent)),
m_verticalSplitAction(new QAction(QIcon(QLatin1String(Constants::ICON_SPLIT_VERTICAL)), EditorManager::tr("Split Side by Side"), parent)),
m_splitNewWindowAction(new QAction(EditorManager::tr("Open in New Window"), parent)),
m_closeSplitButton(new QToolButton(q)),
2010-09-16 12:26:28 +02:00
m_activeToolBar(0),
m_toolBarPlaceholder(new QWidget(q)),
2010-09-16 12:26:28 +02:00
m_defaultToolBar(new QWidget(q)),
m_isStandalone(false)
{
}
/*!
Mimic the look of the text editor toolbar as defined in e.g. EditorView::EditorView
*/
2010-03-09 17:40:55 +01:00
EditorToolBar::EditorToolBar(QWidget *parent) :
2010-09-16 12:26:28 +02:00
Utils::StyledBar(parent), d(new EditorToolBarPrivate(parent, this))
{
2010-03-09 17:40:55 +01:00
QHBoxLayout *toolBarLayout = new QHBoxLayout(this);
toolBarLayout->setMargin(0);
toolBarLayout->setSpacing(0);
2010-09-16 12:26:28 +02:00
toolBarLayout->addWidget(d->m_defaultToolBar);
d->m_toolBarPlaceholder->setLayout(toolBarLayout);
d->m_toolBarPlaceholder->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
2010-09-16 12:26:28 +02:00
d->m_defaultToolBar->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
d->m_activeToolBar = d->m_defaultToolBar;
2010-03-09 17:40:55 +01:00
d->m_lockButton->setAutoRaise(true);
d->m_lockButton->setEnabled(false);
d->m_dragHandle->setProperty("noArrow", true);
d->m_dragHandle->setToolTip(tr("Drag to drag documents between splits"));
d->m_dragHandle->installEventFilter(this);
d->m_dragHandleMenu = new QMenu(d->m_dragHandle);
d->m_dragHandle->setMenu(d->m_dragHandleMenu);
2010-09-16 12:26:28 +02:00
connect(d->m_goBackAction, SIGNAL(triggered()), this, SIGNAL(goBackClicked()));
connect(d->m_goForwardAction, SIGNAL(triggered()), this, SIGNAL(goForwardClicked()));
d->m_editorList->setProperty("hideicon", true);
d->m_editorList->setProperty("notelideasterisk", true);
2010-09-16 12:26:28 +02:00
d->m_editorList->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
d->m_editorList->setMinimumContentsLength(20);
d->m_editorList->setModel(DocumentModel::model());
2010-09-16 12:26:28 +02:00
d->m_editorList->setMaxVisibleItems(40);
d->m_editorList->setContextMenuPolicy(Qt::CustomContextMenu);
d->m_closeEditorButton->setAutoRaise(true);
d->m_closeEditorButton->setIcon(QIcon(QLatin1String(Constants::ICON_BUTTON_CLOSE)));
d->m_closeEditorButton->setEnabled(false);
d->m_closeEditorButton->setProperty("showborder", true);
2010-09-16 12:26:28 +02:00
d->m_toolBarPlaceholder->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
2010-03-09 17:40:55 +01:00
2010-09-16 12:26:28 +02:00
d->m_backButton = new QToolButton(this);
d->m_backButton->setDefaultAction(d->m_goBackAction);
2010-03-09 17:40:55 +01:00
2010-09-16 12:26:28 +02:00
d->m_forwardButton= new QToolButton(this);
d->m_forwardButton->setDefaultAction(d->m_goForwardAction);
if (Utils::HostOsInfo::isMacHost()) {
d->m_horizontalSplitAction->setIconVisibleInMenu(false);
d->m_verticalSplitAction->setIconVisibleInMenu(false);
d->m_splitNewWindowAction->setIconVisibleInMenu(false);
}
d->m_splitButton->setIcon(QIcon(QLatin1String(Constants::ICON_SPLIT_HORIZONTAL)));
d->m_splitButton->setToolTip(tr("Split"));
d->m_splitButton->setPopupMode(QToolButton::InstantPopup);
d->m_splitButton->setProperty("noArrow", true);
QMenu *splitMenu = new QMenu(d->m_splitButton);
splitMenu->addAction(d->m_horizontalSplitAction);
splitMenu->addAction(d->m_verticalSplitAction);
splitMenu->addAction(d->m_splitNewWindowAction);
d->m_splitButton->setMenu(splitMenu);
d->m_closeSplitButton->setAutoRaise(true);
d->m_closeSplitButton->setIcon(QIcon(QLatin1String(Constants::ICON_CLOSE_SPLIT_BOTTOM)));
QHBoxLayout *toplayout = new QHBoxLayout(this);
toplayout->setSpacing(0);
toplayout->setMargin(0);
2010-09-16 12:26:28 +02:00
toplayout->addWidget(d->m_backButton);
toplayout->addWidget(d->m_forwardButton);
toplayout->addWidget(d->m_lockButton);
toplayout->addWidget(d->m_dragHandle);
2010-09-16 12:26:28 +02:00
toplayout->addWidget(d->m_editorList);
toplayout->addWidget(d->m_closeEditorButton);
toplayout->addWidget(d->m_toolBarPlaceholder, 1); // Custom toolbar stretches
toplayout->addWidget(d->m_splitButton);
toplayout->addWidget(d->m_closeSplitButton);
2010-03-09 17:40:55 +01:00
setLayout(toplayout);
2010-03-15 16:02:38 +01:00
// this signal is disconnected for standalone toolbars and replaced with
// a private slot connection
2010-09-16 12:26:28 +02:00
connect(d->m_editorList, SIGNAL(activated(int)), this, SIGNAL(listSelectionActivated(int)));
2010-03-15 16:02:38 +01:00
connect(d->m_editorList, &QComboBox::customContextMenuRequested, [this](QPoint p) {
QMenu menu;
fillListContextMenu(&menu);
menu.exec(d->m_editorList->mapToGlobal(p));
});
connect(d->m_dragHandleMenu, &QMenu::aboutToShow, [this]() {
d->m_dragHandleMenu->clear();
fillListContextMenu(d->m_dragHandleMenu);
});
2010-09-16 12:26:28 +02:00
connect(d->m_lockButton, SIGNAL(clicked()), this, SLOT(makeEditorWritable()));
connect(d->m_closeEditorButton, SIGNAL(clicked()), this, SLOT(closeEditor()), Qt::QueuedConnection);
connect(d->m_horizontalSplitAction, SIGNAL(triggered()),
this, SIGNAL(horizontalSplitClicked()), Qt::QueuedConnection);
connect(d->m_verticalSplitAction, SIGNAL(triggered()),
this, SIGNAL(verticalSplitClicked()), Qt::QueuedConnection);
connect(d->m_splitNewWindowAction, SIGNAL(triggered()),
this, SIGNAL(splitNewWindowClicked()), Qt::QueuedConnection);
connect(d->m_closeSplitButton, SIGNAL(clicked()),
this, SIGNAL(closeSplitClicked()), Qt::QueuedConnection);
connect(ActionManager::command(Constants::CLOSE), SIGNAL(keySequenceChanged()),
2010-03-09 17:40:55 +01:00
this, SLOT(updateActionShortcuts()));
connect(ActionManager::command(Constants::GO_BACK), SIGNAL(keySequenceChanged()),
2010-03-09 17:40:55 +01:00
this, SLOT(updateActionShortcuts()));
connect(ActionManager::command(Constants::GO_FORWARD), SIGNAL(keySequenceChanged()),
2010-03-09 17:40:55 +01:00
this, SLOT(updateActionShortcuts()));
updateActionShortcuts();
}
2010-09-16 12:26:28 +02:00
EditorToolBar::~EditorToolBar()
{
delete d;
2010-09-16 12:26:28 +02:00
}
2010-03-09 17:40:55 +01:00
void EditorToolBar::removeToolbarForEditor(IEditor *editor)
{
QTC_ASSERT(editor, return);
disconnect(editor->document(), SIGNAL(changed()), this, SLOT(checkDocumentStatus()));
2010-03-09 17:40:55 +01:00
QWidget *toolBar = editor->toolBar();
if (toolBar != 0) {
2010-09-16 12:26:28 +02:00
if (d->m_activeToolBar == toolBar) {
d->m_activeToolBar = d->m_defaultToolBar;
d->m_activeToolBar->setVisible(true);
2010-03-09 17:40:55 +01:00
}
2010-09-16 12:26:28 +02:00
d->m_toolBarPlaceholder->layout()->removeWidget(toolBar);
2010-03-09 17:40:55 +01:00
toolBar->setVisible(false);
toolBar->setParent(0);
}
}
void EditorToolBar::setCloseSplitEnabled(bool enable)
{
d->m_closeSplitButton->setVisible(enable);
}
void EditorToolBar::setCloseSplitIcon(const QIcon &icon)
{
d->m_closeSplitButton->setIcon(icon);
}
void EditorToolBar::closeEditor()
2010-03-09 17:40:55 +01:00
{
if (d->m_isStandalone)
EditorManager::slotCloseCurrentEditorOrDocument();
2010-03-09 17:40:55 +01:00
emit closeClicked();
}
2010-03-10 10:50:11 +01:00
void EditorToolBar::addEditor(IEditor *editor)
2010-03-09 17:40:55 +01:00
{
QTC_ASSERT(editor, return);
connect(editor->document(), SIGNAL(changed()), this, SLOT(checkDocumentStatus()));
2010-03-09 17:40:55 +01:00
QWidget *toolBar = editor->toolBar();
2010-09-16 12:26:28 +02:00
if (toolBar && !d->m_isStandalone)
2010-03-09 17:40:55 +01:00
addCenterToolBar(toolBar);
updateDocumentStatus(editor->document());
2010-03-09 17:40:55 +01:00
}
void EditorToolBar::addCenterToolBar(QWidget *toolBar)
{
QTC_ASSERT(toolBar, return);
2010-03-09 17:40:55 +01:00
toolBar->setVisible(false); // will be made visible in setCurrentEditor
2010-09-16 12:26:28 +02:00
d->m_toolBarPlaceholder->layout()->addWidget(toolBar);
2010-03-09 17:40:55 +01:00
updateToolBar(toolBar);
}
void EditorToolBar::updateToolBar(QWidget *toolBar)
{
if (!toolBar)
2010-09-16 12:26:28 +02:00
toolBar = d->m_defaultToolBar;
if (d->m_activeToolBar == toolBar)
2010-03-09 17:40:55 +01:00
return;
toolBar->setVisible(true);
2010-09-16 12:26:28 +02:00
d->m_activeToolBar->setVisible(false);
d->m_activeToolBar = toolBar;
}
2010-03-10 10:50:11 +01:00
void EditorToolBar::setToolbarCreationFlags(ToolbarCreationFlags flags)
{
2010-09-16 12:26:28 +02:00
d->m_isStandalone = flags & FlagsStandalone;
if (d->m_isStandalone) {
connect(EditorManager::instance(), &EditorManager::currentEditorChanged,
this, &EditorToolBar::updateEditorListSelection);
2010-03-15 16:02:38 +01:00
2010-09-16 12:26:28 +02:00
disconnect(d->m_editorList, SIGNAL(activated(int)), this, SIGNAL(listSelectionActivated(int)));
connect(d->m_editorList, SIGNAL(activated(int)), this, SLOT(changeActiveEditor(int)));
d->m_splitButton->setVisible(false);
d->m_closeSplitButton->setVisible(false);
2010-03-12 16:47:46 +01:00
}
2010-03-10 10:50:11 +01:00
}
2010-03-09 17:40:55 +01:00
void EditorToolBar::setMenuProvider(const EditorToolBar::MenuProvider &provider)
{
d->m_menuProvider = provider;
}
2010-03-09 17:40:55 +01:00
void EditorToolBar::setCurrentEditor(IEditor *editor)
{
IDocument *document = editor ? editor->document() : 0;
d->m_editorList->setCurrentIndex(DocumentModel::rowOfDocument(document));
2010-03-09 17:40:55 +01:00
// If we never added the toolbar from the editor, we will never change
// the editor, so there's no need to update the toolbar either.
2010-09-16 12:26:28 +02:00
if (!d->m_isStandalone)
updateToolBar(editor ? editor->toolBar() : 0);
2010-03-09 17:40:55 +01:00
updateDocumentStatus(document);
}
2010-03-09 17:40:55 +01:00
void EditorToolBar::updateEditorListSelection(IEditor *newSelection)
{
2010-03-12 16:47:46 +01:00
if (newSelection)
d->m_editorList->setCurrentIndex(DocumentModel::rowOfDocument(newSelection->document()));
}
2010-03-15 16:02:38 +01:00
void EditorToolBar::changeActiveEditor(int row)
{
EditorManager::activateEditorForEntry(DocumentModel::entryAtRow(row));
}
void EditorToolBar::fillListContextMenu(QMenu *menu)
{
if (d->m_menuProvider) {
d->m_menuProvider(menu);
} else {
IEditor *editor = EditorManager::currentEditor();
DocumentModel::Entry *entry = editor ? DocumentModel::entryForDocument(editor->document())
: 0;
EditorManager::addSaveAndCloseEditorActions(menu, entry, editor);
menu->addSeparator();
EditorManager::addNativeDirAndOpenWithActions(menu, entry);
}
}
2010-03-09 17:40:55 +01:00
void EditorToolBar::makeEditorWritable()
{
if (IDocument *current = EditorManager::currentDocument())
Internal::EditorManagerPrivate::makeFileWritable(current);
2010-03-09 17:40:55 +01:00
}
void EditorToolBar::setCanGoBack(bool canGoBack)
{
2010-09-16 12:26:28 +02:00
d->m_goBackAction->setEnabled(canGoBack);
2010-03-09 17:40:55 +01:00
}
void EditorToolBar::setCanGoForward(bool canGoForward)
{
2010-09-16 12:26:28 +02:00
d->m_goForwardAction->setEnabled(canGoForward);
2010-03-09 17:40:55 +01:00
}
void EditorToolBar::updateActionShortcuts()
{
d->m_closeEditorButton->setToolTip(ActionManager::command(Constants::CLOSE)->stringWithAppendedShortcut(EditorManager::tr("Close Document")));
d->m_goBackAction->setToolTip(ActionManager::command(Constants::GO_BACK)->action()->toolTip());
d->m_goForwardAction->setToolTip(ActionManager::command(Constants::GO_FORWARD)->action()->toolTip());
d->m_closeSplitButton->setToolTip(ActionManager::command(Constants::REMOVE_CURRENT_SPLIT)->stringWithAppendedShortcut(tr("Remove Split")));
}
void EditorToolBar::checkDocumentStatus()
2010-03-09 17:40:55 +01:00
{
IDocument *document = qobject_cast<IDocument *>(sender());
QTC_ASSERT(document, return);
DocumentModel::Entry *entry = DocumentModel::entryAtRow(
d->m_editorList->currentIndex());
2010-03-09 17:40:55 +01:00
if (entry && entry->document && entry->document == document)
updateDocumentStatus(document);
2010-03-09 17:40:55 +01:00
}
void EditorToolBar::updateDocumentStatus(IDocument *document)
2010-03-09 17:40:55 +01:00
{
d->m_closeEditorButton->setEnabled(document != 0);
2010-03-09 17:40:55 +01:00
if (!document) {
d->m_lockButton->setIcon(QIcon());
d->m_lockButton->setEnabled(false);
d->m_lockButton->setToolTip(QString());
d->m_dragHandle->setIcon(QIcon());
2010-09-16 12:26:28 +02:00
d->m_editorList->setToolTip(QString());
return;
2010-03-09 17:40:55 +01:00
}
d->m_editorList->setCurrentIndex(DocumentModel::rowOfDocument(document));
2010-03-09 17:40:55 +01:00
if (document->filePath().isEmpty()) {
d->m_lockButton->setIcon(QIcon());
d->m_lockButton->setEnabled(false);
d->m_lockButton->setToolTip(QString());
} else if (document->isFileReadOnly()) {
d->m_lockButton->setIcon(DocumentModel::lockedIcon());
d->m_lockButton->setEnabled(true);
d->m_lockButton->setToolTip(tr("Make Writable"));
} else {
d->m_lockButton->setIcon(DocumentModel::unlockedIcon());
2010-09-16 12:26:28 +02:00
d->m_lockButton->setEnabled(false);
d->m_lockButton->setToolTip(tr("File is writable"));
}
if (document->filePath().isEmpty())
d->m_dragHandle->setIcon(QIcon());
else
d->m_dragHandle->setIcon(FileIconProvider::icon(QFileInfo(document->filePath())));
d->m_editorList->setToolTip(
document->filePath().isEmpty()
? document->displayName()
: QDir::toNativeSeparators(document->filePath()));
2010-03-09 17:40:55 +01:00
}
bool EditorToolBar::eventFilter(QObject *obj, QEvent *event)
{
if (obj == d->m_dragHandle) {
if (event->type() == QEvent::MouseButtonPress) {
auto me = static_cast<QMouseEvent *>(event);
if (me->buttons() == Qt::LeftButton)
d->m_dragStartPosition = me->pos();
return true; // do not pop up menu already on press
} else if (event->type() == QEvent::MouseButtonRelease) {
d->m_dragHandle->showMenu();
return true;
} else if (event->type() == QEvent::MouseMove) {
auto me = static_cast<QMouseEvent *>(event);
if (me->buttons() != Qt::LeftButton)
return Utils::StyledBar::eventFilter(obj, event);
if ((me->pos() - d->m_dragStartPosition).manhattanLength()
< QApplication::startDragDistance())
return Utils::StyledBar::eventFilter(obj, event);
DocumentModel::Entry *entry = DocumentModel::entryAtRow(
d->m_editorList->currentIndex());
if (!entry) // no document
return Utils::StyledBar::eventFilter(obj, event);
auto drag = new QDrag(this);
auto data = new Utils::FileDropMimeData;
data->addFile(entry->fileName());
drag->setMimeData(data);
Qt::DropAction action = drag->exec(Qt::MoveAction | Qt::CopyAction, Qt::MoveAction);
if (action == Qt::MoveAction)
emit currentDocumentMoved();
return true;
}
}
return Utils::StyledBar::eventFilter(obj, event);
}
2010-03-09 17:40:55 +01:00
void EditorToolBar::setNavigationVisible(bool isVisible)
{
2010-09-16 12:26:28 +02:00
d->m_goBackAction->setVisible(isVisible);
d->m_goForwardAction->setVisible(isVisible);
d->m_backButton->setVisible(isVisible);
d->m_forwardButton->setVisible(isVisible);
}
} // Core