Meladi
26th May 2014, 12:53
Hi all, I hope you can help me with this problem.
I have a QTableView in which the first row contains comboboxes (>100). The comboboxes influence the rest of the contents of the QTableView.
I've noticed that the setModelData() method does get called (and the table gets properly updated), but not immediately after selecting a new item from the combobox, but you need to click somewhere else (lose focus) before it gets called. This is quite inconvenient, and I'd like to know how to skip that click.
My comboboxes are made by calling setItemDelegateForRow for the first row of the table. The implementation of createEditor, setEditorData and setModelData is standard. My comboboxdelegate is derived from QStyledItemDelegate. Here is my code from ComboboxDelegate.cpp:
MainComboBoxDelegate::MainComboBoxDelegate(QObject *parent, MainTableListModel *model) : QStyledItemDelegate(parent)
{
m_model = model;
}
QWidget *MainComboBoxDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem & option , const QModelIndex & index) const
{
if(index.row() == 0 && index.column() != 0)
{
QComboBox* editor = new QComboBox(parent);
editor->setModel(m_model);
editor->setEditable(false);
return editor;
}
else
return QStyledItemDelegate::createEditor(parent, option, index);
}
void MainComboBoxDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
{
if(index.row() == 0)
{
QString temp = index.model()->data(index, Qt::EditRole).toString();
QComboBox *comboBox = static_cast<QComboBox*>(editor);
comboBox->setCurrentIndex(comboBox->findText(temp));
}
}
void MainComboBoxDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
{
if(index.row() == 0)
{
QComboBox *comboBox = static_cast<QComboBox*>(editor);
QString value = comboBox->currentText();
model->setData(index, value, Qt::EditRole);
}
}
void MainComboBoxDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
editor->setGeometry(option.rect);
}
And this is how I create them in my view class:
ui->Main_Table_View->setItemDelegateForRow(0, new MainComboBoxDelegate(ui->Main_Table_View, Maintablelistmodel));
for(int i = 1; i < Maintablemodel->columnCount(); ++i)
ui->Main_Table_View->openPersistentEditor(Maintablemodel->index(0, i));
The Maintablelistmodel is a QSortFilterProxyModel which contains only one column.
As the comboboxes get created this way, it's hard to pick up signals from them manually, but it may be an option.
Is there something I'm doing wrong, or is it some kind of limitation? Am I missing something?
Any help is highly appreciated.
I have a QTableView in which the first row contains comboboxes (>100). The comboboxes influence the rest of the contents of the QTableView.
I've noticed that the setModelData() method does get called (and the table gets properly updated), but not immediately after selecting a new item from the combobox, but you need to click somewhere else (lose focus) before it gets called. This is quite inconvenient, and I'd like to know how to skip that click.
My comboboxes are made by calling setItemDelegateForRow for the first row of the table. The implementation of createEditor, setEditorData and setModelData is standard. My comboboxdelegate is derived from QStyledItemDelegate. Here is my code from ComboboxDelegate.cpp:
MainComboBoxDelegate::MainComboBoxDelegate(QObject *parent, MainTableListModel *model) : QStyledItemDelegate(parent)
{
m_model = model;
}
QWidget *MainComboBoxDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem & option , const QModelIndex & index) const
{
if(index.row() == 0 && index.column() != 0)
{
QComboBox* editor = new QComboBox(parent);
editor->setModel(m_model);
editor->setEditable(false);
return editor;
}
else
return QStyledItemDelegate::createEditor(parent, option, index);
}
void MainComboBoxDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
{
if(index.row() == 0)
{
QString temp = index.model()->data(index, Qt::EditRole).toString();
QComboBox *comboBox = static_cast<QComboBox*>(editor);
comboBox->setCurrentIndex(comboBox->findText(temp));
}
}
void MainComboBoxDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
{
if(index.row() == 0)
{
QComboBox *comboBox = static_cast<QComboBox*>(editor);
QString value = comboBox->currentText();
model->setData(index, value, Qt::EditRole);
}
}
void MainComboBoxDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
editor->setGeometry(option.rect);
}
And this is how I create them in my view class:
ui->Main_Table_View->setItemDelegateForRow(0, new MainComboBoxDelegate(ui->Main_Table_View, Maintablelistmodel));
for(int i = 1; i < Maintablemodel->columnCount(); ++i)
ui->Main_Table_View->openPersistentEditor(Maintablemodel->index(0, i));
The Maintablelistmodel is a QSortFilterProxyModel which contains only one column.
As the comboboxes get created this way, it's hard to pick up signals from them manually, but it may be an option.
Is there something I'm doing wrong, or is it some kind of limitation? Am I missing something?
Any help is highly appreciated.