/**************************************************************************** ** ** 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 "symbolsfindfilter.h" #include "cppmodelmanager.h" #include "cpptoolsconstants.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Core; using namespace Utils; 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(), &ProgressManager::taskStarted, this, &SymbolsFindFilter::onTaskStarted); connect(ProgressManager::instance(), &ProgressManager::allTasksFinished, this, &SymbolsFindFilter::onAllTasksFinished); } QString SymbolsFindFilter::id() const { return QLatin1String(Constants::SYMBOLS_FIND_FILTER_ID); } QString SymbolsFindFilter::displayName() const { return QString(Constants::SYMBOLS_FIND_FILTER_DISPLAY_NAME); } bool SymbolsFindFilter::isEnabled() const { return m_enabled; } void SymbolsFindFilter::cancel() { auto 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) { auto 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); } void SymbolsFindFilter::findAll(const QString &txt, FindFlags findFlags) { SearchResultWindow *window = SearchResultWindow::instance(); SearchResult *search = window->startNewSearch(label(), toolTip(findFlags), txt); search->setSearchAgainSupported(true); connect(search, &SearchResult::activated, this, &SymbolsFindFilter::openEditor); connect(search, &SearchResult::cancelled, this, &SymbolsFindFilter::cancel); connect(search, &SearchResult::paused, this, &SymbolsFindFilter::setPaused); connect(search, &SearchResult::searchAgainRequested, this, &SymbolsFindFilter::searchAgain); connect(this, &IFindFilter::enabledChanged, search, &SearchResult::setSearchAgainEnabled); 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(QVariant::fromValue(parameters)); startSearch(search); } void SymbolsFindFilter::startSearch(SearchResult *search) { SymbolSearcher::Parameters parameters = search->userData().value(); QSet projectFileNames; if (parameters.scope == SymbolSearcher::SearchProjectsOnly) { for (ProjectExplorer::Project *project : ProjectExplorer::SessionManager::projects()) projectFileNames += Utils::transform(project->files(ProjectExplorer::Project::AllFiles), &Utils::FilePath::toString); } auto watcher = new QFutureWatcher; m_watchers.insert(watcher, search); connect(watcher, &QFutureWatcherBase::finished, this, &SymbolsFindFilter::finish); connect(watcher, &QFutureWatcherBase::resultsReadyAt, this, &SymbolsFindFilter::addResults); SymbolSearcher *symbolSearcher = m_manager->indexingSupport()->createSymbolSearcher(parameters, projectFileNames); connect(watcher, &QFutureWatcherBase::finished, symbolSearcher, &QObject::deleteLater); watcher->setFuture(Utils::runAsync(m_manager->sharedThreadPool(), &SymbolSearcher::runSearch, symbolSearcher)); FutureProgress *progress = ProgressManager::addTask(watcher->future(), tr("Searching for Symbol"), Core::Constants::TASK_SEARCH); connect(progress, &FutureProgress::clicked, search, &SearchResult::popup); } void SymbolsFindFilter::addResults(int begin, int end) { auto watcher = static_cast *>(sender()); 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, SearchResult::AddSorted); } void SymbolsFindFilter::finish() { auto watcher = static_cast *>(sender()); SearchResult *search = m_watchers.value(watcher); if (search) search->finishSearch(watcher->isCanceled()); m_watchers.remove(watcher); watcher->deleteLater(); } void SymbolsFindFilter::openEditor(const 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 = static_cast( settings->value(QLatin1String(SETTINGS_SYMBOLTYPES), int(SearchSymbols::AllTypes)).toInt()); m_scope = static_cast( 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(Id type) { if (type == CppTools::Constants::TASK_INDEX) { m_enabled = true; emit enabledChanged(m_enabled); } } void SymbolsFindFilter::searchAgain() { auto search = qobject_cast(sender()); QTC_ASSERT(search, return); search->restart(); startSearch(search); } QString SymbolsFindFilter::label() const { return tr("C++ Symbols:"); } QString SymbolsFindFilter::toolTip(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"), types.join(", "), IFindFilter::descriptionForFindFlags(findFlags)); } // #pragma mark -- SymbolsFindFilterConfigWidget SymbolsFindFilterConfigWidget::SymbolsFindFilterConfigWidget(SymbolsFindFilter *filter) : m_filter(filter) { connect(m_filter, &SymbolsFindFilter::symbolsToSearchChanged, this, &SymbolsFindFilterConfigWidget::getState); auto layout = new QGridLayout(this); setLayout(layout); layout->setContentsMargins(0, 0, 0, 0); auto 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, &QAbstractButton::clicked, this, &SymbolsFindFilterConfigWidget::setState); connect(m_typeMethods, &QAbstractButton::clicked, this, &SymbolsFindFilterConfigWidget::setState); connect(m_typeEnums, &QAbstractButton::clicked, this, &SymbolsFindFilterConfigWidget::setState); connect(m_typeDeclarations, &QAbstractButton::clicked, this, &SymbolsFindFilterConfigWidget::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, &QAbstractButton::clicked, this, &SymbolsFindFilterConfigWidget::setState); connect(m_searchGlobal, &QAbstractButton::clicked, this, &SymbolsFindFilterConfigWidget::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