mirror of
https://github.com/LC044/WeChatMsg
synced 2024-11-16 06:51:19 +08:00
293 lines
11 KiB
Python
293 lines
11 KiB
Python
import os
|
||
import sys
|
||
import time
|
||
from typing import List
|
||
|
||
from PyQt5 import QtWidgets
|
||
from PyQt5.QtCore import QTimer, QThread, pyqtSignal, QObject
|
||
from PyQt5.QtGui import QTextCursor
|
||
from PyQt5.QtWidgets import QApplication, QDialog, QCheckBox, QMessageBox, QCalendarWidget
|
||
|
||
from app.DataBase import micro_msg_db, misc_db
|
||
from app.DataBase.output_pc import Output
|
||
from app.components import ScrollBar
|
||
from app.components.calendar_dialog import CalendarDialog
|
||
from app.components.export_contact_item import ContactQListWidgetItem
|
||
from app.person import Contact
|
||
from app.ui.menu.exportUi import Ui_Dialog
|
||
from app.ui.menu.export_time_range import TimeRangeDialog
|
||
|
||
types = {
|
||
'文本': 1,
|
||
'图片': 3,
|
||
'语音': 34,
|
||
'视频': 43,
|
||
'表情包': 47,
|
||
'音乐与音频': 4903,
|
||
'文件': 4906,
|
||
'分享卡片': 4905,
|
||
'拍一拍等系统消息': 10000
|
||
}
|
||
file_format = {
|
||
'Docx': Output.DOCX,
|
||
'TXT': Output.TXT,
|
||
'HTML': Output.HTML,
|
||
'CSV': Output.CSV,
|
||
}
|
||
Stylesheet = """
|
||
QPushButton{
|
||
background-color: #ffffff;
|
||
}
|
||
QPushButton:hover {
|
||
background-color: lightgray;
|
||
}
|
||
"""
|
||
|
||
|
||
class EmittingStr(QObject):
|
||
textWritten = pyqtSignal(str) # 定义一个发送str的信号
|
||
|
||
def write(self, text):
|
||
self.textWritten.emit(str(text))
|
||
|
||
|
||
class ExportDialog(QDialog, Ui_Dialog):
|
||
def __init__(self, contact=None, title="选择导出的类型", file_type="html", parent=None):
|
||
super(ExportDialog, self).__init__(parent)
|
||
self.setupUi(self)
|
||
self.contacts: List[Contact] = []
|
||
self.setStyleSheet(Stylesheet)
|
||
self.contact = contact
|
||
self.btn_select_all.clicked.connect(self.select_all)
|
||
self.select_all_flag = False
|
||
self.btn_start.clicked.connect(self.export_data)
|
||
self.comboBox_time.activated.connect(self.set_export_date)
|
||
# 下面将输出重定向到textBrowser中
|
||
sys.stdout = EmittingStr(textWritten=self.outputWritten)
|
||
# sys.stderr = EmittingStr(textWritten=self.outputWritten)
|
||
scroll_bar = ScrollBar()
|
||
self.textBrowser.setVerticalScrollBar(scroll_bar)
|
||
self.export_choices = {"文本": True, "图片": True, "语音": False, "视频": False, "表情包": False,
|
||
'音乐与音频': False, '分享卡片': False, '文件': False,
|
||
'拍一拍等系统消息': True} # 定义导出的数据类型,默认全部选择
|
||
self.setWindowTitle(title)
|
||
self.checkBox_word.setEnabled(False)
|
||
self.checkBox_word.setText('Docx(暂时不可用)')
|
||
self.resize(800, 600)
|
||
self.worker = None # 导出线程
|
||
for export_type, default_state in self.export_choices.items():
|
||
checkbox = QCheckBox(export_type)
|
||
checkbox.setObjectName('message_type')
|
||
checkbox.setChecked(default_state)
|
||
self.verticalLayout_2.addWidget(checkbox)
|
||
spacerItem1 = QtWidgets.QSpacerItem(20, 40, QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Expanding)
|
||
self.verticalLayout_2.addItem(spacerItem1)
|
||
self.timer = QTimer(self)
|
||
self.time = 0
|
||
self.total_msg_num = 99999 # 总的消息个数
|
||
self.num = 0 # 当前完成的消息个数
|
||
self.timer.timeout.connect(self.update_elapsed_time)
|
||
self.show_thread = ShowContactThread()
|
||
self.show_thread.showSingal.connect(self.show_contact)
|
||
# self.show_thread.load_finish_signal.connect(self.stop_loading)
|
||
self.show_thread.start()
|
||
self.listWidget.setVerticalScrollBar(ScrollBar())
|
||
# self.listWidget.currentRowChanged.connect(self.setCurrentIndex)
|
||
self.listWidget.itemClicked.connect(self.setCurrentIndex)
|
||
self.visited = set()
|
||
self.now_index = 0
|
||
self.time_range = None
|
||
|
||
def set_export_date(self):
|
||
date_range = self.comboBox_time.currentText()
|
||
if date_range == '全部时间':
|
||
pass
|
||
elif date_range == '最近三个月':
|
||
from datetime import datetime, timedelta
|
||
|
||
# 获取今天的日期和时间
|
||
today = datetime.now()
|
||
|
||
# 获取今天的日期
|
||
today_date = today.date()
|
||
|
||
# 获取今天的24:00:00的时间戳
|
||
today_midnight = datetime.combine(today_date, datetime.min.time()) + timedelta(days=1)
|
||
today_midnight_timestamp = int(today_midnight.timestamp())
|
||
|
||
# 获取三个月前的日期
|
||
three_months_ago = today - timedelta(days=90)
|
||
|
||
# 获取三个月前的00:00:00的时间戳
|
||
three_months_ago_date = three_months_ago.date()
|
||
three_months_ago_midnight = datetime.combine(three_months_ago_date, datetime.min.time())
|
||
three_months_ago_midnight_timestamp = int(three_months_ago_midnight.timestamp())
|
||
self.time_range = (three_months_ago_midnight_timestamp, today_midnight_timestamp)
|
||
|
||
elif date_range == '自定义时间':
|
||
self.time_range_view = TimeRangeDialog(parent=self)
|
||
self.time_range_view.date_range_signal.connect(self.set_time_range)
|
||
self.time_range_view.show()
|
||
self.comboBox_time.setCurrentIndex(0)
|
||
# QMessageBox.warning(self,
|
||
# "别急别急",
|
||
# "马上就实现该功能"
|
||
# )
|
||
|
||
def set_time_range(self, time_range):
|
||
self.time_range = time_range
|
||
self.comboBox_time.setCurrentIndex(2)
|
||
|
||
def outputWritten(self, text):
|
||
cursor = self.textBrowser.textCursor()
|
||
cursor.movePosition(QTextCursor.End)
|
||
cursor.insertText(text)
|
||
self.textBrowser.setTextCursor(cursor)
|
||
self.textBrowser.ensureCursorVisible()
|
||
|
||
def export_data(self):
|
||
self.btn_start.setEnabled(False)
|
||
# 在这里获取用户选择的导出数据类型
|
||
selected_types = {types[export_type]: checkbox.isChecked() for export_type, checkbox in
|
||
zip(self.export_choices.keys(), self.findChildren(QCheckBox, 'message_type'))}
|
||
|
||
# 在这里根据用户选择的数据类型执行导出操作
|
||
print("选择的数据类型:", selected_types)
|
||
|
||
file_types = []
|
||
for checkbox in [self.checkBox_txt, self.checkBox_csv, self.checkBox_html, self.checkBox_word]:
|
||
if checkbox.isChecked():
|
||
file_types.append(file_format[checkbox.text()])
|
||
select_contacts = []
|
||
count = self.listWidget.count()
|
||
# 遍历listwidget中的内容
|
||
for i in range(count):
|
||
item = self.listWidget.item(i)
|
||
if item.is_select:
|
||
select_contacts.append(self.contacts[i])
|
||
# 在这里根据用户选择的数据类型执行导出操作
|
||
print("选择的文件格式:", file_types)
|
||
self.worker = Output(select_contacts, type_=Output.Batch, message_types=selected_types, sub_type=file_types,
|
||
time_range=self.time_range)
|
||
# self.worker.progressSignal.connect(self.update_progress)
|
||
self.worker.okSignal.connect(self.export_finished)
|
||
self.worker.rangeSignal.connect(self.set_total_msg_num)
|
||
self.worker.nowContact.connect(self.update_progress)
|
||
self.worker.start()
|
||
# 启动定时器,每1000毫秒更新一次任务进度
|
||
self.timer.start(1000)
|
||
self.start_time = time.time()
|
||
# self.accept() # 使用accept关闭对话框
|
||
# 绑定点击槽函数 点击显示对应item中的name
|
||
|
||
def set_total_msg_num(self, num):
|
||
self.total_msg_num = num
|
||
# b''+num +(1,1)
|
||
|
||
def setCurrentIndex(self, item):
|
||
# print(row)
|
||
# row = self.listWidget.it
|
||
# item = self.listWidget.item(row)
|
||
item.select()
|
||
item.dis_select()
|
||
# self.now_index = row
|
||
|
||
def select_all(self):
|
||
self.select_all_flag = not self.select_all_flag
|
||
print('全选', self.select_all_flag)
|
||
if self.select_all_flag:
|
||
count = self.listWidget.count()
|
||
# 遍历listwidget中的内容
|
||
for i in range(count):
|
||
item = self.listWidget.item(i)
|
||
item.force_select()
|
||
self.btn_select_all.setText('全不选')
|
||
else:
|
||
count = self.listWidget.count()
|
||
# 遍历listwidget中的内容
|
||
for i in range(count):
|
||
item = self.listWidget.item(i)
|
||
item.force_dis_select()
|
||
self.btn_select_all.setText('全选')
|
||
|
||
def export_finished(self):
|
||
self.btn_start.setEnabled(True)
|
||
self.time = 0
|
||
end_time = time.time()
|
||
print(f'总耗时:{end_time - self.start_time}s')
|
||
reply = QMessageBox(self)
|
||
reply.setIcon(QMessageBox.Information)
|
||
reply.setWindowTitle('OK')
|
||
reply.setText(f"导出聊天记录成功\n在./data/目录下(跟exe文件在一起)\n{os.getcwd()}\\data\\")
|
||
reply.addButton("确认", QMessageBox.AcceptRole)
|
||
reply.addButton("取消", QMessageBox.RejectRole)
|
||
api = reply.exec_()
|
||
# 在任务完成时重置sys.stdout
|
||
sys.stdout = sys.__stdout__
|
||
self.accept()
|
||
|
||
def close(self):
|
||
sys.stdout = sys.__stdout__
|
||
del self.worker
|
||
super().close()
|
||
|
||
def show_contact(self, contact):
|
||
# return
|
||
# print(contact.remark)
|
||
contact_item = ContactQListWidgetItem(contact.remark, contact.smallHeadImgUrl, contact.smallHeadImgBLOG)
|
||
self.listWidget.addItem(contact_item)
|
||
self.listWidget.setItemWidget(contact_item, contact_item.widget)
|
||
self.contacts.append(contact)
|
||
|
||
def update_elapsed_time(self):
|
||
self.time += 1
|
||
self.label_time.setText(f"耗时: {self.time}s")
|
||
|
||
def update_progress(self, remark):
|
||
self.num += 1
|
||
progress_percentage = int((self.num) / self.total_msg_num * 100)
|
||
self.progressBar.setValue(progress_percentage)
|
||
self.label_process.setText(f"导出进度: {progress_percentage}% {remark}")
|
||
|
||
|
||
class ShowContactThread(QThread):
|
||
showSingal = pyqtSignal(Contact)
|
||
load_finish_signal = pyqtSignal(bool)
|
||
|
||
# heightSingal = pyqtSignal(int)
|
||
def __init__(self):
|
||
super().__init__()
|
||
|
||
def run(self) -> None:
|
||
contact_info_lists = micro_msg_db.get_contact()
|
||
for contact_info_list in contact_info_lists:
|
||
# UserName, Alias,Type,Remark,NickName,PYInitial,RemarkPYInitial,ContactHeadImgUrl.smallHeadImgUrl,ContactHeadImgUrl,bigHeadImgUrl
|
||
contact_info = {
|
||
'UserName': contact_info_list[0],
|
||
'Alias': contact_info_list[1],
|
||
'Type': contact_info_list[2],
|
||
'Remark': contact_info_list[3],
|
||
'NickName': contact_info_list[4],
|
||
'smallHeadImgUrl': contact_info_list[7]
|
||
}
|
||
contact = Contact(contact_info)
|
||
contact.smallHeadImgBLOG = misc_db.get_avatar_buffer(contact.wxid)
|
||
contact.set_avatar(contact.smallHeadImgBLOG)
|
||
self.showSingal.emit(contact)
|
||
# print(contact.wxid)
|
||
# pprint(contact.__dict__)
|
||
self.load_finish_signal.emit(True)
|
||
|
||
|
||
if __name__ == '__main__':
|
||
import sys
|
||
|
||
app = QApplication(sys.argv)
|
||
dialog = ExportDialog()
|
||
result = dialog.exec_() # 使用exec_()获取用户的操作结果
|
||
if result == QDialog.Accepted:
|
||
print("用户点击了导出按钮")
|
||
else:
|
||
print("用户点击了取消按钮")
|
||
sys.exit(app.exec_())
|