record-camera-and-screen/RecordWindow.py

498 lines
20 KiB
Python
Raw Normal View History

2018-04-17 08:11:26 +00:00
import sys, datetime
import PyQt5
from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtCore import Qt, QCoreApplication
from PyQt5.QtWidgets import QMessageBox
import RecordVideo,RecordType
from RecordVideo import *
from RecordType import *
import SettingWindow
from SettingWindow import *
import Shortcut
from Shortcut import *
import RecordTrayIcon
from RecordTrayIcon import *
import RecordConfig
from RecordConfig import *
2018-04-18 09:15:20 +00:00
import RecordHelp
from RecordHelp import *
2018-04-17 08:11:26 +00:00
class RecordWindow(QtWidgets.QWidget):
def __init__(self, parent = None):
super(RecordWindow,self).__init__(parent)
self.setupUi()
self.load_modules()
#更新设置
self.need_update_config = False
# self.need_hide = True
2018-04-17 08:11:26 +00:00
#初始化状态
print('初始化状态...')
self.update_state()
def load_modules(self):
#录制
self.rv=RecordVideo()
#鼠标拖动
self.m_drag = False
#托盘图标
self.rti = RecordTrayIcon(self)
self.rti.update_state(self.recording, self.record_type)
2018-06-25 01:20:17 +00:00
self.sc = Shortcut()
self.update_setting(True)
# self.rc = RecordConfig()
# self.file_dir = self.rc.config.get('record','file_dir')
# self.debugCameraAction.triggered.connect(self.rv.debug_camera)
2018-04-17 08:11:26 +00:00
def closeEvent(self, event):
# print('close window.')
# self.close_signal.emit()
# self.close()
if self.recording:
self.stop_record(force = True)
2018-04-17 08:11:26 +00:00
# question = QMessageBox(self)
# question.setText('系统正在录制中,确定要退出吗?')
# question.setWindowTitle('提示')
# question.setIcon(QMessageBox.Question)
# question.addButton(QMessageBox.Yes)
# question.addButton(QMessageBox.No)
# question.setDefaultButton(QMessageBox.No)
# ret = question.exec()
# if ret == QMessageBox.Yes:
# self.stop_record()
# QCoreApplication.instance().quit()
# else:
# question = QMessageBox()
# question.setText('确定要退出吗?')
# question.setWindowTitle('提示')
# question.setIcon(QMessageBox.Question)
# question.addButton(QMessageBox.Yes)
# question.addButton(QMessageBox.No)
# question.setDefaultButton(QMessageBox.No)
# ret = question.exec()
# if ret == QMessageBox.Yes:
print('软件将退出.')
2018-04-17 08:11:26 +00:00
QCoreApplication.instance().quit()
# event.ignore()
def setupUi(self):
self.setObjectName("RecordWindow")
self.resize(94, 81)
self.move(1100,600)
self.pushButton = QtWidgets.QPushButton(self)
self.pushButton.setGeometry(QtCore.QRect(0, 30, 91, 51))
self.pushButton.setObjectName("pushButton")
# self.register_slot(self.pushButton.clicked, self.record)
2018-04-17 08:11:26 +00:00
# self.pushButton.mousePressEvent.connect(self.mousePressEvent)
# print(dir(self.pushButton))
#添加右键菜单
self.createContextMenu()
#添加计时器
self.lcd = QLCDNumber(self)
self.lcd.setDigitCount(10)
self.lcd.setMode(QLCDNumber.Dec)
self.lcd.setGeometry(QtCore.QRect(0, 0, 91, 31))
self.lcd.setSegmentStyle(QLCDNumber.Flat)
self.init_lcd()
#新建一个QTimer对象
self.timer = QTimer()
self.timer.setInterval(1000)
self.timer.timeout.connect(self.onTimerOut)
self.retranslateUi()
#调整窗体属性
self.setWindowFlags(QtCore.Qt.FramelessWindowHint | QtCore.Qt.Tool)
# self.setFocusPolicy(QtCore.Qt.StrongFocus)
self.installEventFilter(self)
def retranslateUi(self):
_translate = QtCore.QCoreApplication.translate
self.setWindowTitle(_translate("RecordWindow", "RecordWindow"))
self.pushButton.setText(_translate("RecordWindow", "开始"))
def init_lcd(self):
self.lcd.display('0:00:00')
def start_timer(self):
self.start_time = datetime.now()
self.init_lcd()
self.timer.start()
def stop_timer(self):
self.timer.stop()
self.init_lcd()
# 刷新录制时间
2018-04-18 02:13:29 +00:00
def onTimerOut(self):
print('on timer out monitor record status :%s' % self.recording)
if self.recording:
self.lcd.display(self.get_display_time(self.start_time))
else:
# self.stop_timer()
self.stop_record(force = self.exception_exit)
if self.exception_exit:
time.sleep(2)
self.record(self.record_type)
2018-04-17 08:11:26 +00:00
def get_display_time(self,old_time):
delta_time = datetime.now() - old_time
delta_time_str = str(delta_time)
pos = delta_time_str.find('.')
time_text = delta_time_str[0:pos]
return time_text
#打开文件目录
def open_file_dir(self):
dirpath = self.file_dir
if os.path.isdir(dirpath):
os.startfile(dirpath)
else:
print('错误的文件目录:%s' % dirpath)
#显示菜单
def showContextMenu(self):
self.contextMenu.exec_(QtGui.QCursor.pos())
#添加右键菜单
def createContextMenu(self):
#更改右键菜单为自定义
print('初始化右键菜单...')
self.setContextMenuPolicy(Qt.CustomContextMenu)
self.customContextMenuRequested.connect(self.showContextMenu)
self.contextMenu = QtWidgets.QMenu(self)
#停止/开始录制
self.recordSwitchAction = self.contextMenu.addAction('开始/停止录制')
#开始录制摄像头
self.recordCameraAction = self.contextMenu.addAction('开始录制摄像头')
#录制屏幕
self.recordScreenAction = self.contextMenu.addAction('开始录制屏幕')
#分隔栏
self.separatorAction = self.contextMenu.addAction('分隔栏')
self.separatorAction.setSeparator(True)
self.openFileDirAction = self.contextMenu.addAction('打开文件目录')
self.openFileDirAction.triggered.connect(self.open_file_dir)
#调试摄像头
# self.debugCameraAction = self.contextMenu.addAction('调试摄像头')
# self.debugCameraAction.setVisible(False)
2018-04-17 08:11:26 +00:00
#设置
self.settingAction=self.contextMenu.addAction('设置')
self.sw = SettingWindow()
self.settingAction.triggered.connect(self.show_setting)
2018-04-17 08:11:26 +00:00
#帮助
2018-04-18 09:15:20 +00:00
self.aboutAction=self.contextMenu.addAction('帮助')
self.rh = RecordHelp()
self.aboutAction.triggered.connect(self.rh.showWindow)
2018-04-17 08:11:26 +00:00
#退出
self.exitAction = self.contextMenu.addAction('退出')
self.exitAction.triggered.connect(self.close)
'''
功能事件
'''
@property
def recording(self):
return self.rv.recording
@property
def exception_exit(self):
return self.rv.exception_exit
2018-04-17 08:11:26 +00:00
@property
def record_type(self):
return self.rv.record_type
def register_slot(self, event_obj, new_action, disconnect = True):
if disconnect:
try:
event_obj.disconnect()
except Exception as e:
pass
event_obj.connect(new_action)
def update_state(self):
2018-04-17 08:11:26 +00:00
if self.recording:
print('录制状态:录制中.')
if self.record_type == RecordType.Camera:
print('正在录制摄像头.')
else:
print('正在录制屏幕.')
print('recording:%s' % self.recording)
print('record_type:%s' % self.record_type)
if self.recording:
self.recordSwitchAction.setText('停止录制')
self.register_slot(self.pushButton.clicked, self.stop_record)
self.register_slot(self.recordSwitchAction.triggered, self.stop_record)
#正在录制摄像头
if self.record_type == RecordType.Camera:
self.pushButton.setText('正在录制摄像头\n点击停止')
self.recordCameraAction.setText('停止录制摄像头')
self.register_slot(self.recordCameraAction.triggered, self.stop_record)
self.recordScreenAction.setText('开始录制屏幕')
self.register_slot(self.recordScreenAction.triggered, lambda: self.record(RecordType.Screen))
#正在录制屏幕
if self.record_type == RecordType.Screen:
self.pushButton.setText('正在录制屏幕\n点击停止')
self.recordScreenAction.setText('停止录制屏幕')
self.register_slot(self.recordScreenAction.triggered, self.stop_record)
self.recordCameraAction.setText('开始录制摄像头')
self.register_slot(self.recordCameraAction.triggered, lambda: self.record(RecordType.Camera))
else:
self.pushButton.setText('开始')
self.recordSwitchAction.setText('开始录制')
self.register_slot(self.pushButton.clicked, lambda: self.record(self.record_type))
self.register_slot(self.recordSwitchAction.triggered, lambda: self.record(self.record_type))
self.recordScreenAction.setText('开始录制屏幕')
self.register_slot(self.recordScreenAction.triggered, lambda: self.record(RecordType.Screen))
self.recordCameraAction.setText('开始录制摄像头')
self.register_slot(self.recordCameraAction.triggered, lambda: self.record(RecordType.Camera))
self.rti.update_state(self.recording, self.record_type)
2018-04-18 02:13:29 +00:00
def stop_record(self, force = False):
#force应用在两种情况
#1.timer实时刷新ffmpeg进程状态当出现异常退时force=true,即force=exception_exit。
#2.应用退出时
2018-04-18 02:13:29 +00:00
if self.recording or force:
2018-04-17 08:11:26 +00:00
self.stop_timer()
if self.record_type == RecordType.Camera:
print('停止录制摄像头.')
if self.record_type == RecordType.Screen:
print('停止录制屏幕.')
try:
self.rv.stop_record()
except KeyboardInterrupt as e:
print(e)
finally:
self.update_state()
2018-04-18 02:13:29 +00:00
# else:
# exit_tip = '系统即将退出'
# print(exit_tip)
# 退出系统
# self.close()
2018-04-17 08:11:26 +00:00
def record(self, rtype):
if self.recording:
if rtype == self.record_type:
return True
print('检测到正在录制,录制类型将切换...')
self.stop_record()
2018-06-16 14:06:56 +00:00
if self.rv.check_device():
#开始录制
if rtype == RecordType.Camera:
print('开始录制摄像头...')
self.rv.record_camera()
elif rtype == RecordType.Screen:
print('开始录制屏幕...')
self.rv.record_screen()
self.start_timer()
self.update_state()
else:
# self.need_hide = False
# question = QMessageBox.information(self, '提示', '检测到录制设备缺失,无法进行录制,请先完善设备设置。', QMessageBox.Yes)
# self.need_hide = True
question = QMessageBox()
question.setText('检测到录制设备缺失,无法进行录制,请先完善设备设置。')
question.setWindowTitle('提示')
question.setIcon(QMessageBox.Question)
question.addButton(QMessageBox.Yes)
tmp_btn = question.button(QMessageBox.Yes)
tmp_btn.setText('确定')
question.adjustSize()
screen_center = QApplication.desktop().screenGeometry()
question.move(((screen_center.width() - question.width()) /2), ((screen_center.height() - question.height())/2))
2018-06-16 14:06:56 +00:00
# question.addButton(QMessageBox.No)
# question.setDefaultButton(QMessageBox.No)
ret = question.exec()
2018-06-16 14:06:56 +00:00
# if ret == QMessageBox.Yes:
# print('软件将退出.')
def show_setting(self):
# QWidget.connect(self.sw, update_setting(bool), self, self.update_setting(bool))
self.sw.update_setting.connect(self.update_setting)
self.sw.showSettingWindow()
def update_setting(self, changed):
if changed:
print('update setting..')
self.rv.load_config()
2018-06-25 01:20:17 +00:00
self.rc = RecordConfig()
self.load_shortcut()
2018-06-25 01:20:17 +00:00
self.file_dir = self.rc.config.get('record','file_dir')
2018-04-17 08:11:26 +00:00
''''
鼠标拖动窗体
'''
def mousePressEvent(self, e):
if not isinstance(self,RecordWindow):
e.ignore()
else:
if e.button() == Qt.LeftButton:
self.m_drag = True
self.m_DragPosition = e.globalPos() - self.pos()
e.accept()
self.setCursor(QtGui.QCursor(Qt.OpenHandCursor))
def mouseReleaseEvent(self, e):
if not isinstance(self,RecordWindow):
e.ignore()
else:
if e.button() == Qt.LeftButton:
self.m_drag = False
self.setCursor(QtGui.QCursor(Qt.ArrowCursor))
def mouseMoveEvent(self, e):
if not isinstance(self,RecordWindow):
e.ignore()
else:
if Qt.LeftButton and self.m_drag:
self.move(e.globalPos() - self.m_DragPosition)
e.accept()
# print('(x:%d/y:%d)' % (self.x(),self.y()))
'''
快捷键监听
'''
2018-06-25 01:20:17 +00:00
def load_shortcut(self):
self.sc.clear()
camera_key_group = self.rc.config.get('shortcut','camera')
screen_key_group = self.rc.config.get('shortcut','screen')
stop_record_key_group = self.rc.config.get('shortcut','stop')
camera_shortcut = [int(key) for key in camera_key_group.split(',')]
screen_shortcut = [int(key) for key in screen_key_group.split(',')]
stop_shortcut = [int(key) for key in stop_record_key_group.split(',')]
print('camera shortcut: %s' % camera_shortcut)
print('screen shortcut: %s' % screen_shortcut)
print('stop shortcut: %s' % stop_shortcut)
if camera_key_group:
self.sc.add(1, camera_shortcut, lambda: self.record(RecordType.Camera))
if screen_key_group:
self.sc.add(2, screen_shortcut, lambda: self.record(RecordType.Screen))
if stop_record_key_group:
self.sc.add(3, stop_shortcut, self.stop_record)
2018-04-17 08:11:26 +00:00
def monitor_shortcut(self):
if self.rv.check_run_state():
2018-04-17 08:11:26 +00:00
2018-06-25 01:20:17 +00:00
self.load_shortcut()
self.sc.monitor()
else:
question = QMessageBox(self)
question.setText('检测到软件非正常运行,以下功能将被禁用(如有疑问,请联系开发商):<br/><ul><li>快捷键</li></ul>')
question.setWindowTitle('限制使用模式')
question.setIcon(QMessageBox.Warning)
question.addButton(QMessageBox.Yes)
# question.setButtonText(QMessageBox.Yes, QString('确定'))
btn = question.button(QMessageBox.Yes)
# print(btn.text())
btn.setText('确定')
# question.addButton('确定', QMessageBox.ButtonRole.AcceptRole)
# print('screen x:%d,y:%d; question x:%d,y:%d.' % (screen_center.x(), screen_center.y(), question.x(), question.y()))
# print(question.frameGeometry())
question.adjustSize()
screen_center = QApplication.desktop().screenGeometry()
#居中x和y的值计算与我想的不一样
#原因question在exec之前无法准确的获取其size。(485,170)是exec之后的实际size。
question.move(((screen_center.width() - 485)/2), ((screen_center.height() - 170)/2))
# question.move((screen_center.width() / 2 - question.width()), (screen_center.height() /2 - question.height()))
# print('screen x:%d,y:%d; question x:%d,y:%d; question width:%d,height:%d.' % (screen_center.width(), screen_center.height(), question.geometry().width(), question.geometry().height(), question.width(), question.height()))
# question.move(QApplication.desktop().screenGeometry().center()- self.rect().center())
ret = question.exec()
2018-04-17 08:11:26 +00:00
'''
窗体事件
'''
def eventFilter(self, obj, event):
etype=event.type()
# print('e type: %s' % etype)
# print('obj is window? %s' % (isinstance(obj,RecordWindow)))
2018-04-17 08:11:26 +00:00
# print('deactivate type id :%d' % QEvent.WindowDeactivate)
if etype == QEvent.WindowDeactivate:
print('丢失焦点')
#代码功能:当窗口失去焦点(不是键盘事件的焦点),窗体隐藏。
#引起的问题:当关闭从此窗口弹出的提示框和子窗口时,此窗口异常退出。
#原因分析提示框和子窗口在弹出时引发WindowDeactivate事件主窗口同时隐藏
#当提示框和子窗口关闭时,因为父窗口已隐藏子窗口似乎是找不到父窗口,或者猜测是误以为父窗口不存在,然后整个应用随即退出。
#解决方案1先用用变量need_hide(bool)来控制是否需要隐藏窗体need_hide=True才隐藏窗体。在打开子窗口和弹出提示窗之前将need_hide=False。
#缺陷:快捷键操作仍然会有同样的问题,因为窗体大多时候是隐藏的。(考虑:把快捷键监控放在托盘图标上?)
#更好的解决方案1.为什么当父窗口是隐藏状态,子窗口关闭,整个应用也随之退出?(关键)
#2.对窗体事件有更准确的了解,换一种更好的方式来实现窗体失去焦点隐藏主窗体。
#最终解决方案1.设置QApplication.setQuitOnLastWindowClosed(False)
#2显示逻辑由此处和RecordTrayIcon.actived事件分别控制此处负责隐藏RecordTrayIcon根据此窗体隐藏状态做判断如果隐藏则显示即单击或双击显示
#如果非隐藏但不是activeWindow则设置activeWindow=True否则隐藏。
2018-04-17 08:11:26 +00:00
self.setVisible(False)
# if self.need_hide:
# self.setVisible(False)
# pass
# else:
# self.need_hide = True
2018-04-17 08:11:26 +00:00
return False
if __name__ == '__main__':
app = QtWidgets.QApplication(sys.argv)
#当这个属性为True应用程序会在最后一个可见窗口关闭时退出。
app.setQuitOnLastWindowClosed(False)
2018-04-17 08:11:26 +00:00
rw = RecordWindow()
rw.monitor_shortcut()
rw.rti.show()
rw.show()
sys.exit(app.exec_())