You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
234 lines
7.2 KiB
234 lines
7.2 KiB
/*
|
|
* Copyright (C) 2014 - 2015 Hong Jen Yee (PCMan) <pcman.tw@gmail.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
#include "dirtreemodel.h"
|
|
#include "dirtreemodelitem.h"
|
|
#include <QDebug>
|
|
#include "core/fileinfojob.h"
|
|
|
|
namespace Fm {
|
|
|
|
DirTreeModel::DirTreeModel(QObject* parent):
|
|
QAbstractItemModel(parent),
|
|
showHidden_(false) {
|
|
}
|
|
|
|
DirTreeModel::~DirTreeModel() {
|
|
}
|
|
|
|
void DirTreeModel::addRoots(Fm::FilePathList rootPaths) {
|
|
auto job = new Fm::FileInfoJob{std::move(rootPaths)};
|
|
job->setAutoDelete(true);
|
|
connect(job, &Fm::FileInfoJob::finished, this, &DirTreeModel::onFileInfoJobFinished, Qt::BlockingQueuedConnection);
|
|
job->runAsync();
|
|
}
|
|
|
|
void DirTreeModel::onFileInfoJobFinished() {
|
|
auto job = static_cast<Fm::FileInfoJob*>(sender());
|
|
for(auto file: job->files()) {
|
|
addRoot(std::move(file));
|
|
}
|
|
}
|
|
|
|
// QAbstractItemModel implementation
|
|
|
|
Qt::ItemFlags DirTreeModel::flags(const QModelIndex& index) const {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
if(item && item->isPlaceHolder()) {
|
|
return Qt::ItemIsEnabled;
|
|
}
|
|
return QAbstractItemModel::flags(index);
|
|
}
|
|
|
|
QVariant DirTreeModel::data(const QModelIndex& index, int role) const {
|
|
if(!index.isValid() || index.column() > 1) {
|
|
return QVariant();
|
|
}
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
if(item) {
|
|
auto info = item->fileInfo_;
|
|
switch(role) {
|
|
case Qt::ToolTipRole:
|
|
return QVariant(item->displayName_);
|
|
case Qt::DisplayRole:
|
|
return QVariant(item->displayName_);
|
|
case Qt::DecorationRole:
|
|
return QVariant(item->icon_);
|
|
case FileInfoRole: {
|
|
QVariant v;
|
|
v.setValue(info);
|
|
return v;
|
|
}
|
|
}
|
|
}
|
|
return QVariant();
|
|
}
|
|
|
|
int DirTreeModel::columnCount(const QModelIndex& /*parent*/) const {
|
|
return 1;
|
|
}
|
|
|
|
int DirTreeModel::rowCount(const QModelIndex& parent) const {
|
|
if(!parent.isValid()) {
|
|
return rootItems_.size();
|
|
}
|
|
DirTreeModelItem* item = itemFromIndex(parent);
|
|
if(item) {
|
|
return item->children_.size();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
QModelIndex DirTreeModel::parent(const QModelIndex& child) const {
|
|
DirTreeModelItem* item = itemFromIndex(child);
|
|
if(item && item->parent_) {
|
|
item = item->parent_; // go to parent item
|
|
if(item) {
|
|
const auto& items = item->parent_ ? item->parent_->children_ : rootItems_;
|
|
auto it = std::find(items.cbegin(), items.cend(), item);
|
|
if(it != items.cend()) {
|
|
int row = it - items.cbegin();
|
|
return createIndex(row, 0, (void*)item);
|
|
}
|
|
}
|
|
}
|
|
return QModelIndex();
|
|
}
|
|
|
|
QModelIndex DirTreeModel::index(int row, int column, const QModelIndex& parent) const {
|
|
if(row >= 0 && column >= 0 && column == 0) {
|
|
if(!parent.isValid()) { // root items
|
|
if(static_cast<size_t>(row) < rootItems_.size()) {
|
|
const DirTreeModelItem* item = rootItems_.at(row);
|
|
return createIndex(row, column, (void*)item);
|
|
}
|
|
}
|
|
else { // child items
|
|
DirTreeModelItem* parentItem = itemFromIndex(parent);
|
|
if(static_cast<size_t>(row) < parentItem->children_.size()) {
|
|
const DirTreeModelItem* item = parentItem->children_.at(row);
|
|
return createIndex(row, column, (void*)item);
|
|
}
|
|
}
|
|
}
|
|
return QModelIndex(); // invalid index
|
|
}
|
|
|
|
bool DirTreeModel::hasChildren(const QModelIndex& parent) const {
|
|
DirTreeModelItem* item = itemFromIndex(parent);
|
|
return item ? !item->isPlaceHolder() : true;
|
|
}
|
|
|
|
QModelIndex DirTreeModel::indexFromItem(DirTreeModelItem* item) const {
|
|
Q_ASSERT(item);
|
|
const auto& items = item->parent_ ? item->parent_->children_ : rootItems_;
|
|
auto it = std::find(items.cbegin(), items.cend(), item);
|
|
if(it != items.cend()) {
|
|
int row = it - items.cbegin();
|
|
return createIndex(row, 0, (void*)item);
|
|
}
|
|
return QModelIndex();
|
|
}
|
|
|
|
// public APIs
|
|
QModelIndex DirTreeModel::addRoot(std::shared_ptr<const Fm::FileInfo> root) {
|
|
DirTreeModelItem* item = new DirTreeModelItem(std::move(root), this);
|
|
int row = rootItems_.size();
|
|
beginInsertRows(QModelIndex(), row, row);
|
|
rootItems_.push_back(item);
|
|
// add_place_holder_child_item(model, item_l, nullptr, FALSE);
|
|
endInsertRows();
|
|
return createIndex(row, 0, (void*)item);
|
|
}
|
|
|
|
DirTreeModelItem* DirTreeModel::itemFromIndex(const QModelIndex& index) const {
|
|
return reinterpret_cast<DirTreeModelItem*>(index.internalPointer());
|
|
}
|
|
|
|
QModelIndex DirTreeModel::indexFromPath(const Fm::FilePath &path) const {
|
|
DirTreeModelItem* item = itemFromPath(path);
|
|
return item ? item->index() : QModelIndex();
|
|
}
|
|
|
|
DirTreeModelItem* DirTreeModel::itemFromPath(const Fm::FilePath &path) const {
|
|
Q_FOREACH(DirTreeModelItem* item, rootItems_) {
|
|
if(item->fileInfo_ && path == item->fileInfo_->path()) {
|
|
return item;
|
|
}
|
|
else {
|
|
DirTreeModelItem* child = item->childFromPath(path, true);
|
|
if(child) {
|
|
return child;
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
|
|
void DirTreeModel::loadRow(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
Q_ASSERT(item);
|
|
if(item && !item->isPlaceHolder()) {
|
|
item->loadFolder();
|
|
}
|
|
}
|
|
|
|
void DirTreeModel::unloadRow(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
if(item && !item->isPlaceHolder()) {
|
|
item->unloadFolder();
|
|
}
|
|
}
|
|
|
|
bool DirTreeModel::isLoaded(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
return item ? item->loaded_ : false;
|
|
}
|
|
|
|
QIcon DirTreeModel::icon(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
return item ? item->icon_ : QIcon();
|
|
}
|
|
|
|
std::shared_ptr<const Fm::FileInfo> DirTreeModel::fileInfo(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
return item ? item->fileInfo_ : nullptr;
|
|
}
|
|
|
|
Fm::FilePath DirTreeModel::filePath(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
return (item && item->fileInfo_) ? item->fileInfo_->path() : Fm::FilePath{};
|
|
}
|
|
|
|
QString DirTreeModel::dispName(const QModelIndex& index) {
|
|
DirTreeModelItem* item = itemFromIndex(index);
|
|
return item ? item->displayName_ : QString();
|
|
}
|
|
|
|
void DirTreeModel::setShowHidden(bool show_hidden) {
|
|
showHidden_ = show_hidden;
|
|
Q_FOREACH(DirTreeModelItem* item, rootItems_) {
|
|
item->setShowHidden(show_hidden);
|
|
}
|
|
}
|
|
|
|
|
|
} // namespace Fm
|