文章目录
- 一、自定义 QTableWidget
- 1、创建 QTableWidget
- 2、设置代理
- 3、给cell设置QWidget
一、自定义 QTableWidget
1、创建 QTableWidget
基于VS+Qt创建一个Qt项目,通过Qt Designer创建QTableWidget,完整代码参考附件,界面如下图:
2、设置代理
给地址列设置代理,要求如下:
QComboBox
作为编辑控件。- 下拉列表内容变更时,发出一个
OnCurrentTextChanged
信号。
代理代码如下:
QWidget *MyAddrDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
{QComboBox *editor = new QComboBox(parent);if (editor == nullptr)return nullptr;QStringList itemList;itemList << QString("北京");itemList << QString("上海");itemList << QString("西安");editor->addItems(itemList);editor->setFrame(false);return editor;
}void MyAddrDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
{QComboBox *combox = static_cast<QComboBox*>(editor);combox->setCurrentIndex(combox->findText(index.model()->data(index, Qt::EditRole).toString()));
}void MyAddrDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
{model->blockSignals(true);QComboBox *combox = static_cast<QComboBox*>(editor);model->setData(index, combox->currentText(), Qt::EditRole);emit OnCurrentTextChanged(index.row(), index.column(), combox->currentText());model->blockSignals(false);
}
关键点:
-
QComboBox
内容变更时,外界如何感知到?代理自定义信号
OnCurrentTextChanged
信号,当更新模型数据时触发这个信号。给"地址"列设置代理时,通过槽函数链接这个信号,如下:// MyAddrDelegate.cpp // 代理更新模型时发送OnCurrentTextChanged信号 void MyAddrDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const {model->blockSignals(true);QComboBox *combox = static_cast<QComboBox*>(editor);model->setData(index, combox->currentText(), Qt::EditRole);emit OnCurrentTextChanged(index.row(), index.column(), combox->currentText());model->blockSignals(false); }// MyTableWidget.cpp // 创建代理时链接信号 void MyTableWidget::InitDelegate() {MyAddrDelegate *pAddrDelegate = new MyAddrDelegate();ui.m_pTableWidget->setItemDelegateForColumn(1, pAddrDelegate);connect(pAddrDelegate, SIGNAL(OnCurrentTextChanged(int, int, QString)), this, SLOT(OnCurrentTextChanged(int, int, QString))); }
需要注意的点:
-
更新模型数据时,如果不需要触发
QTableWidget
本身的信号,模型需要屏蔽信号setModelData()
函数里面的model->setData
操作更新模型数据时,会触发QTableWidget::cellChanged()
等信号,如果不需要触发这些信号,可以调用model->blockSignals
屏蔽信号。
3、给cell设置QWidget
QTableWidget::setCellWidget()
支持设置自定义QWidget
作为cell的内容,如下:
核心代码如下:
// 设置
QTableWidgetItem *pSettingItem = new QTableWidgetItem();
pSettingItem->setText("");
ui.m_pTableWidget->setItem(row, 3, pSettingItem);QPushButton *pBtn = new QPushButton(QString::fromStdWString(data.text));
QVBoxLayout *pLayout = new QVBoxLayout();
QWidget *pWidget = new QWidget(this);pBtn->setFocusPolicy(Qt::FocusPolicy::NoFocus);
pLayout->setContentsMargins(0, 0, 0, 0);
pLayout->addWidget(pBtn);
pWidget->setLayout(pLayout);connect(pBtn, SIGNAL(clicked()), this, SLOT(OnSettingBtnClicked()));ui.m_pTableWidget->setCellWidget(row, 3, pWidget);
关键点:
-
单击按键时,如何判断点击那个cell对应的按键
QTableWidget::indexAt()
函数支持根据位置获取cell对应的索引,槽函数可以获取QPushButton
对象,通过QPushButton
对象的父对象的位置,可以获取到cell对应的索引。如下:void MyTableWidget::OnSettingBtnClicked() {QPushButton *pBtn = qobject_cast<QPushButton *>(sender());if (pBtn == nullptr)return;int x = pBtn->parentWidget()->pos().x();int y = pBtn->parentWidget()->pos().y();int row = ui.m_pTableWidget->indexAt(QPoint(x, y)).row();int col = ui.m_pTableWidget->indexAt(QPoint(x, y)).column();if (row < 0 || row >= ui.m_pTableWidget->rowCount())return;qDebug() << QString("row = %1, col = %2").arg(row).arg(col); }