// 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 "docsettingspage.h" #include "helpconstants.h" #include "helpmanager.h" #include "helptr.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace Utils; namespace Help { namespace Internal { class DocEntry { public: QString name; QString fileName; QString nameSpace; }; bool operator<(const DocEntry &d1, const DocEntry &d2) { return d1.name < d2.name; } class DocModel : public QAbstractListModel { public: using DocEntries = QVector; DocModel() = default; void setEntries(const DocEntries &e) { m_docEntries = e; } int rowCount(const QModelIndex & = QModelIndex()) const override { return m_docEntries.size(); } QVariant data(const QModelIndex &index, int role) const override; void insertEntry(const DocEntry &e); void removeAt(int row); const DocEntry &entryAt(int row) const { return m_docEntries.at(row); } private: DocEntries m_docEntries; }; class DocSettingsPageWidget : public Core::IOptionsPageWidget { public: DocSettingsPageWidget(); private: void apply() final; void addDocumentation(); bool eventFilter(QObject *object, QEvent *event) final; void removeDocumentation(const QList &items); QList currentSelection() const; FilePath m_recentDialogPath; using NameSpaceToPathHash = QMultiHash; NameSpaceToPathHash m_filesToRegister; QHash m_filesToRegisterUserManaged; NameSpaceToPathHash m_filesToUnregister; QListView *m_docsListView = nullptr; QSortFilterProxyModel m_proxyModel; DocModel m_model; }; static DocEntry createEntry(const QString &nameSpace, const QString &fileName, bool userManaged) { DocEntry result; result.name = userManaged ? nameSpace : Tr::tr("%1 (auto-detected)").arg(nameSpace); result.fileName = fileName; result.nameSpace = nameSpace; return result; } QVariant DocModel::data(const QModelIndex &index, int role) const { QVariant result; const int row = index.row(); if (index.isValid() && row < m_docEntries.size()) { switch (role) { case Qt::DisplayRole: result = QVariant(m_docEntries.at(row).name); break; case Qt::ToolTipRole: result = QVariant(QDir::toNativeSeparators(m_docEntries.at(row).fileName)); break; case Qt::UserRole: result = QVariant(m_docEntries.at(row).nameSpace); break; default: break; } } return result; } void DocModel::insertEntry(const DocEntry &e) { const auto it = std::lower_bound(m_docEntries.begin(), m_docEntries.end(), e); const int index = int(it - m_docEntries.begin()); beginInsertRows(QModelIndex(), index, index); m_docEntries.insert(it, e); endInsertRows(); } void DocModel::removeAt(int row) { beginRemoveRows(QModelIndex(), row, row); m_docEntries.removeAt(row); endRemoveRows(); } } // namespace Internal using namespace Help::Internal; DocSettingsPageWidget::DocSettingsPageWidget() { auto groupBox = new QGroupBox(this); auto filterLineEdit = new Utils::FancyLineEdit(groupBox); m_docsListView = new QListView(groupBox); m_docsListView->setObjectName("docsListView"); m_docsListView->setSelectionMode(QAbstractItemView::ExtendedSelection); m_docsListView->setUniformItemSizes(true); auto treeLayout = new QVBoxLayout(); treeLayout->addWidget(filterLineEdit); treeLayout->addWidget(m_docsListView); auto addButton = new QPushButton(groupBox); addButton->setObjectName("addButton"); auto removeButton = new QPushButton(groupBox); removeButton->setObjectName("removeButton"); auto buttonLayout = new QVBoxLayout(); buttonLayout->addWidget(addButton); buttonLayout->addWidget(removeButton); buttonLayout->addStretch(1); auto horizontalLayout = new QHBoxLayout(groupBox); horizontalLayout->addLayout(treeLayout); horizontalLayout->addLayout(buttonLayout); setLayout(new QVBoxLayout); layout()->addWidget(groupBox); setToolTip(Tr::tr("Add and remove compressed help files, .qch.")); groupBox->setTitle(Tr::tr("Registered Documentation")); addButton->setText(Tr::tr("Add...")); removeButton->setText(Tr::tr("Remove")); const QStringList nameSpaces = HelpManager::registeredNamespaces(); const QSet userDocumentationPaths = HelpManager::userDocumentationPaths(); DocModel::DocEntries entries; entries.reserve(nameSpaces.size()); for (const QString &nameSpace : nameSpaces) { const QString filePath = HelpManager::fileFromNamespace(nameSpace); bool user = userDocumentationPaths.contains(filePath); entries.append(createEntry(nameSpace, filePath, user)); m_filesToRegister.insert(nameSpace, filePath); m_filesToRegisterUserManaged.insert(nameSpace, user); } std::stable_sort(entries.begin(), entries.end()); m_model.setEntries(entries); m_proxyModel.setSourceModel(&m_model); m_docsListView->setModel(&m_proxyModel); filterLineEdit->setFiltering(true); connect(filterLineEdit, &QLineEdit::textChanged, &m_proxyModel, &QSortFilterProxyModel::setFilterFixedString); connect(addButton, &QAbstractButton::clicked, this, &DocSettingsPageWidget::addDocumentation); connect(removeButton, &QAbstractButton::clicked, this, [this] { removeDocumentation(currentSelection()); }); m_docsListView->installEventFilter(this); } void DocSettingsPageWidget::addDocumentation() { const FilePaths files = FileUtils::getOpenFilePaths(Core::ICore::dialogParent(), Tr::tr("Add Documentation"), m_recentDialogPath, Tr::tr("Qt Help Files (*.qch)")); if (files.isEmpty()) return; m_recentDialogPath = files.first().canonicalPath(); NameSpaceToPathHash docsUnableToRegister; for (const FilePath &file : files) { const QString filePath = file.cleanPath().toString(); const QString &nameSpace = HelpManager::namespaceFromFile(filePath); if (nameSpace.isEmpty()) { docsUnableToRegister.insert("UnknownNamespace", file.toUserOutput()); continue; } if (m_filesToRegister.contains(nameSpace)) { docsUnableToRegister.insert(nameSpace, file.toUserOutput()); continue; } m_model.insertEntry(createEntry(nameSpace, file.toString(), true /* user managed */)); m_filesToRegister.insert(nameSpace, filePath); m_filesToRegisterUserManaged.insert(nameSpace, true/*user managed*/); // If the files to unregister contains the namespace, grab a copy of all paths added and try to // remove the current file path. Afterwards remove the whole entry and add the clean list back. // Possible outcome: // - might not add the entry back at all if we register the same file again // - might add the entry back with paths to other files with the same namespace // The reason to do this is, if we remove a file with a given namespace/ path and re-add another // file with the same namespace but a different path, we need to unregister the namespace before // we can register the new one. Help engine allows just one registered namespace. if (m_filesToUnregister.contains(nameSpace)) { QSet values = Utils::toSet(m_filesToUnregister.values(nameSpace)); values.remove(filePath); m_filesToUnregister.remove(nameSpace); for (const QString &value : std::as_const(values)) m_filesToUnregister.insert(nameSpace, value); } } QString formatedFail; if (docsUnableToRegister.contains("UnknownNamespace")) { formatedFail += QString::fromLatin1("
  • %1") .arg(Tr::tr("Invalid documentation file:")); const QStringList values = docsUnableToRegister.values("UnknownNamespace"); for (const QString &value : values) formatedFail += QString::fromLatin1("
    • %2
    ").arg(value); formatedFail += "
"; docsUnableToRegister.remove("UnknownNamespace"); } if (!docsUnableToRegister.isEmpty()) { formatedFail += QString::fromLatin1("
  • %1") .arg(Tr::tr("Namespace already registered:")); const NameSpaceToPathHash::ConstIterator cend = docsUnableToRegister.constEnd(); for (NameSpaceToPathHash::ConstIterator it = docsUnableToRegister.constBegin(); it != cend; ++it) { formatedFail += QString::fromLatin1("
    • %1 - %2
    ").arg(it.key(), it.value()); } formatedFail += "
"; } if (!formatedFail.isEmpty()) { QMessageBox::information(Core::ICore::dialogParent(), Tr::tr("Registration Failed"), Tr::tr("Unable to register documentation.") + formatedFail, QMessageBox::Ok); } } void DocSettingsPageWidget::apply() { HelpManager::instance()->unregisterDocumentation(m_filesToUnregister.values()); QStringList files; auto it = m_filesToRegisterUserManaged.constBegin(); while (it != m_filesToRegisterUserManaged.constEnd()) { if (it.value()/*userManaged*/) files << m_filesToRegister.value(it.key()); ++it; } HelpManager::registerUserDocumentation(files); m_filesToUnregister.clear(); } bool DocSettingsPageWidget::eventFilter(QObject *object, QEvent *event) { if (object != m_docsListView) return IOptionsPageWidget::eventFilter(object, event); if (event->type() == QEvent::KeyPress) { auto ke = static_cast(event); switch (ke->key()) { case Qt::Key_Backspace: case Qt::Key_Delete: removeDocumentation(currentSelection()); break; default: break; } } return IOptionsPageWidget::eventFilter(object, event); } void DocSettingsPageWidget::removeDocumentation(const QList &items) { if (items.isEmpty()) return; const QList itemsByDecreasingRow = Utils::sorted(items, [](const QModelIndex &i1, const QModelIndex &i2) { return i1.row() > i2.row(); }); for (const QModelIndex &item : itemsByDecreasingRow) { const int row = item.row(); const QString nameSpace = m_model.entryAt(row).nameSpace; m_filesToRegister.remove(nameSpace); m_filesToRegisterUserManaged.remove(nameSpace); m_filesToUnregister.insert(nameSpace, QDir::cleanPath(HelpManager::fileFromNamespace(nameSpace))); m_model.removeAt(row); } const int newlySelectedRow = qMax(itemsByDecreasingRow.last().row() - 1, 0); const QModelIndex index = m_proxyModel.mapFromSource(m_model.index(newlySelectedRow)); m_docsListView->selectionModel()->select(index, QItemSelectionModel::ClearAndSelect); } QList DocSettingsPageWidget::currentSelection() const { return Utils::transform(m_docsListView->selectionModel()->selectedRows(), [this](const QModelIndex &index) { return m_proxyModel.mapToSource(index); }); } DocSettingsPage::DocSettingsPage() { setId("B.Documentation"); setDisplayName(Tr::tr("Documentation")); setCategory(Help::Constants::HELP_CATEGORY); setWidgetCreator([] { return new DocSettingsPageWidget; }); } } // namespace Help