File indexing completed on 2024-12-01 03:40:51

0001 /*
0002     This file is part of the KDE libraries
0003     SPDX-FileCopyrightText: Stephan Kulow <coolo@kde.org>
0004 
0005     SPDX-License-Identifier: LGPL-2.0-or-later
0006 */
0007 
0008 #include "kfilefiltercombo.h"
0009 #include "kfilefilter.h"
0010 #include "kfilefiltercombo_debug.h"
0011 
0012 #include <KLocalizedString>
0013 #include <QDebug>
0014 #include <QEvent>
0015 #include <QLineEdit>
0016 #include <QMimeDatabase>
0017 
0018 #include <config-kiofilewidgets.h>
0019 
0020 #include <algorithm>
0021 #include <utility>
0022 
0023 class KFileFilterComboPrivate
0024 {
0025 public:
0026     explicit KFileFilterComboPrivate(KFileFilterCombo *qq)
0027         : q(qq)
0028     {
0029     }
0030 
0031     void slotFilterChanged();
0032 
0033     KFileFilterCombo *const q;
0034     // when we have more than 3 mimefilters and no default-filter,
0035     // we don't show the comments of all mimefilters in one line,
0036     // instead we show "All supported files". We have to translate
0037     // that back to the list of mimefilters in currentFilter() tho.
0038     bool m_hasAllSupportedFiles = false;
0039     // true when setMimeFilter was called
0040     bool m_isMimeFilter = false;
0041     QString m_lastFilter;
0042     KFileFilter m_defaultFilter = KFileFilter::fromFilterString(i18nc("Default mime type filter that shows all file types", "*|All Files")).first();
0043 
0044     QList<KFileFilter> m_filters;
0045     bool m_allTypes;
0046 };
0047 
0048 KFileFilterCombo::KFileFilterCombo(QWidget *parent)
0049     : KComboBox(true, parent)
0050     , d(new KFileFilterComboPrivate(this))
0051 {
0052     setTrapReturnKey(true);
0053     setInsertPolicy(QComboBox::NoInsert);
0054     connect(this, &QComboBox::activated, this, &KFileFilterCombo::filterChanged);
0055     connect(this, &KComboBox::returnPressed, this, &KFileFilterCombo::filterChanged);
0056     connect(this, &KFileFilterCombo::filterChanged, this, [this]() {
0057         d->slotFilterChanged();
0058     });
0059     d->m_allTypes = false;
0060 }
0061 
0062 KFileFilterCombo::~KFileFilterCombo() = default;
0063 
0064 void KFileFilterCombo::setFilters(const QList<KFileFilter> &types, const KFileFilter &defaultFilter)
0065 {
0066     clear();
0067     d->m_filters.clear();
0068     QString delim = QStringLiteral(", ");
0069     d->m_hasAllSupportedFiles = false;
0070     bool hasAllFilesFilter = false;
0071     QMimeDatabase db;
0072 
0073     if (types.isEmpty()) {
0074         d->m_filters = {d->m_defaultFilter};
0075         addItem(d->m_defaultFilter.label());
0076 
0077         d->m_lastFilter = currentText();
0078         return;
0079     }
0080 
0081     d->m_allTypes = defaultFilter.isEmpty() && (types.count() > 1);
0082 
0083     if (!types.isEmpty() && types.first().mimePatterns().isEmpty()) {
0084         d->m_allTypes = false;
0085     }
0086 
0087     // If there's MIME types that have the same comment, we will show the extension
0088     // in addition to the MIME type comment
0089     QHash<QString, int> allTypeComments;
0090     for (const KFileFilter &filter : types) {
0091         allTypeComments[filter.label()] += 1;
0092     }
0093 
0094     for (const KFileFilter &filter : types) {
0095         if (!filter.isValid()) {
0096             continue;
0097         }
0098 
0099         const QStringList mimeTypes = filter.mimePatterns();
0100 
0101         const bool isAllFileFilters = std::any_of(mimeTypes.cbegin(), mimeTypes.cend(), [&db](const QString &mimeTypeName) {
0102             const QMimeType type = db.mimeTypeForName(mimeTypeName);
0103 
0104             if (!type.isValid()) {
0105                 qCWarning(KIO_KFILEWIDGETS_KFILEFILTERCOMBO) << mimeTypeName << "is not a valid MIME type";
0106                 return false;
0107             }
0108 
0109             return type.name().startsWith(QLatin1String("all/")) || type.isDefault();
0110         });
0111 
0112         if (isAllFileFilters) {
0113             hasAllFilesFilter = true;
0114             continue;
0115         }
0116 
0117         if (allTypeComments.value(filter.label()) > 1) {
0118             QStringList mimeSuffixes;
0119 
0120             for (const QString &mimeTypeName : filter.mimePatterns()) {
0121                 const QMimeType type = db.mimeTypeForName(mimeTypeName);
0122                 mimeSuffixes << type.suffixes();
0123             }
0124 
0125             const QString label = i18nc("%1 is the mimetype name, %2 is the extensions", "%1 (%2)", filter.label(), mimeSuffixes.join(QLatin1String(", ")));
0126             KFileFilter newFilter(label, filter.filePatterns(), filter.mimePatterns());
0127 
0128             d->m_filters.append(newFilter);
0129             addItem(newFilter.label());
0130         } else {
0131             d->m_filters.append(filter);
0132             addItem(filter.label());
0133         }
0134 
0135         if (filter == defaultFilter) {
0136             setCurrentIndex(count() - 1);
0137         }
0138     }
0139 
0140     if (count() == 1) {
0141         d->m_allTypes = false;
0142     }
0143 
0144     if (d->m_allTypes) {
0145         QStringList allTypes;
0146         for (const KFileFilter &filter : std::as_const(d->m_filters)) {
0147             allTypes << filter.mimePatterns().join(QLatin1Char(' '));
0148         }
0149 
0150         KFileFilter allSupportedFilesFilter;
0151 
0152         if (count() <= 3) { // show the MIME type comments of at max 3 types
0153             QStringList allComments;
0154             for (const KFileFilter &filter : std::as_const(d->m_filters)) {
0155                 allComments << filter.label();
0156             }
0157 
0158             allSupportedFilesFilter = KFileFilter(allComments.join(delim), {}, allTypes);
0159         } else {
0160             allSupportedFilesFilter = KFileFilter(i18n("All Supported Files"), {}, allTypes);
0161             d->m_hasAllSupportedFiles = true;
0162         }
0163 
0164         insertItem(0, allSupportedFilesFilter.label());
0165         d->m_filters.prepend(allSupportedFilesFilter);
0166         setCurrentIndex(0);
0167     }
0168 
0169     if (hasAllFilesFilter) {
0170         addItem(i18n("All Files"));
0171         d->m_filters.append(KFileFilter(i18n("All Files"), {}, {QStringLiteral("application/octet-stream")}));
0172     }
0173 
0174     d->m_lastFilter = currentText();
0175 }
0176 
0177 KFileFilter KFileFilterCombo::currentFilter() const
0178 {
0179     if (currentText() != itemText(currentIndex())) {
0180         // The user edited the text
0181 
0182         const QList<KFileFilter> filter = KFileFilter::fromFilterString(currentText());
0183 
0184         if (!filter.isEmpty()) {
0185             return filter.first();
0186         } else {
0187             return KFileFilter();
0188         }
0189     } else {
0190         if (currentIndex() == -1) {
0191             return KFileFilter();
0192         }
0193 
0194         return d->m_filters[currentIndex()];
0195     }
0196 }
0197 
0198 bool KFileFilterCombo::showsAllTypes() const
0199 {
0200     return d->m_allTypes;
0201 }
0202 
0203 QList<KFileFilter> KFileFilterCombo::filters() const
0204 {
0205     return d->m_filters;
0206 }
0207 
0208 void KFileFilterCombo::setCurrentFilter(const KFileFilter &filter)
0209 {
0210     auto it = std::find(d->m_filters.cbegin(), d->m_filters.cend(), filter);
0211 
0212     if (it == d->m_filters.cend()) {
0213         qCWarning(KIO_KFILEWIDGETS_KFILEFILTERCOMBO) << "Could not find file filter";
0214         setCurrentIndex(-1);
0215         Q_EMIT filterChanged();
0216         return;
0217     }
0218 
0219     setCurrentIndex(std::distance(d->m_filters.cbegin(), it));
0220     Q_EMIT filterChanged();
0221 }
0222 
0223 void KFileFilterComboPrivate::slotFilterChanged()
0224 {
0225     m_lastFilter = q->currentText();
0226 }
0227 
0228 bool KFileFilterCombo::eventFilter(QObject *o, QEvent *e)
0229 {
0230     if (o == lineEdit() && e->type() == QEvent::FocusOut) {
0231         if (currentText() != d->m_lastFilter) {
0232             Q_EMIT filterChanged();
0233         }
0234     }
0235 
0236     return KComboBox::eventFilter(o, e);
0237 }
0238 
0239 void KFileFilterCombo::setDefaultFilter(const KFileFilter &filter)
0240 {
0241     d->m_defaultFilter = filter;
0242 }
0243 
0244 KFileFilter KFileFilterCombo::defaultFilter() const
0245 {
0246     return d->m_defaultFilter;
0247 }
0248 
0249 #include "moc_kfilefiltercombo.cpp"