Q [import sys
import pandas as pd
from PyQt5.QtWidgets import (QApplication, QMainWindow, QWidget, QVBoxLayout, QHBoxLayout,
                             QLabel, QPushButton, QTableWidget, QTableWidgetItem,
                             QFileDialog, QTabWidget, QCheckBox, QGroupBox, QMessageBox,
                             QProgressBar, QScrollArea, QFrame, QDialog, QDialogButtonBox,
                             QVBoxLayout as QVBoxLayout2, QGridLayout)
from PyQt5.QtCore import Qt, QTimer
from datetime import datetime

# 定义密级优先级
SECURITY_LEVELS = {
    '绝密': 3,
    '机密': 2,
    '秘密': 1
}


class ExcelProcessor:
    @staticmethod
    def detect_excel_type(df):
        """识别Excel文件类型"""
        columns = set(df.columns)

        # A类Excel的特征字段
        a_type_columns = {'序号', '密级', '档案名称', '创建日期', '保管部门',
                          '责任人', '存放位置', '保密期限', '解密条件', '页数', '备注'}

        # B类Excel的特征字段
        b_type_columns = {'编码', '密级', '最早日期', '最晚日期', '总页数'}

        if a_type_columns.issubset(columns):
            return 'A'
        elif b_type_columns.issubset(columns):
            return 'B'
        else:
            return None

    @staticmethod
    def process_type_a(df):
        """处理A类Excel数据"""
        # 提取前缀 (如GB-1 → GB)
        df['前缀'] = df['序号'].str.extract(r'([A-Za-z]+)')[0]

        results = []
        for prefix, group in df.groupby('前缀'):
            # 获取最高密级
            max_security = group.loc[group['密级'].map(SECURITY_LEVELS).idxmax(), '密级']

            # 转换日期格式并获取最早和最晚日期
            dates = pd.to_datetime(group['创建日期'])
            min_date = dates.min().strftime('%Y-%m-%d')
            max_date = dates.max().strftime('%Y-%m-%d')

            # 计算总页数
            total_pages = group['页数'].sum()

            results.append({
                '编码': prefix,
                '密级': max_security,
                '最早日期': min_date,
                '最晚日期': max_date,
                '总页数': total_pages
            })

        return pd.DataFrame(results)

    @staticmethod
    def process_type_b(df):
        """处理B类Excel数据"""
        # 直接返回原始数据，但分成两个模板
        template1 = df[['编码', '密级', '最早日期']].copy()
        template2 = df[['最晚日期', '总页数']].copy()
        return template1, template2


class TemplateSelectionDialog(QDialog):
    """优化后的模板选择对话框"""

    def __init__(self, templates, parent=None):
        super().__init__(parent)
        self.setWindowTitle("选择导出模板")
        self.setMinimumWidth(400)

        # 主布局
        layout = QVBoxLayout2(self)

        # 标题和全选按钮
        header_layout = QHBoxLayout()
        self.title_label = QLabel("请选择要导出的模板:")
        self.title_label.setStyleSheet("font-weight: bold; font-size: 14px;")

        self.select_all_btn = QPushButton("全选")
        self.select_all_btn.setFixedWidth(80)
        self.select_all_btn.setStyleSheet("background-color: #f0f0f0;")
        self.select_all_btn.clicked.connect(self.toggle_select_all)

        header_layout.addWidget(self.title_label)
        header_layout.addStretch()
        header_layout.addWidget(self.select_all_btn)

        # 模板选择区域
        self.template_frame = QFrame()
        self.template_frame.setFrameShape(QFrame.StyledPanel)
        self.template_layout = QGridLayout()
        self.template_layout.setSpacing(10)

        self.checkboxes = []
        for i, template in enumerate(templates):
            checkbox = QCheckBox(template)
            checkbox.setChecked(True)
            checkbox.setStyleSheet("QCheckBox { padding: 5px; }")
            self.checkboxes.append(checkbox)
            self.template_layout.addWidget(checkbox, i // 2, i % 2)

        self.template_frame.setLayout(self.template_layout)

        # 按钮区域
        self.button_box = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
        self.button_box.accepted.connect(self.accept)
        self.button_box.rejected.connect(self.reject)

        # 添加到主布局
        layout.addLayout(header_layout)
        layout.addWidget(self.template_frame)
        layout.addWidget(self.button_box)

        # 初始状态
        self.all_selected = True

    def toggle_select_all(self):
        """切换全选状态"""
        self.all_selected = not self.all_selected
        for cb in self.checkboxes:
            cb.setChecked(self.all_selected)

        self.select_all_btn.setText("取消全选" if self.all_selected else "全选")

    def get_selected_templates(self):
        """获取选中的模板"""
        return [cb.text() for cb in self.checkboxes if cb.isChecked()]


class MainWindow(QMainWindow):
    def __init__(self):
        super().__init__()
        self.setWindowTitle("Excel数据处理工具")
        self.setGeometry(100, 100, 1000, 800)

        # 初始化变量
        self.current_file = None
        self.current_df = None
        self.excel_type = None
        self.processed_data = None

        # 创建UI
        self.init_ui()

    def init_ui(self):
        """初始化用户界面"""
        main_widget = QWidget()
        main_layout = QVBoxLayout()

        # 顶部控制区域
        control_group = QGroupBox("文件操作")
        control_layout = QHBoxLayout()

        self.import_btn = QPushButton("导入Excel文件")
        self.import_btn.clicked.connect(self.import_excel)

        self.process_btn = QPushButton("处理数据")
        self.process_btn.clicked.connect(self.process_data)
        self.process_btn.setEnabled(False)

        self.export_btn = QPushButton("导出结果")
        self.export_btn.clicked.connect(self.export_results)
        self.export_btn.setEnabled(False)

        control_layout.addWidget(self.import_btn)
        control_layout.addWidget(self.process_btn)
        control_layout.addWidget(self.export_btn)
        control_group.setLayout(control_layout)

        # 中间显示区域 - 使用分割布局
        display_group = QGroupBox("数据处理")
        display_layout = QVBoxLayout()

        # 原始数据区域
        raw_data_group = QGroupBox("原始数据")
        raw_data_layout = QVBoxLayout()

        self.raw_data_table = QTableWidget()
        raw_data_layout.addWidget(self.raw_data_table)
        raw_data_group.setLayout(raw_data_layout)

        # 处理结果区域
        result_group = QGroupBox("数据汇总")
        result_layout = QVBoxLayout()

        # 结果表格区域 - 使用标签页
        self.result_tabs = QTabWidget()
        self.result_tabs.setTabPosition(QTabWidget.North)
        self.result_tabs.setMovable(True)

        # 进度条
        self.progress_bar = QProgressBar()
        self.progress_bar.setRange(0, 100)
        self.progress_bar.hide()

        result_layout.addWidget(self.result_tabs)
        result_layout.addWidget(self.progress_bar)
        result_group.setLayout(result_layout)

        # 添加到主显示区域
        display_layout.addWidget(raw_data_group)
        display_layout.addWidget(result_group)
        display_group.setLayout(display_layout)

        # 底部状态区域
        self.status_label = QLabel("就绪")
        self.status_label.setAlignment(Qt.AlignCenter)

        # 添加到主布局
        main_layout.addWidget(control_group)
        main_layout.addWidget(display_group)
        main_layout.addWidget(self.status_label)

        main_widget.setLayout(main_layout)
        self.setCentralWidget(main_widget)

    def import_excel(self):
        """导入Excel文件"""
        file_path, _ = QFileDialog.getOpenFileName(
            self, "选择Excel文件", "", "Excel文件 (*.xlsx *.xls)")

        if file_path:
            try:
                self.current_file = file_path
                self.current_df = pd.read_excel(file_path)
                self.excel_type = ExcelProcessor.detect_excel_type(self.current_df)

                if self.excel_type:
                    self.show_data_in_table(self.raw_data_table, self.current_df)
                    self.process_btn.setEnabled(True)
                    self.status_label.setText(f"已导入: {file_path} (类型: {self.excel_type})")
                else:
                    QMessageBox.warning(self, "错误", "无法识别的Excel格式!")
                    self.reset_state()

            except Exception as e:
                QMessageBox.critical(self, "错误", f"读取文件失败: {str(e)}")
                self.reset_state()

    def process_data(self):
        """处理数据"""
        if self.current_df is None or self.excel_type is None:
            return

        try:
            # 显示进度条
            self.progress_bar.show()
            self.progress_bar.setValue(0)
            QApplication.processEvents()  # 强制更新UI

            # 清除旧的结果
            self.clear_results()

            # 模拟处理进度
            self.simulate_progress()

            # 实际处理数据
            QTimer.singleShot(100, self.actual_data_processing)

        except Exception as e:
            QMessageBox.critical(self, "错误", f"数据处理失败: {str(e)}")
            self.reset_state()

    def simulate_progress(self):
        """模拟处理进度"""
        for i in range(1, 101):
            def update_progress(value):
                return lambda: self.progress_bar.setValue(value)

            QTimer.singleShot(i * 20, update_progress(i))

    def actual_data_processing(self):
        """实际数据处理"""
        try:
            if self.excel_type == 'A':
                # 处理A类数据
                processed_df = ExcelProcessor.process_type_a(self.current_df)

                # 创建模板1
                template1 = processed_df[['编码', '密级', '最早日期', '最晚日期', '总页数']]
                self.add_result_tab("模板1", template1)

                self.processed_data = {'模板1': template1}

            elif self.excel_type == 'B':
                # 处理B类数据
                template1, template2 = ExcelProcessor.process_type_b(self.current_df)

                # 创建模板1和模板2
                self.add_result_tab("模板1", template1)
                self.add_result_tab("模板2", template2)

                self.processed_data = {
                    '模板1': template1,
                    '模板2': template2
                }

            self.export_btn.setEnabled(True)
            self.status_label.setText(f"数据处理完成 (类型: {self.excel_type})")
            self.progress_bar.hide()

        except Exception as e:
            QMessageBox.critical(self, "错误", f"数据处理失败: {str(e)}")
            self.reset_state()

    def add_result_tab(self, title, df):
        """添加结果标签页"""
        scroll = QScrollArea()
        scroll.setWidgetResizable(True)

        table = QTableWidget()
        self.show_data_in_table(table, df)

        scroll.setWidget(table)
        self.result_tabs.addTab(scroll, title)

    def clear_results(self):
        """清除结果区域"""
        while self.result_tabs.count() > 0:
            self.result_tabs.removeTab(0)

    def export_results(self):
        """导出结果"""
        if not self.processed_data:
            return

        try:
            # 显示模板选择对话框
            dialog = TemplateSelectionDialog(list(self.processed_data.keys()), self)
            if dialog.exec_() != QDialog.Accepted:
                return

            selected_templates = dialog.get_selected_templates()
            if not selected_templates:
                QMessageBox.warning(self, "警告", "请至少选择一个模板导出!")
                return

            file_path, _ = QFileDialog.getSaveFileName(
                self, "保存结果", "", "Excel文件 (*.xlsx)")

            if file_path:
                with pd.ExcelWriter(file_path) as writer:
                    for template_name in selected_templates:
                        if template_name in self.processed_data:
                            self.processed_data[template_name].to_excel(
                                writer, sheet_name=template_name, index=False)

                self.status_label.setText(f"结果已保存到: {file_path}")
                QMessageBox.information(self, "成功", f"已导出选定的{len(selected_templates)}个模板!")

        except Exception as e:
            QMessageBox.critical(self, "错误", f"导出失败: {str(e)}")

    def show_data_in_table(self, table, df):
        """在表格中显示数据"""
        table.clear()

        # 设置行列数
        rows, cols = df.shape
        table.setRowCount(rows)
        table.setColumnCount(cols)

        # 设置表头
        table.setHorizontalHeaderLabels(df.columns)

        # 填充数据
        for i in range(rows):
            for j in range(cols):
                item = QTableWidgetItem(str(df.iat[i, j]))
                item.setFlags(item.flags() ^ Qt.ItemIsEditable)  # 设置为不可编辑
                table.setItem(i, j, item)

        # 调整列宽
        table.resizeColumnsToContents()

    def reset_state(self):
        """重置状态"""
        self.current_file = None
        self.current_df = None
        self.excel_type = None
        self.processed_data = None

        self.raw_data_table.clear()
        self.clear_results()

        self.progress_bar.hide()

        self.process_btn.setEnabled(False)
        self.export_btn.setEnabled(False)
        self.status_label.setText("就绪")


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

    # 设置应用程序样式
    app.setStyle("Fusion")

    window = MainWindow()
    window.show()
    sys.exit(app.exec_())
123