/**************************************************************************** ** ** 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 ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "symbolsfindfilter.h" #include "cppmodelmanager.h" #include "cpptoolsconstants.h" #include #include #include #include #include #include #include #include #include #include #include using namespace Core; namespace CppTools { namespace Internal { const char SETTINGS_GROUP[] = "CppSymbols"; const char SETTINGS_SYMBOLTYPES[] = "SymbolsToSearchFor"; const char SETTINGS_SEARCHSCOPE[] = "SearchScope"; SymbolsFindFilter::SymbolsFindFilter(CppModelManager *manager) : m_manager(manager), m_enabled(true), m_symbolsToSearch(SearchSymbols::AllTypes), m_scope(SymbolSearcher::SearchProjectsOnly) { // for disabling while parser is running connect(ProgressManager::instance(), SIGNAL(taskStarted(Core::Id)), this, SLOT(onTaskStarted(Core::Id))); connect(ProgressManager::instance(), SIGNAL(allTasksFinished(Core::Id)), this, SLOT(onAllTasksFinished(Core::Id))); } QString SymbolsFindFilter::id() const { return QLatin1String("CppSymbols"); } QString SymbolsFindFilter::displayName() const { return tr("C++ Symbols"); } bool SymbolsFindFilter::isEnabled() const { return m_enabled; } void SymbolsFindFilter::cancel() { Core::SearchResult *search = qobject_cast(sender()); QTC_ASSERT(search, return); QFutureWatcher *watcher = m_watchers.key(search); QTC_ASSERT(watcher, return); watcher->cancel(); } void SymbolsFindFilter::setPaused(bool paused) { Core::SearchResult *search = qobject_cast(sender()); QTC_ASSERT(search, return); QFutureWatcher *watcher = m_watchers.key(search); QTC_ASSERT(watcher, return); if (!paused || watcher->isRunning()) // guard against pausing when the search is finished watcher->setPaused(paused); } Core::FindFlags SymbolsFindFilter::supportedFindFlags() const { return Core::FindCaseSensitively | Core::FindRegularExpression | Core::FindWholeWords; } void SymbolsFindFilter::findAll(const QString &txt, Core::FindFlags findFlags) { Core::SearchResultWindow *window = Core::SearchResultWindow::instance(); Core::SearchResult *search = window->startNewSearch(label(), toolTip(findFlags), txt); search->setSearchAgainSupported(true); connect(search, SIGNAL(activated(Core::SearchResultItem)), this, SLOT(openEditor(Core::SearchResultItem))); connect(search, SIGNAL(cancelled()), this, SLOT(cancel())); connect(search, SIGNAL(paused(bool)), this, SLOT(setPaused(bool))); connect(search, SIGNAL(searchAgainRequested()), this, SLOT(searchAgain())); connect(this, SIGNAL(enabledChanged(bool)), search, SLOT(setSearchAgainEnabled(bool))); window->popup(IOutputPane::ModeSwitch | IOutputPane::WithFocus); SymbolSearcher::Parameters parameters; parameters.text = txt; parameters.flags = findFlags; parameters.types = m_symbolsToSearch; parameters.scope = m_scope; search->setUserData(qVariantFromValue(parameters)); startSearch(search); } void SymbolsFindFilter::startSearch(Core::SearchResult *search) { SymbolSearcher::Parameters parameters = search->userData().value(); QSet projectFileNames; if (parameters.scope == SymbolSearcher::SearchProjectsOnly) { foreach (ProjectExplorer::Project *project, ProjectExplorer::SessionManager::projects()) projectFileNames += project->files(ProjectExplorer::Project::AllFiles).toSet(); } QFutureWatcher *watcher = new QFutureWatcher(); m_watchers.insert(watcher, search); connect(watcher, SIGNAL(finished()), this, SLOT(finish())); connect(watcher, SIGNAL(resultsReadyAt(int,int)), this, SLOT(addResults(int,int))); SymbolSearcher *symbolSearcher = m_manager->indexingSupport()->createSymbolSearcher(parameters, projectFileNames); connect(watcher, SIGNAL(finished()), symbolSearcher, SLOT(deleteLater())); watcher->setFuture(QtConcurrent::run(&SymbolSearcher::runSearch, symbolSearcher)); FutureProgress *progress = ProgressManager::addTask(watcher->future(), tr("Searching for Symbol"), Core::Constants::TASK_SEARCH); connect(progress, SIGNAL(clicked()), search, SLOT(popup())); } void SymbolsFindFilter::addResults(int begin, int end) { QFutureWatcher *watcher = static_cast *>(sender()); Core::SearchResult *search = m_watchers.value(watcher); if (!search) { // search was removed from search history while the search is running watcher->cancel(); return; } QList items; for (int i = begin; i < end; ++i) items << watcher->resultAt(i); search->addResults(items, Core::SearchResult::AddSorted); } void SymbolsFindFilter::finish() { QFutureWatcher *watcher = static_cast *>(sender()); Core::SearchResult *search = m_watchers.value(watcher); if (search) search->finishSearch(watcher->isCanceled()); m_watchers.remove(watcher); watcher->deleteLater(); } void SymbolsFindFilter::openEditor(const Core::SearchResultItem &item) { if (!item.userData.canConvert()) return; IndexItem::Ptr info = item.userData.value(); EditorManager::openEditorAt(info->fileName(), info->line(), info->column()); } QWidget *SymbolsFindFilter::createConfigWidget() { return new SymbolsFindFilterConfigWidget(this); } void SymbolsFindFilter::writeSettings(QSettings *settings) { settings->beginGroup(QLatin1String(SETTINGS_GROUP)); settings->setValue(QLatin1String(SETTINGS_SYMBOLTYPES), (int)m_symbolsToSearch); settings->setValue(QLatin1String(SETTINGS_SEARCHSCOPE), (int)m_scope); settings->endGroup(); } void SymbolsFindFilter::readSettings(QSettings *settings) { settings->beginGroup(QLatin1String(SETTINGS_GROUP)); m_symbolsToSearch = (SearchSymbols::SymbolTypes)settings->value(QLatin1String(SETTINGS_SYMBOLTYPES), (int)SearchSymbols::AllTypes).toInt(); m_scope = (SearchScope)settings->value(QLatin1String(SETTINGS_SEARCHSCOPE), (int)SymbolSearcher::SearchProjectsOnly).toInt(); settings->endGroup(); emit symbolsToSearchChanged(); } void SymbolsFindFilter::onTaskStarted(Id type) { if (type == CppTools::Constants::TASK_INDEX) { m_enabled = false; emit enabledChanged(m_enabled); } } void SymbolsFindFilter::onAllTasksFinished(Core::Id type) { if (type == CppTools::Constants::TASK_INDEX) { m_enabled = true; emit enabledChanged(m_enabled); } } void SymbolsFindFilter::searchAgain() { Core::SearchResult *search = qobject_cast(sender()); QTC_ASSERT(search, return); search->restart(); startSearch(search); } QString SymbolsFindFilter::label() const { return tr("C++ Symbols:"); } QString SymbolsFindFilter::toolTip(Core::FindFlags findFlags) const { QStringList types; if (m_symbolsToSearch & SymbolSearcher::Classes) types.append(tr("Classes")); if (m_symbolsToSearch & SymbolSearcher::Functions) types.append(tr("Functions")); if (m_symbolsToSearch & SymbolSearcher::Enums) types.append(tr("Enums")); if (m_symbolsToSearch & SymbolSearcher::Declarations) types.append(tr("Declarations")); return tr("Scope: %1\nTypes: %2\nFlags: %3") .arg(searchScope() == SymbolSearcher::SearchGlobal ? tr("All") : tr("Projects")) .arg(types.join(tr(", "))) .arg(Core::IFindFilter::descriptionForFindFlags(findFlags)); } // #pragma mark -- SymbolsFindFilterConfigWidget SymbolsFindFilterConfigWidget::SymbolsFindFilterConfigWidget(SymbolsFindFilter *filter) : m_filter(filter) { connect(m_filter, SIGNAL(symbolsToSearchChanged()), this, SLOT(getState())); QGridLayout *layout = new QGridLayout(this); setLayout(layout); layout->setMargin(0); QLabel *typeLabel = new QLabel(tr("Types:")); layout->addWidget(typeLabel, 0, 0); m_typeClasses = new QCheckBox(tr("Classes")); layout->addWidget(m_typeClasses, 0, 1); m_typeMethods = new QCheckBox(tr("Functions")); layout->addWidget(m_typeMethods, 0, 2); m_typeEnums = new QCheckBox(tr("Enums")); layout->addWidget(m_typeEnums, 1, 1); m_typeDeclarations = new QCheckBox(tr("Declarations")); layout->addWidget(m_typeDeclarations, 1, 2); // hacks to fix layouting: typeLabel->setMinimumWidth(80); typeLabel->setAlignment(Qt::AlignRight); m_typeClasses->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); m_typeMethods->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); connect(m_typeClasses, SIGNAL(clicked(bool)), this, SLOT(setState())); connect(m_typeMethods, SIGNAL(clicked(bool)), this, SLOT(setState())); connect(m_typeEnums, SIGNAL(clicked(bool)), this, SLOT(setState())); connect(m_typeDeclarations, SIGNAL(clicked(bool)), this, SLOT(setState())); m_searchProjectsOnly = new QRadioButton(tr("Projects only")); layout->addWidget(m_searchProjectsOnly, 2, 1); m_searchGlobal = new QRadioButton(tr("All files")); layout->addWidget(m_searchGlobal, 2, 2); m_searchGroup = new QButtonGroup(this); m_searchGroup->addButton(m_searchProjectsOnly); m_searchGroup->addButton(m_searchGlobal); connect(m_searchProjectsOnly, SIGNAL(clicked(bool)), this, SLOT(setState())); connect(m_searchGlobal, SIGNAL(clicked(bool)), this, SLOT(setState())); } void SymbolsFindFilterConfigWidget::getState() { SearchSymbols::SymbolTypes symbols = m_filter->symbolsToSearch(); m_typeClasses->setChecked(symbols & SymbolSearcher::Classes); m_typeMethods->setChecked(symbols & SymbolSearcher::Functions); m_typeEnums->setChecked(symbols & SymbolSearcher::Enums); m_typeDeclarations->setChecked(symbols & SymbolSearcher::Declarations); SymbolsFindFilter::SearchScope scope = m_filter->searchScope(); m_searchProjectsOnly->setChecked(scope == SymbolSearcher::SearchProjectsOnly); m_searchGlobal->setChecked(scope == SymbolSearcher::SearchGlobal); } void SymbolsFindFilterConfigWidget::setState() const { SearchSymbols::SymbolTypes symbols; if (m_typeClasses->isChecked()) symbols |= SymbolSearcher::Classes; if (m_typeMethods->isChecked()) symbols |= SymbolSearcher::Functions; if (m_typeEnums->isChecked()) symbols |= SymbolSearcher::Enums; if (m_typeDeclarations->isChecked()) symbols |= SymbolSearcher::Declarations; m_filter->setSymbolsToSearch(symbols); if (m_searchProjectsOnly->isChecked()) m_filter->setSearchScope(SymbolSearcher::SearchProjectsOnly); else m_filter->setSearchScope(SymbolSearcher::SearchGlobal); } } // namespace Internal } // namespace CppTools