1

最近做项目中遇到需要采集产品的条形码和二维码进行保存显示,并上传服务器或企业ERP系统,于是针对无线扫码枪和二维码扫码枪做了数据采集测试。
扫码枪本身是一个输入设备,类似键盘输入。条码都是ASCII字符,最后一个字符为回车键,一般扫码枪连接电脑后,光标焦点放在编辑框,扫码后数据就会自动输入到编辑框,本文主要讲解如何在没有编辑框的时候,通过拦截WM_CHAR消息获取输入条码的具体数据。


1. 软硬件配置

  • 科密无线扫码枪(条形码),型号:YX-38
  • 软件环境 Qt5.9.6

2. 解决思路

1.一般扫码枪是通过串口进行采集,其优点是可实现无焦点采集,可循环存到一个固定的内存地址,缺点是串口必须接一根串口线,用户体验不太好;
2.Qt采用键盘事件来判别条形码输入的具体内容,因此可以通过监听虚拟键盘的输入来判定扫码枪的输入,从而实现无焦点采集。
扫码枪设置.jpg
上图是扫码枪设置的图,可以设置不同的工作模式,在扫码枪超出无线传输的距离范围后,可以使用盘点模式,这样扫码数据就保存到扫码枪,盘点完成后再扫数据上传,这样就可以实现离线的货物盘点了,这一点还是比较人性化的,不过这些工作模式我这里没有一一测试。

3、具体代码实现

3.1 Qt自带的键盘事件QEvent::KeyPress

private slots:
    //申明KeyPress函数
    void keyPressEvent(QKeyEvent* event);

以上是头文件的函数申明,其具体实现代码如下:

void Widget::keyPressEvent(QKeyEvent* event)
{
    switch(event->key())
        {
            case Qt::Key_Down:
            {
                qDebug()<<"ok";
            }
                break;
            case Qt::Key_F1:
            {
                qDebug()<<"ssss";
            }
        }
    if (0x30 <= event->key() && 0x39 >= event->key())
       {
        char st=event->key();
        QString str_ascii = QString(st);
        qDebug()<<str_ascii;
        qDebug()<<event->key();
       }
     else if (0x41 <= event->key() && 0x48 >= event->key())
       {
       qDebug()<<event->key();
       }else
       {
       qDebug()<<event->key();
       }
}

键盘事件中的按键对应值如下表所示:
键盘对应键值.png

3.2 自己写全局的键盘监测函数

使用notify()函数来实现键盘的监控

bool GlobalApplication::notify(QObject *obj, QEvent *e)
{
    const QMetaObject* objMeta = obj->metaObject();
    QString clName = objMeta->className();
//    qDebug()<<clName;

//if(clName == "Form")
//    {
    if(e->type() == QEvent::KeyPress)
        {
            QKeyEvent *keyEvent = static_cast<QKeyEvent *>(e);
            if(keyEvent->key() == 16777220)
            {
                qDebug()<<strinput;
                qDebug() << clName;
                strinput.clear();
            }
            else
            {
                char st=keyEvent->key();
                QString str_ascii = QString(st);
                strinput +=str_ascii;
 //               qDebug()<<strinput;
            }
            return true;
         }
//}

//if(clName == "QWidgetWindow")
//{
//    if(e->type() == QEvent::KeyPress)
//    {
//        QKeyEvent *keyEvent = static_cast<QKeyEvent *>(e);
//        if(keyEvent->key() == Qt::Key_F1)
//        {
//            qDebug() << clName;
//            qDebug() << "F1";
//        }
// //     widget->keyPress(keyEvent);
//     }
//    else if(e->type() == QEvent::MouseButtonPress)
//    {
//        QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(e);
//        if(mouseEvent->buttons() == Qt::LeftButton)
//        qDebug() << "left";
//    }
//}

    return QApplication::notify(obj,e);
}

这个函数也可用作鼠标的监控,其中注释的部分涉及到鼠标的监控。全局监控需要在main.cpp文件里,设定为全局类GlobalApplication

int main(int argc, char *argv[])
{
    GlobalApplication a(argc, argv);
 //   QApplication a(argc, argv);
    Widget w;
    w.show();

    return a.exec();
}

3.3 附上全部代码及测试效果

globalapplication.h
#ifndef GLOBALAPPLICATION_H
#define GLOBALAPPLICATION_H
#include<QApplication>

class GlobalApplication : public QApplication
{
public:
    GlobalApplication(int&argc,char **argv);
    ~GlobalApplication();

    bool notify(QObject*,QEvent *);
    void setWindowInstance(QWidget *wnd);
private:
    QWidget *widget;
       QString strinput;
};

#endif // GLOBALAPPLICATION_H
globalapplication.cpp
#include "globalapplication.h"
#include <QApplication>
#include <QKeyEvent>
#include <QDebug>

GlobalApplication::GlobalApplication(int &argc,char **argv):
QApplication(argc,argv)
{

}

GlobalApplication::~GlobalApplication()
{

}

void GlobalApplication::setWindowInstance(QWidget *wnd)
{
    widget = wnd;
}

bool GlobalApplication::notify(QObject *obj, QEvent *e)
{
    const QMetaObject* objMeta = obj->metaObject();
    QString clName = objMeta->className();
//    qDebug()<<clName;

//if(clName == "Form")
//    {
    if(e->type() == QEvent::KeyPress)
        {
            QKeyEvent *keyEvent = static_cast<QKeyEvent *>(e);
            if(keyEvent->key() == 16777220)
            {
                qDebug()<<strinput;
                qDebug() << clName;
                strinput.clear();
            }
            else
            {
                char st=keyEvent->key();
                QString str_ascii = QString(st);
                strinput +=str_ascii;
 //               qDebug()<<strinput;
            }
            return true;

         }

//}

//if(clName == "QWidgetWindow")
//{
//    if(e->type() == QEvent::KeyPress)
//    {
//        QKeyEvent *keyEvent = static_cast<QKeyEvent *>(e);
//        if(keyEvent->key() == Qt::Key_F1)
//        {
//            qDebug() << clName;
//            qDebug() << "F1";
//        }
// //     widget->keyPress(keyEvent);
//     }
//    else if(e->type() == QEvent::MouseButtonPress)
//    {
//        QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(e);
//        if(mouseEvent->buttons() == Qt::LeftButton)
//        qDebug() << "left";
//    }
//}


    return QApplication::notify(obj,e);

}
widget.h
#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include "dialog.h"
#include "form.h"

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();

private:
    Ui::Widget *ui;
    Dialog *p_dialog;
    Form *p_form;

private slots:

    void on_pushButton_Load_clicked();
    void on_pushButton_test_clicked();
    void keyPressEvent(QKeyEvent* event);
};

#endif // WIDGET_H
widget.cpp
#include "widget.h"
#include "ui_widget.h"
#include "dialog.h"
#include <QDebug>
#include<QKeyEvent>

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
    p_dialog = NULL;
    p_form = NULL;
}

Widget::~Widget()
{
    delete ui;
}

void Widget::on_pushButton_Load_clicked()
{
    if(p_dialog == NULL)
    {
        p_dialog =new Dialog();
        p_dialog->setWindowTitle(QString::fromLocal8Bit("虚拟键盘测试"));
        p_dialog->show();
        qDebug()<<"Test"<<endl;
    }
    else
    {
        p_dialog->show();

    }
}

void Widget::on_pushButton_test_clicked()
{
    if(p_form == NULL)
    {
        p_form =new Form();
        p_form->setWindowTitle(QString::fromLocal8Bit("测试"));
        p_form->show();
//        qDebug()<<"Test"<<endl;
    }
    else
    {
        p_form->show();

    }
}

void Widget::keyPressEvent(QKeyEvent* event)
{
    switch(event->key())
        {
            case Qt::Key_Down:
            {
                qDebug()<<"ok";
            }
                break;
            case Qt::Key_F1:
            {
                qDebug()<<"ssss";
            }
        }


//    if (0x30 <= event->key() && 0x39 >= event->key())
//       {
//        char st=event->key();
//        QString str_ascii = QString(st);
//        qDebug()<<str_ascii;

//           qDebug()<<event->key();
//       }
//       else if (0x41 <= event->key() && 0x48 >= event->key())
//       {

//       qDebug()<<event->key();
//       }
//       else
//       {

//       qDebug()<<event->key();
//       }

}

不同窗口下键盘输入结果.png

键盘输入监控.png

扫码枪扫码结果输出.png

3.4 代码仓库

码云仓库地址
下一节内容 如何区分键盘输入还是扫码枪输入。


coder_Alaric
9 声望7 粉丝