QT从QTableWidgetItem继承到Widget并覆盖'<'运算符
QT Inherit from QTableWidgetItem to Widget and overide '<' operator
我想要一个 QTableWidget
带有自定义 QProgressBar
的某些单元格,我希望它可以对包含这些的列进行排序。
我的自定义 QProgressBar
是从 QProgressBar
和 QTableWidgetItem
继承的,我重写了 '<'
运算符以允许排序。
下面是自定义 QTableWidget
(QTableWidget_custom
)、QTableWidgetItem
(QTableWidgetItem_double
) 和 QProgressBar
(QProgressBar_custom
) 的示例.填充 table 后,我只能对包含 QTableWidgetItem_double
s.
的列进行排序
QTableWidgetItem_double
class QTableWidgetItem_double : public QTableWidgetItem
{
public:
QTableWidgetItem_double(const double _d = 0.0) :QTableWidgetItem(){
d = _d;
this->setTextAlignment(Qt::AlignRight | Qt::AlignVCenter);
setText(QString::number(d, 'f', 4));
}
double d = 0.0;
bool operator <(const QTableWidgetItem& other) const{
QTableWidgetItem_double const *qtwd = dynamic_cast<QTableWidgetItem_double const *>(&other);
return d < qtwd->d;
}
};
QProgressBar_custom
class QProgressBar_custom : public QProgressBar, public QTableWidgetItem
{
public:
QProgressBar_custom() : QProgressBar(){
setTextVisible(true);
setStyleSheet(" QProgressBar { border: 1px solid grey; border-radius: 0px; text-align: right;} QProgressBar::chunk { background-color: rgba(0,255,0,100); width: 1px;}");
setAlignment(Qt::AlignCenter);
}
QProgressBar_custom(const QString txt) : QTableWidgetItem(txt){}
bool operator <(const QTableWidgetItem& other) const{
QProgressBar const *p = dynamic_cast<QProgressBar const*>(&other);
return value() < p->value();
}
private:
double d1 = 0.0;
double d2 = 0.0;
QString txt1 = "";
QString txt2 = "";
public:
void setText(){
QString txtfield = QString::number(d1, 'f', 2) + " " + txt1 + " \t" + QString::number(d2, 'f', 4) + " " + txt2 + " ";
setFormat(txtfield);
}
void setTxt1(const QString& txt){ txt1 = txt; }
void setTxt2(const QString& txt){ txt2 = txt; }
void setAmount1(double d){ d1 = d; }
void setAmount2(double d){ d2 = d; }
void setPercentage(double p){
setValue(int(100.0*p));
setText();
}
};
QTableWidget_custom
class QTableWidget_custom : public QTableWidget
{
public:
QTableWidget_custom(QWidget *parent) : QTableWidget(parent){}
void addCustomProgressBarCell(int row, int col, double d1, double d2, const QString& txt1, const QString& txt2, double p){
QProgressBar_custom* qprbar = new QProgressBar_custom();
qprbar->setAmount1(d1);
qprbar->setAmount2(d2);
qprbar->setTxt1(txt1);
qprbar->setTxt2(txt2);
qprbar->setPercentage(p);
setCellWidget(row, col, qprbar);
}
void addCustomDoubleCell(int row, int col, double d){
QTableWidgetItem_double* qtwd = new QTableWidgetItem_double(d);
setItem(row, col, qtwd);
}
};
正在填充 table
void QTWidgetTableItem::testCustomTableWidgetItem(){
//Some random data:
double data1[10] = { 1.0, 2.0, 34.4, 32.02, 1000.0, 2002.0, 0.0001, 0.02, 0.009, 10.0 };
double data2[10] = { -1.01, 22.3, 54.4, 1232.02, 0.0, -22.0, 0.1231, -0.02, 10.009, 5.0 };
double data3[10] = { 5.0, 201.0, 434.4444, 32.32, 1234.231, 2002.232, 0.111, 0.0422, 0.212, -10.0 };
double data4[10] = { 1.0, 2.0, 5.0, 25.00, 12.2, 100.0, 0.0, 50.0, 65.03, 0.9 };
QString txt1[10] = { "abc", "abc", "abc", "eur", "usd", "php", "cpp", "xxx", "yyy", "zzz" };
QString txt2[10] = { "aaa", "bbb", "ccc", "ddd", "asd", "qwe", "trt", "tr1", "tr2", "tr3" };
//Prepare table:
QTableWidget_custom* table = ui.tableWidget;
table->insertColumn(0); table->setColumnWidth(0, 70);
table->insertColumn(1); table->setColumnWidth(1, 70);
table->insertColumn(2); table->setColumnWidth(2, 170);
table->setSortingEnabled(false);
//Add data to table:
for (int i = 0; i < 10; i++){
table->insertRow(i);
table->addCustomDoubleCell(i, 0, data1[i]);
table->addCustomDoubleCell(i, 1, data2[i]);
table->addCustomProgressBarCell(i, 2, data3[i], data4[i], txt1[i], txt2[i], data4[i] / 100.0);
}
table->setSortingEnabled(true);
}
结果
如果我单击第 1 列或第 2 列的水平 header,table 会使用 QTableWidgetItem_double
的重载运算符正确排序。如果我点击第三个 header,没有任何反应。永远不会调用重载函数。
只有 QTableWidgetItem
在订购时被调用,在你的情况下 QProgressBar_custom
被添加为 cellWidget
s,而不是 QTableWidgetItem
s,你必须这样做它如下:
void addCustomProgressBarCell(int row, int col, double d1, double d2, const QString& txt1, const QString& txt2, double p){
QProgressBar_custom *qprbar = new QProgressBar_custom;
qprbar->setAmount1(d1);
qprbar->setAmount2(d2);
qprbar->setTxt1(txt1);
qprbar->setTxt2(txt2);
qprbar->setPercentage(p);
setCellWidget(row, col, qprbar);
setItem(row, col, qprbar); // add this line
}
一些评论:
如果项目很多,使用委托比为每个项目实例化小部件的性能更高。委托的paint
方法可以使用QStyle::drawControl
。稍后将提供完整的示例。它不是特别复杂,而且效果很好。
为了性能和可读性,首选 QString
格式字符串而不是串联。使用 QStringLiteral
而不是强制转换 C 字符串。
auto const txtField = QStringLiteral("%1 %2 \t%3 %4 ")
.arg(d1, 'f', 2) .arg(txt1)
.arg(f2, 'f', 4) .arg(txt2);
对于QObject
派生的类,qobject_cast
是另一种选择。为了使其健壮并且不会破坏 table 视图中的内部数据结构,比较运算符必须始终 return 一致的结果(或者根本没有结果 - 即断言):
bool operator <(const QTableWidgetItem& other) const {
auto const *p = qobject_cast<const QProgressBar*>(&other);
if (true)
// let's always succeed
return p ? value() < p->value()
: static_cast<const QTableWidgetItem*>(this) < &other;
else {
// catch errors
Q_ASSERT(p);
return value() < p->value();
}
}
我想要一个 QTableWidget
带有自定义 QProgressBar
的某些单元格,我希望它可以对包含这些的列进行排序。
我的自定义 QProgressBar
是从 QProgressBar
和 QTableWidgetItem
继承的,我重写了 '<'
运算符以允许排序。
下面是自定义 QTableWidget
(QTableWidget_custom
)、QTableWidgetItem
(QTableWidgetItem_double
) 和 QProgressBar
(QProgressBar_custom
) 的示例.填充 table 后,我只能对包含 QTableWidgetItem_double
s.
QTableWidgetItem_double
class QTableWidgetItem_double : public QTableWidgetItem
{
public:
QTableWidgetItem_double(const double _d = 0.0) :QTableWidgetItem(){
d = _d;
this->setTextAlignment(Qt::AlignRight | Qt::AlignVCenter);
setText(QString::number(d, 'f', 4));
}
double d = 0.0;
bool operator <(const QTableWidgetItem& other) const{
QTableWidgetItem_double const *qtwd = dynamic_cast<QTableWidgetItem_double const *>(&other);
return d < qtwd->d;
}
};
QProgressBar_custom
class QProgressBar_custom : public QProgressBar, public QTableWidgetItem
{
public:
QProgressBar_custom() : QProgressBar(){
setTextVisible(true);
setStyleSheet(" QProgressBar { border: 1px solid grey; border-radius: 0px; text-align: right;} QProgressBar::chunk { background-color: rgba(0,255,0,100); width: 1px;}");
setAlignment(Qt::AlignCenter);
}
QProgressBar_custom(const QString txt) : QTableWidgetItem(txt){}
bool operator <(const QTableWidgetItem& other) const{
QProgressBar const *p = dynamic_cast<QProgressBar const*>(&other);
return value() < p->value();
}
private:
double d1 = 0.0;
double d2 = 0.0;
QString txt1 = "";
QString txt2 = "";
public:
void setText(){
QString txtfield = QString::number(d1, 'f', 2) + " " + txt1 + " \t" + QString::number(d2, 'f', 4) + " " + txt2 + " ";
setFormat(txtfield);
}
void setTxt1(const QString& txt){ txt1 = txt; }
void setTxt2(const QString& txt){ txt2 = txt; }
void setAmount1(double d){ d1 = d; }
void setAmount2(double d){ d2 = d; }
void setPercentage(double p){
setValue(int(100.0*p));
setText();
}
};
QTableWidget_custom
class QTableWidget_custom : public QTableWidget
{
public:
QTableWidget_custom(QWidget *parent) : QTableWidget(parent){}
void addCustomProgressBarCell(int row, int col, double d1, double d2, const QString& txt1, const QString& txt2, double p){
QProgressBar_custom* qprbar = new QProgressBar_custom();
qprbar->setAmount1(d1);
qprbar->setAmount2(d2);
qprbar->setTxt1(txt1);
qprbar->setTxt2(txt2);
qprbar->setPercentage(p);
setCellWidget(row, col, qprbar);
}
void addCustomDoubleCell(int row, int col, double d){
QTableWidgetItem_double* qtwd = new QTableWidgetItem_double(d);
setItem(row, col, qtwd);
}
};
正在填充 table
void QTWidgetTableItem::testCustomTableWidgetItem(){
//Some random data:
double data1[10] = { 1.0, 2.0, 34.4, 32.02, 1000.0, 2002.0, 0.0001, 0.02, 0.009, 10.0 };
double data2[10] = { -1.01, 22.3, 54.4, 1232.02, 0.0, -22.0, 0.1231, -0.02, 10.009, 5.0 };
double data3[10] = { 5.0, 201.0, 434.4444, 32.32, 1234.231, 2002.232, 0.111, 0.0422, 0.212, -10.0 };
double data4[10] = { 1.0, 2.0, 5.0, 25.00, 12.2, 100.0, 0.0, 50.0, 65.03, 0.9 };
QString txt1[10] = { "abc", "abc", "abc", "eur", "usd", "php", "cpp", "xxx", "yyy", "zzz" };
QString txt2[10] = { "aaa", "bbb", "ccc", "ddd", "asd", "qwe", "trt", "tr1", "tr2", "tr3" };
//Prepare table:
QTableWidget_custom* table = ui.tableWidget;
table->insertColumn(0); table->setColumnWidth(0, 70);
table->insertColumn(1); table->setColumnWidth(1, 70);
table->insertColumn(2); table->setColumnWidth(2, 170);
table->setSortingEnabled(false);
//Add data to table:
for (int i = 0; i < 10; i++){
table->insertRow(i);
table->addCustomDoubleCell(i, 0, data1[i]);
table->addCustomDoubleCell(i, 1, data2[i]);
table->addCustomProgressBarCell(i, 2, data3[i], data4[i], txt1[i], txt2[i], data4[i] / 100.0);
}
table->setSortingEnabled(true);
}
结果
如果我单击第 1 列或第 2 列的水平 header,table 会使用 QTableWidgetItem_double
的重载运算符正确排序。如果我点击第三个 header,没有任何反应。永远不会调用重载函数。
只有 QTableWidgetItem
在订购时被调用,在你的情况下 QProgressBar_custom
被添加为 cellWidget
s,而不是 QTableWidgetItem
s,你必须这样做它如下:
void addCustomProgressBarCell(int row, int col, double d1, double d2, const QString& txt1, const QString& txt2, double p){
QProgressBar_custom *qprbar = new QProgressBar_custom;
qprbar->setAmount1(d1);
qprbar->setAmount2(d2);
qprbar->setTxt1(txt1);
qprbar->setTxt2(txt2);
qprbar->setPercentage(p);
setCellWidget(row, col, qprbar);
setItem(row, col, qprbar); // add this line
}
一些评论:
如果项目很多,使用委托比为每个项目实例化小部件的性能更高。委托的
paint
方法可以使用QStyle::drawControl
。稍后将提供完整的示例。它不是特别复杂,而且效果很好。为了性能和可读性,首选
QString
格式字符串而不是串联。使用QStringLiteral
而不是强制转换 C 字符串。auto const txtField = QStringLiteral("%1 %2 \t%3 %4 ") .arg(d1, 'f', 2) .arg(txt1) .arg(f2, 'f', 4) .arg(txt2);
对于
QObject
派生的类,qobject_cast
是另一种选择。为了使其健壮并且不会破坏 table 视图中的内部数据结构,比较运算符必须始终 return 一致的结果(或者根本没有结果 - 即断言):bool operator <(const QTableWidgetItem& other) const { auto const *p = qobject_cast<const QProgressBar*>(&other); if (true) // let's always succeed return p ? value() < p->value() : static_cast<const QTableWidgetItem*>(this) < &other; else { // catch errors Q_ASSERT(p); return value() < p->value(); } }