整合營銷服務商

          電腦端+手機端+微信端=數據同步管理

          免費咨詢熱線:

          Python入坑系列-pyside6桌面編程之掌握控

          Python入坑系列-pyside6桌面編程之掌握控件類使用技巧

          過本文章,可以掌握以下內容:

          1. Pyside6控件類的屬性與方法
          2. Pyside6的控件為什么看不到實現過程
          3. Pyside6如何查看相關類的屬性和方法作用

          對于一個Python小白來說,在學習Pyside6的控件時,照著網上的一些示例,可以將很快將控件運行出來。但在看控件python源代碼時,卻很是迷惑,本文就以QWidget為例,進行展開解讀。

          QWidget python源代碼

          如上圖是QWidget部分Python源代碼,你能看懂多少?

          1. 大部分方法都包含self參數,其作用是什么?
          2. 為什么方法體很多都是空實現,用pass或者簡單返回,并未看到具體實現,為何能正常使用方法?
          3. 定義的枚舉和屬性都是none,為何能使用枚舉類,以及枚舉類的枚舉值在哪?

          一、Pyside6類的屬性與方法

          要了解控件大部分方法為什么都包含self參數,只需要了解Python類、實例的屬性和方法即可。

          實例屬性與類屬性

          類的對象,就是類的一個實例。類的實例屬性被對象所有,包含在每個對象之中,不同的對象之間,互不干擾。類的類屬性被類所有,被包含在類中,是所有的類對象共享。一般情況下,實例屬性會在__init__ 方法中聲明并初始化,并且使用self 來綁定。而類屬性是在類作用域中被聲明,并且不使用self 來綁定。

          python類、實例與屬性關系

          實例方法,類方法,靜態方法

          Python 類中有三種方法:

          • 實例方法:屬于對象,方法中都有一個self 參數(代表對象本身)。實例方法只能由對象調用,不能通過類名訪問。實例方法中可以訪問實例屬性和類屬性。
          • 類方法:屬于類,方法中都有一個cls 參數(代表類本身)。類方法即可以通過類名訪問,也可以通過對象訪問,類方法中只能訪問類屬性,不能訪問實例屬性。定義類方法需要用到裝飾器@classmethod
          • 靜態方法:沒有self 參數,也沒有cls 參數。因此,在靜態方法中,既不能訪問類屬性,也不能訪問實例屬性。靜態方法可以使用類名訪問,也可以使用對象訪問。定義靜態方法需要用到裝飾器@staticmethod

          注意:Python 解釋器在構造類與對象時,類是先于對象產生的。因此,類屬性與類方法是先于實例屬性與實例方法產生的。所以當類方法產生時,還沒有實例屬性,因此,類方法中不能訪問實例屬性。

          專有方法

          以雙下劃線__開頭且結尾的方法__xxx__,就是專有方法,也稱魔法方法。這些方法都被Python 賦予了特殊的含義,用戶可以根據需要,來實現這些方法。比如QWidget類重寫了Python object的方法:

          • __delattr__:Python 中的一個特殊方法,用于自定義刪除對象屬性的行為。當你使用執行del obj.attribute語句刪除對象的屬性時,__delattr__ 方法會被調用,
          • __init__:負責初始化實例,在實例創建之后調用。它是一個實例方法,用于設置實例屬性和執行其他初始化操作
          • __new__:負責創建實例,在實例化對象時首先調用。它是一個靜態方法,返回類的一個新實例
          • __repr__:返回對象的字符串表示,通常用于調試
          • __setattr__:用于設置對象的屬性值,在賦值時自動調用

          二、Pyside6的調用過程

          在開發Pyside6的時候,很多源代碼只包含方法名、pass 關鍵字和空的枚舉類,那pyside6的代碼在實例化以及調用方法時,整個過程是怎樣的呢。

          在 PySide6 中,Python和Qt的C++庫之間的關聯是通過綁定技術實現的。具體來說,PySide6使用Shiboken,這是一種綁定生成工具,它自動將C++類和函數映射到Python。通過這種方式,PySide6可以在Python中使用Qt的功能。關鍵技術概念:

          • Shiboken:這是一個自動化工具,用于將 C++ 代碼綁定到Python。Shiboken分析Qt 的 C++頭文件,生成相應的 Python 綁定代碼。這使得我們可以在 Python 中使用 Qt 的 C++ 類和方法。
          • 類型轉換:Shiboken 處理C++和Python類型之間的轉換。對于每個C++類型,Shiboken生成相應的Python類型包裝器,并處理兩者之間的數據轉換。
          • 方法和屬性綁定:Shiboken生成的綁定代碼將C++類的方法和屬性映射到Python類。這包括普通方法、虛方法、信號和槽。

          在 PySide6中,staticMetaObject是一個靜態屬性,存在于所有繼承自QObject的類中。它的主要作用是提供對Qt元對象系統(Meta-Object System)的訪問。這個元對象系統在 Qt 中用于實現信號與槽機制、屬性系統以及其他反射(reflection)功能。

          元對象系統(Meta-Object System)

          Qt 的元對象系統是一種運行時類型信息系統,提供以下功能:

          • 信號與槽機制:用于對象之間的通信。
          • 動態屬性系統:允許在運行時查詢和操作對象的屬性。
          • 反射:允許在運行時查詢類的元數據(例如類名、方法列表等)。

          staticMetaObject 的作用

          staticMetaObject 是一個包含類的元數據的對象,提供了對以下內容的訪問:

          • 類名:對象的類名。
          • 信號和槽:類中定義的信號和槽。
          • 屬性:類中定義的屬性。
          • 方法:類中定義的方法。

          staticMetaObject在PySide6中提供了對Qt元對象系統的訪問,允許在運行時查詢類的元數據。它是實現動態屬性訪問、信號與槽機制以及反射功能的基礎,極大地增強了代碼的靈活性和動態特性。

          三、正確使用Pyside6控件類

          如何正確使用Pyside6的控件,我們需要了解Qt for Python相關文檔

          訪問QT官網地址

          訪問官網:https://doc.qt.io/qtforpython-6/search.html,選擇對應pyside的版本。

          QT官網

          搜索控件類

          比如想了解Qwidget類,在上面搜索框中輸入關鍵字回車。

          控件類文檔

          查看控件類信息

          基于上面搜索結果,點擊進去,可以看到類的詳細信息,包括類的繼承關系、屬性、方法、枚舉類、信號??梢酝ㄟ^右邊列表快速定位屬性和方法

          查找控件類

          針對枚舉類,拷貝Python源代碼到界面搜索,即可看到使用方法,如Qwidget類定義的RenderFlag=None,可以搜索RenderFlag,就可以知道有哪些枚舉值。

          枚舉類

          、前言

          隨著移動互聯網的盛行,現在手機APP大行其道,每個人的手機沒有十幾個APP都不好意思說自己是現代人,各種聊天、購物、直播、小視頻等APP,有個陌生人社交的APP叫探探,本人用過幾次,當然不是去為了找對象,而是純粹為了好玩研究下他的U設計和軟件邏輯流程等,其中有個雷達控件,單擊以后可以搜索附近的異性進行配對,這個雷達控件的效果蠻好的,于是手癢琢磨著用Qt來實現一個,畢竟自己寫了150多個控件了,已經上癮了,對各種效果都如魚得水,看到各種效果都不自然的想到編碼思路等。
          這個控件的核心其實就是外圍的那個掃描圈和發散的掃描線,中間變大變小恢復正常的圓形頭像,外圍的掃描圈采用錐形漸變顏色,通過透明度控制形成掃描效果,核心方法就是drawPie,至于擴散圈,需要識別到單擊以后將擴散圈存入隊列,因為可能會單擊多次,產生多個擴散圈,至于中間頭像的動態效果,采用三個QPropertyAnimation來實現,一個負責變大,一個負責變小,一個負責恢復正常,然后三個動畫加入到QSequentialAnimationGroup動畫序列中,按照順序執行。

          二、實現的功能

          • 1:可設置中間圖像
          • 2:可設置圖像的邊框寬度+邊框顏色,產生圓形圖像效果
          • 3:可設置掃描線的最大半徑
          • 4:可設置掃描線的邊框寬度
          • 5:可設置擴散圈的線條寬度
          • 6:可設置掃描線的每次移動的步長
          • 7:可設置擴散圈的每次移動的步長
          • 8:可設置掃描線的顏色
          • 9:可設置擴散圈的顏色

          三、效果圖

          四、頭文件代碼

          #ifndef SCANTANTAN_H
          #define SCANTANTAN_H
          
          /**
           * 探探雷達控件 作者:東門吹雪(QQ:709102202) 整理:feiyangqingyun(QQ:517216493) 2019-10-01
           * 1:可設置中間圖像
           * 2:可設置圖像的邊框寬度+邊框顏色,產生圓形圖像效果
           * 3:可設置掃描線的最大半徑
           * 4:可設置掃描線的邊框寬度
           * 5:可設置擴散圈的線條寬度
           * 6:可設置掃描線的每次移動的步長
           * 7:可設置擴散圈的每次移動的步長
           * 8:可設置掃描線的顏色
           * 9:可設置擴散圈的顏色
           */
          
          #include <QWidget>
          
          class QSequentialAnimationGroup;
          
          #ifdef quc
          #if (QT_VERSION < QT_VERSION_CHECK(5,7,0))
          #include <QtDesigner/QDesignerExportWidget>
          #else
          #include <QtUiPlugin/QDesignerExportWidget>
          #endif
          
          class QDESIGNER_WIDGET_EXPORT ScanTanTan : public QWidget
          #else
          class ScanTanTan : public QWidget
          #endif
          
          {
              Q_OBJECT
              Q_PROPERTY(QPixmap image READ getImage WRITE setImage)
              Q_PROPERTY(int imageBorderWidth READ getImageBorderWidth WRITE setImageBorderWidth)
              Q_PROPERTY(QColor imageBorderColor READ getImageBorderColor WRITE setImageBorderColor)
          
              Q_PROPERTY(int scanRadius READ getScanRadius WRITE setScanRadius)
              Q_PROPERTY(int scanWidth READ getScanWidth WRITE setScanWidth)
              Q_PROPERTY(int ringWidth READ getRingWidth WRITE setRingWidth)
          
              Q_PROPERTY(int scanStep READ getScanStep WRITE setScanStep)
              Q_PROPERTY(int ringStep READ getRingStep WRITE setRingStep)
          
              Q_PROPERTY(QColor scanColor READ getScanColor WRITE setScanColor)
              Q_PROPERTY(QColor ringColor READ getRingColor WRITE setRingColor)
          
          public:
              struct RingData {
                  int radius;     //半徑
                  float width;    //畫筆粗細
                  int alpha;      //透明度
              };
          
              explicit ScanTanTan(QWidget *parent=0);
          
          protected:
              void mousePressEvent(QMouseEvent *);
              void mouseReleaseEvent(QMouseEvent *);
              void paintEvent(QPaintEvent *);
              void drawScan(QPainter *painter);
              void drawRing(QPainter *painter);
              void drawImage(QPainter *painter);
          
          private slots:
              void changeScan();
              void changeRing();
              void updateImage(const QVariant &value);
              double twoPtDistance(const QPointF &pt1, const QPointF &pt2);
          
          private:
              QPixmap image;          //中間圖片
              int imageBorderWidth;   //圖片邊框寬度
              QColor imageBorderColor;//圖片邊框顏色
          
              int scanRadius;         //掃描線最大半徑
              int scanWidth;          //掃描線邊框寬度
              int ringWidth;          //擴散圈線條寬度
          
              int scanStep;           //掃描線每次移動的步長
              int ringStep;           //擴散圈每次移動的步長
          
              QColor scanColor;       //掃描線顏色
              QColor ringColor;       //擴散圈顏色
          
              bool isPressed;         //鼠標是否按下
              int ringRadius;         //擴散圈半徑
              int imageRadius;        //圖片半徑
              int scanDeg;            //當前掃描線角度
          
              //擴散圈集合,鼠標可能按下多次則產生多個擴散圈,用隊列存起來
              QList<RingData> rings;
              //動畫組合,用于中間圖片的變大放小
              QSequentialAnimationGroup *animationGroup;
          
          public:
              QPixmap getImage()          const;
              int getImageBorderWidth()   const;
              QColor getImageBorderColor()const;
          
              int getScanRadius()         const;
              int getScanWidth()          const;
              int getRingWidth()          const;
          
              int getScanStep()           const;
              int getRingStep()           const;
          
              QColor getScanColor()       const;
              QColor getRingColor()       const;
          
              QSize sizeHint()            const;
              QSize minimumSizeHint()     const;
          
          public Q_SLOTS:
              //設置圖片+圖片邊框寬度+圖片邊框顏色
              void setImage(const QPixmap &image);
              void setImageBorderWidth(int imageBorderWidth);
              void setImageBorderColor(const QColor &imageBorderColor);
          
              //設置掃描線最大半徑+掃描線邊框寬度+擴散圈線條寬度
              void setScanRadius(int scanRadius);
              void setScanWidth(int scanWidth);
              void setRingWidth(int ringWidth);
          
              //設置掃描線步長+擴散圈步長
              void setScanStep(int scanStep);
              void setRingStep(int ringStep);
          
              //設置掃描線顏色+擴散圈顏色
              void setScanColor(const QColor &scanColor);
              void setRingColor(const QColor &ringColor);
          };
          
          #endif // SCANTANTAN_H
          
          
          

          【領QT開發教程學習資料,點擊下方鏈接莬費領取↓↓,先碼住不迷路~】

          點擊這里:「鏈接」

          五、核心代碼

          void ScanTanTan::paintEvent(QPaintEvent *)
          {
              int width=this->width();
              int height=this->height();
              int side=qMin(width, height);
          
              //繪制準備工作,啟用反鋸齒,平移坐標軸中心,等比例縮放
              QPainter painter(this);
              painter.setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform);
              painter.translate(width / 2, height / 2);
              painter.scale(side / 200.0, side / 200.0);
          
              //繪制掃描線
              drawScan(&painter);
              //繪制擴散圈
              drawRing(&painter);
              //繪制中間圖片
              drawImage(&painter);
          }
          
          void ScanTanTan::drawScan(QPainter *painter)
          {
              painter->save();
          
              //錐形漸變顏色,通過透明度控制形成掃描效果
              QConicalGradient conicalGradient(0, 0, scanDeg);
              QColor color=scanColor;
              color.setAlpha(50);
              conicalGradient.setColorAt(0, color);
              color.setAlpha(0);
              conicalGradient.setColorAt(1, color);
          
              //設置畫筆畫刷
              QPen pen;
              pen.setWidth(scanWidth);
              pen.setBrush(conicalGradient);
              painter->setPen(pen);
              painter->setBrush(conicalGradient);
          
              //繪制餅圓
              QRect rect(-scanRadius, -scanRadius, scanRadius * 2, scanRadius * 2);
              painter->drawPie(rect, scanDeg * 16, 360 * 16);
          
              painter->restore();
          }
          
          void ScanTanTan::drawRing(QPainter *painter)
          {
              painter->save();
              painter->setBrush(Qt::NoBrush);
          
              //繪制所有擴散圈,擴散圈其實就是個沒有背景顏色的圓形
              for (int i=0; i < rings.count(); i++) {
                  RingData ring=rings.at(i);
                  int radius=ring.radius;
                  float width=ring.width;
                  int alpha=255 - ring.alpha;
                  QColor color=ringColor;
                  color.setAlpha(alpha);
          
                  QPen pen;
                  pen.setWidthF(width);
                  pen.setColor(color);
                  painter->setPen(pen);
                  painter->drawEllipse(-radius, -radius, radius * 2, radius * 2);
              }
          
              painter->restore();
          }
          
          void ScanTanTan::drawImage(QPainter *painter)
          {
              painter->save();
          
              //設置圓形遮罩路徑,產生圓形頭像效果
              QPainterPath path;
              path.addEllipse(QPoint(0, 0), imageRadius, imageRadius);
              painter->setClipPath(path);
          
              //繪制圖片
              QRect rect(-imageRadius, -imageRadius, imageRadius * 2, imageRadius * 2);
              painter->drawPixmap(rect, image);
          
              //繪制圖片邊緣圓形
              QPen pen;
              pen.setWidth(imageBorderWidth);
              pen.setColor(imageBorderColor);
              painter->setPen(pen);
              painter->setBrush(Qt::NoBrush);
          
              //以下兩種方法二選一,其實繪制360度的圓弧=繪制無背景的圓形
              //painter->drawArc(rect, 0, 360 * 16);
              painter->drawEllipse(rect);
          
              painter->restore();
          }
          
          

          六、控件介紹

          1. 超過160個精美控件,涵蓋了各種儀表盤、進度條、進度球、指南針、曲線圖、標尺、溫度計、導航條、導航欄,flatui、高亮按鈕、滑動選擇器、農歷等。遠超qwt集成的控件數量。
          2. 每個類都可以獨立成一個單獨的控件,零耦合,每個控件一個頭文件和一個實現文件,不依賴其他文件,方便單個控件以源碼形式集成到項目中,較少代碼量。qwt的控件類環環相扣,高度耦合,想要使用其中一個控件,必須包含所有的代碼。
          3. 全部純Qt編寫,QWidget+QPainter繪制,支持Qt4.6到Qt5.13的任何Qt版本,支持mingw、msvc、gcc等編譯器,支持任意操作系統比如windows+linux+mac+嵌入式linux等,不亂碼,可直接集成到Qt Creator中,和自帶的控件一樣使用,大部分效果只要設置幾個屬性即可,極為方便。
          4. 每個控件都有一個對應的單獨的包含該控件源碼的DEMO,方便參考使用。同時還提供一個所有控件使用的集成的DEMO。
          5. 每個控件的源代碼都有詳細中文注釋,都按照統一設計規范編寫,方便學習自定義控件的編寫。
          6. 每個控件默認配色和demo對應的配色都非常精美。
          7. 超過130個可見控件,6個不可見控件。
          8. 部分控件提供多種樣式風格選擇,多種指示器樣式選擇。
          9. 所有控件自適應窗體拉伸變化。
          10. 集成自定義控件屬性設計器,支持拖曳設計,所見即所得,支持導入導出xml格式。
          11. 自帶activex控件demo,所有控件可以直接運行在ie瀏覽器中。
          12. 集成fontawesome圖形字體+阿里巴巴iconfont收藏的幾百個圖形字體,享受圖形字體帶來的樂趣。
          13. 所有控件最后生成一個動態庫文件(dll或者so等),可以直接集成到qtcreator中拖曳設計使用。
          14. 目前已經有qml版本,后期會考慮出pyqt版本,如果用戶需求量很大的話。
          15. 自定義控件插件開放動態庫使用(永久免費),無任何后門和限制,請放心使用。
          16. 目前已提供26個版本的dll,其中包括了qt5.12.3 msvc2017 32+64 mingw 32+64 的。
          17. 不定期增加控件和完善控件,不定期更新SDK,歡迎各位提出建議,謝謝!
          18. Qt入門書籍推薦霍亞飛的《Qt Creator快速入門》《Qt5編程入門》,Qt進階書籍推薦官方的《C++ GUI Qt4編程》。
          19. 強烈推薦程序員自我修養和規劃系列書《大話程序員》《程序員的成長課》《解憂程序員》,受益匪淺,受益終生!
          20. SDK地址:https://gitee.com/feiyangqingyun/QUCSDK

          原文鏈接:https://www.cnblogs.com/feiyangqingyun/p/11632586.html

          【領QT開發教程學習資料,點擊下方鏈接莬費領取↓↓,先碼住不迷路~】

          點擊這里:Qt資料領取(視頻教程+文檔+代碼+項目實戰)

          、前言

          代碼行數統計主要用來統計項目中的所有文件的代碼行數,其中包括空行、注釋行、代碼行,可以指定過濾拓展名,比如只想統計.cpp的文件,也可以指定文件或者指定目錄進行統計。寫完這個工具第一件事情就是統計了一下自己寫過的最大的項目大概多少行代碼,看下是不是傳說中的一行代碼一塊錢,這個最大的項目從2010年開始的,到現在差不多快10年了,是自己在現在公司寫過的最大的項目,一直在升級更新完善,途中重構過兩次,大的結構改動,統計了下好像有15W行左右的代碼,純代碼大概在10W,其余是空行和注釋行,著實把自己嚇了一跳,還算是中型項目了,然后又統計了下自定義控件的所有代碼,我勒個去,總代碼23W行,純代碼17W行呢,哎呀我去!

          開源地址:https://gitee.com/feiyangqingyun/QWidgetDemo https://github.com/feiyangqingyun/QWidgetDemo

          二、實現的功能

          • 1:可分別統計代碼行/空行/注釋行
          • 2:支持指定過濾拓展名
          • 3:支持指定文件或者指定目錄進行統計
          • 4:分步顯示統計結果,不卡主界面
          • 5:分別展示每個統計過的文件的大小/總行數/代碼行數等

          三、效果圖

          四、頭文件代碼

          #ifndef FRMCOUNTCODE_H
          #define FRMCOUNTCODE_H
          
          #include <QWidget>
          
          namespace Ui {
          class frmCountCode;
          }
          
          class frmCountCode : public QWidget
          {
              Q_OBJECT
          
          public:
              explicit frmCountCode(QWidget *parent=0);
              ~frmCountCode();
          
          private:
              Ui::frmCountCode *ui;
              QStringList listFile;
          
          private:
              void initForm();
              bool checkFile(const QString &fileName);
              void countCode(const QString &filePath);
              void countCode(const QStringList &files);
              void countCode(const QString &fileName, int &lineCode, int &lineBlank, int &lineNotes);
          
          private slots:
              void on_btnOpenFile_clicked();
              void on_btnOpenPath_clicked();
              void on_btnClear_clicked();
          };
          
          #endif // FRMCOUNTCODE_H
          
          
          

          【領QT開發教程學習資料,點擊下方鏈接莬費領取↓↓,先碼住不迷路~】

          點擊這里:Qt資料領取(視頻教程+文檔+代碼+項目實戰)

          五、核心代碼

          #pragma execution_character_set("utf-8")
          
          #include "frmcountcode.h"
          #include "ui_frmcountcode.h"
          #include "qfile.h"
          #include "qtextstream.h"
          #include "qfiledialog.h"
          #include "qfileinfo.h"
          #include "qdebug.h"
          
          frmCountCode::frmCountCode(QWidget *parent) : QWidget(parent), ui(new Ui::frmCountCode)
          {
              ui->setupUi(this);
              this->initForm();
              on_btnClear_clicked();
          }
          
          frmCountCode::~frmCountCode()
          {
              delete ui;
          }
          
          void frmCountCode::initForm()
          {
              QStringList headText;
              headText << "文件名" << "類型" << "大小" << "總行數" << "代碼行數" << "注釋行數" << "空白行數" << "路徑";
              QList<int> columnWidth;
              columnWidth << 130 << 50 << 70 << 80 << 70 << 70 << 70 << 150;
          
              int columnCount=headText.count();
              ui->tableWidget->setColumnCount(columnCount);
              ui->tableWidget->setHorizontalHeaderLabels(headText);
              ui->tableWidget->setSelectionBehavior(QAbstractItemView::SelectRows);
              ui->tableWidget->setEditTriggers(QAbstractItemView::NoEditTriggers);
              ui->tableWidget->setSelectionMode(QAbstractItemView::SingleSelection);
              ui->tableWidget->verticalHeader()->setVisible(false);
              ui->tableWidget->horizontalHeader()->setStretchLastSection(true);
              ui->tableWidget->horizontalHeader()->setHighlightSections(false);
              ui->tableWidget->verticalHeader()->setDefaultSectionSize(20);
              ui->tableWidget->verticalHeader()->setHighlightSections(false);
          
              for (int i=0; i < columnCount; i++) {
                  ui->tableWidget->setColumnWidth(i, columnWidth.at(i));
              }
          
              //設置前景色
              ui->txtCount->setStyleSheet("color:#17A086;");
              ui->txtSize->setStyleSheet("color:#CA5AA6;");
              ui->txtRow->setStyleSheet("color:#CD1B19;");
              ui->txtCode->setStyleSheet("color:#22A3A9;");
              ui->txtNote->setStyleSheet("color:#D64D54;");
              ui->txtBlank->setStyleSheet("color:#A279C5;");
          
              //設置字體加粗
              QFont font;
              font.setBold(true);
              if (font.pointSize() > 0) {
                  font.setPointSize(font.pointSize() + 1);
              } else {
                  font.setPixelSize(font.pixelSize() + 2);
              }
          
              ui->txtCount->setFont(font);
              ui->txtSize->setFont(font);
              ui->txtRow->setFont(font);
              ui->txtCode->setFont(font);
              ui->txtNote->setFont(font);
              ui->txtBlank->setFont(font);
          
          #if (QT_VERSION > QT_VERSION_CHECK(4,7,0))
              ui->txtFilter->setPlaceholderText("中間空格隔開,例如 *.h *.cpp *.c");
          #endif
          }
          
          bool frmCountCode::checkFile(const QString &fileName)
          {
              if (fileName.startsWith("moc_") || fileName.startsWith("ui_") || fileName.startsWith("qrc_")) {
                  return false;
              }
          
              QFileInfo file(fileName);
              QString suffix="*." + file.suffix();
              QString filter=ui->txtFilter->text().trimmed();
              QStringList filters=filter.split(" ");
              return filters.contains(suffix);
          }
          
          void frmCountCode::countCode(const QString &filePath)
          {
              QDir dir(filePath);
              foreach (QFileInfo fileInfo , dir.entryInfoList()) {
                  if (fileInfo.isFile()) {
                      QString strFileName=fileInfo.fileName();
                      if (checkFile(strFileName)) {
                          listFile << fileInfo.filePath();
                      }
                  } else {
                      if(fileInfo.fileName()=="." || fileInfo.fileName()=="..") {
                          continue;
                      }
          
                      //遞歸找出文件
                      countCode(fileInfo.absoluteFilePath());
                  }
              }
          }
          
          void frmCountCode::countCode(const QStringList &files)
          {
              int lineCode;
              int lineBlank;
              int lineNotes;
              int count=files.count();
              on_btnClear_clicked();
              ui->tableWidget->setRowCount(count);
          
              quint32 totalLines=0;
              quint32 totalBytes=0;
              quint32 totalCodes=0;
              quint32 totalNotes=0;
              quint32 totalBlanks=0;
          
              for (int i=0; i < count; i++) {
                  QFileInfo fileInfo(files.at(i));
                  countCode(fileInfo.filePath(), lineCode, lineBlank, lineNotes);
                  int lineAll=lineCode + lineBlank + lineNotes;
          
                  QTableWidgetItem *itemName=new QTableWidgetItem;
                  itemName->setText(fileInfo.fileName());
          
                  QTableWidgetItem *itemSuffix=new QTableWidgetItem;
                  itemSuffix->setText(fileInfo.suffix());
          
                  QTableWidgetItem *itemSize=new QTableWidgetItem;
                  itemSize->setText(QString::number(fileInfo.size()));
          
                  QTableWidgetItem *itemLine=new QTableWidgetItem;
                  itemLine->setText(QString::number(lineAll));
          
                  QTableWidgetItem *itemCode=new QTableWidgetItem;
                  itemCode->setText(QString::number(lineCode));
          
                  QTableWidgetItem *itemNote=new QTableWidgetItem;
                  itemNote->setText(QString::number(lineNotes));
          
                  QTableWidgetItem *itemBlank=new QTableWidgetItem;
                  itemBlank->setText(QString::number(lineBlank));
          
                  QTableWidgetItem *itemPath=new QTableWidgetItem;
                  itemPath->setText(fileInfo.filePath());
          
                  itemSuffix->setTextAlignment(Qt::AlignCenter);
                  itemSize->setTextAlignment(Qt::AlignCenter);
                  itemLine->setTextAlignment(Qt::AlignCenter);
                  itemCode->setTextAlignment(Qt::AlignCenter);
                  itemNote->setTextAlignment(Qt::AlignCenter);
                  itemBlank->setTextAlignment(Qt::AlignCenter);
          
                  ui->tableWidget->setItem(i, 0, itemName);
                  ui->tableWidget->setItem(i, 1, itemSuffix);
                  ui->tableWidget->setItem(i, 2, itemSize);
                  ui->tableWidget->setItem(i, 3, itemLine);
                  ui->tableWidget->setItem(i, 4, itemCode);
                  ui->tableWidget->setItem(i, 5, itemNote);
                  ui->tableWidget->setItem(i, 6, itemBlank);
                  ui->tableWidget->setItem(i, 7, itemPath);
          
                  totalBytes  +=fileInfo.size();
                  totalLines  +=lineAll;
                  totalCodes  +=lineCode;
                  totalNotes  +=lineNotes;
                  totalBlanks +=lineBlank;
          
                  if (i % 100==0) {
                      qApp->processEvents();
                  }
              }
          
              //顯示統計結果
              listFile.clear();
              ui->txtCount->setText(QString::number(count));
              ui->txtSize->setText(QString::number(totalBytes));
              ui->txtRow->setText(QString::number(totalLines));
              ui->txtCode->setText(QString::number(totalCodes));
              ui->txtNote->setText(QString::number(totalNotes));
              ui->txtBlank->setText(QString::number(totalBlanks));
          
              //計算百分比
              double percent=0.0;
              //代碼行所占百分比
              percent=((double)totalCodes / totalLines) * 100;
              ui->labPercentCode->setText(QString("%1%").arg(percent, 5, 'f', 2, QChar(' ')));
              //注釋行所占百分比
              percent=((double)totalNotes / totalLines) * 100;
              ui->labPercentNote->setText(QString("%1%").arg(percent, 5, 'f', 2, QChar(' ')));
              //空行所占百分比
              percent=((double)totalBlanks / totalLines) * 100;
              ui->labPercentBlank->setText(QString("%1%").arg(percent, 5, 'f', 2, QChar(' ')));
          }
          
          void frmCountCode::countCode(const QString &fileName, int &lineCode, int &lineBlank, int &lineNotes)
          {
              lineCode=lineBlank=lineNotes=0;
              QFile file(fileName);
              if (file.open(QFile::ReadOnly)) {
                  QTextStream out(&file);
                  QString line;
                  bool isNote=false;
                  while (!out.atEnd()) {
                      line=out.readLine();
          
                      //移除前面的空行
                      if (line.startsWith(" ")) {
                          line.remove(" ");
                      }
          
                      //判斷當前行是否是注釋
                      if (line.startsWith("/*")) {
                          isNote=true;
                      }
          
                      //注釋部分
                      if (isNote) {
                          lineNotes++;
                      } else {
                          if (line.startsWith("//")) {    //注釋行
                              lineNotes++;
                          } else if (line.isEmpty()) {    //空白行
                              lineBlank++;
                          } else {                        //代碼行
                              lineCode++;
                          }
                      }
          
                      //注釋結束
                      if (line.endsWith("*/")) {
                          isNote=false;
                      }
                  }
              }
          }
          
          void frmCountCode::on_btnOpenFile_clicked()
          {
              QString filter=QString("代碼文件(%1)").arg(ui->txtFilter->text().trimmed());
              QStringList files=QFileDialog::getOpenFileNames(this, "選擇文件", "./", filter);
              if (files.size() > 0) {
                  ui->txtFile->setText(files.join("|"));
                  countCode(files);
              }
          }
          
          void frmCountCode::on_btnOpenPath_clicked()
          {
              QString path=QFileDialog::getExistingDirectory(this, "選擇目錄", "./",  QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks);
              if (!path.isEmpty()) {
                  ui->txtPath->setText(path);
                  listFile.clear();
                  countCode(path);
                  countCode(listFile);
              }
          }
          
          void frmCountCode::on_btnClear_clicked()
          {
              ui->txtCount->setText("0");
              ui->txtSize->setText("0");
              ui->txtRow->setText("0");
          
              ui->txtCode->setText("0");
              ui->txtNote->setText("0");
              ui->txtBlank->setText("0");
          
              ui->labPercentCode->setText("0%");
              ui->labPercentNote->setText("0%");
              ui->labPercentBlank->setText("0%");
              ui->tableWidget->setRowCount(0);
          }
          
          
          

          六、控件介紹

          1. 超過160個精美控件,涵蓋了各種儀表盤、進度條、進度球、指南針、曲線圖、標尺、溫度計、導航條、導航欄,flatui、高亮按鈕、滑動選擇器、農歷等。遠超qwt集成的控件數量。
          2. 每個類都可以獨立成一個單獨的控件,零耦合,每個控件一個頭文件和一個實現文件,不依賴其他文件,方便單個控件以源碼形式集成到項目中,較少代碼量。qwt的控件類環環相扣,高度耦合,想要使用其中一個控件,必須包含所有的代碼。
          3. 全部純Qt編寫,QWidget+QPainter繪制,支持Qt4.6到Qt5.13的任何Qt版本,支持mingw、msvc、gcc等編譯器,支持任意操作系統比如windows+linux+mac+嵌入式linux等,不亂碼,可直接集成到Qt Creator中,和自帶的控件一樣使用,大部分效果只要設置幾個屬性即可,極為方便。
          4. 每個控件都有一個對應的單獨的包含該控件源碼的DEMO,方便參考使用。同時還提供一個所有控件使用的集成的DEMO。
          5. 每個控件的源代碼都有詳細中文注釋,都按照統一設計規范編寫,方便學習自定義控件的編寫。
          6. 每個控件默認配色和demo對應的配色都非常精美。
          7. 超過130個可見控件,6個不可見控件。
          8. 部分控件提供多種樣式風格選擇,多種指示器樣式選擇。
          9. 所有控件自適應窗體拉伸變化。
          10. 集成自定義控件屬性設計器,支持拖曳設計,所見即所得,支持導入導出xml格式。
          11. 自帶activex控件demo,所有控件可以直接運行在ie瀏覽器中。
          12. 集成fontawesome圖形字體+阿里巴巴iconfont收藏的幾百個圖形字體,享受圖形字體帶來的樂趣。
          13. 所有控件最后生成一個動態庫文件(dll或者so等),可以直接集成到qtcreator中拖曳設計使用。
          14. 目前已經有qml版本,后期會考慮出pyqt版本,如果用戶需求量很大的話。
          15. 自定義控件插件開放動態庫使用(永久免費),無任何后門和限制,請放心使用。
          16. 目前已提供32個版本的dll,其中qt_5_7_0_mingw530_32這個版本會一直保證最新的完整的。
          17. 不定期增加控件和完善控件,不定期更新SDK,歡迎各位提出建議,謝謝!
          18. Qt入門書籍推薦霍亞飛的《Qt Creator快速入門》《Qt5編程入門》,Qt進階書籍推薦官方的《C++ GUI Qt4編程》。
          19. 強烈推薦程序員自我修養和規劃系列書《大話程序員》《程序員的成長課》《解憂程序員》,受益匪淺,受益終生!
          20. SDK地址:https://gitee.com/feiyangqingyun/QUCSDK https://github.com/feiyangqingyun/qucsdk

          原文鏈接:https://www.cnblogs.com/feiyangqingyun/p/11669523.html

          【領QT開發教程學習資料,點擊下方鏈接莬費領取↓↓,先碼住不迷路~】

          點擊這里:「鏈接」


          主站蜘蛛池模板: 国产精品va无码一区二区| 成人免费区一区二区三区| 蜜桃臀无码内射一区二区三区| 在线日韩麻豆一区| 一区二区三区在线|欧| 亲子乱av一区二区三区| 97人妻无码一区二区精品免费| 综合无码一区二区三区| 亚洲免费一区二区| 国产AV天堂无码一区二区三区| 亚洲综合一区二区三区四区五区| 少妇激情av一区二区| 久久精品一区二区三区不卡| 国精产品一区二区三区糖心| 精品少妇一区二区三区在线| 男人免费视频一区二区在线观看| 精品无人乱码一区二区三区 | 亚洲av不卡一区二区三区| 精品国产一区二区22| 国产伦精品一区二区三区视频猫咪| 日韩国产一区二区| 午夜精品一区二区三区在线视| 国产成人精品亚洲一区 | 中字幕一区二区三区乱码| 文中字幕一区二区三区视频播放| 国产精品熟女一区二区| 久久福利一区二区| 无码播放一区二区三区| 中文字幕AV一区二区三区| 久久久久久人妻一区二区三区| 久久久无码精品人妻一区| 免费观看日本污污ww网站一区| 波多野结衣中文一区| 精品日韩在线视频一区二区三区| 亚洲一区爱区精品无码| 亚洲国产美国国产综合一区二区 | 亚洲第一区精品观看| 国产精品污WWW一区二区三区| 国产午夜精品一区二区三区小说 | 在线观看一区二区三区视频| 麻豆一区二区三区蜜桃免费|