/**************************************************************************** ** ** 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 "basefilefind.h" #include "textdocument.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Utils; using namespace Core; namespace TextEditor { namespace Internal { namespace { class InternalEngine : public TextEditor::SearchEngine { public: InternalEngine() : m_widget(new QWidget) {} ~InternalEngine() override { delete m_widget;} QString title() const override { return TextEditor::SearchEngine::tr("Internal"); } QString toolTip() const override { return QString(); } QWidget *widget() const override { return m_widget; } QVariant parameters() const override { return QVariant(); } void readSettings(QSettings * /*settings*/) override {} void writeSettings(QSettings * /*settings*/) const override {} QFuture executeSearch( const TextEditor::FileFindParameters ¶meters, BaseFileFind *baseFileFind) override { auto func = parameters.flags & FindRegularExpression ? Utils::findInFilesRegExp : Utils::findInFiles; return func(parameters.text, baseFileFind->files(parameters.nameFilters, parameters.exclusionFilters, parameters.additionalParameters), textDocumentFlagsForFindFlags(parameters.flags), TextDocument::openedTextDocumentContents()); } Core::IEditor *openEditor(const Core::SearchResultItem &/*item*/, const TextEditor::FileFindParameters &/*parameters*/) override { return nullptr; } private: QWidget *m_widget; }; } // namespace class SearchEnginePrivate { public: bool isEnabled = true; }; class CountingLabel : public QLabel { public: CountingLabel(); void updateCount(int count); }; class BaseFileFindPrivate { public: ~BaseFileFindPrivate() { delete m_internalSearchEngine; } QPointer m_currentFindSupport; QLabel *m_resultLabel = nullptr; // models in native path format QStringListModel m_filterStrings; QStringListModel m_exclusionStrings; // current filter in portable path format QString m_filterSetting; QString m_exclusionSetting; QPointer m_filterCombo; QPointer m_exclusionCombo; QVector m_searchEngines; SearchEngine *m_internalSearchEngine; int m_currentSearchEngineIndex = -1; }; } // namespace Internal static void syncComboWithSettings(QComboBox *combo, const QString &setting) { if (!combo) return; const QString &nativeSettings = QDir::toNativeSeparators(setting); int index = combo->findText(nativeSettings); if (index < 0) combo->setEditText(nativeSettings); else combo->setCurrentIndex(index); } static void updateComboEntries(QComboBox *combo, bool onTop) { int index = combo->findText(combo->currentText()); if (index < 0) { if (onTop) combo->insertItem(0, combo->currentText()); else combo->addItem(combo->currentText()); combo->setCurrentIndex(combo->findText(combo->currentText())); } } using namespace Internal; SearchEngine::SearchEngine(QObject *parent) : QObject(parent), d(new SearchEnginePrivate) { } SearchEngine::~SearchEngine() { delete d; } bool SearchEngine::isEnabled() const { return d->isEnabled; } void SearchEngine::setEnabled(bool enabled) { if (enabled == d->isEnabled) return; d->isEnabled = enabled; emit enabledChanged(d->isEnabled); } BaseFileFind::BaseFileFind() : d(new BaseFileFindPrivate) { d->m_internalSearchEngine = new InternalEngine; addSearchEngine(d->m_internalSearchEngine); } BaseFileFind::~BaseFileFind() { delete d; } bool BaseFileFind::isEnabled() const { return true; } QStringList BaseFileFind::fileNameFilters() const { if (d->m_filterCombo) return splitFilterUiText(d->m_filterCombo->currentText()); return QStringList(); } QStringList BaseFileFind::fileExclusionFilters() const { if (d->m_exclusionCombo) return splitFilterUiText(d->m_exclusionCombo->currentText()); return QStringList(); } SearchEngine *BaseFileFind::currentSearchEngine() const { if (d->m_searchEngines.isEmpty() || d->m_currentSearchEngineIndex == -1) return nullptr; return d->m_searchEngines[d->m_currentSearchEngineIndex]; } QVector BaseFileFind::searchEngines() const { return d->m_searchEngines; } void BaseFileFind::setCurrentSearchEngine(int index) { if (d->m_currentSearchEngineIndex == index) return; d->m_currentSearchEngineIndex = index; emit currentSearchEngineChanged(); } static void displayResult(QFutureWatcher *watcher, SearchResult *search, int index) { FileSearchResultList results = watcher->resultAt(index); QList items; foreach (const FileSearchResult &result, results) { SearchResultItem item; item.path = QStringList() << QDir::toNativeSeparators(result.fileName); item.mainRange.begin.line = result.lineNumber; item.mainRange.begin.column = result.matchStart; item.mainRange.end = item.mainRange.begin; item.mainRange.end.column += result.matchLength; item.text = result.matchingLine; item.useTextEditorFont = true; item.userData = result.regexpCapturedTexts; items << item; } search->addResults(items, SearchResult::AddOrdered); } void BaseFileFind::runNewSearch(const QString &txt, FindFlags findFlags, SearchResultWindow::SearchMode searchMode) { d->m_currentFindSupport = nullptr; if (d->m_filterCombo) updateComboEntries(d->m_filterCombo, true); if (d->m_exclusionCombo) updateComboEntries(d->m_exclusionCombo, true); QString tooltip = toolTip(); SearchResult *search = SearchResultWindow::instance()->startNewSearch( label(), tooltip.arg(IFindFilter::descriptionForFindFlags(findFlags)), txt, searchMode, SearchResultWindow::PreserveCaseEnabled, QString::fromLatin1("TextEditor")); search->setTextToReplace(txt); search->setSearchAgainSupported(true); FileFindParameters parameters; parameters.text = txt; parameters.flags = findFlags; parameters.nameFilters = fileNameFilters(); parameters.exclusionFilters = fileExclusionFilters(); parameters.additionalParameters = additionalParameters(); parameters.searchEngineParameters = currentSearchEngine()->parameters(); parameters.searchEngineIndex = d->m_currentSearchEngineIndex; search->setUserData(qVariantFromValue(parameters)); connect(search, &SearchResult::activated, this, &BaseFileFind::openEditor); if (searchMode == SearchResultWindow::SearchAndReplace) connect(search, &SearchResult::replaceButtonClicked, this, &BaseFileFind::doReplace); connect(search, &SearchResult::visibilityChanged, this, &BaseFileFind::hideHighlightAll); connect(search, &SearchResult::searchAgainRequested, this, &BaseFileFind::searchAgain); connect(this, &BaseFileFind::enabledChanged, search, &SearchResult::requestEnabledCheck); connect(search, &SearchResult::requestEnabledCheck, this, &BaseFileFind::recheckEnabled); runSearch(search); } void BaseFileFind::runSearch(SearchResult *search) { FileFindParameters parameters = search->userData().value(); auto label = new CountingLabel; connect(search, &SearchResult::countChanged, label, &CountingLabel::updateCount); auto statusLabel = new CountingLabel; connect(search, &SearchResult::countChanged, statusLabel, &CountingLabel::updateCount); SearchResultWindow::instance()->popup(IOutputPane::Flags(IOutputPane::ModeSwitch|IOutputPane::WithFocus)); auto watcher = new QFutureWatcher(); watcher->setPendingResultsLimit(1); // search is deleted if it is removed from search panel connect(search, &QObject::destroyed, watcher, &QFutureWatcherBase::cancel); connect(search, &SearchResult::cancelled, watcher, &QFutureWatcherBase::cancel); connect(search, &SearchResult::paused, watcher, [watcher](bool paused) { if (!paused || watcher->isRunning()) // guard against pausing when the search is finished watcher->setPaused(paused); }); connect(watcher, &QFutureWatcherBase::resultReadyAt, search, [watcher, search](int index) { displayResult(watcher, search, index); }); // auto-delete: connect(watcher, &QFutureWatcherBase::finished, watcher, &QObject::deleteLater); connect(watcher, &QFutureWatcherBase::finished, search, [watcher, search]() { search->finishSearch(watcher->isCanceled()); }); watcher->setFuture(executeSearch(parameters)); FutureProgress *progress = ProgressManager::addTask(watcher->future(), tr("Searching"), Constants::TASK_SEARCH); progress->setWidget(label); progress->setStatusBarWidget(statusLabel); connect(progress, &FutureProgress::clicked, search, &SearchResult::popup); } void BaseFileFind::findAll(const QString &txt, FindFlags findFlags) { runNewSearch(txt, findFlags, SearchResultWindow::SearchOnly); } void BaseFileFind::replaceAll(const QString &txt, FindFlags findFlags) { runNewSearch(txt, findFlags, SearchResultWindow::SearchAndReplace); } void BaseFileFind::addSearchEngine(SearchEngine *searchEngine) { d->m_searchEngines.push_back(searchEngine); if (d->m_searchEngines.size() == 1) // empty before, make sure we have a current engine setCurrentSearchEngine(0); } void BaseFileFind::doReplace(const QString &text, const QList &items, bool preserveCase) { QStringList files = replaceAll(text, items, preserveCase); if (!files.isEmpty()) { Utils::FadingIndicator::showText(ICore::mainWindow(), tr("%n occurrences replaced.", nullptr, items.size()), Utils::FadingIndicator::SmallText); DocumentManager::notifyFilesChangedInternally(files); SearchResultWindow::instance()->hide(); } } static QComboBox *createCombo(QAbstractItemModel *model) { auto combo = new QComboBox; combo->setEditable(true); combo->setModel(model); combo->setMaxCount(10); combo->setMinimumContentsLength(10); combo->setSizeAdjustPolicy(QComboBox::AdjustToMinimumContentsLengthWithIcon); combo->setInsertPolicy(QComboBox::InsertAtBottom); combo->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); return combo; } static QLabel *createLabel(const QString &text) { auto filePatternLabel = new QLabel(text); filePatternLabel->setMinimumWidth(80); filePatternLabel->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Preferred); filePatternLabel->setAlignment(Qt::AlignRight | Qt::AlignVCenter); return filePatternLabel; } QList> BaseFileFind::createPatternWidgets() { QLabel *filterLabel = createLabel(msgFilePatternLabel()); d->m_filterCombo = createCombo(&d->m_filterStrings); d->m_filterCombo->setToolTip(msgFilePatternToolTip()); filterLabel->setBuddy(d->m_filterCombo); syncComboWithSettings(d->m_filterCombo, d->m_filterSetting); QLabel *exclusionLabel = createLabel(msgExclusionPatternLabel()); d->m_exclusionCombo = createCombo(&d->m_exclusionStrings); d->m_exclusionCombo->setToolTip(msgFilePatternToolTip()); exclusionLabel->setBuddy(d->m_exclusionCombo); syncComboWithSettings(d->m_exclusionCombo, d->m_exclusionSetting); return { qMakePair(filterLabel, d->m_filterCombo), qMakePair(exclusionLabel, d->m_exclusionCombo) }; } void BaseFileFind::writeCommonSettings(QSettings *settings) { std::function fromNativeSeparators = [](const QStringList &files) { return Utils::transform(files, &QDir::fromNativeSeparators); }; settings->setValue("filters", fromNativeSeparators(d->m_filterStrings.stringList())); if (d->m_filterCombo) settings->setValue("currentFilter", QDir::fromNativeSeparators(d->m_filterCombo->currentText())); settings->setValue("exclusionFilters", fromNativeSeparators(d->m_exclusionStrings.stringList())); if (d->m_exclusionCombo) settings->setValue("currentExclusionFilter", QDir::fromNativeSeparators(d->m_exclusionCombo->currentText())); foreach (SearchEngine *searchEngine, d->m_searchEngines) searchEngine->writeSettings(settings); settings->setValue("currentSearchEngineIndex", d->m_currentSearchEngineIndex); } void BaseFileFind::readCommonSettings(QSettings *settings, const QString &defaultFilter, const QString &defaultExclusionFilter) { std::function toNativeSeparators = [](const QStringList &files) { return Utils::transform(files, &QDir::toNativeSeparators); }; QStringList filters = settings->value("filters").toStringList(); if (filters.isEmpty()) filters << defaultFilter; const QVariant currentFilter = settings->value("currentFilter"); d->m_filterSetting = currentFilter.isValid() ? currentFilter.toString() : filters.first(); d->m_filterStrings.setStringList(toNativeSeparators(filters)); if (d->m_filterCombo) syncComboWithSettings(d->m_filterCombo, d->m_filterSetting); QStringList exclusionFilters = settings->value("exclusionFilters").toStringList(); if (exclusionFilters.isEmpty()) exclusionFilters << defaultExclusionFilter; const QVariant currentExclusionFilter = settings->value("currentExclusionFilter"); d->m_exclusionSetting = currentExclusionFilter.isValid() ? currentExclusionFilter.toString() : exclusionFilters.first(); d->m_exclusionStrings.setStringList(toNativeSeparators(exclusionFilters)); if (d->m_exclusionCombo) syncComboWithSettings(d->m_exclusionCombo, d->m_exclusionSetting); foreach (SearchEngine* searchEngine, d->m_searchEngines) searchEngine->readSettings(settings); const int currentSearchEngineIndex = settings->value("currentSearchEngineIndex", 0).toInt(); syncSearchEngineCombo(currentSearchEngineIndex); } void BaseFileFind::openEditor(const SearchResultItem &item) { auto result = qobject_cast(sender()); FileFindParameters parameters = result->userData().value(); IEditor *openedEditor = d->m_searchEngines[parameters.searchEngineIndex]->openEditor(item, parameters); if (!openedEditor) EditorManager::openEditorAtSearchResult(item, EditorManager::DoNotSwitchToDesignMode); if (d->m_currentFindSupport) d->m_currentFindSupport->clearHighlights(); d->m_currentFindSupport = nullptr; if (!openedEditor) return; // highlight results if (auto findSupport = Aggregation::query(openedEditor->widget())) { d->m_currentFindSupport = findSupport; d->m_currentFindSupport->highlightAll(parameters.text, parameters.flags); } } void BaseFileFind::hideHighlightAll(bool visible) { if (!visible && d->m_currentFindSupport) d->m_currentFindSupport->clearHighlights(); } void BaseFileFind::searchAgain() { auto search = qobject_cast(sender()); search->restart(); runSearch(search); } void BaseFileFind::recheckEnabled() { auto search = qobject_cast(sender()); if (!search) return; search->setSearchAgainEnabled(isEnabled()); } QStringList BaseFileFind::replaceAll(const QString &text, const QList &items, bool preserveCase) { if (items.isEmpty()) return QStringList(); RefactoringChanges refactoring; QHash > changes; foreach (const SearchResultItem &item, items) changes[QDir::fromNativeSeparators(item.path.first())].append(item); // Checking for files without write permissions QHashIterator > it(changes); QSet roFiles; while (it.hasNext()) { it.next(); const QFileInfo fileInfo(it.key()); if (!fileInfo.isWritable()) roFiles.insert(it.key()); } // Query the user for permissions if (!roFiles.isEmpty()) { ReadOnlyFilesDialog roDialog(roFiles.toList(), ICore::mainWindow()); roDialog.setShowFailWarning(true, tr("Aborting replace.")); if (roDialog.exec() == ReadOnlyFilesDialog::RO_Cancel) return QStringList(); } it.toFront(); while (it.hasNext()) { it.next(); const QString fileName = it.key(); const QList changeItems = it.value(); ChangeSet changeSet; RefactoringFilePtr file = refactoring.file(fileName); QSet > processed; foreach (const SearchResultItem &item, changeItems) { const QPair &p = qMakePair(item.mainRange.begin.line, item.mainRange.begin.column); if (processed.contains(p)) continue; processed.insert(p); QString replacement; if (item.userData.canConvert() && !item.userData.toStringList().isEmpty()) { replacement = Utils::expandRegExpReplacement(text, item.userData.toStringList()); } else if (preserveCase) { const QString originalText = (item.mainRange.length() == 0) ? item.text : item.mainRange.mid(text); replacement = Utils::matchCaseReplacement(originalText, text); } else { replacement = text; } const int start = file->position(item.mainRange.begin.line, item.mainRange.begin.column + 1); const int end = file->position(item.mainRange.end.line, item.mainRange.end.column + 1); changeSet.replace(start, end, replacement); } file->setChangeSet(changeSet); file->apply(); } return changes.keys(); } QVariant BaseFileFind::getAdditionalParameters(SearchResult *search) { return search->userData().value().additionalParameters; } QFuture BaseFileFind::executeSearch(const FileFindParameters ¶meters) { return d->m_searchEngines[parameters.searchEngineIndex]->executeSearch(parameters, this); } namespace Internal { CountingLabel::CountingLabel() { setAlignment(Qt::AlignCenter); // ### TODO this setup should be done by style QFont f = font(); f.setBold(true); f.setPointSizeF(StyleHelper::sidebarFontSize()); setFont(f); setPalette(StyleHelper::sidebarFontPalette(palette())); setProperty("_q_custom_style_disabled", QVariant(true)); updateCount(0); } void CountingLabel::updateCount(int count) { setText(BaseFileFind::tr("%n found.", nullptr, count)); } } // namespace Internal } // namespace TextEditor