我试图让QTreeView像QFileSystemModel一样运行,QFileSystemModel是一个文件夹层次结构,您可以在其中浏览该特定节点的子元素。唯一的问题是我希望它基于字符串数据,例如使用字符串列表" my / path / file.ext"。我尝试过使用QStringListModel,但它只是将它们全部放在同一级别上。
答案 0 :(得分:1)
QStringListModel
提供字符串列表,而不是字符串树。它不会以任何方式解释字符串。
您可以重复使用QStandardItemModel
。 QStandardItem
已经是树节点。可以使用下面的类将您的路径插入QStandardItemModel
。
class TreeManipulator {
Q_DISABLE_COPY(TreeManipulator)
static QChar const m_sep;
QMap<QString, QStandardItem*> m_items;
QStandardItem * m_root;
QString normalizedPath(const QString & path) {
QStringList p(path.split(m_sep, QString::SkipEmptyParts));
return p.join(sep);
}
QStandardItem * find(QStandardItem * item, const QString & child) {
for (int i = 0; i < item->rowCount(); ++i)
if (item->child(i)->text() == child) return item->child(i);
return nullptr;
}
public:
explicit TreeManipulator(QStandardItem * root) : m_root(root) {}
explicit TreeManipulator(QStandardItemModel * model) :
m_root(model->invisibleRootItem()) {}
QStandardItem * find(QString path) const {
path = normalizedPath(path);
if (m_items.contains(path)) return m_items[path];
return nullptr;
}
QStandardItem * addItem(QString path) {
QStandardItem * item = m_root;
QStringList p(path.split(m_sep, QString::SkipEmptyParts));
path = normalizedPath(path);
if (m_items.contains(path)) return m_items[path];
while (!p.isEmpty()) {
QString elt = p.takeFirst();
QStandardItem * child = find(item, elt);
if (!child)
item.appendRow((child = new QStandardItem(elt)));
item = child;
}
m_items.insert(path, item);
return item;
}
};
QChar const TreeManipulator::m_sep(QLatin1Char('/'));