dataChanged 信号不适用于 ComboBoxDelegate

Posted

技术标签:

【中文标题】dataChanged 信号不适用于 ComboBoxDelegate【英文标题】:dataChanged signal does not work with ComboBoxDelegate 【发布时间】:2015-05-13 13:16:42 【问题描述】:

我的问题如下:

有一个QTableView和一个QStandardItemModel这样使用:

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]);


   

【问题讨论】:

您在表格视图中看到编辑器设置的新值了吗? connect 函数是否返回 true 【参考方案1】:

您的委托实现的问题是,当组合索引更改时,您没有发出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 信号会在组合框的索引发生变化时发出。

【讨论】:

太棒了!感谢您的支持,它现在确实有效。

以上是关于dataChanged 信号不适用于 ComboBoxDelegate的主要内容,如果未能解决你的问题,请参考以下文章

如何处理圆图的Datachange信号?

无法在 QAbstractItemModel 的 dataChanged 信号槽中获取发送者对象

PyQT 列表视图没有响应 datachanged 信号

QTableView 似乎对 dataChanged 信号没有反应

MapItemView 在 dataChanged 信号后未更新

如果由同一应用程序触发,则丢弃 QClipboard::dataChanged() 信号