Python读写文本URL蓝牙WIFI自动连接电子名片位置坐标智能海报等NDEF标签

本示例使用的发卡器:https://item.taobao.com/item.htm?id=615391857885&spm=a1z10.5-c.w4002-21818769070.11.60ad789erlonvk         
        近场通信(Near Field Communication,简称NFC),是一种新兴的技术,使用了NFC技术的设备(例如移动电话)可以在彼此靠近的情况下进行数据交换,是由非接触式射频识别(RFID)及互连互通技术整合演变而来的,通过在单一芯片上集成感应式读卡器、感应式卡片和点对点通信的功能,利用移动终端实现移动支付、电子票务、门禁、移动身份识别、防伪等应用。近场通信技术是由诺基亚(Nokia)、飞利浦(Philips)和索尼(Sony)共同制定的标准 ,在ISO 18092、ECMA 340和ETSI TS 102 190框架下推动标准化,同时也兼容应用广泛的ISO 14443 、Type-A、ISO 15693、B以及Felica标准非接触式智能卡 的基础架构。

       NDEF 全称 NFC data exchange format 即 nfc 数据交换格式,是一种标准化的数据格式,可用于在任何兼容的NFC设备与另一个NFC设备或标签之间交换信息。数据格式由NDEF消息和NDEF记录组成 。

from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtWidgets import QApplication, QWidget, QDesktopWidget,QMessageBox
from PyQt5.QtGui import QFont
from NDEFTagTools import Ui_Frame
import sys
import struct  # struct的pack函数把任意数据类型变成字符串
import ctypes  # 调用DLL动态库要有这个引用

#'为防止测试时忘记卡片密码,本系统统一使用密码为 197402020111 ,MifareClassic的密码6字节 ,forumtype2标签的密码是前4个字节,密码为 19740202
oldpicckey = bytes.fromhex('197402020111')
newpicckey = bytes.fromhex('197402020111')

class MainWindow(QtWidgets.QMainWindow, Ui_Frame):
    def __init__(self, parent=None):
        super(MainWindow, self).__init__(parent)
        self.setupUi(self)

        self.pushButton_clearinf.clicked.connect(self.pushButton_clearinf_clicked)
        self.pushButton_beep.clicked.connect(self.pushButton_beep_clicked)
        self.pushButton_getdevnum.clicked.connect(self.pushButton_getdevnum_clicked)
        self.pushButton_clearbuff.clicked.connect(self.pushButton_clearbuff_clicked)
        self.pushButton_writetag.clicked.connect(self.pushButton_writetag_clicked)
        self.comboBox_seleapp.currentTextChanged.connect(self.comboBox_seleapp_currentindexchanged)
        self.pushButton_text.clicked.connect(self.pushButton_text_clicked)
        self.pushButton_App.clicked.connect(self.pushButton_App_clicked)
        self.pushButton_Url.clicked.connect(self.pushButton_Url_clicked)
        self.pushButton_Map.clicked.connect(self.pushButton_Map_clicked)
        self.pushButton_Bluetooth.clicked.connect(self.pushButton_Bluetooth_clicked)
        self.pushButton_Wifi.clicked.connect(self.pushButton_Wifi_clicked)
        self.pushButton_Card.clicked.connect(self.pushButton_Card_clicked)
        self.pushButton_Data.clicked.connect(self.pushButton_Data_clicked)
        self.pushButton_Call.clicked.connect(self.pushButton_Call_clicked)
        self.pushButton_readtag.clicked.connect(self.pushButton_readtag_clicked)
        self.pushButton_cleatag.clicked.connect(self.pushButton_cleatag_clicked)
        self.checkBox_later.toggled.connect(self.checkBox_later_toggled)


    def pushButton_clearinf_clicked(self):
        self.listWidget.clear()
        self.plainTextEdit_taginf.setPlainText('')

    def checkBox_later_toggled(self):
        if self.checkBox_later.isChecked():
            answ=QMessageBox.warning(self,'警告','      ISO15693协议标签锁定保护后就不能再解锁修改!请问确定要开启锁定保护功能吗?',QMessageBox.Yes|QMessageBox.No)
            if answ==QMessageBox.No:
                self.checkBox_later.setChecked(False)


    def ListErrInf(self,Errcode):
        if (Errcode == 8):
            self.ListBottom('错误代码:8,未寻到卡,请重新拿开卡后再放到感应区!')
        elif (Errcode == 1):
            self.ListBottom('错误代码:1,0~2块都没读出来,可能刷卡太块。但卡序列号已被读出来!')
        elif (Errcode == 2):
            self.ListBottom('错误代码:2,第0块已被读出,但1~2块读取失败。卡序列号已被读出来!')
        elif (Errcode == 3):
            self.ListBottom('错误代码:3,第0、1块已被读出,但2块读取失败。卡序列号已被读出来!')
        elif (Errcode == 9):
            self.ListBottom('错误代码:9,有多张卡在感应区,寻卡过程中防冲突失败,读序列吗错误!')
        elif (Errcode == 10):
            self.ListBottom('错误代码:10,该卡可能已被休眠,无法选中卡片!')
        elif (Errcode == 11):
            self.ListBottom('错误代码:11,密码装载失败!')
        elif (Errcode == 12):
            self.ListBottom('错误代码:12,标签密码认证错误!')
        elif (Errcode == 13):
            self.ListBottom('错误代码:13,读指定块失败,原因是刷卡太快或本块所对应的区还没通过密码认证!')
        elif (Errcode == 14):
            self.ListBottom('错误代码:14,写标签失败,可能需要验证密码!')
        elif (Errcode == 21):
            self.ListBottom('错误代码:21,没有动态库!')
        elif (Errcode == 22):
            self.ListBottom('错误代码:22,动态库或驱动程序异常!')
        elif (Errcode == 23):
            self.ListBottom('错误代码:23,(表示未检测到免驱动读写器!)(有驱动读写器表示驱动程序错误或未检测到有驱读写器!)')
        elif (Errcode == 24):
            self.ListBottom('错误代码:24,操作超时,一般是动态库没有反映!')
        elif (Errcode == 25):
            self.ListBottom('错误代码:25,发送字数不够!')
        elif (Errcode == 26):
            self.ListBottom('错误代码:26,发送的CRC错!')
        elif (Errcode == 27):
            self.ListBottom('错误代码:27,接收的字数不够!')
        elif (Errcode == 28):
            self.ListBottom('错误代码:28,接收的CRC错!')
        elif (Errcode == 45):
            self.ListBottom('错误代码:45,此卡不支持更改UID号或UID块已被锁定!')
        elif (Errcode == 46):
            self.ListBottom('错误代码:46,标签存储空间不足!')
        elif (Errcode == 254):
            self.ListBottom('错误代码:254,标签存储空间不足!')
        elif (Errcode == 255):
            self.ListBottom('错误代码:255,暂不支持长度大于255字节的信息写入!')
        else:
            self.ListBottom('未知错误,错误代码:' + str(Errcode))

    def comboBox_seleapp_currentindexchanged(self):
        packstr=self.comboBox_seleapp.currentText().strip()
        FieldsList = packstr.split('|')
        self.lineEdit_appname.setText(FieldsList[1])

    def ListBottom(self,dispinf):
        if self.listWidget.count()>50:
            self.listWidget.clear()
        self.listWidget.addItem(dispinf)
        self.listWidget.scrollToBottom()
        seleid = self.listWidget.count() - 1
        self.listWidget.item(seleid).setSelected(True)

    def pushButton_beep_clicked(self):
        status = Objdll.pcdbeep(50) % 256
        if status == 0:
            self.ListBottom("读写器已执行响声指令!")
        else:
            self.ListErrInf(status)

    def pushButton_getdevnum_clicked(self):
        devno = bytes(4)  # 声明4个字节缓冲
        status = Objdll.pcdgetdevicenumber(devno) % 256
        if (status == 0):
            Objdll.pcdbeep(38)
            SerialNum = ''
            for num in range(0, len(devno)):
                SerialNum = SerialNum + '%03d' % (devno[num])
                if (num < len(devno) - 1):
                    SerialNum = SerialNum + '-'
            self.ListBottom('读取设备编号成功,设备编号:' + SerialNum)
        else:
            self.ListErrInf(status)

    def pushButton_clearbuff_clicked(self):
        Objdll.tagbuf_forumtype4_clear()
        Objdll.tagbuf_clear()
        self.ListBottom('NDEF写卡缓冲区数据清除成功!' )

    def checkcardtype(self):        #检测NFC标签类型
        myctrlword=0
        devno = bytes(4)        # 声明4个字节缓冲
        mypiccserial= bytes(8)  # 声明8个字节缓冲
        mypicckey=bytes(6)      # 声明6个字节缓冲
        mypicdata=bytes(48)
        mypiccseriallen=bytes(2)

        try:
            status = Objdll.pcdgetdevicenumber(devno) % 256
            if (status == 0):
                status = Objdll.piccreadex_ntag(myctrlword, mypiccserial, mypicckey, 4, 1, mypicdata) % 256
                if (status == 0):
                    return 1        #forumtype2 Ntag2标签
                else:
                    status = Objdll.iso15693readex(myctrlword,0,1,1, mypiccserial,mypicdata) % 256
                    if (status == 0):
                        return 2    #forumtype5 15693标签
                    else:
                        myctrlword = 23
                        mypicckey=bytes([255,255,255,255,255,255])
                        status = Objdll.piccreadex(myctrlword,mypiccserial,0,1,mypicckey,mypicdata) % 256
                        if (status == 0):
                            return 3    #MifareClissic标签,出厂全新状态
                        else:
                            mypicckey = bytes([160, 161, 162, 163, 164, 165])
                            status = Objdll.piccreadex(myctrlword, mypiccserial, 0, 1, mypicckey, mypicdata) % 256
                            if (status == 0):
                                return 3        #MifareClissic标签,已经写有NDEF数据
                            else:
                                myctrlword = 0
                                status = Objdll.forumtype4request(myctrlword, mypiccserial, mypiccseriallen) % 256
                                if (status == 0 or status == 52):
                                    return 4    #forumtype4 标签
                                else:
                                    return -1   #其他未知型号标签
            else:
                return 23   #未连接到发卡器

        except:
            return 23

    def WriteTag(self,funcname):     #将NDEF缓冲区数据写入各种不同类型的NFC标签,清除标签内NDEF数据
        mypiccserial = bytes(8)
        mypiccseriallen=bytes(1)

        try:
            cardtyep = self.checkcardtype()  #判断发卡器上的标签类型
            if(cardtyep==1):        #写forumtype2 Ntag2标签
                if self.checkBox_before.isChecked():
                    myctrlword = 16
                else:
                    myctrlword = 0
                status = Objdll.forumtype2_write_ndeftag(myctrlword, mypiccserial, oldpicckey) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'ForumType2,UID:'
                    for num in range(0, 7):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('NDEF缓冲区数据写入标签成功!标签类型:'+cardno)
                    if (self.checkBox_before.isChecked() and not self.checkBox_later.isChecked()) or (not self.checkBox_before.isChecked() and self.checkBox_later.isChecked() or (self.checkBox_before.isChecked() and self.checkBox_later.isChecked())):
                        self.NtagkeyEn()
                else:
                    self.ListErrInf(status)

            elif(cardtyep==2):       #写forumtype5 15693标签
                myctrlword = 0
                afi=0
                status = Objdll.forumtype5_write_ndeftag(myctrlword,afi, mypiccserial) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'ForumType5,UID:'
                    for num in range(0, 8):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    if self.checkBox_later.isChecked():    #15693卡锁定块数据后只能读取不可再修改,为防止卡片锁死,请谨慎锁定
                        Objdll.iso15693lockblock(0,1,mypiccserial)
                    self.ListBottom('NDEF缓冲区数据写入标签成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif (cardtyep == 3):  # 写MifareClassic标签
                if(funcname=='NDEF_Clear'):
                    if self.checkBox_before.isChecked():
                        myctrlword = 210
                    else:
                        myctrlword = 146
                    status = Objdll.piccclear_ndeftag(myctrlword, mypiccserial, oldpicckey) % 256
                else:
                    if self.checkBox_before.isChecked():
                        myctrlword = 210
                    else:
                        myctrlword = 146
                    if self.checkBox_later.isChecked():  #写标签后给标签加密保护
                        myctrlword=myctrlword+4
                    status = Objdll.piccwrite_ndeftag(myctrlword, mypiccserial, oldpicckey,newpicckey) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'MifareClass,UID:'
                    for num in range(0, 4):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('NDEF缓冲区数据写入标签成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif(cardtyep==4):       #写forumtype4 标签
                if self.checkBox_before.isChecked():
                    myctrlword = 64
                else:
                    myctrlword = 0
                status = Objdll.forumtype4_write_ndeftag(myctrlword, mypiccserial,mypiccseriallen,newpicckey) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'ForumType4,UID:'
                    for num in range(0, mypiccseriallen[0]):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('NDEF缓冲区数据写入标签成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif (cardtyep == 23):
                self.ListErrInf(23)
            else:
                self.ListErrInf(8)

        except:
            self.ListErrInf(8)

    def ReadTag(self,funcname):      # 读取各种不同类型的NFC标签的NDEF数据
        mypiccserial = bytes(8)
        mypiccseriallen = bytes(1)
        mypiccdata = bytes(2048)
        revstrlen = bytes(2)
        recordnumber = bytes(2)

        try:
            cardtyep = self.checkcardtype()  #判断发卡器上的标签类型
            if (cardtyep == 1):         # 读forumtype2 Ntag2标签
                if self.checkBox_before.isChecked():
                    myctrlword = 16
                else:
                    myctrlword = 0
                status = Objdll.forumtype2_read_ndeftag(myctrlword, mypiccserial, oldpicckey) % 256
                if(status==0):
                    Objdll.pcdbeep(38)
                    cardno = 'ForumType2,UID:'
                    for num in range(0, 7):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('读取NDEF标签数据成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif(cardtyep==2):       #读forumtype5 15693标签
                myctrlword = 0
                afi=0
                status = Objdll.forumtype5_read_ndeftag(myctrlword,afi, mypiccserial) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'ForumType5,'
                    for num in range(0, 8):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('读取NDEF标签数据成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif (cardtyep == 3):  # 读MifareClassic标签
                if self.checkBox_before.isChecked():
                    myctrlword = 208
                else:
                    myctrlword = 144
                status = Objdll.piccread_ndeftag(myctrlword, mypiccserial, oldpicckey) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'MifareClass,'
                    for num in range(0, 4):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('读取NDEF标签数据成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif(cardtyep==4):       #读forumtype4 标签
                if self.checkBox_before.isChecked():
                    myctrlword = 64
                else:
                    myctrlword = 0
                status = Objdll.forumtype4_read_ndeftag(myctrlword, mypiccserial,mypiccseriallen,oldpicckey) % 256
                if (status == 0):
                    Objdll.pcdbeep(38)
                    cardno = 'ForumType4,'
                    for num in range(0, mypiccseriallen[0]):
                        cardno = cardno + '%02X' % (mypiccserial[num])
                    self.ListBottom('读取NDEF标签数据成功!标签类型:' + cardno)
                else:
                    self.ListErrInf(status)

            elif (cardtyep == 23):
                self.ListErrInf(23)
            else:
                self.ListErrInf(8)

            if (status == 0):
                Objdll.tagbuf_read(mypiccdata,revstrlen,recordnumber)
                strlen=revstrlen[0]+revstrlen[1]*256
                strinf=mypiccdata[0:strlen]
                ndefstr=strinf.decode('gbk')
                self.plainTextEdit_taginf.setPlainText(ndefstr)

        except:
            self.ListErrInf(8)

    def NtagkeyEn(self):    #开启或关闭Ntag标签的密钥保护功能
        mypiccserial=bytes(8)

        if self.checkBox_before.isChecked():    #操作已经加密保护的ntag2x卡
            myctrlword=16
        else:
            myctrlword=0

        if self.checkBox_later.isChecked(): #需要开启Ntag卡的加密保护功能
            mypiccdata=bytes.fromhex('0000000400000000')
            mypiccdata=mypiccdata + newpicckey[0:4]
            mypiccdata = mypiccdata +bytes.fromhex('16160000')
            myctrlword=myctrlword+7
            status = Objdll.piccinit_ntag(myctrlword, mypiccserial, oldpicckey, mypiccdata)
            if status == 0:
                cardno = 'ForumType2,UID:'
                for num in range(0, 7):
                    cardno = cardno + '%02X' % (mypiccserial[num])
                self.ListBottom('NDEF标签加密锁定保护成功!标签类型:' + cardno)
            else:
                self.ListBottom('NDEF标签加密锁定保护失败!错误代码%d:' %(status))
        else:
            mypiccdata = bytes.fromhex('000000FF000000000000000000000000')      #关闭Ntag卡的密钥保护功能
            myctrlword = myctrlword + 3
            status=Objdll.piccinit_ntag(myctrlword,mypiccserial,oldpicckey,mypiccdata)
            if status == 0:
                cardno = 'ForumType2,UID:'
                for num in range(0, 7):
                    cardno = cardno + '%02X' % (mypiccserial[num])
                self.ListBottom('NDEF标签解密去除锁定保护成功!标签类型:' + cardno)
            else:
                self.ListBottom('NDEF标签解密去除锁定保护失败!错误代码%d:' %(status))

    def pushButton_writetag_clicked(self):
        self.WriteTag('NDEF缓冲区数据')

    def pushButton_text_clicked(self):
        languagecodestr = "en".encode('gbk')
        textstr=self.lineEdit_text.text().strip().encode('gbk')
        if len(textstr)<1:
            QMessageBox.critical(self,'提示','请输入要加入写卡缓冲区的文本信息!',QMessageBox.Yes)
            self.lineEdit_text.setFocus()
            return
        else:
            status = Objdll.tagbuf_addtext(languagecodestr, len(languagecodestr), textstr, len(textstr)) % 256
            if status == 0:
                self.ListBottom("文本信息加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
            else:
                self.ListBottom("文本信息加入NDEF写卡缓冲区失败!")

    def pushButton_App_clicked(self):
        packagestr = self.lineEdit_appname.text().strip().encode('gbk')
        if len(packagestr)<1:
            QMessageBox.critical(self,'提示','请输入要加入写卡缓冲区的自启动APP程序包名!',QMessageBox.Yes)
            self.lineEdit_appname.setFocus()
            return
        else:
            status = Objdll.tagbuf_addapp(packagestr, len(packagestr)) % 256
            if status == 0:
                self.ListBottom("自启动APP名称加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
            else:
                self.ListBottom("自启动APP名称加入NDEF写卡缓冲区失败!")

    def pushButton_Url_clicked(self):
        languagecodestr = "en".encode('gbk')
        titlestr=self.lineEdit_title.text().strip().encode('gbk')
        uriheaderindex = self.comboBox_headindex.currentIndex()
        uristr=self.lineEdit_uri.text().strip().encode('gbk')
        if len(uristr)<1:
            QMessageBox.critical(self,'提示','请输入要加入写卡缓冲区的网址URI!',QMessageBox.Yes)
            self.lineEdit_uri.setFocus()
            return
        else:
            status = Objdll.tagbuf_adduri(languagecodestr, len(languagecodestr), titlestr, len(titlestr), uriheaderindex,uristr, len(uristr)) % 256
            if status == 0:
                self.ListBottom("网址URL加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
            else:
                self.ListBottom("网址URL加入NDEF写卡缓冲区失败!")

    def pushButton_Map_clicked(self):
        languagecodestr = "en".encode('gbk')
        titlestr=self.lineEdit_mapname.text().strip().encode('gbk')
        latitude=self.lineEdit_latitude.text().strip()
        longitude=self.lineEdit_longitude.text().strip()
        mapcoord = ("geo:" + latitude + "," + longitude).encode('gbk')
        status = Objdll.tagbuf_adduri(languagecodestr, len(languagecodestr), titlestr, len(titlestr), 0, mapcoord,len(mapcoord)) % 256
        if status == 0:
            self.ListBottom("地图坐标加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
        else:
            self.ListBottom("地图坐标加入NDEF写卡缓冲区失败!")

    def pushButton_Bluetooth_clicked(self):
        try:
            macname = self.lineEdit_Bluetoothname.text().strip() .encode('gbk')
            maclist = self.lineEdit_Bluetoothmac.text().strip().split(':')
            if (len(maclist) == 6):
                macbuf = bytes(
                    [eval('0x' + maclist[0]), eval('0x' + maclist[1]), eval('0x' + maclist[2]), eval('0x' + maclist[3]),
                     eval('0x' + maclist[4]), eval('0x' + maclist[5])])
                status = Objdll.tagbuf_addbluetooth(macname, len(macname), macbuf) % 256
                if status == 0:
                    self.ListBottom("蓝牙连接信息加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
                else:
                    self.ListBottom("蓝牙连接信息加入NDEF写卡缓冲区失败!")
            else:
                QMessageBox.critical(self, '提示', '蓝牙MAC地址输入错误!', QMessageBox.Yes)
                self.lineEdit_Bluetoothmac.setFocus()
        except:
            QMessageBox.critical(self, '提示', '蓝牙MAC地址输入错误!', QMessageBox.Yes)
            self.lineEdit_Bluetoothmac.setFocus()

    def pushButton_Wifi_clicked(self):
        ssidstr = self.lineEdit_hostname.text().strip().encode('gbk')
        if len(ssidstr)<1:
            QMessageBox.critical(self,'提示','请输入要加入写卡缓冲区的热点名称!',QMessageBox.Yes)
            self.lineEdit_hostname.setFocus()
            return

        authtype=self.comboBox_auth.currentIndex()
        crypttype=self.comboBox_Calculation.currentIndex()
        keystr=self.lineEdit_password.text().strip().encode('gbk')
        status = Objdll.tagbuf_addwifi(ssidstr, len(ssidstr), authtype, crypttype, keystr, len(keystr)) % 256
        if status == 0:
            self.ListBottom("WIFI热点信息加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
        else:
            self.ListBottom("WIFI热点信息加入NDEF写卡缓冲区失败!")

    def pushButton_Card_clicked(self):
        infostr = "BEGIN:VCARD" + "\n"
        infostr = infostr + "VERSION:3.0" + "\n"
        infostr = infostr + "FN:"+self.lineEdit_name.text().strip()+ "\n"
        infostr = infostr + "TEL:" +self.lineEdit_tel.text().strip()+ "\n"
        infostr = infostr + "ORG:" +self.lineEdit_unilname.text().strip()+ "\n"
        infostr = infostr + "ADR:" +self.lineEdit_address.text().strip()+ "\n"
        infostr = infostr + "EMAIL:" +self.lineEdit_email.text().strip()+ "\n"
        infostr = infostr + "URL:" +self.lineEdit_net.text().strip()+ "\n"
        infostr = infostr + "END:VCARD"
        businesscard = infostr.encode('gbk')
        status = Objdll.tagbuf_addbusinesscard(businesscard, len(businesscard)) % 256
        if status == 0:
            self.ListBottom("电子名片信息加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
        else:
            self.ListBottom("电子名片信息加入NDEF写卡缓冲区失败!")

    def pushButton_Data_clicked(self):
        typestr = self.lineEdit_datatype.text().strip().encode('gbk')
        datastr = self.lineEdit_data.text().strip().encode('gbk')
        status = Objdll.tagbuf_adddata(typestr, len(typestr), datastr, len(datastr)) % 256
        if status == 0:
            self.ListBottom("自定义数据类型加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
        else:
            self.ListBottom("自定义数据类型加入NDEF写卡缓冲区失败!")

    def pushButton_Call_clicked(self):
        languagecodestr = "en".encode('gbk')
        titlestr = "".encode('gbk')
        CallNumber=self.lineEdit_callnum.text().strip().encode('gbk')
        status = Objdll.tagbuf_adduri(languagecodestr, len(languagecodestr), titlestr, len(titlestr), 5, CallNumber,len(CallNumber)) % 256
        if status == 0:
            self.ListBottom("呼叫电话加入NDEF写卡缓冲区成功,可继续添加标签信息到缓冲区,也可以将缓冲区现有信息写入标签!")
        else:
            self.ListBottom("呼叫电话加入NDEF写卡缓冲区失败!")

    def pushButton_readtag_clicked(self):
        #Objdll.tagbuf_forumtype4_clear()
        #Objdll.tagbuf_clear()
        self.plainTextEdit_taginf.setPlainText('')
        self.ReadTag('NDEF_ReadInfo')

    def pushButton_cleatag_clicked(self):
        Objdll.tagbuf_forumtype4_clear()
        Objdll.tagbuf_clear()
        self.checkBox_later.setChecked(False)
        self.WriteTag('NDEF_Clear')


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    mainWindow = MainWindow()

    screen = QDesktopWidget().screenGeometry()
    size = mainWindow.geometry()
    # 获得窗口相关坐标
    newLeft = (screen.width() - size.width()) // 2
    newTop = (screen.height() - size.height()) // 2
    # 移动窗口使其居中
    mainWindow.pushButton_clearbuff.setStyleSheet("color:red;")
    mainWindow.pushButton_writetag.setStyleSheet("color:blue;")
    mainWindow.move(newLeft, newTop)

    if sys.platform == 'linux' or sys.platform == 'mac':
        # Linux加载当前目录下的so库
        dllfile = sys.path[0] + '/libOURMIFARE.so'
        Objdll = ctypes.cdll.LoadLibrary(dllfile)
    else:
        # windows系统加载当前目录下的DLL库
        dllfile = sys.path[0] + '\OUR_MIFARE.dll'
        Objdll = ctypes.windll.LoadLibrary(dllfile)

    mainWindow.show()
    sys.exit(app.exec_())

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/569324.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

雨云 湖北十堰 8272CL 高防高性能云服务器测评

雨云 湖北十堰 高防云服务器&#xff0c;铂金8272CL高性能处理器&#xff0c;2核2G 10兆 400G防御&#xff0c;仅需60元/月&#xff1b;8核16G 20兆 400G高防&#xff0c;仅需170元/月&#xff0c;年付8折1632元/年&#xff08;约136元/月&#xff09;。 企业级纯NVME固态硬盘高…

javase__进阶 day18 多线程02

1. 线程池 1.1 线程状态介绍 当线程被创建并启动以后&#xff0c;它既不是一启动就进入了执行状态&#xff0c;也不是一直处于执行状态。线程对象在不同的时期有不同的状态。那么Java中的线程存在哪几种状态呢&#xff1f;Java中的线程 状态被定义在了java.lang.Thread.Stat…

Nginx 防盗链

原文&#xff1a;https://blog.iyatt.com/?p14998 基于 Nginx 1.18 服务器默认配置文件路径&#xff1a;/etc/nginx/sites-available/default 屏蔽非指定域名的解析 我这里如果发现请求的地址不是我的 iyatt.com&#xff0c;就会返回 403 比如有人用其它域名指向我的服务器…

基于 Spring Boot 博客系统开发(四)

基于 Spring Boot 博客系统开发&#xff08;四&#xff09; 本系统是简易的个人博客系统开发&#xff0c;为了更加熟练地掌握 SprIng Boot 框架及相关技术的使用。&#x1f33f;&#x1f33f;&#x1f33f; 基于 Spring Boot 博客系统开发&#xff08;三&#xff09;&#x1f…

MySQL库表占用空间排序

在进行数据库备份恢复时&#xff0c;经常会碰到耗时很长的问题。大概率是因为某些库表的占用空间太大。 以下语句按照库表占用空间大小&#xff0c;进行降序排序&#xff1a; SELECT table_schema AS Database,table_name AS Table,ROUND((data_length index_length) / 1024…

【C语言】结构体,联合体,枚举--->自定义类型详解!

&#x1f525;博客主页&#x1f525;&#xff1a;【 坊钰_CSDN博客 】 欢迎各位点赞&#x1f44d;评论✍收藏⭐ 目录 1. 结构体 1.1 结构体定义 1.2 结构体的声明 1.3 结构体变量的定义和初始化 1.4 结构体的特殊声明->匿名声明 1.5 结构体的自应用 2. 结构体内存对齐…

上海·得物技术沙龙-「无线技术」专场报名开启!

本次无线沙龙聚焦于最新的技术趋势和实践&#xff0c;将在上海/线上为你带来四个令人期待的演讲话题&#xff0c;包括&#xff1a;《快手主App启动接口带宽优化实践》、《得物App视频体验优化实践》、《Chromium内核架构和网络库优化介绍》、《得物App发热监控实践》。相信这些…

本地部署Llama3教程,断网也能用啦!

4月18日&#xff0c;Meta在官方博客官宣了Llama3&#xff0c;标志着人工智能领域迈向了一个重要的飞跃。经过笔者的个人体验&#xff0c;Llama3 8B效果已经超越GPT-3.5&#xff0c;最为重要的是&#xff0c;Llama3是开源的&#xff0c;我们可以自己部署&#xff01; 本文和大家…

C++感受9-Hello Object 生死版•上

你好对象&#xff01; 认识C中基础中的基础类型&#xff1b;创建用户自定义的复合类型&#xff1b;创建新类型的对象&#xff1b;定制新类型对象的生死过程 零、面向对象启蒙 之前我们一直在问候世界&#xff0c;从这节课开始&#xff0c;我们的问候对象就是“对象&#xff08…

验证 python解释器是否安装成功

一. 简介 前一篇文章学习了下载并安装 python解释器&#xff0c;文章如下&#xff1a; windows系统下python解释器安装-CSDN博客 本文验证 python解释器是否安装成功。 二. 验证 python解释器是否安装成功 1. 首先&#xff0c;打开 Windows系统的 "cmd" 界面。…

Netty学习——实战篇8 Handler链调用、TCP粘包和拆包

1 Handler链调用-需求 使用自定义的编码器和解码器来说明Netty的Handler调用机制。客户端发送long类型数据到服务端&#xff1b;服务端发送long到客户端。 2 Handler链调用-实战 2.1 NettyServer.java public class NettyServer {public static void main(String[] args) {…

C语言——小知识和小细节16

一、左旋字符串 例如字符串 abcd &#xff0c;左旋一个就是 bcda &#xff0c;左旋两个就是 cdab 。 方法一&#xff1a;循环 #include <stdio.h> #include <string.h>void func(char* str, int n) {int i 0;int j 0;int len (int)strlen(str);n % len;//超出…

算法打卡day48|动态规划篇16| Leetcode 583. 两个字符串的删除操作、72. 编辑距离

算法题 Leetcode 583. 两个字符串的删除操作 题目链接:583. 两个字符串的删除操作 大佬视频讲解&#xff1a;583. 两个字符串的删除操作视频讲解 个人思路 本题和115.不同的子序列相比&#xff0c;变为了两个字符串都可以删除&#xff0c;整体思路是不变的&#xff0c;依旧…

vue 表格获取当前行索引,加颜色

vue 表格获取当前行索引&#xff0c;加颜色 <span styledisplay:inline-block;width:10px;height:10px;border-radius:50% :style"{background:color[scope.$index]}" />//定义颜色color: [#5387F7, #A794E0, #F3543C, #999999, #77D3F8, #FFA1B4, #26CEBA, #…

ElementUI RUOYI 深色适配

1. 切换按钮&#xff1a;随便找个页面放上去 页面触发逻辑如下 a. html 按钮结构&#xff08;可自定义&#xff09; <el-switchstyle"margin-top: 4px; margin-left: 8px; margin-right: 8px"v-model"isDark"inline-promptactive-icon"Moon"…

使用 Gradio 的“热重载”模式快速开发 AI 应用

在这篇文章中&#xff0c;我将展示如何利用 Gradio 的热重载模式快速构建一个功能齐全的 AI 应用。但在进入正题之前&#xff0c;让我们先了解一下什么是重载模式以及 Gradio 为什么要采用自定义的自动重载逻辑。如果你已熟悉 Gradio 并急于开始构建&#xff0c;请直接跳转到第…

vite和webpacke的常规配置

文章目录 1、vite和webpacke的区分2、vite的常规配置介绍主要部分介绍vite基本配置示例 3、webpacke的常规配置介绍主要部分介绍Webpack 基本配置示例 1、vite和webpacke的区分 相同点&#xff1a; 都是构建工具&#xff0c;用于资源打包 &#xff1b; 都有应用到摇树原理 tre…

网络研讨会 | 数据中心中的人工智能

人工智能&#xff08;AI&#xff09;是嵌入式开发人员必须解决的最复杂的技术之一。将其集成到您的系统中会带来很多问题而不是很多答案。行业媒体Embedded Computing Design特地推出“工程师的人工智能集成指南”月度网络研讨会系列&#xff0c;目的是尽可能地简化嵌入式计算设…

「共沐书香 分享喜“阅”」世界读书日交流活动在国际数字影像产业园圆满举行

人间最美四月天&#xff0c;正是读书好时节。4月23日&#xff0c;在数媒大厦的春日里&#xff0c;我们共同迎来了第29个“世界读书日"。由数字影像联合工会委员会、树莓科技&#xff08;成都&#xff09;集团有限公司工会委员会主办&#xff0c;成都树莓信息技术有限公司、…

Linux(韦东山)

linux和windows的差别 推荐学习路线 先学习 应用程序 然后&#xff1a; 驱动程序基础 最后&#xff1a;项目 韦东山课程学习顺序 看完第六篇之后&#xff0c;还可以继续做更多的官网的项目 入门之后&#xff0c;根据自己的需要学习bootloader / 驱动大全 / LVGL