projecttreewidget.cpp 15.5 KB
Newer Older
hjk's avatar
hjk committed
1
/****************************************************************************
con's avatar
con committed
2
**
3
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
hjk's avatar
hjk committed
4
** Contact: http://www.qt-project.org/legal
con's avatar
con committed
5
**
hjk's avatar
hjk committed
6
** This file is part of Qt Creator.
con's avatar
con committed
7
**
hjk's avatar
hjk committed
8 9 10 11 12 13 14
** 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.
15
**
16
** GNU Lesser General Public License Usage
hjk's avatar
hjk committed
17 18 19 20 21 22 23 24 25
** 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
con's avatar
con committed
26 27
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
hjk's avatar
hjk committed
28
****************************************************************************/
hjk's avatar
hjk committed
29

con's avatar
con committed
30
#include "projecttreewidget.h"
hjk's avatar
hjk committed
31

con's avatar
con committed
32
#include "projectexplorer.h"
33
#include "projectnodes.h"
34 35
#include "project.h"
#include "session.h"
con's avatar
con committed
36 37
#include "projectmodels.h"

38
#include <coreplugin/actionmanager/command.h>
39
#include <coreplugin/coreconstants.h>
con's avatar
con committed
40 41
#include <coreplugin/icore.h>
#include <coreplugin/editormanager/editormanager.h>
42
#include <coreplugin/editormanager/ieditor.h>
Eike Ziller's avatar
Eike Ziller committed
43
#include <coreplugin/find/treeviewfind.h>
dt's avatar
dt committed
44

45
#include <utils/navigationtreeview.h>
hjk's avatar
hjk committed
46

47 48
#include <QDebug>
#include <QSettings>
con's avatar
con committed
49

50
#include <QStyledItemDelegate>
51 52 53 54
#include <QVBoxLayout>
#include <QToolButton>
#include <QAction>
#include <QMenu>
con's avatar
con committed
55

56
using namespace Core;
con's avatar
con committed
57 58 59 60
using namespace ProjectExplorer;
using namespace ProjectExplorer::Internal;

namespace {
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77

class ProjectTreeItemDelegate : public QStyledItemDelegate
{
public:
    ProjectTreeItemDelegate(QObject *parent) : QStyledItemDelegate(parent)
    { }

    void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
    {
        QStyleOptionViewItem opt = option;
        if (!index.data(ProjectExplorer::Project::EnabledRole).toBool())
            opt.state &= ~QStyle::State_Enabled;
        QStyledItemDelegate::paint(painter, opt, index);
    }
};

bool debug = false;
con's avatar
con committed
78 79
}

80
class ProjectTreeView : public Utils::NavigationTreeView
con's avatar
con committed
81 82 83 84 85 86
{
public:
    ProjectTreeView()
    {
        setEditTriggers(QAbstractItemView::EditKeyPressed);
        setContextMenuPolicy(Qt::CustomContextMenu);
87 88
        m_context = new IContext(this);
        m_context->setContext(Context(ProjectExplorer::Constants::C_PROJECT_TREE));
89
        m_context->setWidget(this);
90
        ICore::addContextObject(m_context);
dt's avatar
dt committed
91 92 93
    }
    ~ProjectTreeView()
    {
94
        ICore::removeContextObject(m_context);
dt's avatar
dt committed
95
        delete m_context;
con's avatar
con committed
96
    }
dt's avatar
dt committed
97 98

private:
99
    IContext *m_context;
con's avatar
con committed
100 101 102 103 104 105 106
};

/*!
  /class ProjectTreeWidget

  Shows the projects in form of a tree.
  */
hjk's avatar
hjk committed
107
ProjectTreeWidget::ProjectTreeWidget(QWidget *parent)
con's avatar
con committed
108 109 110 111 112
        : QWidget(parent),
          m_explorer(ProjectExplorerPlugin::instance()),
          m_view(0),
          m_model(0),
          m_filterProjectsAction(0),
113
          m_autoSync(false),
114
          m_autoExpand(true)
con's avatar
con committed
115
{
hjk's avatar
hjk committed
116 117
    m_model = new FlatModel(SessionManager::sessionNode(), this);
    Project *pro = SessionManager::startupProject();
118 119
    if (pro)
        m_model->setStartupProject(pro->rootProjectNode());
120
    NodesWatcher *watcher = new NodesWatcher(this);
hjk's avatar
hjk committed
121
    SessionManager::sessionNode()->registerWatcher(watcher);
122

Robert Loehning's avatar
Robert Loehning committed
123 124 125 126
    connect(watcher, SIGNAL(foldersAboutToBeRemoved(FolderNode*,QList<FolderNode*>)),
            this, SLOT(foldersAboutToBeRemoved(FolderNode*,QList<FolderNode*>)));
    connect(watcher, SIGNAL(filesAboutToBeRemoved(FolderNode*,QList<FileNode*>)),
            this, SLOT(filesAboutToBeRemoved(FolderNode*,QList<FileNode*>)));
con's avatar
con committed
127 128 129

    m_view = new ProjectTreeView;
    m_view->setModel(m_model);
130
    m_view->setItemDelegate(new ProjectTreeItemDelegate(this));
con's avatar
con committed
131 132 133 134
    setFocusProxy(m_view);
    initView();

    QVBoxLayout *layout = new QVBoxLayout();
Eike Ziller's avatar
Eike Ziller committed
135
    layout->addWidget(Core::TreeViewFind::createSearchableWrapper(m_view));
con's avatar
con committed
136 137 138
    layout->setContentsMargins(0, 0, 0, 0);
    setLayout(layout);

Leena Miettinen's avatar
Leena Miettinen committed
139
    m_filterProjectsAction = new QAction(tr("Simplify Tree"), this);
con's avatar
con committed
140 141 142 143
    m_filterProjectsAction->setCheckable(true);
    m_filterProjectsAction->setChecked(false); // default is the traditional complex tree
    connect(m_filterProjectsAction, SIGNAL(toggled(bool)), this, SLOT(setProjectFilter(bool)));

Leena Miettinen's avatar
Leena Miettinen committed
144
    m_filterGeneratedFilesAction = new QAction(tr("Hide Generated Files"), this);
con's avatar
con committed
145 146 147 148 149 150 151
    m_filterGeneratedFilesAction->setCheckable(true);
    m_filterGeneratedFilesAction->setChecked(true);
    connect(m_filterGeneratedFilesAction, SIGNAL(toggled(bool)), this, SLOT(setGeneratedFilesFilter(bool)));

    // connections
    connect(m_model, SIGNAL(modelReset()),
            this, SLOT(initView()));
Robert Loehning's avatar
Robert Loehning committed
152 153 154 155 156 157
    connect(m_view, SIGNAL(activated(QModelIndex)),
            this, SLOT(openItem(QModelIndex)));
    connect(m_view->selectionModel(), SIGNAL(currentChanged(QModelIndex,QModelIndex)),
            this, SLOT(handleCurrentItemChange(QModelIndex)));
    connect(m_view, SIGNAL(customContextMenuRequested(QPoint)),
            this, SLOT(showContextMenu(QPoint)));
hjk's avatar
hjk committed
158 159 160

    QObject *sessionManager = SessionManager::instance();
    connect(sessionManager, SIGNAL(singleProjectAdded(ProjectExplorer::Project*)),
Robert Loehning's avatar
Robert Loehning committed
161
            this, SLOT(handleProjectAdded(ProjectExplorer::Project*)));
hjk's avatar
hjk committed
162
    connect(sessionManager, SIGNAL(startupProjectChanged(ProjectExplorer::Project*)),
Robert Loehning's avatar
Robert Loehning committed
163
            this, SLOT(startupProjectChanged(ProjectExplorer::Project*)));
con's avatar
con committed
164

hjk's avatar
hjk committed
165
    connect(sessionManager, SIGNAL(aboutToLoadSession(QString)),
166
            this, SLOT(disableAutoExpand()));
hjk's avatar
hjk committed
167
    connect(sessionManager, SIGNAL(sessionLoaded(QString)),
168
            this, SLOT(loadExpandData()));
hjk's avatar
hjk committed
169
    connect(sessionManager, SIGNAL(aboutToSaveSession()),
170 171
            this, SLOT(saveExpandData()));

172
    m_toggleSync = new QToolButton;
173
    m_toggleSync->setIcon(QIcon(QLatin1String(Core::Constants::ICON_LINK)));
174 175 176 177 178
    m_toggleSync->setCheckable(true);
    m_toggleSync->setChecked(autoSynchronization());
    m_toggleSync->setToolTip(tr("Synchronize with Editor"));
    connect(m_toggleSync, SIGNAL(clicked(bool)), this, SLOT(toggleAutoSynchronization()));

179
    setAutoSynchronization(true);
180 181
}

182 183 184 185 186 187 188 189
void ProjectTreeWidget::disableAutoExpand()
{
    m_autoExpand = false;
}

void ProjectTreeWidget::loadExpandData()
{
    m_autoExpand = true;
hjk's avatar
hjk committed
190
    QStringList data = SessionManager::value(QLatin1String("ProjectTree.ExpandData")).toStringList();
191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
    recursiveLoadExpandData(m_view->rootIndex(), data.toSet());
}

void ProjectTreeWidget::recursiveLoadExpandData(const QModelIndex &index, const QSet<QString> &data)
{
    if (data.contains(m_model->nodeForIndex(index)->path())) {
        m_view->expand(index);
        int count = m_model->rowCount(index);
        for (int i = 0; i < count; ++i)
            recursiveLoadExpandData(index.child(i, 0), data);
    }
}

void ProjectTreeWidget::saveExpandData()
{
    QStringList data;
    recursiveSaveExpandData(m_view->rootIndex(), &data);
    // TODO if there are multiple ProjectTreeWidgets, the last one saves the data
hjk's avatar
hjk committed
209
    SessionManager::setValue(QLatin1String("ProjectTree.ExpandData"), data);
210 211 212 213 214 215 216 217 218 219 220 221 222
}

void ProjectTreeWidget::recursiveSaveExpandData(const QModelIndex &index, QStringList *data)
{
    Q_ASSERT(data);
    if (m_view->isExpanded(index)) {
        data->append(m_model->nodeForIndex(index)->path());
        int count = m_model->rowCount(index);
        for (int i = 0; i < count; ++i)
            recursiveSaveExpandData(index.child(i, 0), data);
    }
}

223 224 225
void ProjectTreeWidget::foldersAboutToBeRemoved(FolderNode *, const QList<FolderNode*> &list)
{
    Node *n = m_explorer->currentNode();
226
    while (n) {
227 228
        if (FolderNode *fn = qobject_cast<FolderNode *>(n)) {
            if (list.contains(fn)) {
229 230 231 232 233
                ProjectNode *pn = n->projectNode();
                // Make sure the node we are switching too isn't going to be removed also
                while (list.contains(pn))
                    pn = pn->parentFolderNode()->projectNode();
                m_explorer->setCurrentNode(pn);
234 235 236 237 238 239 240 241 242 243
                break;
            }
        }
        n = n->parentFolderNode();
    }
}

void ProjectTreeWidget::filesAboutToBeRemoved(FolderNode *, const QList<FileNode*> &list)
{
    if (FileNode *fileNode = qobject_cast<FileNode *>(m_explorer->currentNode())) {
244
        if (list.contains(fileNode))
245 246 247 248
            m_explorer->setCurrentNode(fileNode->projectNode());
    }
}

249 250 251
QToolButton *ProjectTreeWidget::toggleSync()
{
    return m_toggleSync;
con's avatar
con committed
252 253 254 255 256 257 258 259 260 261 262 263 264 265
}

void ProjectTreeWidget::toggleAutoSynchronization()
{
    setAutoSynchronization(!m_autoSync);
}

bool ProjectTreeWidget::autoSynchronization() const
{
    return m_autoSync;
}

void ProjectTreeWidget::setAutoSynchronization(bool sync, bool syncNow)
{
266
    m_toggleSync->setChecked(sync);
con's avatar
con committed
267 268 269 270 271 272 273 274
    if (sync == m_autoSync)
        return;

    m_autoSync = sync;

    if (debug)
        qDebug() << (m_autoSync ? "Enabling auto synchronization" : "Disabling auto synchronization");
    if (m_autoSync) {
Robert Loehning's avatar
Robert Loehning committed
275 276
        connect(m_explorer, SIGNAL(currentNodeChanged(ProjectExplorer::Node*,ProjectExplorer::Project*)),
                this, SLOT(setCurrentItem(ProjectExplorer::Node*,ProjectExplorer::Project*)));
con's avatar
con committed
277
        if (syncNow)
278
            setCurrentItem(m_explorer->currentNode(), ProjectExplorerPlugin::currentProject());
con's avatar
con committed
279
    } else {
Robert Loehning's avatar
Robert Loehning committed
280 281
        disconnect(m_explorer, SIGNAL(currentNodeChanged(ProjectExplorer::Node*,ProjectExplorer::Project*)),
                this, SLOT(setCurrentItem(ProjectExplorer::Node*,ProjectExplorer::Project*)));
con's avatar
con committed
282 283 284
    }
}

285 286 287 288 289
void ProjectTreeWidget::collapseAll()
{
    m_view->collapseAll();
}

con's avatar
con committed
290 291
void ProjectTreeWidget::editCurrentItem()
{
292 293
    if (m_view->selectionModel()->currentIndex().isValid())
        m_view->edit(m_view->selectionModel()->currentIndex());
con's avatar
con committed
294 295 296 297 298
}

void ProjectTreeWidget::setCurrentItem(Node *node, Project *project)
{
    if (debug)
299 300
        qDebug() << "ProjectTreeWidget::setCurrentItem(" << (project ? project->displayName() : QLatin1String("0"))
                 << ", " <<  (node ? node->path() : QLatin1String("0")) << ")";
301

302
    if (!project)
con's avatar
con committed
303 304 305 306
        return;

    const QModelIndex mainIndex = m_model->indexForNode(node);

307 308 309 310 311
    if (mainIndex.isValid()) {
        if (mainIndex != m_view->selectionModel()->currentIndex()) {
            m_view->setCurrentIndex(mainIndex);
            m_view->scrollTo(mainIndex);
        }
312 313 314 315
    } else {
        if (debug)
            qDebug() << "clear selection";
        m_view->clearSelection();
con's avatar
con committed
316
    }
317

con's avatar
con committed
318 319 320 321 322
}

void ProjectTreeWidget::handleCurrentItemChange(const QModelIndex &current)
{
    Node *node = m_model->nodeForIndex(current);
dt's avatar
dt committed
323
    // node might be 0. that's okay
con's avatar
con committed
324 325 326 327 328 329 330 331 332 333
    bool autoSync = autoSynchronization();
    setAutoSynchronization(false);
    m_explorer->setCurrentNode(node);
    setAutoSynchronization(autoSync, false);
}

void ProjectTreeWidget::showContextMenu(const QPoint &pos)
{
    QModelIndex index = m_view->indexAt(pos);
    Node *node = m_model->nodeForIndex(index);
334
    m_explorer->showContextMenu(this, m_view->mapToGlobal(pos), node);
con's avatar
con committed
335 336 337 338 339 340
}

void ProjectTreeWidget::handleProjectAdded(ProjectExplorer::Project *project)
{
    Node *node = project->rootProjectNode();
    QModelIndex idx = m_model->indexForNode(node);
341 342
    if (m_autoExpand) // disabled while session restoring
        m_view->setExpanded(idx, true);
343
    m_view->setCurrentIndex(idx);
con's avatar
con committed
344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366
}

void ProjectTreeWidget::startupProjectChanged(ProjectExplorer::Project *project)
{
    if (project) {
        ProjectNode *node = project->rootProjectNode();
        m_model->setStartupProject(node);
    } else {
        m_model->setStartupProject(0);
    }
}

void ProjectTreeWidget::initView()
{
    QModelIndex sessionIndex = m_model->index(0, 0);

    // hide root folder
    m_view->setRootIndex(sessionIndex);

    while (m_model->canFetchMore(sessionIndex))
        m_model->fetchMore(sessionIndex);

    // expand top level projects
hjk's avatar
hjk committed
367
    for (int i = 0; i < m_model->rowCount(sessionIndex); ++i)
con's avatar
con committed
368 369
        m_view->expand(m_model->index(i, 0, sessionIndex));

370
    setCurrentItem(m_explorer->currentNode(), ProjectExplorerPlugin::currentProject());
con's avatar
con committed
371 372 373 374 375
}

void ProjectTreeWidget::openItem(const QModelIndex &mainIndex)
{
    Node *node = m_model->nodeForIndex(mainIndex);
376 377
    if (node->nodeType() != FileNodeType)
        return;
Eike Ziller's avatar
Eike Ziller committed
378
    IEditor *editor = EditorManager::openEditor(node->path());
379
    if (editor && node->line() >= 0)
380
        editor->gotoLine(node->line());
con's avatar
con committed
381 382 383 384 385 386 387 388 389 390 391 392 393 394
}

void ProjectTreeWidget::setProjectFilter(bool filter)
{
    m_model->setProjectFilterEnabled(filter);
    m_filterProjectsAction->setChecked(filter);
}

void ProjectTreeWidget::setGeneratedFilesFilter(bool filter)
{
    m_model->setGeneratedFilesFilterEnabled(filter);
    m_filterGeneratedFilesAction->setChecked(filter);
}

395 396 397 398 399 400 401 402 403 404 405
bool ProjectTreeWidget::generatedFilesFilter()
{
    return m_model->generatedFilesFilterEnabled();
}

bool ProjectTreeWidget::projectFilter()
{
    return m_model->projectFilterEnabled();
}


hjk's avatar
hjk committed
406
ProjectTreeWidgetFactory::ProjectTreeWidgetFactory()
con's avatar
con committed
407 408 409 410 411 412 413
{
}

ProjectTreeWidgetFactory::~ProjectTreeWidgetFactory()
{
}

414
QString ProjectTreeWidgetFactory::displayName() const
con's avatar
con committed
415 416 417 418
{
    return tr("Projects");
}

419 420 421 422 423
int ProjectTreeWidgetFactory::priority() const
{
    return 100;
}

424
Id ProjectTreeWidgetFactory::id() const
425
{
426
    return "Projects";
427 428 429
}

QKeySequence ProjectTreeWidgetFactory::activationSequence() const
con's avatar
con committed
430
{
431
    return QKeySequence(UseMacShortcuts ? tr("Meta+X") : tr("Alt+X"));
con's avatar
con committed
432 433
}

434
NavigationView ProjectTreeWidgetFactory::createWidget()
con's avatar
con committed
435
{
436
    NavigationView n;
hjk's avatar
hjk committed
437
    ProjectTreeWidget *ptw = new ProjectTreeWidget;
con's avatar
con committed
438 439 440
    n.widget = ptw;

    QToolButton *filter = new QToolButton;
441
    filter->setIcon(QIcon(QLatin1String(Core::Constants::ICON_FILTER)));
Leena Miettinen's avatar
Leena Miettinen committed
442
    filter->setToolTip(tr("Filter Tree"));
con's avatar
con committed
443
    filter->setPopupMode(QToolButton::InstantPopup);
Eike Ziller's avatar
Eike Ziller committed
444
    filter->setProperty("noArrow", true);
con's avatar
con committed
445 446 447 448 449
    QMenu *filterMenu = new QMenu(filter);
    filterMenu->addAction(ptw->m_filterProjectsAction);
    filterMenu->addAction(ptw->m_filterGeneratedFilesAction);
    filter->setMenu(filterMenu);

Roberto Raggi's avatar
Roberto Raggi committed
450
    n.dockToolBarWidgets << filter << ptw->toggleSync();
con's avatar
con committed
451 452 453
    return n;
}

454 455 456 457
void ProjectTreeWidgetFactory::saveSettings(int position, QWidget *widget)
{
    ProjectTreeWidget *ptw = qobject_cast<ProjectTreeWidget *>(widget);
    Q_ASSERT(ptw);
458
    QSettings *settings = ICore::settings();
459 460 461 462
    const QString baseKey = QLatin1String("ProjectTreeWidget.") + QString::number(position);
    settings->setValue(baseKey + QLatin1String(".ProjectFilter"), ptw->projectFilter());
    settings->setValue(baseKey + QLatin1String(".GeneratedFilter"), ptw->generatedFilesFilter());
    settings->setValue(baseKey + QLatin1String(".SyncWithEditor"), ptw->autoSynchronization());
463 464 465 466 467 468
}

void ProjectTreeWidgetFactory::restoreSettings(int position, QWidget *widget)
{
    ProjectTreeWidget *ptw = qobject_cast<ProjectTreeWidget *>(widget);
    Q_ASSERT(ptw);
469
    QSettings *settings = ICore::settings();
470 471 472 473
    const QString baseKey = QLatin1String("ProjectTreeWidget.") + QString::number(position);
    ptw->setProjectFilter(settings->value(baseKey + QLatin1String(".ProjectFilter"), false).toBool());
    ptw->setGeneratedFilesFilter(settings->value(baseKey + QLatin1String(".GeneratedFilter"), true).toBool());
    ptw->setAutoSynchronization(settings->value(baseKey +  QLatin1String(".SyncWithEditor"), true).toBool());
474
}