// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include "currentdocumentfind.h" #include "../coreplugintr.h" #include #include #include #include #include #include namespace Core::Internal { CurrentDocumentFind::CurrentDocumentFind() : m_currentFind(nullptr) { connect(qApp, &QApplication::focusChanged, this, &CurrentDocumentFind::updateCandidateFindFilter); } void CurrentDocumentFind::removeConnections() { disconnect(qApp, nullptr, this, nullptr); removeFindSupportConnections(); } void CurrentDocumentFind::resetIncrementalSearch() { QTC_ASSERT(m_currentFind, return); m_currentFind->resetIncrementalSearch(); } void CurrentDocumentFind::clearHighlights() { QTC_ASSERT(m_currentFind, return); m_currentFind->clearHighlights(); } bool CurrentDocumentFind::isEnabled() const { return m_currentFind && (!m_currentWidget || m_currentWidget->isVisible()); } IFindSupport *CurrentDocumentFind::candidate() const { return m_candidateFind; } bool CurrentDocumentFind::supportsReplace() const { if (!m_currentFind) return false; return m_currentFind->supportsReplace(); } bool CurrentDocumentFind::supportsSelectAll() const { if (!m_currentFind) return false; return m_currentFind->supportsSelectAll(); } FindFlags CurrentDocumentFind::supportedFindFlags() const { QTC_ASSERT(m_currentFind, return {}); return m_currentFind->supportedFindFlags(); } QString CurrentDocumentFind::currentFindString() const { if (!m_currentFind) return QString(); return m_currentFind->currentFindString(); } QString CurrentDocumentFind::completedFindString() const { QTC_ASSERT(m_currentFind, return QString()); return m_currentFind->completedFindString(); } void CurrentDocumentFind::highlightAll(const QString &txt, FindFlags findFlags) { if (!m_currentFind) return; m_currentFind->highlightAll(txt, findFlags); } IFindSupport::Result CurrentDocumentFind::findIncremental(const QString &txt, FindFlags findFlags) { QTC_ASSERT(m_currentFind, return IFindSupport::NotFound); return m_currentFind->findIncremental(txt, findFlags); } IFindSupport::Result CurrentDocumentFind::findStep(const QString &txt, FindFlags findFlags) { QTC_ASSERT(m_currentFind, return IFindSupport::NotFound); return m_currentFind->findStep(txt, findFlags); } void CurrentDocumentFind::selectAll(const QString &txt, FindFlags findFlags) { QTC_ASSERT(m_currentFind && m_currentFind->supportsSelectAll(), return); m_currentFind->selectAll(txt, findFlags); } void CurrentDocumentFind::replace(const QString &before, const QString &after, FindFlags findFlags) { QTC_ASSERT(m_currentFind, return); m_currentFind->replace(before, after, findFlags); } bool CurrentDocumentFind::replaceStep(const QString &before, const QString &after, FindFlags findFlags) { QTC_ASSERT(m_currentFind, return false); return m_currentFind->replaceStep(before, after, findFlags); } int CurrentDocumentFind::replaceAll(const QString &before, const QString &after, FindFlags findFlags) { QTC_ASSERT(m_currentFind, return 0); QTC_CHECK(m_currentWidget); int count = m_currentFind->replaceAll(before, after, findFlags); Utils::FadingIndicator::showText(m_currentWidget, Tr::tr("%n occurrences replaced.", nullptr, count), Utils::FadingIndicator::SmallText); return count; } void CurrentDocumentFind::defineFindScope() { if (!m_currentFind) return; m_currentFind->defineFindScope(); } void CurrentDocumentFind::clearFindScope() { QTC_ASSERT(m_currentFind, return); m_currentFind->clearFindScope(); } void CurrentDocumentFind::updateCandidateFindFilter(QWidget *old, QWidget *now) { Q_UNUSED(old) QWidget *candidate = now; QPointer impl = nullptr; while (!impl && candidate) { impl = Aggregation::query(candidate); if (!impl) candidate = candidate->parentWidget(); } if (candidate == m_candidateWidget && impl == m_candidateFind) { // trigger update of action state since a changed focus can still require disabling the // Find/Replace action emit changed(); return; } if (m_candidateWidget) disconnect(Aggregation::Aggregate::parentAggregate(m_candidateWidget), &Aggregation::Aggregate::changed, this, &CurrentDocumentFind::candidateAggregationChanged); m_candidateWidget = candidate; m_candidateFind = impl; if (m_candidateWidget) connect(Aggregation::Aggregate::parentAggregate(m_candidateWidget), &Aggregation::Aggregate::changed, this, &CurrentDocumentFind::candidateAggregationChanged); emit candidateChanged(); } void CurrentDocumentFind::acceptCandidate() { if (!m_candidateFind || m_candidateFind == m_currentFind) return; removeFindSupportConnections(); if (m_currentFind) m_currentFind->clearHighlights(); if (m_currentWidget) disconnect(Aggregation::Aggregate::parentAggregate(m_currentWidget), &Aggregation::Aggregate::changed, this, &CurrentDocumentFind::aggregationChanged); m_currentWidget = m_candidateWidget; connect(Aggregation::Aggregate::parentAggregate(m_currentWidget), &Aggregation::Aggregate::changed, this, &CurrentDocumentFind::aggregationChanged); m_currentFind = m_candidateFind; if (m_currentFind) { connect(m_currentFind.data(), &IFindSupport::changed, this, &CurrentDocumentFind::changed); connect(m_currentFind.data(), &QObject::destroyed, this, &CurrentDocumentFind::clearFindSupport); } if (m_currentWidget) m_currentWidget->installEventFilter(this); emit changed(); } void CurrentDocumentFind::removeFindSupportConnections() { if (m_currentFind) { disconnect(m_currentFind.data(), &IFindSupport::changed, this, &CurrentDocumentFind::changed); disconnect(m_currentFind.data(), &IFindSupport::destroyed, this, &CurrentDocumentFind::clearFindSupport); } if (m_currentWidget) m_currentWidget->removeEventFilter(this); } void CurrentDocumentFind::clearFindSupport() { removeFindSupportConnections(); m_currentWidget = nullptr; m_currentFind = nullptr; emit changed(); } bool CurrentDocumentFind::setFocusToCurrentFindSupport() { if (m_currentFind && m_currentWidget) { QWidget *w = m_currentWidget->focusWidget(); if (!w) w = m_currentWidget; w->setFocus(); return true; } return false; } bool CurrentDocumentFind::eventFilter(QObject *obj, QEvent *event) { if (m_currentWidget && obj == m_currentWidget) { if (event->type() == QEvent::Hide || event->type() == QEvent::Show) emit changed(); } return QObject::eventFilter(obj, event); } void CurrentDocumentFind::aggregationChanged() { if (m_currentWidget) { QPointer currentFind = Aggregation::query(m_currentWidget); if (currentFind != m_currentFind) { // There's a change in the find support if (currentFind) { m_candidateWidget = m_currentWidget; m_candidateFind = currentFind; acceptCandidate(); } else { clearFindSupport(); } } } } void CurrentDocumentFind::candidateAggregationChanged() { if (m_candidateWidget && m_candidateWidget != m_currentWidget) { m_candidateFind = Aggregation::query(m_candidateWidget); emit candidateChanged(); } } } // Core::Internal