簡體   English   中英

QT從QTableWidgetItem繼承到Widget並覆蓋“ <”運算符

[英]QT Inherit from QTableWidgetItem to Widget and overide '<' operator

我想要一個具有某些單元格的QTableWidget作為自定義QProgressBar ,並且希望可以對包含這些單元格的列進行排序。

我自定義的QProgressBar繼承自QProgressBarQTableWidgetItem ,並且我覆蓋了'<'運算符以進行排序。

以下是自定義QTableWidgetQTableWidget_custom ), QTableWidgetItemQTableWidgetItem_double )和QProgressBarQProgressBar_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.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM