来用Qt实现一个桌面弹幕程序吧



  • 前言

    为什么做? 因为有趣呀~

    最近一段时间呢,经常上 B站 的 杰洛君 看到 up主 们在直播的时候用一个桌面程序能在桌面上显示弹幕飞过,感觉非常有趣,于是就学着用Qt做了个桌面弹幕程序啦~

    接下来的日子就打算写文章记录这一过程,也算是巩固知识吧。

    这个是教程?

    这个能称之为教程吗? 并不,和你想象地不一样 杰洛君 不是精通Qt程序开发的人,相反是大菜鸟一个(杰洛君大失败 (T__T)) 不过还是觉得记录下来比较好,所以接下来的文章还请各路大神们多多指教啦 (^-^)V 啦啦啦~

    这也是杰洛君第一次在Qt dream写文章啦,不过接下来的文章不会那么死板的,因为我也不希望自己的文章写成API文档一样,希望其中你能感受到我的心意,杰洛君是很希望和大家交流滴(^o^)/。

    大坑慢填

    这个程序说大不大,说小又不小,因为完整的应用包括了 1.桌面端 2.服务端 3.网页端 甚至可能拓展到 利用QML 实现 Android端,当然那也要等到我学会才行呀 Kira☆~。

    写作压力好大呀,杰洛君大危机!即便如此我还是会慢慢把它写好,因为杰洛君的脑子不好使,拥有着YD(《电波老师》梗--不感兴趣的东西就没有动力做),所以真正把技术学到会用全凭兴趣。我会渣油完成这个坑。

    不看广告看疗效!

    程序的界面

    0_1456276667637_pic1.png

    大弹幕时代哦,你还在等待什么呢?

    0_1456276697936_pic2.png

    控制版面啦啦啦~

    0_1456276770196_IMG_20151114_203026.jpg

    晚会现场使用效果

    感觉如何?(快夸我快夸我)

    什么?!你说这个界面丑哭了?

    以玻璃心著称的杰洛君会去哭上两天的,毕竟杰洛君是来分享的,不是来羞耻play的(T__T)555。嘛嘛,界面不对你口味没关系,不要在意这些细节 Don't mind

    接下来……

    这个程序用到的技术包括 c++/Qt5,python/Tornado,以及Html等前端的一些东西,结构简单,就是网页发送内容,桌面程序显示弹幕。虽然程序不复杂,记录其包含的过程还是很多的,可谓任重道远呀。

    因为杰洛君也有点忙,所以虽然会尽量快地填坑,但有时难免跳票。如果心急了可以催一催,但请不要太频繁,我会报警的(>^ω^<)(认真)

    声音:额。。。我看完文章还是想问 -- 杰洛飞是谁呀?
    回答:一点都不可爱喜欢二次元的声优控+怪蜀黍一枚。



  • 这个可厉害了。我一定会支持的。有机会的话,我们详细聊一下,如果你比较了解QML的话,可以教你将你的创意和我的应用结合起来。



  • 速度更新博文吧,带上构思和源码,快要等不急了!PS:我并不喜欢二次元,什么鬼!



  • @jiangcaiyang 昨天听了你的公开课,看到了你的App,我就知道是二次元同道中人呀!感觉这个论坛十分有趣,所以就来写文了,这个东西实现起来很简单的。简单版本700行代码不到,多多交流呀kira ☆~



  • @Anson 好的,好的,哈哈~



  • 厉害,佩服!支持!😄



  • @Accelerated 没有啦~在群里看到你很热心帮大家解决问题,还是你更加厉害,快带我飞😄



  • @杰洛飞
    我之前看到你的那个弹幕开发的,牛掰了!我解决点小问题。你创造的是大技术!☺



  • 真不错!膜拜!同为二次元宅男[手动滑稽]



  • 好好好!
    赶紧更新博客吧。我的类似项目github

    https://github.com/qyvlik/DarkFlameMaster

    科科



  • @大神 这个真的很简单的,不值得膜拜,多多交流[手动滑稽] hiahia~



  • @qyvlik 这个好厉害呀,可惜我学Qt有点晚接触QML,现在只会简单的写写Rectangle什么的。这个桌面程序还是很传统地用Widget实现的。



  • @杰洛飞
    挺好的。QML实现的话,可能会有一些小问题,可以看那个项目的文档试一试。


Log in to reply
 

走马观花

最近的回复

  • 还是很正常的Python脚本哈。
    不过有点好奇,你在实际场景中用QUiLoader多吗?🎯

    read more
  • 好棒,高产啊。真的很棒!🐱 能将自己有用的知识分享出来真不错!

    read more
  • 自动翻译器的qt部分

    接下来我们要实现qt窗口部分,这里遇到一个很尴尬的事情,qt for python的开发环境要求按照python,但我安装的是Anaconda,使用Jupyter开发,安完了PySide2,Qt找不到这个模块,用Jupyter呢,又提示找不到qt.qpa.plugin,打开环境变量查看os.environ,

    'QT_QPA_PLATFORM_PLUGIN_PATH': 'C:\\ProgramData\\Anaconda3\\lib\\site-packages\\PySide2\\plugins\\platform'platform竟然少个s,我也是醉了,由于不想在py文件追加环境变量配置,我很机智的把文件夹platforms复制了一份,改名platform,这样是不好的,但是好用呀。

    追加main.py文件,导入需要的包 import sys import random from PySide2.QtWidgets import QApplication, QWidget, QTextEdit, QHBoxLayout from PySide2.QtUiTools import QUiLoader from PySide2.QtCore import QFile, QDir, QTimer, Slot, Qt from PySide2.QtGui import QClipboard from Spider import * 使用线程创建翻译器 #创建翻译器 class WorkerThread(QThread): resultReady = Signal() def __init__(self): super().__init__() def __del__(self): pass def run(self): global translation translation = Translation() self.resultReady.emit() 翻译过程也使用线程 #翻译线程 class WorkerThreadTrans(QThread): resultReady = Signal(str) def __init__(self): super().__init__() self.last = '' def setText(self, value): self.value = value def run(self): self.result = '' while(1): self.getClipborad() time.sleep(1) @Slot() def getClipborad(self): global translation board = QApplication.clipboard() self.str = board.text() if self.str == '' or self.str == self.last: pass else: try: self.last = self.str self.result = translation.translate(self.str) except: pass finally: pass self.resultReady.emit(self.result) 实现窗口类 #窗口 class Form(QWidget): def __init__(self): super().__init__() self.textEdit = QTextEdit() self.layout = QHBoxLayout() self.layout.addWidget(self.textEdit) self.setLayout(self.layout) self.setWindowTitle("百度翻译器") self.resize(400, 300) self.setWindowFlags(Qt.WindowStaysOnTopHint) self.icon = QIcon() self.icon.addPixmap(QPixmap('favicon.ico'), QIcon.Normal, QIcon.Off) self.setWindowIcon(self.icon) self.create = False self.thread = WorkerThread() self.thread.finished.connect(self.thread.deleteLater) self.thread.resultReady.connect(self.createTrans) self.thread.start() self.threadTrans = WorkerThreadTrans() self.threadTrans.finished.connect(self.threadTrans.deleteLater) self.threadTrans.resultReady.connect(self.setTextValue) def __del__(self): self.threadTrans.quit() self.threadTrans.wait() if self.create: del translation @Slot() def createTrans(self): self.create = True self.threadTrans.start() @Slot() def setTextValue(self, value): self.textEdit.setPlainText(value) main if __name__ == "__main__": app = QApplication(sys.argv) translation = None window = Form() window.show() sys.exit(app.exec_())

    为了防止窗口启动卡顿,运行卡顿,我们开辟了两个线程。我们还把窗口置顶,为窗口设置了图标。我们重载了窗口关闭事件,用于把线程关闭。

    程序运行效果
    1564980342307.png

    打包

    具体打包操作请看帖子 https://www.jianshu.com/p/046e690c0f12

    打包命令

    pyinstaller -F -w -i favicon.ico --icon=favicon.ico main.py -p C:\ProgramData\Anaconda3\Lib\site-packages\PySide2

    为了平台显示我们把C:\ProgramData\Anaconda3\Lib\site-packages\PySide2\plugins\platforms\qwindows.dll也拷贝过来
    1564993349890.png

    去除selenium隐藏控制台解决办法

    参考帖子 https://www.cnblogs.com/TurboWay/p/9300105.html

    修改C:\ProgramData\Anaconda3\Lib\site-packages\selenium\webdriver\common\service.py源码

    程序地址

    read more
  • image.png

    自动翻译器的python部分 一、设计思路

    1.qt提取剪贴板/鼠标选中内容作为翻译内容

    2.使用python向百度翻译提交翻译内容,然后取回翻译结果

    3.使用qt显示翻译结果

    二、实现步骤

    我们已经安装了Jupyter作为开发环境

    先分析百度翻译提交接口

    有三种接口方式可以使用

    地址栏https://fanyi.baidu.com/translate?aldtype=16047&query=&keyfrom=baidu&smartresult=dict&lang=auto2zh#en/zh/world world就是要查询的单词 https://fanyi.baidu.com/sug 这个是百度自动识别的单词下拉项
    1564905012628.png https://fanyi.baidu.com/v2transapi 这个是真正百度翻译的接口

    这三种接口各有优缺点:

    第一种url简单,使用方便,缺点就是返回的是一整个网页,需要从网页中提取翻译的内容,但是该网页是动态渲染的,里面并没有我们需要的信息 第二种并不是一个真实的翻译,只是百度检索出类似的情况,不一定是需要的,而且如果是一句话的翻译,这个是空的 第三种是真正的翻译,但是需要提交详细数据,下面就是要提交的数据,其他数据还好,这个sign比较麻烦,他是js动态生成的,是加密的,我们无法模拟 from: en to: zh query: world transtype: realtime simple_means_flag: 3 sign: 335290.130699 token: fcd815f24ac02a1ddc7c485f38c8efe8

    综合考虑,这三种我们都要放弃。

    针对动态渲染的网页,Python提供了许多模拟浏览器运行的库,比如Selenium

    使用Selenium

    首先命令行安装selenium

    pip install selenium -i http://pypi.douban.com/simple/ --trusted-host pypi.douban.com 导入模块 import time from selenium import webdriver from selenium.webdriver.common.by import By from selenium.webdriver.common.keys import Keys from selenium.webdriver.support import expected_conditions as EC from selenium.webdriver.support.wait import WebDriverWait from selenium.webdriver.common.desired_capabilities import DesiredCapabilities 封装类 class Translation: def __init__(self): self.options = webdriver.ChromeOptions() self.options.add_argument('headless') # 后台运行 # self.options.add_experimental_option("excludeSwitches", ["ignore-certificate-errors"]) # 禁止图片的加载 self.prefs = {"profile.managed_default_content_settings.images":2} self.options.add_experimental_option("prefs",self.prefs) self.browser = webdriver.Chrome(executable_path='chromedriver.exe', options=self.options)#, desired_capabilities=self.desired_capabilities) self.load = False def __del__(self): self.browser.close() def translate(self, words): try: if not self.load : self.load = True self.browser.get('https://fanyi.baidu.com/translate#en/zh/') self.input= self.browser.find_element_by_id('baidu_translate_input') self.input.clear() self.input.send_keys(words) self.button = self.browser.find_element_by_id('translate-button') self.button.click() time.sleep(1) self.out = self.browser.find_element_by_class_name('output-bd') print (self.out.text) return self.out.text finally: # print ("translate [ {0} ] error.".format(words)) return '' 测试类 if __name__ == '__main__': words = [ '0', 'Taylor was nominated for a Primetime Emmy Award last year for portraying Minnie in the latest Mickey Mouse TV show rendition ', '"I really want whoever comes after us to be aware of the history and the tradition, and to love the characters as much as we do," Taylor said about herself and Allwine, according to Disney.', '"Minnie Mouse lost her voice with the passing of Russi Taylor," Bob Iger, Disney Chairman and CEO, wrote on Twitter.' ] print ('---start-') translation = Translation() for w in words: print ('----', w) result = translation.translate(w) print (result) del translation 测试结果 ---start- ---- 0 ---- Taylor was nominated for a Primetime Emmy Award last year for portraying Minnie in the latest Mickey Mouse TV show rendition 泰勒去年因在最新的米奇老鼠电视节目“表演”中饰演米妮而获得艾美奖的提名。 ---- "I really want whoever comes after us to be aware of the history and the tradition, and to love the characters as much as we do," Taylor said about herself and Allwine, according to Disney. “我真的希望任何一个追随我们的人都能意识到历史和传统,并且像我们一样热爱这些角色,”根据迪士尼的说法,泰勒在谈到自己和奥尔温时说。 ---- "Minnie Mouse lost her voice with the passing of Russi Taylor," Bob Iger, Disney Chairman and CEO, wrote on Twitter. 迪斯尼董事长兼首席执行官鲍勃•伊格尔在Twitter上写道:“米妮•老鼠在路西•泰勒去世后失去了声音。”

    之所以第一个数据是0,是因为未知原因第一个翻译时候,网页会刷新,导致得不到翻译结果,所以需要屏蔽。

    read more

关注我们

微博
QQ群