/**************************************************************************** ** ** Copyright (C) 2017 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the examples of Qt for Device Creation. ** ** $QT_BEGIN_LICENSE:BSD$ ** 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. ** ** BSD License Usage ** Alternatively, you may use this file under the terms of the BSD license ** as follows: ** ** "Redistribution and use in source and binary forms, with or without ** modification, are permitted provided that the following conditions are ** met: ** * Redistributions of source code must retain the above copyright ** notice, this list of conditions and the following disclaimer. ** * Redistributions in binary form must reproduce the above copyright ** notice, this list of conditions and the following disclaimer in ** the documentation and/or other materials provided with the ** distribution. ** * Neither the name of The Qt Company Ltd nor the names of its ** contributors may be used to endorse or promote products derived ** from this software without specific prior written permission. ** ** ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT ** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR ** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT ** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, ** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT ** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE ** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "applistmodel.h" #include "appparser.h" #include "applog.h" #include #include static QHash modelRoles() { QHash roles; roles[AppListModel::App] = "appEntry"; roles[AppListModel::IconName] = "iconName"; roles[AppListModel::ApplicationName] = "applicationName"; roles[AppListModel::ApplicationId] = "applicationId"; roles[AppListModel::ExeuctableName] = "executableName"; roles[AppListModel::ExecutablePath] = "executablePath"; roles[AppListModel::SourceFileName] = "sourceFileName"; return roles; } QHash AppListModel::m_roles = modelRoles(); int AppListModel::rowCount(const QModelIndex& index) const { if (index.isValid()) return 0; return m_rows.count(); } QVariant AppListModel::data(const QModelIndex& index, int role) const { if (!index.isValid() || index.parent().isValid()) return QVariant(); auto entry = m_rows[index.row()]; switch (role) { case App: return QVariant::fromValue(entry); case IconName: return entry.iconName; case ApplicationName: return entry.appName; case ApplicationId: return entry.appId; case ExeuctableName: return entry.executableName; case ExecutablePath: return entry.executablePath; case SourceFileName: return entry.sourceFileName; default: qCWarning(apps) << "Unhandled role" << role; return QVariant(); } } QHash AppListModel::roleNames() const { return m_roles; } /*! * Returns true if the directory can be watched * * Parse all JSON application files from the given directory * and monitor it for changes. */ bool AppListModel::addAndWatchDir(const QString& dirName) { auto watcher = new QFileSystemWatcher(this); connect(watcher, &QFileSystemWatcher::directoryChanged, this, &AppListModel::addDir); auto res = watcher->addPath(dirName); addDir(dirName); qCDebug(apps) << "addAndWatchDir" << dirName << "result: " << res; return res; } void AppListModel::addFile(const QString& fileName) { beginResetModel(); doAddFile(fileName); endResetModel(); } void AppListModel::addDir(const QString& dirName) { QDirIterator dirIt(dirName, QDir::Files | QDir::NoDotAndDotDot | QDir::Readable); // We want to deal with the corner case of some files being // removed. One option would have been to keep a marker bit // inside the AppEntry but that might leak too much of the // implementation to the outside. Let's keep a list of file // names that came from the directory and remove items from // the list as we re-parse them. Once we are done emit the // signals. QVector deletionCandidates = entriesWithPrefix(dirName); beginResetModel(); while (dirIt.hasNext()) { auto fileName = dirIt.next(); if (!doAddFile(fileName)) continue; deletionCandidates.removeAll(fileName); } QVector removedApps = removeEntries(deletionCandidates); endResetModel(); // Announce the apps we removed after the model has been updated for (const auto& app : removedApps) { qCDebug(apps) << "Going to remove entry for " << app.sourceFileName; emit appRemoved(app); } } bool AppListModel::doAddFile(const QString& fileName) { bool ok; auto newEntry = AppParser::parseFile(fileName, &ok); if (!ok) return false; for (int i = 0; i < m_rows.count(); ++i) { if (m_rows[i].sourceFileName == fileName) { m_rows[i] = newEntry; return true; } } m_rows.push_back(newEntry); return true; } QVector AppListModel::entriesWithPrefix(const QString& prefix) const { QVector entries; for (const AppEntry& entry : m_rows) if (entry.sourceFileName.startsWith(prefix)) entries.push_back(entry.sourceFileName); return entries; } QVector AppListModel::removeEntries(const QVector& fileNames) { QVector removedEntries(fileNames.size()); // Rare but quadratic. The actual removal. for (const auto &toRemoveFile: fileNames) { for (int i = 0; i < m_rows.size(); ++i) { if (m_rows[i].sourceFileName != toRemoveFile) continue; removedEntries.append(m_rows[i]); m_rows.removeAt(i); break; } } return removedEntries; } QVariant AppListModel::findApplicationId(const QString& appId) const { for (const auto& entry : m_rows) { if (entry.appId == appId) return QVariant::fromValue(entry); } return QVariant(); }