#!/usr/bin/env python # -*- coding: utf-8 -*- """业务操作处理模块""" import os import time import datetime import json import logging import tkinter as tk from tkinter import messagebox from threading import Thread from app.config.settings import ConfigManager from app.services.ocr_service import OCRService from app.services.order_service import OrderService from app.core.utils.log_utils import get_logger from .logging_ui import add_to_log, init_gui_logger, dispose_gui_logger, GUILogHandler from .ui_widgets import ProgressReporter from .error_utils import show_error_dialog, get_error_suggestion logger = get_logger(__name__) from .result_previews import show_ocr_result_preview, show_excel_result_preview, show_merge_result_preview from .user_settings import add_recent_file from .command_runner import get_running_task, set_running_task from .file_operations import select_file, select_excel_file, validate_unit_price_against_item_data def _ask_and_merge_purchase_orders(order_service, log_widget, add_to_recent=False): """弹窗询问是否合并采购单,返回合并结果路径或 None。 用于 run_pipeline_directly 和 batch_process_orders_with_status 的共享逻辑。 """ try: purchase_orders = order_service.get_purchase_orders() if len(purchase_orders) == 0: add_to_log(log_widget, "没有找到采购单文件,跳过合并步骤\n", "info") elif len(purchase_orders) == 1: add_to_log(log_widget, f"只有1个采购单文件,无需合并: {os.path.basename(purchase_orders[0])}\n", "info") else: add_to_log(log_widget, f"找到{len(purchase_orders)}个采购单文件\n", "info") file_list = "\n".join([f"• {os.path.basename(f)}" for f in purchase_orders]) merge_choice = messagebox.askyesnocancel( "采购单合并选择", f"发现{len(purchase_orders)}个采购单文件:\n\n{file_list}\n\n是否需要合并这些采购单?\n\n• 选择'是':合并所有采购单\n• 选择'否':保持文件分离\n• 选择'取消':跳过此步骤", icon='question' ) if merge_choice is True: add_to_log(log_widget, "开始合并采购单...\n", "info") merge_result = order_service.merge_all_purchase_orders() if merge_result: add_to_log(log_widget, "采购单合并完成\n", "success") if add_to_recent: try: add_recent_file(merge_result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") return merge_result else: add_to_log(log_widget, "合并失败\n", "warning") elif merge_choice is False: add_to_log(log_widget, "用户选择不合并采购单,保持文件分离\n", "info") else: add_to_log(log_widget, "用户取消合并操作\n", "info") except Exception as e: add_to_log(log_widget, f"合并过程出现问题: {str(e)}\n", "warning") return None def process_single_image_with_status(log_widget, status_bar): status_bar.set_status("选择图片中...") file_path = select_file(log_widget, [("图片文件", "*.jpg *.jpeg *.png *.bmp"), ("所有文件", "*.*")], "选择图片") if not file_path: status_bar.set_status("操作已取消") add_to_log(log_widget, "未选择文件,操作已取消\n", "warning") return def run_in_thread(): try: status_bar.set_running(True) status_bar.set_status("开始处理图片...") gui_handler = GUILogHandler(log_widget) gui_handler.setLevel(logging.INFO) formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s') gui_handler.setFormatter(formatter) root_logger = logging.getLogger() for handler in root_logger.handlers[:]: if isinstance(handler, logging.StreamHandler): root_logger.removeHandler(handler) root_logger.addHandler(gui_handler) root_logger.setLevel(logging.INFO) ocr_service = OCRService() add_to_log(log_widget, f"开始处理图片: {file_path}\n", "info") try: add_recent_file(file_path) except Exception as e: logger.debug(f"添加最近文件失败: {e}") excel_path = ocr_service.process_image(file_path) if excel_path: add_to_log(log_widget, "图片OCR处理完成\n", "success") preview_output = f"采购单已保存到: {excel_path}\n" show_excel_result_preview(preview_output) try: add_recent_file(excel_path) except Exception as e: logger.debug(f"添加最近文件失败: {e}") else: add_to_log(log_widget, "图片OCR处理失败\n", "error") except Exception as e: add_to_log(log_widget, f"处理单个图片时出错: {str(e)}\n", "error") sugg = get_error_suggestion(str(e)) if sugg: show_error_dialog("OCR处理错误", str(e), sugg) finally: try: root_logger = logging.getLogger() for handler in root_logger.handlers[:]: if isinstance(handler, GUILogHandler): root_logger.removeHandler(handler) handler.close() except Exception as e: logger.debug(f"清理日志处理器失败: {e}") status_bar.set_running(False) status_bar.set_status("就绪") thread = Thread(target=run_in_thread) thread.daemon = True thread.start() def run_pipeline_directly(log_widget, status_bar): """直接运行完整处理流程""" if get_running_task() is not None: messagebox.showinfo("任务进行中", "请等待当前任务完成后再执行新的操作。") return def run_in_thread(): set_running_task("pipeline") if status_bar: status_bar.set_running(True) status_bar.set_status("开始完整处理流程...") start_time = datetime.datetime.now() start_perf = time.perf_counter() log_widget.configure(state=tk.NORMAL) log_widget.delete(1.0, tk.END) log_widget.insert(tk.END, "执行命令: 完整处理流程\n", "command") log_widget.insert(tk.END, f"开始时间: {start_time.strftime('%Y-%m-%d %H:%M:%S')}\n", "time") log_widget.insert(tk.END, "=" * 50 + "\n\n", "separator") log_widget.configure(state=tk.DISABLED) try: config = ConfigManager() gui_handler = init_gui_logger(log_widget) ocr_service = OCRService(config) order_service = OrderService(config) reporter = ProgressReporter(status_bar) reporter.running() reporter.set("开始OCR批量处理...", 10) total, success = ocr_service.batch_process(progress_cb=lambda p: reporter.set("OCR处理中...", p)) if total == 0: add_to_log(log_widget, "没有找到需要处理的图片\n", "warning") if status_bar: status_bar.set_status("未找到图片文件") return elif success == 0: add_to_log(log_widget, "OCR处理没有成功处理任何新文件\n", "warning") else: add_to_log(log_widget, f"OCR处理完成,共处理 {success}/{total} 个文件\n", "success") try: processed_map = {} config = ConfigManager() pjson = config.get('Paths', 'processed_record', fallback='data/processed_files.json') if os.path.exists(pjson): with open(pjson, 'r', encoding='utf-8') as f: processed_map = json.load(f) outputs = list(processed_map.values()) for p in outputs[-10:]: if p: add_recent_file(os.path.abspath(p)) except Exception as e: logger.debug(f"加载已处理文件记录失败: {e}") reporter.set("开始Excel处理...", 92) add_to_log(log_widget, "开始Excel处理...\n", "info") result = order_service.process_excel() if not result: add_to_log(log_widget, "Excel处理失败\n", "error") else: add_to_log(log_widget, "Excel处理完成\n", "success") try: add_recent_file(result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") try: validate_unit_price_against_item_data(result, log_widget) except Exception as e: logger.debug(f"单价校验失败: {e}") reporter.set("检查是否需要合并采购单...", 80) _ask_and_merge_purchase_orders(order_service, log_widget, add_to_recent=True) end_time = datetime.datetime.now() duration_sec = max(0.0, time.perf_counter() - start_perf) add_to_log(log_widget, f"\n{'=' * 50}\n", "separator") add_to_log(log_widget, "完整处理流程执行完毕!\n", "success") add_to_log(log_widget, f"结束时间: {end_time.strftime('%Y-%m-%d %H:%M:%S')}\n", "time") add_to_log(log_widget, f"耗时: {duration_sec:.2f} 秒\n", "time") reporter.set("处理完成", 100) except Exception as e: add_to_log(log_widget, f"执行过程中发生错误: {str(e)}\n", "error") import traceback add_to_log(log_widget, f"详细错误信息: {traceback.format_exc()}\n", "error") finally: dispose_gui_logger() reporter.done() set_running_task(None) if status_bar: status_bar.set_running(False) status_bar.set_status("就绪") thread = Thread(target=run_in_thread) thread.daemon = True thread.start() def batch_ocr_with_status(log_widget, status_bar): """OCR批量识别""" def run_in_thread(): try: reporter = ProgressReporter(status_bar) reporter.running() reporter.set("正在进行OCR批量识别...", 10) add_to_log(log_widget, "开始OCR批量识别\n", "info") init_gui_logger(log_widget) ocr_service = OCRService() result = ocr_service.batch_process() if result: add_to_log(log_widget, "OCR批量识别完成\n", "success") show_ocr_result_preview("OCR批量识别成功完成") reporter.set("批量识别完成", 100) try: processed_map = {} config = ConfigManager() pjson = config.get('Paths', 'processed_record', fallback='data/processed_files.json') if os.path.exists(pjson): with open(pjson, 'r', encoding='utf-8') as f: processed_map = json.load(f) outputs = list(processed_map.values()) for p in outputs[-10:]: if p: add_recent_file(p) inputs = list(processed_map.keys()) for p in inputs[-10:]: if p: add_recent_file(p) except Exception as e: logger.debug(f"加载已处理文件记录失败: {e}") else: add_to_log(log_widget, "OCR批量识别失败\n", "error") except Exception as e: add_to_log(log_widget, f"OCR批量识别出错: {str(e)}\n", "error") sugg = get_error_suggestion(str(e)) if sugg: show_error_dialog("OCR处理错误", str(e), sugg) finally: dispose_gui_logger() reporter.done() thread = Thread(target=run_in_thread) thread.daemon = True thread.start() def batch_process_orders_with_status(log_widget, status_bar): """批量处理订单(仅Excel处理,包含合并确认)""" def run_in_thread(): try: reporter = ProgressReporter(status_bar) reporter.running() reporter.set("正在批量处理订单...", 10) add_to_log(log_widget, "开始批量处理订单\n", "info") init_gui_logger(log_widget) order_service = OrderService() add_to_log(log_widget, "开始Excel处理...\n", "info") try: latest_input = order_service.get_latest_excel() if latest_input: add_recent_file(latest_input) except Exception as e: logger.debug(f"获取最新Excel失败: {e}") result = order_service.process_excel(progress_cb=lambda p: reporter.set("Excel处理中...", p)) if result: add_to_log(log_widget, "Excel处理完成\n", "success") try: validate_unit_price_against_item_data(result, log_widget) except Exception as e: logger.debug(f"单价校验失败: {e}") reporter.set("检查是否需要合并采购单...", 70) add_to_log(log_widget, "检查是否需要合并采购单...\n", "info") _ask_and_merge_purchase_orders(order_service, log_widget) add_to_log(log_widget, "批量处理订单完成\n", "success") reporter.set("批量处理订单完成", 100) show_excel_result_preview(f"采购单已保存到: {result}\n") try: add_recent_file(result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") else: add_to_log(log_widget, "批量处理订单失败\n", "error") except Exception as e: add_to_log(log_widget, f"批量处理订单时出错: {str(e)}\n", "error") sugg = get_error_suggestion(str(e)) if sugg: show_error_dialog("Excel处理错误", str(e), sugg) finally: dispose_gui_logger() reporter.done() thread = Thread(target=run_in_thread) thread.daemon = True thread.start() def merge_orders_with_status(log_widget, status_bar): """合并采购单""" def run_in_thread(): try: reporter = ProgressReporter(status_bar) reporter.running() reporter.set("正在合并采购单...", 10) add_to_log(log_widget, "开始合并采购单\n", "info") init_gui_logger(log_widget) order_service = OrderService() result = order_service.merge_all_purchase_orders(progress_cb=lambda p: reporter.set("合并处理中...", p)) if result: add_to_log(log_widget, "采购单合并完成\n", "success") show_merge_result_preview(f"已保存到: {result}\n") try: add_recent_file(result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") try: validate_unit_price_against_item_data(result, log_widget) except Exception as e: logger.debug(f"单价校验失败: {e}") else: add_to_log(log_widget, "采购单合并失败\n", "error") except Exception as e: add_to_log(log_widget, f"采购单合并出错: {str(e)}\n", "error") sugg = get_error_suggestion(str(e)) if sugg: show_error_dialog("合并错误", str(e), sugg) finally: dispose_gui_logger() reporter.done() thread = Thread(target=run_in_thread) thread.daemon = True thread.start() def process_excel_file_with_status(log_widget, status_bar): """处理Excel文件""" def run_in_thread(): try: status_bar.set_running(True) status_bar.set_status("选择Excel文件中...") file_path = select_excel_file(log_widget) if file_path: status_bar.set_status("开始处理Excel文件...") add_to_log(log_widget, f"开始处理Excel文件: {file_path}\n", "info") else: status_bar.set_status("操作已取消") add_to_log(log_widget, "未选择文件,操作已取消\n", "warning") return init_gui_logger(log_widget) order_service = OrderService() if file_path: try: add_recent_file(file_path) except Exception as e: logger.debug(f"添加最近文件失败: {e}") result = order_service.process_excel(file_path, progress_cb=lambda p: status_bar.set_status("Excel处理中...", p)) else: try: latest_input = order_service.get_latest_excel() if latest_input: add_recent_file(latest_input) except Exception as e: logger.debug(f"获取最新Excel失败: {e}") result = order_service.process_excel(progress_cb=lambda p: status_bar.set_status("Excel处理中...", p)) if result: add_to_log(log_widget, "Excel文件处理完成\n", "success") show_excel_result_preview(f"采购单已保存到: {result}\n") try: add_recent_file(result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") try: validate_unit_price_against_item_data(result, log_widget) except Exception as e: logger.debug(f"单价校验失败: {e}") else: add_to_log(log_widget, "Excel文件处理失败\n", "error") except Exception as e: add_to_log(log_widget, f"Excel文件处理出错: {str(e)}\n", "error") msg = str(e) suggestion = None if 'openpyxl' in msg or 'engine' in msg: suggestion = "安装依赖:pip install openpyxl" elif 'xlrd' in msg: suggestion = "安装依赖:pip install xlrd" if suggestion: show_error_dialog("Excel处理错误", msg, suggestion) finally: dispose_gui_logger() status_bar.set_running(False) status_bar.set_status("就绪") thread = Thread(target=run_in_thread) thread.daemon = True thread.start() def process_dropped_file(log_widget, status_bar, file_path): try: ext = os.path.splitext(file_path)[1].lower() if ext in ['.jpg', '.jpeg', '.png', '.bmp']: def _run_img(): try: reporter = ProgressReporter(status_bar) reporter.running() init_gui_logger(log_widget) add_to_log(log_widget, f"开始一键处理图片: {file_path}\n", "info") try: add_recent_file(file_path) except Exception as e: logger.debug(f"添加最近文件失败: {e}") # 步骤1: OCR识别 reporter.set("OCR识别中...", 10) ocr_service = OCRService() excel_path = ocr_service.process_image(file_path) if not excel_path: add_to_log(log_widget, "图片OCR处理失败\n", "error") return add_to_log(log_widget, f"OCR识别完成: {excel_path}\n", "success") # 步骤2: Excel处理 reporter.set("Excel处理中...", 40) order_service = OrderService() result = order_service.process_excel(excel_path, progress_cb=lambda p: reporter.set("Excel处理中...", p)) if not result: add_to_log(log_widget, "Excel处理失败\n", "error") return add_to_log(log_widget, f"Excel处理完成: {result}\n", "success") try: add_recent_file(result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") try: validate_unit_price_against_item_data(result, log_widget) except Exception as e: logger.debug(f"单价校验失败: {e}") # 步骤3: 合并采购单 reporter.set("检查合并采购单...", 80) _ask_and_merge_purchase_orders(order_service, log_widget, add_to_recent=True) reporter.set("处理完成", 100) add_to_log(log_widget, "一键处理完成!\n", "success") finally: dispose_gui_logger() reporter.done() t = Thread(target=_run_img) t.daemon = True t.start() elif ext in ['.xlsx', '.xls']: def _run_xls(): try: reporter = ProgressReporter(status_bar) reporter.running() init_gui_logger(log_widget) order_service = OrderService() add_to_log(log_widget, f"开始一键处理Excel文件: {file_path}\n", "info") try: add_recent_file(file_path) except Exception as e: logger.debug(f"添加最近文件失败: {e}") # 步骤1: Excel处理 reporter.set("Excel处理中...", 20) result = order_service.process_excel(file_path, progress_cb=lambda p: reporter.set("Excel处理中...", p)) if not result: add_to_log(log_widget, "Excel文件处理失败\n", "error") return add_to_log(log_widget, f"Excel处理完成: {result}\n", "success") try: add_recent_file(result) except Exception as e: logger.debug(f"添加最近文件失败: {e}") try: validate_unit_price_against_item_data(result, log_widget) except Exception as e: logger.debug(f"单价校验失败: {e}") # 步骤2: 合并采购单 reporter.set("检查合并采购单...", 80) _ask_and_merge_purchase_orders(order_service, log_widget, add_to_recent=True) reporter.set("处理完成", 100) add_to_log(log_widget, "一键处理完成!\n", "success") finally: dispose_gui_logger() reporter.done() t = Thread(target=_run_xls) t.daemon = True t.start() else: add_to_log(log_widget, f"不支持的文件类型: {file_path}\n", "warning") except Exception as e: add_to_log(log_widget, f"处理拖拽文件失败: {str(e)}\n", "error")