[英]QT Inherit from QTableWidgetItem to Widget and overide '<' operator
我想要一個具有某些單元格的QTableWidget
作為自定義QProgressBar
,並且希望可以對包含這些單元格的列進行排序。
我自定義的QProgressBar
繼承自QProgressBar
和QTableWidgetItem
,並且我覆蓋了'<'
運算符以進行排序。
以下是自定義QTableWidget
( QTableWidget_custom
), QTableWidgetItem
( QTableWidgetItem_double
)和QProgressBar
( QProgressBar_custom
)的示例。 填充表格后,我只能對包含QTableWidgetItem_double
的列進行排序。
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);
}
};
填表
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列的水平標題,則會使用QTableWidgetItem_double
的重載運算符對表進行正確排序。 如果單擊第三個標題,則不會發生任何事情。 重載函數從不調用。
訂購時,僅調用QTableWidgetItem
,在您的情況下, QProgressBar_custom
作為cellWidget
添加,而不作為QTableWidgetItem
,您必須執行以下操作:
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
是另一種選擇。 為了使其健壯並且不破壞表視圖中的內部數據結構,比較運算符必須始終返回一致的結果(或根本沒有結果-即斷言):
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(); } }
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.