/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the Qt Designer of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** 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://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: 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. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "appfontdialog.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE enum {FileNameRole = Qt::UserRole + 1, IdRole = Qt::UserRole + 2 }; enum { debugAppFontWidget = 0 }; static const char fontFileKeyC[] = "fontFiles"; // AppFontManager: Singleton that maintains the mapping of loaded application font // ids to the file names (which are not stored in QFontDatabase) // and provides API for loading/unloading fonts as well for saving/restoring settings. class AppFontManager { Q_DISABLE_COPY(AppFontManager) AppFontManager(); public: static AppFontManager &instance(); void save(QDesignerSettingsInterface *s, const QString &prefix) const; void restore(const QDesignerSettingsInterface *s, const QString &prefix); // Return id or -1 int add(const QString &fontFile, QString *errorMessage); bool remove(int id, QString *errorMessage); bool remove(const QString &fontFile, QString *errorMessage); bool removeAt(int index, QString *errorMessage); // Store loaded fonts as pair of file name and Id typedef QPair FileNameFontIdPair; typedef QList FileNameFontIdPairs; const FileNameFontIdPairs &fonts() const; private: FileNameFontIdPairs m_fonts; }; AppFontManager::AppFontManager() { } AppFontManager &AppFontManager::instance() { static AppFontManager rc; return rc; } void AppFontManager::save(QDesignerSettingsInterface *s, const QString &prefix) const { // Store as list of file names QStringList fontFiles; const FileNameFontIdPairs::const_iterator cend = m_fonts.constEnd(); for (FileNameFontIdPairs::const_iterator it = m_fonts.constBegin(); it != cend; ++it) fontFiles.push_back(it->first); s->beginGroup(prefix); s->setValue(QLatin1String(fontFileKeyC), fontFiles); s->endGroup(); if (debugAppFontWidget) qDebug() << "AppFontManager::saved" << fontFiles.size() << "fonts under " << prefix; } void AppFontManager::restore(const QDesignerSettingsInterface *s, const QString &prefix) { QString key = prefix; key += QLatin1Char('/'); key += QLatin1String(fontFileKeyC); const QStringList fontFiles = s->value(key, QStringList()).toStringList(); if (debugAppFontWidget) qDebug() << "AppFontManager::restoring" << fontFiles.size() << "fonts from " << prefix; if (!fontFiles.empty()) { QString errorMessage; const QStringList::const_iterator cend = fontFiles.constEnd(); for (QStringList::const_iterator it = fontFiles.constBegin(); it != cend; ++it) if (add(*it, &errorMessage) == -1) qWarning("%s", qPrintable(errorMessage)); } } int AppFontManager::add(const QString &fontFile, QString *errorMessage) { const QFileInfo inf(fontFile); if (!inf.isFile()) { *errorMessage = QCoreApplication::translate("AppFontManager", "'%1' is not a file.").arg(fontFile); return -1; } if (!inf.isReadable()) { *errorMessage = QCoreApplication::translate("AppFontManager", "The font file '%1' does not have read permissions.").arg(fontFile); return -1; } const QString fullPath = inf.absoluteFilePath(); // Check if already loaded const FileNameFontIdPairs::const_iterator cend = m_fonts.constEnd(); for (FileNameFontIdPairs::const_iterator it = m_fonts.constBegin(); it != cend; ++it) { if (it->first == fullPath) { *errorMessage = QCoreApplication::translate("AppFontManager", "The font file '%1' is already loaded.").arg(fontFile); return -1; } } const int id = QFontDatabase::addApplicationFont(fullPath); if (id == -1) { *errorMessage = QCoreApplication::translate("AppFontManager", "The font file '%1' could not be loaded.").arg(fontFile); return -1; } if (debugAppFontWidget) qDebug() << "AppFontManager::add" << fontFile << id; m_fonts.push_back(FileNameFontIdPair(fullPath, id)); return id; } bool AppFontManager::remove(int id, QString *errorMessage) { const int count = m_fonts.size(); for (int i = 0; i < count; i++) if (m_fonts[i].second == id) return removeAt(i, errorMessage); *errorMessage = QCoreApplication::translate("AppFontManager", "'%1' is not a valid font id.").arg(id); return false; } bool AppFontManager::remove(const QString &fontFile, QString *errorMessage) { const int count = m_fonts.size(); for (int i = 0; i < count; i++) if (m_fonts[i].first == fontFile) return removeAt(i, errorMessage); *errorMessage = QCoreApplication::translate("AppFontManager", "There is no loaded font matching the id '%1'.").arg(fontFile); return false; } bool AppFontManager::removeAt(int index, QString *errorMessage) { Q_ASSERT(index >= 0 && index < m_fonts.size()); const QString fontFile = m_fonts[index].first; const int id = m_fonts[index].second; if (debugAppFontWidget) qDebug() << "AppFontManager::removeAt" << index << '(' << fontFile << id << ')'; if (!QFontDatabase::removeApplicationFont(id)) { *errorMessage = QCoreApplication::translate("AppFontManager", "The font '%1' (%2) could not be unloaded.").arg(fontFile).arg(id); return false; } m_fonts.removeAt(index); return true; } const AppFontManager::FileNameFontIdPairs &AppFontManager::fonts() const { return m_fonts; } // ------------- AppFontModel class AppFontModel : public QStandardItemModel { Q_DISABLE_COPY(AppFontModel) public: AppFontModel(QObject *parent = 0); void init(const AppFontManager &mgr); void add(const QString &fontFile, int id); int idAt(const QModelIndex &idx) const; }; AppFontModel::AppFontModel(QObject * parent) : QStandardItemModel(parent) { setHorizontalHeaderLabels(QStringList(AppFontWidget::tr("Fonts"))); } void AppFontModel::init(const AppFontManager &mgr) { typedef AppFontManager::FileNameFontIdPairs FileNameFontIdPairs; const FileNameFontIdPairs &fonts = mgr.fonts(); const FileNameFontIdPairs::const_iterator cend = fonts.constEnd(); for (FileNameFontIdPairs::const_iterator it = fonts.constBegin(); it != cend; ++it) add(it->first, it->second); } void AppFontModel::add(const QString &fontFile, int id) { const QFileInfo inf(fontFile); // Root item with base name QStandardItem *fileItem = new QStandardItem(inf.completeBaseName()); const QString fullPath = inf.absoluteFilePath(); fileItem->setData(fullPath, FileNameRole); fileItem->setToolTip(fullPath); fileItem->setData(id, IdRole); fileItem->setFlags(Qt::ItemIsSelectable|Qt::ItemIsEnabled); appendRow(fileItem); const QStringList families = QFontDatabase::applicationFontFamilies(id); const QStringList::const_iterator cend = families.constEnd(); for (QStringList::const_iterator it = families.constBegin(); it != cend; ++it) { QStandardItem *familyItem = new QStandardItem(*it); familyItem->setToolTip(fullPath); familyItem->setFont(QFont(*it)); familyItem->setFlags(Qt::ItemIsEnabled); fileItem->appendRow(familyItem); } } int AppFontModel::idAt(const QModelIndex &idx) const { if (const QStandardItem *item = itemFromIndex(idx)) return item->data(IdRole).toInt(); return -1; } // ------------- AppFontWidget AppFontWidget::AppFontWidget(QWidget *parent) : QGroupBox(parent), m_view(new QTreeView), m_addButton(new QToolButton), m_removeButton(new QToolButton), m_removeAllButton(new QToolButton), m_model(new AppFontModel(this)) { m_model->init(AppFontManager::instance()); m_view->setModel(m_model); m_view->setSelectionMode(QAbstractItemView::ExtendedSelection); m_view->expandAll(); connect(m_view->selectionModel(), SIGNAL(selectionChanged(QItemSelection,QItemSelection)), this, SLOT(selectionChanged(QItemSelection,QItemSelection))); m_addButton->setToolTip(tr("Add font files")); m_addButton->setIcon(qdesigner_internal::createIconSet(QString::fromUtf8("plus.png"))); connect(m_addButton, SIGNAL(clicked()), this, SLOT(addFiles())); m_removeButton->setEnabled(false); m_removeButton->setToolTip(tr("Remove current font file")); m_removeButton->setIcon(qdesigner_internal::createIconSet(QString::fromUtf8("minus.png"))); connect(m_removeButton, SIGNAL(clicked()), this, SLOT(slotRemoveFiles())); m_removeAllButton->setToolTip(tr("Remove all font files")); m_removeAllButton->setIcon(qdesigner_internal::createIconSet(QString::fromUtf8("editdelete.png"))); connect(m_removeAllButton, SIGNAL(clicked()), this, SLOT(slotRemoveAll())); QHBoxLayout *hLayout = new QHBoxLayout; hLayout->addWidget(m_addButton); hLayout->addWidget(m_removeButton); hLayout->addWidget(m_removeAllButton); hLayout->addItem(new QSpacerItem(0, 0,QSizePolicy::MinimumExpanding)); QVBoxLayout *vLayout = new QVBoxLayout; vLayout->addWidget(m_view); vLayout->addLayout(hLayout); setLayout(vLayout); } void AppFontWidget::addFiles() { const QStringList files = QFileDialog::getOpenFileNames(this, tr("Add Font Files"), QString(), tr("Font files (*.ttf)")); if (files.empty()) return; QString errorMessage; AppFontManager &fmgr = AppFontManager::instance(); const QStringList::const_iterator cend = files.constEnd(); for (QStringList::const_iterator it = files.constBegin(); it != cend; ++it) { const int id = fmgr.add(*it, &errorMessage); if (id != -1) { m_model->add(*it, id); } else { QMessageBox::critical(this, tr("Error Adding Fonts"), errorMessage); } } m_view->expandAll(); } static void removeFonts(const QModelIndexList &selectedIndexes, AppFontModel *model, QWidget *dialogParent) { if (selectedIndexes.empty()) return; // Reverse sort top level rows and remove AppFontManager &fmgr = AppFontManager::instance(); QVector rows; rows.reserve(selectedIndexes.size()); QString errorMessage; const QModelIndexList::const_iterator cend = selectedIndexes.constEnd(); for (QModelIndexList::const_iterator it = selectedIndexes.constBegin(); it != cend; ++it) { const int id = model->idAt(*it); if (id != -1) { if (fmgr.remove(id, &errorMessage)) { rows.push_back(it->row()); } else { QMessageBox::critical(dialogParent, AppFontWidget::tr("Error Removing Fonts"), errorMessage); } } } qStableSort(rows.begin(), rows.end()); for (int i = rows.size() - 1; i >= 0; i--) model->removeRow(rows[i]); } void AppFontWidget::slotRemoveFiles() { removeFonts(m_view->selectionModel()->selectedIndexes(), m_model, this); } void AppFontWidget::slotRemoveAll() { const int count = m_model->rowCount(); if (!count) return; const QMessageBox::StandardButton answer = QMessageBox::question(this, tr("Remove Fonts"), tr("Would you like to remove all fonts?"), QMessageBox::Yes|QMessageBox::No, QMessageBox::No); if (answer == QMessageBox::No) return; QModelIndexList topLevels; for (int i = 0; i < count; i++) topLevels.push_back(m_model->index(i, 0)); removeFonts(topLevels, m_model, this); } void AppFontWidget::selectionChanged(const QItemSelection &selected, const QItemSelection & /*deselected*/) { m_removeButton->setEnabled(!selected.indexes().empty()); } void AppFontWidget::save(QDesignerSettingsInterface *s, const QString &prefix) { AppFontManager::instance().save(s, prefix); } void AppFontWidget::restore(const QDesignerSettingsInterface *s, const QString &prefix) { AppFontManager::instance().restore(s, prefix); } // ------------ AppFontDialog AppFontDialog::AppFontDialog(QWidget *parent) : QDialog(parent), m_appFontWidget(new AppFontWidget) { setAttribute(Qt::WA_DeleteOnClose, true); setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint); setWindowTitle(tr("Additional Fonts")); setModal(false); QVBoxLayout *vl = new QVBoxLayout; vl->addWidget(m_appFontWidget); QDialogButtonBox *bb = new QDialogButtonBox(QDialogButtonBox::Close); QDialog::connect(bb, SIGNAL(rejected()), this, SLOT(reject())); vl->addWidget(bb); setLayout(vl); } QT_END_NAMESPACE