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的主要内容,如果未能解决你的问题,请参考以下文章
无法在 QAbstractItemModel 的 dataChanged 信号槽中获取发送者对象
QTableView 似乎对 dataChanged 信号没有反应