一、QListWidget、QTableWidget和QTreeWidget

QTableWidget的item默认是可以编辑,其他两个的item默认是不可编辑。如果要将QTableWidget设置成不可编辑,可以使用

tableWidget->setEditTriggers(QAbstractItemView::NoEditTriggers);

则tablewidget的所有item都是不可编辑的;

QListWidget和QTreeWidget默认是不可编辑的,如果想要将item变成可以编辑的,可以简单地将item设置成可以编辑的,如下所示

item = new QTreeWidgetItem(parent);
        item->setFlags(Qt::ItemIsEditable|item->flags());

则整个的该item是可以编辑的。如果只是想某一列可以编辑,其中一种方法是使用信号和槽。双击可编辑的方式;

treeWidget = new QTreeWidget;
    treeWidget->setColumnCount(2);
        。
        。
        。
    connect(treeWidget, SIGNAL(itemDoubleClicked(QTreeWidgetItem*,int)), this, SLOT(editFileFilter(QTreeWidgetItem*,int)));


// slot
void SettingsViewer::editFileFilter(QTreeWidgetItem *item, int column)
{
    if (column == 0)
        item->setFlags(Qt::ItemIsEditable | item->flags());
    else
        item->setFlags(item->flags() & ~(Qt::ItemIsEditable) );
}

如上所示自己编写一个slot,当双击的时候触发设置标记

 

二、QStringListModel、QDirModel

QStringListModel 

通过QModelIndex index = model->index(row);指定的第几行获得对应的QModelIndex;

要删除行,可以通过

model->removeRows(listView->currentIndex().row(), 1);
    model->removeRow(listView->currentIndex().row());

第一个要在指定的行号后面增加对应删除多少行;下面则一次只能删除一行。ps:上面两行代码都是删除一行;

可以将model装进listview里面

model = new QStringListModel(this);
    model->setStringList(leaders);

    listView = new QListView;
    listView->setModel(model);
    listView->setEditTriggers(QAbstractItemView::AnyKeyPressed
                              |QAbstractItemView::DoubleClicked);

第二行将QStringList leaders的数据装进model;然后将model给listview来显示(如第四行代码所示)

 

 QDirModel

model = new QDirModel;
    model->setReadOnly(false);
    model->setSorting(QDir::DirsFirst|QDir::IgnoreCase|QDir::Name);

    treeView = new QTreeView;
    treeView->setModel(model);

QDirModel装载的是文件目录;可以对文件和目录进行操作;

QModelIndex index(const QString &path, int column = 0) const;
QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const Q_DECL_OVERRIDE;

文件查找有两种方式来获取QModelIndex; 如下所示

QModelIndex index = model->index(QDir::currentPath());

文件删除

if (model->fileInfo(index).isDir())
        ok = model->rmdir(index);
    else
        ok = model->remove(index);

如果是目录则需要用rmdir的方式进行删除;否则通过remove进行删除文件,如上所示

创建文件夹

if (!model->mkdir(index, dirName).isValid())

 

QSortFilterProxyModel

QRegExp;指定column还有sort,increase或者decrease等;其中sort也可以自定义排序;

virtual bool lessThan(const QModelIndex &source_left, const QModelIndex &source_right) const;

通过重写此代码进行重新自定义排序;

virtual bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const;
    virtual bool filterAcceptsColumn(int source_column, const QModelIndex &source_parent) const;

可以通过重写这两个函数进行过滤指定的source_row是否需要显示,指定的source_column是否需要显示;

bool HideColumnProxy::filterAcceptsColumn(int source_column, const QModelIndex &source_parent) const
{
    if(source_column >= 3)
        return false;
    return true;
}

如上,则只显示从0~2的column号

 

三、QAbstractTableModel

当子类化QAbstractTableModel时,必须实现rowCount(),columnCount,和data()函数。因为在QAbstractItemModel中,该三个函数为纯虚函数。QAbstractItemModel另外的两个纯虚函数在QAbstractTableModel中已经被实现了,无需一定要再重写。

headerData函数,例如:

QVariant CurrencyModel::headerData(int section, Qt::Orientation orientation, int role) const
{
    if (role != Qt::DisplayRole)
        return QVariant();
    return currencyAt(section);
}

当需要更新列表的数据的时候,在qt4使用reset()函数,但是在Qt5中,却没有该函数;通过查看QAbstractItemModel的头文件可以看到:

#if QT_DEPRECATED_SINCE(5,0)
    QT_DEPRECATED void reset()
    {
        beginResetModel();
        endResetModel();
    }
#endif

    void beginResetModel();
    void endResetModel();

则要更新的话,需要自己用begin和end函数替代reset函数。

当需要更新数据的时候,可以重写该函数

Q_INVOKABLE virtual bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::EditRole);

如下例子

bool CityModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
    if (index.isValid() && index.row() != index.column()
            && role == Qt::EditRole) {
        int offset = offsetOf(index.row(), index.column());
        distances[offset] = value.toInt();

        QModelIndex transposedIndex = createIndex(index.column(),index.row());

        emit dataChanged(index, index);
        emit dataChanged(transposedIndex, transposedIndex);

        return true;
    }

    return false;
}

首先index必须有效,其他两个判断则是根据实际情况进行判断的。

然后createIndex则是因为当index改变的时候,另一个index也需要改变,所以这里create了一个index;因为当在指定的范围内发生改变的时候需要emit dataChanged;因为index跟transposedIndex都发生了改变,所以都需要emit;

This signal is emitted whenever the data in an existing item changes.

返回的true或者false表示设置数据成功与否。

 

QTableWidgetItem qtablewidgetitem设置可编辑_sed

 

 

 

 

 

 

 

问题及解决:

在重写QDialog的done函数的时候,忘记在函数的最后调用父类的done函数,导致窗口无法关闭;只能通过杀死进程的方式;QDialog::done(result);