1

我的问题如下:

有aQTableView和aQStandardItemModel这样使用:

ui->tableView->setModel(model);
model->setItem(myrow, mycolumn, myQStandardItem);

和一个组合框委托:

ComboBoxDelegate* mydelegate = new ComboBoxDelegate();
ui->tableView->setItemDelegateForColumn(mycolumn,mydelegate);

每次更改表格单元格的值(通过组合框)时,我都需要捕获新值和刚刚修改的单元格的索引。我dataChaged以这种方式使用与模型关联的信号:

connect(model,SIGNAL(dataChanged(QModelIndex&,QModelIndex&)),this,SLOT(GetChangedValue(QModelIndex&)));

但它不起作用,GetChangedValue尽管组合框已更改其值,但它从不调用该方法。我跳过任何步骤吗?

下面是代码ComboBoxDelegate

class ComboBoxDelegate : public QStyledItemDelegate
{
    Q_OBJECT
public:

    ComboBoxDelegate(QVector<QString>& ItemsToCopy,QObject *parent = 0);
    ~ComboBoxDelegate();
     void setItemData(QVector<QString>& ItemsToCopy);

    QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const ;
    void setEditorData(QWidget *editor, const QModelIndex &index) const;
    void setModelData(QWidget *editor, QAbstractItemModel *model, const  QModelIndex &index) const;
    void updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const;



    private:
     QVector<QString> Items;

};

ComboBoxDelegate::ComboBoxDelegate(QVector<QString>&  ItemsToCopy,QObject  *parent)
:QStyledItemDelegate(parent)
{
    setItemData(ItemsToCopy);
}


ComboBoxDelegate::~ComboBoxDelegate()
{
}


QWidget *ComboBoxDelegate::createEditor(QWidget *parent, const     QStyleOptionViewItem &option, const QModelIndex &index) const
{

    QComboBox* editor = new QComboBox(parent);
    editor->setEditable(true);



    for (int i = 0; i < Items.size(); ++i)
    {
        editor->addItem(Items[i]);
    }


    editor->setStyleSheet("combobox-popup: 0;");

    return editor;
}




void ComboBoxDelegate::setEditorData(QWidget *editor, const QModelIndex  &index) const
{
    QComboBox *comboBox = static_cast<QComboBox*>(editor);
    QString currentText = index.data(Qt::EditRole).toString();
    int cbIndex = comboBox->findText(currentText);
    comboBox->setCurrentIndex(cbIndex);
}



void ComboBoxDelegate::setModelData(QWidget *editor, QAbstractItemModel    *model, const QModelIndex &index) const
 {
    QComboBox *comboBox = static_cast<QComboBox*>(editor);
    model->setData(index, comboBox->currentText(), Qt::EditRole);

}



void ComboBoxDelegate::updateEditorGeometry(QWidget *editor, const   QStyleOptionViewItem &option, const QModelIndex &/* index */) const
{
    editor->setGeometry(option.rect);

}



void ComboBoxDelegate::setItemData(QVector<QString>&  ItemsToCopy)
{
    for (int row = 0; row < ItemsToCopy.size(); ++row)
    {

       Items.push_back(ItemsToCopy[row]);


   }

}

4

1 回答 1

3

您的委托实现的问题是当组合索引更改时您不会发出commitData信号。它在 Qt 文档中说明:

当编辑器小部件完成数据编辑并想要将其写回模型时,必须发出此信号。

您可以将组合框作为委托类的成员,并将currentIndexChanged组合框的信号连接到某个发出的插槽commitData

#include <QItemDelegate>

#include <QComboBox>

class ComboBoxDelegate: public QItemDelegate
{
 Q_OBJECT
public:
    ComboBoxDelegate(QObject *parent = 0);

    QWidget *createEditor( QWidget *parent,
                            const QStyleOptionViewItem &option,
                            const QModelIndex &index ) const;

    void setEditorData( QWidget *editor,
                            const QModelIndex &index ) const;

    void setModelData( QWidget *editor,
                            QAbstractItemModel *model,
                            const QModelIndex &index ) const;

    void updateEditorGeometry( QWidget *editor,
                            const QStyleOptionViewItem &option,
                            const QModelIndex &index ) const;

    QStringList comboItems;

    mutable QComboBox *combo;

private slots:

    void setData(int val);

};

ComboBoxDelegate::ComboBoxDelegate(QObject *parent ):QItemDelegate(parent)
{
        comboItems<<"Item 1"<<"Item 2"<<"Item 3";
}

QWidget *ComboBoxDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
    combo = new QComboBox( parent );
    QObject::connect(combo,SIGNAL(currentIndexChanged(int)),this,SLOT(setData(int)));
    combo->addItems(comboItems);
    combo->setMaxVisibleItems(comboItems.count());
    return combo;
}

void ComboBoxDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
{
    QString text = index.model()->data( index, Qt::DisplayRole ).toString();

    int comboIndex = comboItems.indexOf(QRegExp(text));

    if(comboIndex>=0)
        (static_cast<QComboBox*>( editor ))->setCurrentIndex(comboIndex);
}

void ComboBoxDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
{
    model->setData( index, static_cast<QComboBox*>( editor )->currentText() );
}


void ComboBoxDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
    editor->setGeometry( option.rect );
}

void ComboBoxDelegate::setData(int val)
{
    emit commitData(combo);
    //emit closeEditor(combo);
}

如您所见currentIndexChanged,组合框的信号连接到setData将数据提交给模型的插槽。此外,您应该将组合框声明为可变的,以便将createEditor其更新为常量。const如果一个数据成员被声明为可变的,那么从一个成员函数给这个数据成员赋值是合法的。

现在,dataChanged当组合框的索引更改时,将发出信号。

于 2015-05-14T04:25:08.920 回答