/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** 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 The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include "outlinefactory.h" #include #include #include #include #include #include #include #include #include #include namespace TextEditor { static QList g_outlineWidgetFactories; static QPointer g_outlineFactory; IOutlineWidgetFactory::IOutlineWidgetFactory() { g_outlineWidgetFactories.append(this); } IOutlineWidgetFactory::~IOutlineWidgetFactory() { g_outlineWidgetFactories.removeOne(this); } void IOutlineWidgetFactory::updateOutline() { if (QTC_GUARD(!g_outlineFactory.isNull())) emit g_outlineFactory->updateOutline(); } namespace Internal { OutlineWidgetStack::OutlineWidgetStack(OutlineFactory *factory) : QStackedWidget(), m_factory(factory), m_syncWithEditor(true), m_sorted(false) { QLabel *label = new QLabel(tr("No outline available"), this); label->setAlignment(Qt::AlignCenter); // set background to be white label->setAutoFillBackground(true); label->setBackgroundRole(QPalette::Base); addWidget(label); m_toggleSync = new QToolButton(this); m_toggleSync->setIcon(Utils::Icons::LINK_TOOLBAR.icon()); m_toggleSync->setCheckable(true); m_toggleSync->setChecked(true); m_toggleSync->setToolTip(tr("Synchronize with Editor")); connect(m_toggleSync, &QAbstractButton::clicked, this, &OutlineWidgetStack::toggleCursorSynchronization); m_filterButton = new QToolButton(this); // The ToolButton needs a parent because updateFilterMenu() sets // it visible. That would open a top-level window if the button // did not have a parent in that moment. m_filterButton->setIcon(Utils::Icons::FILTER.icon()); m_filterButton->setToolTip(tr("Filter tree")); m_filterButton->setPopupMode(QToolButton::InstantPopup); m_filterButton->setProperty("noArrow", true); m_filterMenu = new QMenu(m_filterButton); m_filterButton->setMenu(m_filterMenu); m_toggleSort = new QToolButton(this); m_toggleSort->setIcon(Utils::Icons::SORT_ALPHABETICALLY_TOOLBAR.icon()); m_toggleSort->setCheckable(true); m_toggleSort->setChecked(false); m_toggleSort->setToolTip(tr("Sort Alphabetically")); connect(m_toggleSort, &QAbstractButton::clicked, this, &OutlineWidgetStack::toggleSort); connect(Core::EditorManager::instance(), &Core::EditorManager::currentEditorChanged, this, &OutlineWidgetStack::updateEditor); connect(factory, &OutlineFactory::updateOutline, this, &OutlineWidgetStack::updateCurrentEditor); updateCurrentEditor(); } OutlineWidgetStack::~OutlineWidgetStack() = default; QToolButton *OutlineWidgetStack::toggleSyncButton() { return m_toggleSync; } QToolButton *OutlineWidgetStack::filterButton() { return m_filterButton; } QToolButton *OutlineWidgetStack::sortButton() { return m_toggleSort; } void OutlineWidgetStack::saveSettings(QSettings *settings, int position) { const QString baseKey = QStringLiteral("Outline.%1.").arg(position); settings->setValue(baseKey + QLatin1String("SyncWithEditor"), toggleSyncButton()->isChecked()); for (auto iter = m_widgetSettings.constBegin(); iter != m_widgetSettings.constEnd(); ++iter) settings->setValue(baseKey + iter.key(), iter.value()); } void OutlineWidgetStack::restoreSettings(QSettings *settings, int position) { const QString baseKey = QStringLiteral("Outline.%1.").arg(position); bool syncWithEditor = true; m_widgetSettings.clear(); foreach (const QString &longKey, settings->allKeys()) { if (!longKey.startsWith(baseKey)) continue; const QString key = longKey.mid(baseKey.length()); if (key == QLatin1String("SyncWithEditor")) { syncWithEditor = settings->value(longKey).toBool(); continue; } m_widgetSettings.insert(key, settings->value(longKey)); } toggleSyncButton()->setChecked(syncWithEditor); if (auto outlineWidget = qobject_cast(currentWidget())) outlineWidget->restoreSettings(m_widgetSettings); } bool OutlineWidgetStack::isCursorSynchronized() const { return m_syncWithEditor; } void OutlineWidgetStack::toggleCursorSynchronization() { m_syncWithEditor = !m_syncWithEditor; if (auto outlineWidget = qobject_cast(currentWidget())) outlineWidget->setCursorSynchronization(m_syncWithEditor); } void OutlineWidgetStack::toggleSort() { m_sorted = !m_sorted; if (auto outlineWidget = qobject_cast(currentWidget())) outlineWidget->setSorted(m_sorted); } void OutlineWidgetStack::updateFilterMenu() { m_filterMenu->clear(); if (auto outlineWidget = qobject_cast(currentWidget())) { foreach (QAction *filterAction, outlineWidget->filterMenuActions()) { m_filterMenu->addAction(filterAction); } } m_filterButton->setVisible(!m_filterMenu->actions().isEmpty()); } void OutlineWidgetStack::updateCurrentEditor() { updateEditor(Core::EditorManager::currentEditor()); } void OutlineWidgetStack::updateEditor(Core::IEditor *editor) { IOutlineWidget *newWidget = nullptr; if (editor) { for (IOutlineWidgetFactory *widgetFactory : qAsConst(g_outlineWidgetFactories)) { if (widgetFactory->supportsEditor(editor)) { newWidget = widgetFactory->createWidget(editor); m_toggleSort->setVisible(widgetFactory->supportsSorting()); break; } } } if (newWidget != currentWidget()) { // delete old widget if (auto outlineWidget = qobject_cast(currentWidget())) { QVariantMap widgetSettings = outlineWidget->settings(); for (auto iter = widgetSettings.constBegin(); iter != widgetSettings.constEnd(); ++iter) m_widgetSettings.insert(iter.key(), iter.value()); removeWidget(outlineWidget); delete outlineWidget; } if (newWidget) { newWidget->restoreSettings(m_widgetSettings); newWidget->setCursorSynchronization(m_syncWithEditor); m_toggleSort->setChecked(newWidget->isSorted()); addWidget(newWidget); setCurrentWidget(newWidget); setFocusProxy(newWidget); } updateFilterMenu(); } } OutlineFactory::OutlineFactory() { QTC_CHECK(g_outlineFactory.isNull()); g_outlineFactory = this; setDisplayName(tr("Outline")); setId("Outline"); setPriority(600); } Core::NavigationView OutlineFactory::createWidget() { Core::NavigationView n; auto placeHolder = new OutlineWidgetStack(this); n.widget = placeHolder; n.dockToolBarWidgets.append(placeHolder->filterButton()); n.dockToolBarWidgets.append(placeHolder->sortButton()); n.dockToolBarWidgets.append(placeHolder->toggleSyncButton()); return n; } void OutlineFactory::saveSettings(QSettings *settings, int position, QWidget *widget) { auto widgetStack = qobject_cast(widget); Q_ASSERT(widgetStack); widgetStack->saveSettings(settings, position); } void OutlineFactory::restoreSettings(QSettings *settings, int position, QWidget *widget) { auto widgetStack = qobject_cast(widget); Q_ASSERT(widgetStack); widgetStack->restoreSettings(settings, position); } } // namespace Internal } // namespace TextEditor