過本文章,可以掌握以下內容:
對于一個Python小白來說,在學習Pyside6的控件時,照著網上的一些示例,可以將很快將控件運行出來。但在看控件python源代碼時,卻很是迷惑,本文就以QWidget為例,進行展開解讀。
QWidget python源代碼
如上圖是QWidget部分Python源代碼,你能看懂多少?
一、Pyside6類的屬性與方法
要了解控件大部分方法為什么都包含self參數,只需要了解Python類、實例的屬性和方法即可。
類的對象,就是類的一個實例。類的實例屬性被對象所有,包含在每個對象之中,不同的對象之間,互不干擾。類的類屬性被類所有,被包含在類中,是所有的類對象共享。一般情況下,實例屬性會在__init__ 方法中聲明并初始化,并且使用self 來綁定。而類屬性是在類作用域中被聲明,并且不使用self 來綁定。
python類、實例與屬性關系
Python 類中有三種方法:
注意:Python 解釋器在構造類與對象時,類是先于對象產生的。因此,類屬性與類方法是先于實例屬性與實例方法產生的。所以當類方法產生時,還沒有實例屬性,因此,類方法中不能訪問實例屬性。
以雙下劃線__開頭且結尾的方法__xxx__,就是專有方法,也稱魔法方法。這些方法都被Python 賦予了特殊的含義,用戶可以根據需要,來實現這些方法。比如QWidget類重寫了Python object的方法:
二、Pyside6的調用過程
在開發Pyside6的時候,很多源代碼只包含方法名、pass 關鍵字和空的枚舉類,那pyside6的代碼在實例化以及調用方法時,整個過程是怎樣的呢。
在 PySide6 中,Python和Qt的C++庫之間的關聯是通過綁定技術實現的。具體來說,PySide6使用Shiboken,這是一種綁定生成工具,它自動將C++類和函數映射到Python。通過這種方式,PySide6可以在Python中使用Qt的功能。關鍵技術概念:
在 PySide6中,staticMetaObject是一個靜態屬性,存在于所有繼承自QObject的類中。它的主要作用是提供對Qt元對象系統(Meta-Object System)的訪問。這個元對象系統在 Qt 中用于實現信號與槽機制、屬性系統以及其他反射(reflection)功能。
Qt 的元對象系統是一種運行時類型信息系統,提供以下功能:
staticMetaObject 是一個包含類的元數據的對象,提供了對以下內容的訪問:
staticMetaObject在PySide6中提供了對Qt元對象系統的訪問,允許在運行時查詢類的元數據。它是實現動態屬性訪問、信號與槽機制以及反射功能的基礎,極大地增強了代碼的靈活性和動態特性。
三、正確使用Pyside6控件類
如何正確使用Pyside6的控件,我們需要了解Qt for Python相關文檔
訪問官網: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動畫序列中,按照順序執行。
#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();
}
原文鏈接: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
#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);
}
原文鏈接:https://www.cnblogs.com/feiyangqingyun/p/11669523.html
【領QT開發教程學習資料,點擊下方鏈接莬費領取↓↓,先碼住不迷路~】
點擊這里:「鏈接」
*請認真填寫需求信息,我們會在24小時內與您取得聯系。