Epson L4150 L4160 L4170 Resetter Adjustment Program ★ <Fast>
root.mainloop() if == " main ": main()
def disconnect_printer(self): """Disconnect from printer""" if self.serial_port and self.serial_port.is_open: self.serial_port.close() self.connected = False self.serial_port = None self.status_label.config(text="● Disconnected", foreground="red") self.connect_btn.config(state=tk.NORMAL) self.disconnect_btn.config(state=tk.DISABLED) self.reset_pad_btn.config(state=tk.DISABLED) self.reset_all_btn.config(state=tk.DISABLED) self.log_message("Disconnected from printer")
def send_command(self, command, response_length=0): """Send command to printer and return response""" if not self.serial_port or not self.serial_port.is_open: raise Exception("Printer not connected") self.serial_port.write(command) self.serial_port.flush() if response_length > 0: time.sleep(0.2) response = self.serial_port.read(response_length) return response return None Epson L4150 L4160 L4170 Resetter Adjustment Program
def log_message(self, message): """Add message to log with timestamp""" timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S") log_entry = f"[{timestamp}] {message}\n" self.log_text.insert(tk.END, log_entry) self.log_text.see(tk.END) # Limit log size if int(self.log_text.index('end-1c').split('.')[0]) > 1000: self.log_text.delete(1.0, 500.0) class EpsonResetterAdvanced: """Advanced features for Epson resetter"""
# Help menu help_menu = tk.Menu(menubar, tearoff=0) menubar.add_cascade(label="Help", menu=help_menu) help_menu.add_command(label="Instructions", command=lambda: messagebox.showinfo("Instructions", "1. Connect printer via USB\n" "2. Select COM port and baud rate (9600)\n" "3. Click Connect\n" "4. Check current waste ink counter values\n" "5. Replace waste ink pads if necessary\n" "6. Click 'Reset Waste Ink Pad Counter'\n" "7. Verify counters are reset to 0\n\n" "Supported models: Epson L4150, L4160, L4170")) help_menu.add_command(label="About", command=lambda: messagebox.showinfo("About", "Epson Resetter Adjustment Program v2.0\n\n" "For Epson L4150/L4160/L4170 printers\n\n" "Use at your own risk. Always replace waste ink pads before resetting.")) Click Connect\n" "4
def __init__(self, parent): self.parent = parent self.serial_port = None self.current_model = None self.connected = False self.setup_ui() def setup_ui(self): """Setup the main user interface""" self.parent.title("Epson Resetter Adjustment Program v2.0") self.parent.geometry("800x600") self.parent.resizable(True, True) # Set style style = ttk.Style() style.theme_use('clam') # Main container main_frame = ttk.Frame(self.parent, padding="10") main_frame.grid(row=0, column=0, sticky=(tk.W, tk.E, tk.N, tk.S)) # Connection Frame conn_frame = ttk.LabelFrame(main_frame, text="Printer Connection", padding="10") conn_frame.grid(row=0, column=0, columnspan=2, sticky=(tk.W, tk.E), pady=5) ttk.Label(conn_frame, text="COM Port:").grid(row=0, column=0, sticky=tk.W) self.port_combo = ttk.Combobox(conn_frame, values=self.get_serial_ports(), width=15) self.port_combo.grid(row=0, column=1, padx=5) ttk.Label(conn_frame, text="Baud Rate:").grid(row=0, column=2, sticky=tk.W, padx=(20,0)) self.baud_combo = ttk.Combobox(conn_frame, values=[9600, 19200, 38400, 115200], width=10) self.baud_combo.set(9600) self.baud_combo.grid(row=0, column=3, padx=5) self.connect_btn = ttk.Button(conn_frame, text="Connect", command=self.connect_printer) self.connect_btn.grid(row=0, column=4, padx=10) self.disconnect_btn = ttk.Button(conn_frame, text="Disconnect", command=self.disconnect_printer, state=tk.DISABLED) self.disconnect_btn.grid(row=0, column=5) # Status indicator self.status_label = ttk.Label(conn_frame, text="● Disconnected", foreground="red") self.status_label.grid(row=0, column=6, padx=(20,0)) # Printer Info Frame info_frame = ttk.LabelFrame(main_frame, text="Printer Information", padding="10") info_frame.grid(row=1, column=0, columnspan=2, sticky=(tk.W, tk.E), pady=5) ttk.Label(info_frame, text="Model:").grid(row=0, column=0, sticky=tk.W) self.model_var = tk.StringVar() ttk.Label(info_frame, textvariable=self.model_var, font=("Arial", 10, "bold")).grid(row=0, column=1, sticky=tk.W) ttk.Label(info_frame, text="Firmware:").grid(row=0, column=2, sticky=tk.W, padx=(20,0)) self.firmware_var = tk.StringVar() ttk.Label(info_frame, textvariable=self.firmware_var).grid(row=0, column=3, sticky=tk.W) ttk.Label(info_frame, text="Serial:").grid(row=1, column=0, sticky=tk.W) self.serial_var = tk.StringVar() ttk.Label(info_frame, textvariable=self.serial_var).grid(row=1, column=1, sticky=tk.W) # Counter Display Frame counter_frame = ttk.LabelFrame(main_frame, text="Waste Ink Counters", padding="10") counter_frame.grid(row=2, column=0, columnspan=2, sticky=(tk.W, tk.E, tk.N, tk.S), pady=5) # Create treeview for counters columns = ('Counter', 'Current Value', 'Maximum', 'Percentage') self.counter_tree = ttk.Treeview(counter_frame, columns=columns, show='headings', height=5) for col in columns: self.counter_tree.heading(col, text=col) self.counter_tree.column(col, width=150) scrollbar = ttk.Scrollbar(counter_frame, orient=tk.VERTICAL, command=self.counter_tree.yview) self.counter_tree.configure(yscrollcommand=scrollbar.set) self.counter_tree.grid(row=0, column=0, sticky=(tk.W, tk.E, tk.N, tk.S)) scrollbar.grid(row=0, column=1, sticky=(tk.N, tk.S)) ttk.Button(counter_frame, text="Refresh Counters", command=self.get_counters).grid(row=1, column=0, pady=10) # Reset Actions Frame reset_frame = ttk.LabelFrame(main_frame, text="Reset Operations", padding="10") reset_frame.grid(row=3, column=0, columnspan=2, sticky=(tk.W, tk.E), pady=5) # Reset buttons self.reset_pad_btn = ttk.Button(reset_frame, text="Reset Waste Ink Pad Counter", command=self.reset_pad_counter, state=tk.DISABLED, style="Danger.TButton") self.reset_pad_btn.grid(row=0, column=0, padx=5, pady=5) self.reset_all_btn = ttk.Button(reset_frame, text="Reset All Counters", command=self.reset_all_counters, state=tk.DISABLED) self.reset_all_btn.grid(row=0, column=1, padx=5, pady=5) # Progress bar self.progress = ttk.Progressbar(main_frame, mode='indeterminate') self.progress.grid(row=4, column=0, columnspan=2, sticky=(tk.W, tk.E), pady=10) # Log Frame log_frame = ttk.LabelFrame(main_frame, text="Operation Log", padding="10") log_frame.grid(row=5, column=0, columnspan=2, sticky=(tk.W, tk.E, tk.N, tk.S), pady=5) self.log_text = tk.Text(log_frame, height=10, wrap=tk.WORD) scroll_log = ttk.Scrollbar(log_frame, orient=tk.VERTICAL, command=self.log_text.yview) self.log_text.configure(yscrollcommand=scroll_log.set) self.log_text.grid(row=0, column=0, sticky=(tk.W, tk.E, tk.N, tk.S)) scroll_log.grid(row=0, column=1, sticky=(tk.N, tk.S)) # Configure grid weights self.parent.columnconfigure(0, weight=1) self.parent.rowconfigure(0, weight=1) main_frame.columnconfigure(0, weight=1) main_frame.rowconfigure(2, weight=1) main_frame.rowconfigure(5, weight=1) counter_frame.columnconfigure(0, weight=1) counter_frame.rowconfigure(0, weight=1) log_frame.columnconfigure(0, weight=1) log_frame.rowconfigure(0, weight=1) # Create custom styles style.configure("Danger.TButton", foreground="red") def get_serial_ports(self): """Get list of available serial ports""" ports = serial.tools.list_ports.comports() return [port.device for port in ports]
def get_counters(self): """Get waste ink counter values""" if not self.connected: messagebox.showwarning("Warning", "Printer not connected") return def fetch(): self.progress.start() try: # Clear existing items for item in self.counter_tree.get_children(): self.counter_tree.delete(item) # Request counter data response = self.send_command(self.CMD_GET_COUNTER, 32) if response and len(response) >= 20: # Parse counters (mock parsing - actual implementation varies) pad_counter1 = int.from_bytes(response[8:12], byteorder='little') pad_counter2 = int.from_bytes(response[12:16], byteorder='little') max_counter = 15000 # Typical max for these models # Add pad counter 1 percent1 = (pad_counter1 / max_counter) * 100 self.counter_tree.insert('', 'end', values=( 'Waste Ink Pad 1', f"{pad_counter1:,}", f"{max_counter:,}", f"{percent1:.1f}%" )) # Add pad counter 2 percent2 = (pad_counter2 / max_counter) * 100 self.counter_tree.insert('', 'end', values=( 'Waste Ink Pad 2', f"{pad_counter2:,}", f"{max_counter:,}", f"{percent2:.1f}%" )) # Change color based on percentage for item, percent in zip(self.counter_tree.get_children(), [percent1, percent2]): if percent > 90: self.counter_tree.tag_configure('critical', background='#ffcccc') self.counter_tree.item(item, tags=('critical',)) elif percent > 70: self.counter_tree.tag_configure('warning', background='#ffffcc') self.counter_tree.item(item, tags=('warning',)) self.log_message(f"Counters retrieved - Pad1: {pad_counter1}, Pad2: {pad_counter2}") else: self.log_message("Failed to retrieve counter data") except Exception as e: self.log_message(f"Error reading counters: {str(e)}") messagebox.showerror("Error", f"Failed to read counters: {str(e)}") finally: self.progress.stop() threading.Thread(target=fetch, daemon=True).start() Click 'Reset Waste Ink Pad Counter'\n" "7
def connect_printer(self): """Connect to the printer""" if not self.port_combo.get(): messagebox.showerror("Error", "Please select a COM port") return try: self.serial_port = serial.Serial( port=self.port_combo.get(), baudrate=int(self.baud_combo.get()), bytesize=serial.EIGHTBITS, parity=serial.PARITY_NONE, stopbits=serial.STOPBITS_ONE, timeout=3 ) # Initialize printer self.send_command(self.CMD_INITIALIZE) time.sleep(0.5) # Get printer info self.get_printer_info() self.connected = True self.status_label.config(text="● Connected", foreground="green") self.connect_btn.config(state=tk.DISABLED) self.disconnect_btn.config(state=tk.NORMAL) self.reset_pad_btn.config(state=tk.NORMAL) self.reset_all_btn.config(state=tk.NORMAL) self.log_message("Connected to printer successfully") self.get_counters() except serial.SerialException as e: messagebox.showerror("Connection Error", f"Failed to connect: {str(e)}") self.log_message(f"Connection failed: {str(e)}") except Exception as e: messagebox.showerror("Error", f"Unexpected error: {str(e)}")
# File menu file_menu = tk.Menu(menubar, tearoff=0) menubar.add_cascade(label="File", menu=file_menu) file_menu.add_command(label="Refresh Ports", command=lambda: app.port_combo.configure(values=app.get_serial_ports())) file_menu.add_separator() file_menu.add_command(label="Exit", command=root.quit)
import tkinter as tk from tkinter import ttk, messagebox, filedialog import serial import serial.tools.list_ports import time import threading import struct import hashlib import os import json from datetime import datetime class EpsonResetter: """Epson Printer Resetter for L4150/L4160/L4170"""
def get_printer_info(self): """Get printer information""" try: # Send status request response = self.send_command(self.CMD_STATUS, 64) if response and len(response) >= 32: # Parse model from response model_str = response[8:24].decode('ascii', errors='ignore').strip('\x00') # Detect model for model in self.SUPPORTED_MODELS.keys(): if model in model_str: self.current_model = model self.model_var.set(model) break else: self.model_var.set("Unknown") # Parse firmware firmware = f"{response[24]:02d}.{response[25]:02d}" self.firmware_var.set(firmware) # Parse serial (mock parsing) serial_num = response[16:24].hex().upper() self.serial_var.set(serial_num) self.log_message(f"Printer detected: {self.model_var.get()} (FW: {firmware})") except Exception as e: self.log_message(f"Failed to get printer info: {str(e)}")