QRunnable and the QThreadPool

https://www.learnpyqt.com/courses/concurrent-execution/multithreading-pyqt-applications-qthreadpool/

using this method

from PyQt5.QtGui import *
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *

import time
import traceback, sys


class WorkerSignals(QObject):
'''
Defines the signals available from a running worker thread.

Supported signals are:

finished
No data

error
`tuple` (exctype, value, traceback.format_exc() )

result
`object` data returned from processing, anything

progress
`int` indicating % progress

'''
finished = pyqtSignal()
error = pyqtSignal(tuple)
result = pyqtSignal(object)
progress = pyqtSignal(int)


class Worker(QRunnable):
'''
Worker thread

Inherits from QRunnable to handler worker thread setup, signals and wrap-up.

:param callback: The function callback to run on this worker thread. Supplied args and
kwargs will be passed through to the runner.
:type callback: function
:param args: Arguments to pass to the callback function
:param kwargs: Keywords to pass to the callback function

'''

def __init__(self, fn, *args, **kwargs):
super(Worker, self).__init__()

# Store constructor arguments (re-used for processing)
self.fn = fn
self.args = args
self.kwargs = kwargs
self.signals = WorkerSignals()

# Add the callback to our kwargs
self.kwargs['progress_callback'] = self.signals.progress

@pyqtSlot()
def run(self):
'''
Initialise the runner function with passed args, kwargs.
'''

# Retrieve args/kwargs here; and fire processing using them
try:
result = self.fn(*self.args, **self.kwargs)
except:
traceback.print_exc()
exctype, value = sys.exc_info()[:2]
self.signals.error.emit((exctype, value, traceback.format_exc()))
else:
self.signals.result.emit(result) # Return the result of the processing
finally:
self.signals.finished.emit() # Done



class MainWindow(QMainWindow):


def __init__(self, *args, **kwargs):
super(MainWindow, self).__init__(*args, **kwargs)

self.counter = 0

layout = QVBoxLayout()

self.l = QLabel("Start")
b = QPushButton("DANGER!")
b.pressed.connect(self.oh_no)

layout.addWidget(self.l)
layout.addWidget(b)

w = QWidget()
w.setLayout(layout)

self.setCentralWidget(w)

self.show()

self.threadpool = QThreadPool()
print("Multithreading with maximum %d threads" % self.threadpool.maxThreadCount())

self.timer = QTimer()
self.timer.setInterval(1000)
self.timer.timeout.connect(self.recurring_timer)
self.timer.start()

def progress_fn(self, n):
print("%d%% done" % n)

def execute_this_fn(self, progress_callback):
for n in range(0, 5):
time.sleep(1)
progress_callback.emit(n*100/4)

return "Done."

def print_output(self, s):
print(s)

def thread_complete(self):
print("THREAD COMPLETE!")

def oh_no(self):
# Pass the function to execute
worker = Worker(self.execute_this_fn) # Any other args, kwargs are passed to the run function
worker.signals.result.connect(self.print_output)
worker.signals.finished.connect(self.thread_complete)
worker.signals.progress.connect(self.progress_fn)

# Execute
self.threadpool.start(worker)


def recurring_timer(self):
self.counter +=1
self.l.setText("Counter: %d" % self.counter)


app = QApplication([])
window = MainWindow()
app.exec_()

QThread

class GetStockCode(QThread):
signal_str = pyqtSignal(str)

def __init__(self):
super().__init__()

def run(self):
self.signal_str.emit(r'开始获取股票代码...')
if len(result.inserted_ids) != len(data_arr):
assert False
self.signal_str.emit(r'插入代码表数据失败.请检查...')
else:
self.signal_str.emit(r'插入代码表数据成功...')

def __init__(self, parent=None):
self.GetStockCode = GetStockCode()
self.ui.btn_update_stock_code.clicked.connect(self.on_btn_stock_code)

def on_btn_trade_date(self):
self.GetTradeDate.start()

移动光标

//VK_RIGHT
QTextCursor cursor = m_textEditIn->textCursor();//得到当前text的光标
cursor.movePosition(QTextCursor::Right,QTextCursor::MoveAnchor);
m_textEditIn->setTextCursor(cursor);

删除前一个字符

QTextCursor cursor=ui->textEdit->textCursor();//得到当前text的光标
if(cursor.hasSelection())//如果有选中,则取消,以免受受影响
cursor.clearSelection();
cursor.deletePreviousChar();//删除前一个字符
ui->textEdit->setTextCursor(cursor);//让光标移到删除后的位置

背景颜色

m_textEditIn->setStyleSheet("background:black");

获取文字

QString strTextIn = ui.textEditIn->toPlainText();

键值相关

其中\033 是ESC 的8进制表示形式

切换输入法

// 输入法变量
HKL g_hCurKL;

/*
0x0409的专业名称是Language Identifier,详见:
https://msdn.microsoft.com/en-us/library/dd318693(v=vs.85).aspx
*/
void disableIME()
{
LOGI((QString("disableIME enter:")).toStdString().c_str());
g_hCurKL = GetKeyboardLayout(0);
LoadKeyboardLayout((LPCWSTR)QString("0x0409").utf16(), KLF_ACTIVATE);
}

void enableIME()
{
LOGI((QString("enableIME enter:")).toStdString().c_str());
ActivateKeyboardLayout(g_hCurKL, 0);
}

特殊按键

web site

组合键

void MyWidget::keyPressEvent(QKeyEvent *event)
{
// 捕捉 Ctrl + Alt + A 组合键;
if (event->key() == Qt::Key_A && event->modifiers() == (Qt::AltModifier | Qt::ControlModifier))
{
// 开始截图;
startCatureImage();
}
...
}

键盘事件,输入法状态下

bool remoteCom::eventFilter(QObject *obj, QEvent *ev)
{
if (QEvent::KeyPress == ev->type())
{
//英文输入法
QKeyEvent *keyEvent = dynamic_cast<QKeyEvent *>(ev);
switch (keyEvent->key())
{
case Qt::Key_0:
case Qt::Key_Backspace:
break;
default:
break;
}
}else if(QEvent::InputMethod== ev->type())
{
// 中文或其他输入法
QInputMethodEvent *keyEvent = dynamic_cast<QInputMethodEvent *>(ev);
QString strInput = keyEvent->commitString();
//strInput 是键盘输入值
}
return true;
}

发送键盘按下消息

QKeyEvent tabKey(QEvent::KeyPress, Qt::Key_Right, Qt::NoModifier);
QCoreApplication::sendEvent(m_textEditIn, &tabKey);


/*
https://www.codeproject.com/Articles/7305/Keyboard-Events-Simulation-using-keybd-event-funct
将Qt::key的值转为windows bVk的值,即调用keybd_event需要的键盘值
keybd_event(key, 0x45, KEYEVENTF_EXTENDEDKEY, 0);//按键按下
keybd_event(key, 0x45, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, 0);//按键松开
*/
int translateKeyCode(int key)
{
int k = key;
bool legal = true;
if(k >= Qt::Key_0 && k <= Qt::Key_9)
{
}
else if(k >= Qt::Key_A && k <= Qt::Key_Z)
{
}
else if(k >= Qt::Key_F1 && k <= Qt::Key_F24)
{
k &= 0x000000ff;
k += 0x40;
}
else if(k == Qt::Key_Tab)
{
k = 0x09;
}
else if(k == Qt::Key_Backspace)
{
k = 0x08;
}
else if(k == Qt::Key_Return)
{
k = 0x0d;
}
else if(k <= Qt::Key_Down && k >= Qt::Key_Left)
{
int off = k - Qt::Key_Left;
k = 0x25 + off;
}
else if(k == Qt::Key_Shift)
{
k = 0x10;
}
else if(k == Qt::Key_Control)
{
k = 0x11;
}
else if(k == Qt::Key_Alt)
{
k = 0x12;
}
else if(k == Qt::Key_Meta)
{
k = 0x5b;
}
else if(k == Qt::Key_Insert)
{
k = 0x2d;
}
else if(k == Qt::Key_Delete)
{
k = 0x2e;
}
else if(k == Qt::Key_Home)
{
k = 0x24;
}
else if(k == Qt::Key_End)
{
k = 0x23;
}
else if(k == Qt::Key_PageUp)
{
k = 0x21;
}
else if(k == Qt::Key_Down)
{
k = 0x22;
}
else if(k == Qt::Key_CapsLock)
{
k = 0x14;
}
else if(k == Qt::Key_NumLock)
{
k = 0x90;
}
else if(k == Qt::Key_Space)
{
k = 0x20;
}
else
legal = false;

if(!legal)
return 0;
return k;
}

显示中文

QTextCodec::setCodecForTr(QTextCodec::codecForLocale());	

多语言支持

为了在使用多语言,必须做两件事情:
1,把想要翻译的字符串放入tr()中,如 QString str=tr(“Legal”);
2, 在开始的时候载入预先制作好的翻译文件(.qm)。
1、生成TS文件
QT的国际化需要TS文件,我们可以通过以下方法创建TS文件
QT 工具 create new translation file
2、编辑TS文件
用QT自带的linguist工具,对代码里需要翻译的内容进行翻译,完成后进行发布,一般会在TS当前目录下生成QM文件
在QRC文件加入QM文件:
双击QRC文件,按照以下配置进行配置,主要用于load()函数

update 翻译文件
在应用程序代码中修改后添加某些翻译字符串后,“.ts”文件不会自动更新,需要在相应的“.ts”文件的右键菜单中点击“lupdate”项,才会更新。
lupdate xx.cpp -ts xx.ts

int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QLocale myLocale;
QString locale = myLocale.name();

QTextCodec * textCodec = QTextCodec::codecForName("System");
QTextCodec::setCodecForLocale(textCodec);

g_pTranslator = new QTranslator(&a);
bool bRet = g_pTranslator->load("remotecom_en.qm");
if (!bRet)
{
qDebug(QString("can't load %1 translation").arg(locale).toLatin1().data());
}
a.installTranslator(g_pTranslator);
...
}

// 点击菜单切换语言
public: signals:
void languagechange(int nType);

void remoteCom::menu_chinese()
{
emit languagechange(1);
bool bRet = g_pTranslator->load(g_appPath + "\\language\\remotecom_zh.qm");
if (!bRet)
{
qDebug(QString("can't load remotecom_zh.qm translation").toLatin1().data());
}
qApp->installTranslator(g_pTranslator);
this->ui.retranslateUi(this);
}

void remoteCom::menu_english()
{
emit languagechange(0);
bool bRet = g_pTranslator->load(g_appPath + "\\language\\remotecom_en.qm");
if (!bRet)
{
qDebug(QString("can't load remotecom_en translation").toLatin1().data());
}
qApp->installTranslator(g_pTranslator);
this->ui.retranslateUi(this);

}

// 其他窗口响应语言改变信号
public slots:
void languageChange(int type);

void Config::languageChange( int type )
{
switch (type)
{
case 0:
{
QTranslator translator;
bool bRet = translator.load(g_appPath + "\\language\\remotecom_en.qm");
if (!bRet)
{
qDebug(QString("can't load remotecom_en translation").toLatin1().data());
}
qApp->installTranslator(&translator);
this->ui.retranslateUi(this);
}
break;
case 1:
{
QTranslator translator;
bool bRet = translator.load(g_appPath + "\\language\\remotecom_zh.qm");
if (!bRet)
{
qDebug(QString("can't load remotecom_en translation").toLatin1().data());
}
qApp->installTranslator(&translator);
this->ui.retranslateUi(this);
}
break;
default:
qDebug("Config::languageChange type undefined.");
break;
}
}

系统时间

QDateTime dateTime(QDateTime::currentDateTime());
QString qStr = dateTime.toString("yyyy-MM-dd hh:mm::ss ddd");

Timer

在Qt中使用定时器有两种方法,QTimer类的定时器和基于QObiect类的定时器。定时器的精确性依赖于操作系统和硬件(主要是操作系统),大多数平台支持20ms的精确度,实测Win10的精度基本就是20毫秒左右(硬件高配)。

阅读全文 »

路径是否存在

QString tmpPath = "D:\\zhengjun\\remoteCom\\Win32\\Debug";
QDir dir(tmpPath);
if (dir.exists())
{
QString fileTest = tmpPath + "\\QTTest.cfg";
if(QFile::exists(fileTest))
QFile::remove(fileTest);

QFile f(fileTest);
f.open(QIODevice::ReadWrite | QIODevice::Text);
QTextStream ft(&f);
ft<<param;
f.close();
}
阅读全文 »

Qt提供了qInstallMsgHandler 方法用来定制消息发生后如何来处理。
qInstallMsgHandler 是一个回调函数,主要是由qDebug、qWarnng、qCritical、qFatal这些函数进行触发,
也就是说,qDeubg这些函数处理的消息文本会被qInstallMsgHandler 所指向的回调函数截获,这样就允许用户自己来处理这些消息文本。
例如,你完全可以将这些消息文本输出并保存到相关的日志文件中。请看下面的示例!

qDebug : 调试信息提示
qWarning: 一般的警告提示
qCritical: 严重错误提示
qFatal: 致命错误提示

注意,release版本的日志中QMessageLogContext内容为空,解决方案请参考
在pro文件中添加一个定义:

DEFINES += QT_MESSAGELOGCONTEXT
阅读全文 »

string to qstring

qs = QString::fromStdString(s);

数字转字符串

int i = 42;
QString s = QString::number(i);

字符串转数字

QString Abcd = "123.5 Kb";
Abcd.split(" ")[0].toInt(); //convert the first part to Int
Abcd.split(" ")[0].toDouble(); //convert the first part to double
Abcd.split(" ")[0].toFloat(); //convert the first part to float

bool flag;
double v = Abcd.split(" ")[0].toDouble(&flag);
if(flag){
// use v
}

To char*

QString str1 = "Test";
QByteArray ba = str1.toLocal8Bit();
const char *c_str2 = ba.data();

QString转QByteArray


char convertCharToHex(char ch)
{
if((ch >= '0') && (ch <= '9'))
return ch-0x30;
else if((ch >= 'A') && (ch <= 'F'))
return ch-'A'+10;
else if((ch >= 'a') && (ch <= 'f'))
return ch-'a'+10;
else return (-1);
}

void convertStringToHex(const QString &str, QByteArray &byteData)
{
int hexdata,lowhexdata;
int hexdatalen = 0;
int len = str.length();
byteData.resize(len/2);
char lstr,hstr;
for(int i=0; i<len; )
{
//char lstr,
hstr=str[i].toLatin1();
if(hstr == ' ')
{
i++;
continue;
}
i++;
if(i >= len)
break;
lstr = str[i].toLatin1();
hexdata = convertCharToHex(hstr);
lowhexdata = convertCharToHex(lstr);
if((hexdata == 16) || (lowhexdata == 16))
break;
else
hexdata = hexdata*16+lowhexdata;
i++;
byteData[hexdatalen] = (char)hexdata;
hexdatalen++;
}
byteData.resize(hexdatalen);
}

QString testData="a503930000000200000008746ed16a156655a6";
QByteArray arr = m_socket->readAll();
arr = testData.toUtf8().toHex();
convertStringToHex(testData,arr);
char* p = (char*)arr.data();

string to qstring

qs = QString::fromStdString(s);

数字转字符串

int i = 42;
QString s = QString::number(i);

字符串转数字

QString Abcd = "123.5 Kb";
Abcd.split(" ")[0].toInt(); //convert the first part to Int
Abcd.split(" ")[0].toDouble(); //convert the first part to double
Abcd.split(" ")[0].toFloat(); //convert the first part to float

bool flag;
double v = Abcd.split(" ")[0].toDouble(&flag);
if(flag){
// use v
}

To char*

QString str1 = "Test";
QByteArray ba = str1.toLocal8Bit();
const char *c_str2 = ba.data();

QString转QByteArray


char convertCharToHex(char ch)
{
if((ch >= '0') && (ch <= '9'))
return ch-0x30;
else if((ch >= 'A') && (ch <= 'F'))
return ch-'A'+10;
else if((ch >= 'a') && (ch <= 'f'))
return ch-'a'+10;
else return (-1);
}

void convertStringToHex(const QString &str, QByteArray &byteData)
{
int hexdata,lowhexdata;
int hexdatalen = 0;
int len = str.length();
byteData.resize(len/2);
char lstr,hstr;
for(int i=0; i<len; )
{
//char lstr,
hstr=str[i].toLatin1();
if(hstr == ' ')
{
i++;
continue;
}
i++;
if(i >= len)
break;
lstr = str[i].toLatin1();
hexdata = convertCharToHex(hstr);
lowhexdata = convertCharToHex(lstr);
if((hexdata == 16) || (lowhexdata == 16))
break;
else
hexdata = hexdata*16+lowhexdata;
i++;
byteData[hexdatalen] = (char)hexdata;
hexdatalen++;
}
byteData.resize(hexdatalen);
}

QString testData="a503930000000200000008746ed16a156655a6";
QByteArray arr = m_socket->readAll();
arr = testData.toUtf8().toHex();
convertStringToHex(testData,arr);
char* p = (char*)arr.data();

char* to QByteArray

databuf = QByteArray((char*)buf, 10);