Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
1 неделю назад
8 часов назад
1 неделю назад
1 день назад
1 неделю назад
1 день назад
1 неделю назад
1 день назад
1 неделю назад
1 день назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
8 часов назад
1 неделю назад
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700
  1. #!/usr/bin/env python3
  2. """
  3. Bag2 – GUI to show FPSMS job orders by plan date.
  4. Uses the public API GET /py/job-orders (no login required).
  5. UI tuned for aged users: larger font, Traditional Chinese labels, prev/next date.
  6. Run: python Bag2.py
  7. """
  8. import json
  9. import os
  10. import select
  11. import socket
  12. import sys
  13. import tempfile
  14. import threading
  15. import time
  16. import tkinter as tk
  17. from datetime import date, datetime, timedelta
  18. from tkinter import messagebox, ttk
  19. from typing import Callable, Optional
  20. import requests
  21. try:
  22. import serial
  23. except ImportError:
  24. serial = None # type: ignore
  25. try:
  26. import win32print # type: ignore[import]
  27. import win32ui # type: ignore[import]
  28. import win32con # type: ignore[import]
  29. import win32gui # type: ignore[import]
  30. except ImportError:
  31. win32print = None # type: ignore[assignment]
  32. win32ui = None # type: ignore[assignment]
  33. win32con = None # type: ignore[assignment]
  34. win32gui = None # type: ignore[assignment]
  35. try:
  36. from PIL import Image, ImageDraw, ImageFont, ImageOps
  37. try:
  38. from PIL import ImageWin # type: ignore
  39. except Exception:
  40. ImageWin = None # type: ignore[assignment]
  41. import qrcode
  42. _HAS_PIL_QR = True
  43. except ImportError:
  44. Image = None # type: ignore[assignment]
  45. ImageDraw = None # type: ignore[assignment]
  46. ImageFont = None # type: ignore[assignment]
  47. ImageOps = None # type: ignore[assignment]
  48. ImageWin = None # type: ignore[assignment]
  49. qrcode = None # type: ignore[assignment]
  50. _HAS_PIL_QR = False
  51. DEFAULT_BASE_URL = os.environ.get("FPSMS_BASE_URL", "http://localhost:8090/api")
  52. # When run as PyInstaller exe, save settings next to the exe; otherwise next to script
  53. if getattr(sys, "frozen", False):
  54. _SETTINGS_DIR = os.path.dirname(sys.executable)
  55. else:
  56. _SETTINGS_DIR = os.path.dirname(os.path.abspath(__file__))
  57. # Bag2 has its own settings file so it doesn't share with Bag1.
  58. SETTINGS_FILE = os.path.join(_SETTINGS_DIR, "bag2_settings.json")
  59. LASER_COUNTER_FILE = os.path.join(_SETTINGS_DIR, "last_batch_count.txt")
  60. DEFAULT_SETTINGS = {
  61. "api_ip": "localhost",
  62. "api_port": "8090",
  63. "dabag_ip": "",
  64. "dabag_port": "3008",
  65. "laser_ip": "192.168.17.10",
  66. "laser_port": "45678",
  67. # For 標簽機 on Windows, this is the Windows printer name, e.g. "TSC TTP-246M Pro"
  68. "label_com": "TSC TTP-246M Pro",
  69. }
  70. def load_settings() -> dict:
  71. """Load settings from JSON file; return defaults if missing or invalid."""
  72. try:
  73. if os.path.isfile(SETTINGS_FILE):
  74. with open(SETTINGS_FILE, "r", encoding="utf-8") as f:
  75. data = json.load(f)
  76. return {**DEFAULT_SETTINGS, **data}
  77. except Exception:
  78. pass
  79. return dict(DEFAULT_SETTINGS)
  80. def save_settings(settings: dict) -> None:
  81. """Save settings to JSON file."""
  82. with open(SETTINGS_FILE, "w", encoding="utf-8") as f:
  83. json.dump(settings, f, indent=2, ensure_ascii=False)
  84. def build_base_url(api_ip: str, api_port: str) -> str:
  85. ip = (api_ip or "localhost").strip()
  86. port = (api_port or "8090").strip()
  87. return f"http://{ip}:{port}/api"
  88. def try_printer_connection(printer_name: str, sett: dict) -> bool:
  89. """Try to connect to the selected printer (TCP IP:port or COM). Returns True if OK."""
  90. if printer_name == "打袋機 DataFlex":
  91. ip = (sett.get("dabag_ip") or "").strip()
  92. port_str = (sett.get("dabag_port") or "9100").strip()
  93. if not ip:
  94. return False
  95. try:
  96. port = int(port_str)
  97. s = socket.create_connection((ip, port), timeout=PRINTER_SOCKET_TIMEOUT)
  98. s.close()
  99. return True
  100. except (socket.error, ValueError, OSError):
  101. return False
  102. if printer_name == "激光機":
  103. ip = (sett.get("laser_ip") or "").strip()
  104. port_str = (sett.get("laser_port") or "45678").strip()
  105. if not ip:
  106. return False
  107. try:
  108. port = int(port_str)
  109. s = socket.create_connection((ip, port), timeout=PRINTER_SOCKET_TIMEOUT)
  110. s.close()
  111. return True
  112. except (socket.error, ValueError, OSError):
  113. return False
  114. if printer_name == "標簽機":
  115. target = (sett.get("label_com") or "").strip()
  116. if not target:
  117. return False
  118. # On Windows, allow using a Windows printer name (e.g. "TSC TTP-246M Pro")
  119. # as an alternative to a COM port. If it doesn't look like a COM port,
  120. # try opening it via the Windows print spooler.
  121. if os.name == "nt" and not target.upper().startswith("COM"):
  122. if win32print is None:
  123. return False
  124. try:
  125. handle = win32print.OpenPrinter(target)
  126. win32print.ClosePrinter(handle)
  127. return True
  128. except Exception:
  129. return False
  130. # Fallback: treat as serial COM port (original behaviour)
  131. if serial is None:
  132. return False
  133. try:
  134. ser = serial.Serial(target, timeout=1)
  135. ser.close()
  136. return True
  137. except (serial.SerialException, OSError):
  138. return False
  139. return False
  140. # Larger font for aged users (point size)
  141. FONT_SIZE = 16
  142. FONT_SIZE_BUTTONS = 15
  143. FONT_SIZE_QTY = 12 # smaller for 需求數量 under batch no.
  144. FONT_SIZE_META = 11 # single-line 需求/已印 (compact list)
  145. # Less vertical padding so ~30 rows fit more comfortably
  146. LIST_ROW_PADY = 2
  147. LIST_ROW_IPADY = 5
  148. FONT_SIZE_ITEM = 20 # item code and item name (larger for readability)
  149. FONT_FAMILY = "Microsoft JhengHei UI" # Traditional Chinese; fallback to TkDefaultFont
  150. FONT_SIZE_ITEM_CODE = 20 # item code (larger for readability)
  151. FONT_SIZE_ITEM_NAME = 26 # item name (bigger than item code)
  152. # Column widths: fixed frame widths so 品號/品名 columns line up across rows
  153. LEFT_COL_WIDTH_PX = 300 # 工單 + 需求/已印 block
  154. ITEM_CODE_WRAP = 140 # Label wraplength (px)
  155. # Narrower than wrap+padding so short codes sit closer to 品名 (still aligned across rows)
  156. CODE_COL_WIDTH_PX = ITEM_CODE_WRAP + 6
  157. ITEM_NAME_WRAP = 640 # item name wraps in remaining space
  158. # Light blue theme (softer than pure grey)
  159. BG_TOP = "#E8F4FC"
  160. BG_LIST = "#D4E8F7"
  161. BG_ROOT = "#E1F0FF"
  162. BG_ROW = "#C5E1F5"
  163. BG_ROW_SELECTED = "#6BB5FF" # highlighted when selected (for printing)
  164. # Connection status bar
  165. BG_STATUS_ERROR = "#FFCCCB" # red background when disconnected
  166. FG_STATUS_ERROR = "#B22222" # red text
  167. BG_STATUS_OK = "#90EE90" # light green when connected
  168. FG_STATUS_OK = "#006400" # green text
  169. RETRY_MS = 30 * 1000 # 30 seconds reconnect
  170. REFRESH_MS = 60 * 1000 # 60 seconds refresh when connected
  171. PRINTER_CHECK_MS = 60 * 1000 # 1 minute when printer OK
  172. PRINTER_RETRY_MS = 30 * 1000 # 30 seconds when printer failed
  173. PRINTER_SOCKET_TIMEOUT = 3
  174. DATAFLEX_SEND_TIMEOUT = 10 # seconds when sending ZPL to DataFlex
  175. def _zpl_escape(s: str) -> str:
  176. """Escape text for ZPL ^FD...^FS (backslash and caret)."""
  177. return s.replace("\\", "\\\\").replace("^", "\\^")
  178. def generate_zpl_dataflex(
  179. batch_no: str,
  180. item_code: str,
  181. item_name: str,
  182. item_id: Optional[int] = None,
  183. stock_in_line_id: Optional[int] = None,
  184. lot_no: Optional[str] = None,
  185. font_regular: str = "E:STXihei.ttf",
  186. font_bold: str = "E:STXihei.ttf",
  187. ) -> str:
  188. """
  189. Row 1 (from zero): QR code, then item name (rotated 90°).
  190. Row 2: Batch/lot (left), item code (right).
  191. Label and QR use lotNo from API when present, else batch_no (Bxxxxx).
  192. """
  193. desc = _zpl_escape((item_name or "—").strip())
  194. code = _zpl_escape((item_code or "—").strip())
  195. label_line = (lot_no or batch_no or "").strip()
  196. label_esc = _zpl_escape(label_line)
  197. # QR payload: prefer JSON {"itemId":..., "stockInLineId":...} when both present; else fall back to lot/batch text
  198. if item_id is not None and stock_in_line_id is not None:
  199. qr_payload = json.dumps({"itemId": item_id, "stockInLineId": stock_in_line_id})
  200. else:
  201. qr_payload = label_line if label_line else batch_no.strip()
  202. qr_value = _zpl_escape(qr_payload)
  203. return f"""^XA
  204. ^CI28
  205. ^PW700
  206. ^LL500
  207. ^PO N
  208. ^FO10,20
  209. ^BQN,2,4^FDQA,{qr_value}^FS
  210. ^FO170,20
  211. ^A@R,72,72,{font_regular}^FD{desc}^FS
  212. ^FO0,200
  213. ^A@R,72,72,{font_regular}^FD{label_esc}^FS
  214. ^FO55,200
  215. ^A@R,88,88,{font_bold}^FD{code}^FS
  216. ^XZ"""
  217. def generate_zpl_label_small(
  218. batch_no: str,
  219. item_code: str,
  220. item_name: str,
  221. item_id: Optional[int] = None,
  222. stock_in_line_id: Optional[int] = None,
  223. lot_no: Optional[str] = None,
  224. font: str = "MingLiUHKSCS",
  225. ) -> str:
  226. """
  227. ZPL for 標簽機. Row 1: item name. Row 2: QR left | item code + lot no (or batch) right.
  228. QR contains {"itemId": xxx, "stockInLineId": xxx} when both present; else batch_no.
  229. Unicode (^CI28); font set for Big-5 (e.g. MingLiUHKSCS).
  230. """
  231. desc = _zpl_escape((item_name or "—").strip())
  232. code = _zpl_escape((item_code or "—").strip())
  233. label_line2 = (lot_no or batch_no or "—").strip()
  234. label_line2_esc = _zpl_escape(label_line2)
  235. if item_id is not None and stock_in_line_id is not None:
  236. qr_data = _zpl_escape(json.dumps({"itemId": item_id, "stockInLineId": stock_in_line_id}))
  237. else:
  238. qr_data = f"QA,{batch_no}"
  239. return f"""^XA
  240. ^CI28
  241. ^PW500
  242. ^LL500
  243. ^FO10,15
  244. ^FB480,3,0,L,0
  245. ^A@N,38,38,{font}^FD{desc}^FS
  246. ^FO10,110
  247. ^BQN,2,6^FD{qr_data}^FS
  248. ^FO150,110
  249. ^A@N,48,48,{font}^FD{code}^FS
  250. ^FO150,175
  251. ^A@N,40,40,{font}^FD{label_line2_esc}^FS
  252. ^XZ"""
  253. # Label image size (pixels) for 標簽機 image printing.
  254. # Enlarged for readability (approx +90% scale).
  255. LABEL_IMAGE_W = 720
  256. LABEL_IMAGE_H = 530
  257. LABEL_PADDING = 23
  258. LABEL_FONT_NAME_SIZE = 42
  259. LABEL_FONT_CODE_SIZE = 49
  260. LABEL_FONT_BATCH_SIZE = 34
  261. LABEL_QR_SIZE = 210
  262. def _get_chinese_font(size: int) -> Optional["ImageFont.FreeTypeFont"]:
  263. """Return a Chinese-capable font for PIL, or None to use default."""
  264. if ImageFont is None:
  265. return None
  266. # Prefer real font files on Windows (font *names* may fail and silently fallback).
  267. if os.name == "nt":
  268. fonts_dir = os.path.join(os.environ.get("WINDIR", r"C:\Windows"), "Fonts")
  269. for rel in (
  270. "msjh.ttc", # Microsoft JhengHei
  271. "msjhl.ttc", # Microsoft JhengHei Light
  272. "msjhbd.ttc", # Microsoft JhengHei Bold
  273. "mingliu.ttc",
  274. "mingliub.ttc",
  275. "kaiu.ttf",
  276. "msyh.ttc", # Microsoft YaHei
  277. "msyhbd.ttc",
  278. "simhei.ttf",
  279. "simsun.ttc",
  280. ):
  281. p = os.path.join(fonts_dir, rel)
  282. try:
  283. if os.path.exists(p):
  284. return ImageFont.truetype(p, size)
  285. except (OSError, IOError):
  286. continue
  287. # Fallback: try common font names (may still work depending on Pillow build)
  288. for name in (
  289. "Microsoft JhengHei UI",
  290. "Microsoft JhengHei",
  291. "MingLiU",
  292. "MingLiU_HKSCS",
  293. "Microsoft YaHei",
  294. "SimHei",
  295. "SimSun",
  296. ):
  297. try:
  298. return ImageFont.truetype(name, size)
  299. except (OSError, IOError):
  300. continue
  301. try:
  302. return ImageFont.load_default()
  303. except Exception:
  304. return None
  305. def render_label_to_image(
  306. batch_no: str,
  307. item_code: str,
  308. item_name: str,
  309. item_id: Optional[int] = None,
  310. stock_in_line_id: Optional[int] = None,
  311. lot_no: Optional[str] = None,
  312. ) -> "Image.Image":
  313. """
  314. Render 標簽機 label as a PIL Image (white bg, black text + QR).
  315. Use this image for printing so Chinese displays correctly; words are drawn bigger.
  316. Requires Pillow and qrcode. Raises RuntimeError if not available.
  317. """
  318. if not _HAS_PIL_QR or Image is None or qrcode is None:
  319. raise RuntimeError("Pillow and qrcode are required for image labels. Run: pip install Pillow qrcode[pil]")
  320. img = Image.new("RGB", (LABEL_IMAGE_W, LABEL_IMAGE_H), "white")
  321. draw = ImageDraw.Draw(img)
  322. # QR payload (same as ZPL)
  323. if item_id is not None and stock_in_line_id is not None:
  324. qr_data = json.dumps({"itemId": item_id, "stockInLineId": stock_in_line_id})
  325. else:
  326. qr_data = f"QA,{batch_no}"
  327. # Draw QR top-left area
  328. qr = qrcode.QRCode(box_size=4, border=2)
  329. qr.add_data(qr_data)
  330. qr.make(fit=True)
  331. qr_img = qr.make_image(fill_color="black", back_color="white")
  332. _resample = getattr(Image, "Resampling", Image).NEAREST
  333. qr_img = qr_img.resize((LABEL_QR_SIZE, LABEL_QR_SIZE), _resample)
  334. img.paste(qr_img, (LABEL_PADDING, LABEL_PADDING))
  335. # Fonts (bigger for readability)
  336. font_name = _get_chinese_font(LABEL_FONT_NAME_SIZE)
  337. font_code = _get_chinese_font(LABEL_FONT_CODE_SIZE)
  338. font_batch = _get_chinese_font(LABEL_FONT_BATCH_SIZE)
  339. x_right = LABEL_PADDING + LABEL_QR_SIZE + LABEL_PADDING
  340. y_line = LABEL_PADDING
  341. # Line 1: item name (wrap within remaining width)
  342. name_str = (item_name or "—").strip()
  343. max_name_w = LABEL_IMAGE_W - x_right - LABEL_PADDING
  344. if font_name:
  345. # Wrap rule: after 7 "words" (excl. parentheses). ()() not counted; +=*/. and A–Z/a–z count as 0.5.
  346. def _wrap_text(text: str, font, max_width: int) -> list:
  347. ignore = set("()()")
  348. half = set("+=*/.abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ")
  349. max_count = 6.5
  350. lines: list[str] = []
  351. current: list[str] = []
  352. count = 0.0
  353. for ch in text:
  354. if ch == "\n":
  355. lines.append("".join(current).strip())
  356. current = []
  357. count = 0.0
  358. continue
  359. ch_count = 0.0 if ch in ignore else (0.5 if ch in half else 1.0)
  360. if count + ch_count > max_count and current:
  361. lines.append("".join(current).strip())
  362. current = []
  363. count = 0.0
  364. current.append(ch)
  365. count += ch_count
  366. if current:
  367. lines.append("".join(current).strip())
  368. # Max 2 rows for item name. If still long, keep everything in row 2.
  369. if len(lines) > 2:
  370. lines = [lines[0], "".join(lines[1:]).strip()]
  371. # Safety: if any line still exceeds pixel width, wrap by width as well.
  372. if hasattr(draw, "textbbox"):
  373. out: list[str] = []
  374. for ln in lines:
  375. buf: list[str] = []
  376. for ch in ln:
  377. buf.append(ch)
  378. bbox = draw.textbbox((0, 0), "".join(buf), font=font)
  379. if bbox[2] - bbox[0] > max_width and len(buf) > 1:
  380. out.append("".join(buf[:-1]).strip())
  381. buf = [buf[-1]]
  382. if buf:
  383. out.append("".join(buf).strip())
  384. out = [x for x in out if x]
  385. if len(out) > 2:
  386. out = [out[0], "".join(out[1:]).strip()]
  387. return out
  388. lines = [x for x in lines if x]
  389. if len(lines) > 2:
  390. lines = [lines[0], "".join(lines[1:]).strip()]
  391. return lines
  392. lines = _wrap_text(name_str, font_name, max_name_w)
  393. for i, ln in enumerate(lines):
  394. draw.text((x_right, y_line + i * (LABEL_FONT_NAME_SIZE + 4)), ln, font=font_name, fill="black")
  395. y_line += len(lines) * (LABEL_FONT_NAME_SIZE + 4) + 8
  396. else:
  397. draw.text((x_right, y_line), name_str[:30], fill="black")
  398. y_line += LABEL_FONT_NAME_SIZE + 12
  399. # Item code (bigger)
  400. code_str = (item_code or "—").strip()
  401. if font_code:
  402. draw.text((x_right, y_line), code_str, font=font_code, fill="black")
  403. else:
  404. draw.text((x_right, y_line), code_str, fill="black")
  405. y_line += LABEL_FONT_CODE_SIZE + 6
  406. # Batch/lot line
  407. batch_str = (lot_no or batch_no or "—").strip()
  408. if font_batch:
  409. draw.text((x_right, y_line), batch_str, font=font_batch, fill="black")
  410. else:
  411. draw.text((x_right, y_line), batch_str, fill="black")
  412. return img
  413. def _image_to_zpl_gfa(pil_image: "Image.Image") -> str:
  414. """
  415. Convert a PIL image into ZPL ^GFA (ASCII hex) so we can print Chinese reliably
  416. on ZPL printers (USB/Windows printer or COM) without relying on GDI drivers.
  417. """
  418. if Image is None or ImageOps is None:
  419. raise RuntimeError("Pillow is required for image-to-ZPL conversion.")
  420. # Convert to 1-bit monochrome bitmap. Invert so '1' bits represent black in ZPL.
  421. img_bw = ImageOps.invert(pil_image.convert("L")).convert("1")
  422. w, h = img_bw.size
  423. bytes_per_row = (w + 7) // 8
  424. raw = img_bw.tobytes()
  425. total = bytes_per_row * h
  426. # Ensure length matches expected (Pillow should already pack per row).
  427. if len(raw) != total:
  428. raw = raw[:total].ljust(total, b"\x00")
  429. hex_data = raw.hex().upper()
  430. return f"""^XA
  431. ^PW{w}
  432. ^LL{h}
  433. ^FO0,0
  434. ^GFA,{total},{total},{bytes_per_row},{hex_data}
  435. ^FS
  436. ^XZ"""
  437. def send_image_to_label_printer(printer_name: str, pil_image: "Image.Image") -> None:
  438. """
  439. Send a PIL Image to 標簽機 via Windows GDI (so Chinese and graphics print correctly).
  440. Only supported when target is a Windows printer name (not COM port). Requires pywin32.
  441. """
  442. dest = (printer_name or "").strip()
  443. if not dest:
  444. raise ValueError("Label printer destination is empty.")
  445. if os.name != "nt" or dest.upper().startswith("COM"):
  446. raise RuntimeError("Image printing is only supported for a Windows printer name (e.g. TSC TTP-246M Pro).")
  447. if win32print is None or win32ui is None or win32con is None or win32gui is None:
  448. raise RuntimeError("pywin32 is required. Run: pip install pywin32")
  449. dc = win32ui.CreateDC()
  450. dc.CreatePrinterDC(dest)
  451. dc.StartDoc("FPSMS Label")
  452. dc.StartPage()
  453. try:
  454. bmp_w = pil_image.width
  455. bmp_h = pil_image.height
  456. # Scale-to-fit printable area (important for smaller physical labels).
  457. try:
  458. page_w = int(dc.GetDeviceCaps(win32con.HORZRES))
  459. page_h = int(dc.GetDeviceCaps(win32con.VERTRES))
  460. except Exception:
  461. page_w, page_h = bmp_w, bmp_h
  462. if page_w <= 0 or page_h <= 0:
  463. page_w, page_h = bmp_w, bmp_h
  464. scale = min(page_w / max(1, bmp_w), page_h / max(1, bmp_h))
  465. out_w = max(1, int(bmp_w * scale))
  466. out_h = max(1, int(bmp_h * scale))
  467. x0 = max(0, (page_w - out_w) // 2)
  468. y0 = max(0, (page_h - out_h) // 2)
  469. # Most reliable: render via Pillow ImageWin directly to printer DC.
  470. if ImageWin is not None:
  471. dib = ImageWin.Dib(pil_image.convert("RGB"))
  472. dib.draw(dc.GetHandleOutput(), (x0, y0, x0 + out_w, y0 + out_h))
  473. else:
  474. # Fallback: Draw image to printer DC via temp BMP (GDI uses BMP)
  475. with tempfile.NamedTemporaryFile(suffix=".bmp", delete=False) as f:
  476. tmp_bmp = f.name
  477. try:
  478. pil_image.save(tmp_bmp, "BMP")
  479. hbm = win32gui.LoadImage(
  480. 0, tmp_bmp, win32con.IMAGE_BITMAP, 0, 0,
  481. win32con.LR_LOADFROMFILE | win32con.LR_CREATEDIBSECTION,
  482. )
  483. if hbm == 0:
  484. raise RuntimeError("Failed to load label image as bitmap.")
  485. try:
  486. mem_dc = win32ui.CreateDCFromHandle(win32gui.CreateCompatibleDC(dc.GetSafeHdc()))
  487. bmp = getattr(win32ui, "CreateBitmapFromHandle", lambda h: win32ui.PyCBitmap.FromHandle(h))(hbm)
  488. mem_dc.SelectObject(bmp)
  489. dc.StretchBlt((x0, y0), (out_w, out_h), mem_dc, (0, 0), (bmp_w, bmp_h), win32con.SRCCOPY)
  490. finally:
  491. win32gui.DeleteObject(hbm)
  492. finally:
  493. try:
  494. os.unlink(tmp_bmp)
  495. except OSError:
  496. pass
  497. finally:
  498. dc.EndPage()
  499. dc.EndDoc()
  500. def send_zpl_to_dataflex(ip: str, port: int, zpl: str) -> None:
  501. """Send ZPL to DataFlex printer via TCP. Raises on connection/send error."""
  502. sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  503. sock.settimeout(DATAFLEX_SEND_TIMEOUT)
  504. try:
  505. sock.connect((ip, port))
  506. sock.sendall(zpl.encode("utf-8"))
  507. finally:
  508. sock.close()
  509. def send_zpl_to_label_printer(target: str, zpl: str) -> None:
  510. """
  511. Send ZPL to 標簽機.
  512. On Windows, if target is not a COM port (e.g. "TSC TTP-246M Pro"),
  513. send raw ZPL to the named Windows printer via the spooler.
  514. Otherwise, treat target as a serial COM port (original behaviour).
  515. """
  516. dest = (target or "").strip()
  517. if not dest:
  518. raise ValueError("Label printer destination is empty.")
  519. # Unicode (^CI28); send UTF-8 to 標簽機
  520. raw_bytes = zpl.encode("utf-8")
  521. # Windows printer name path (USB printer installed as normal printer)
  522. if os.name == "nt" and not dest.upper().startswith("COM"):
  523. if win32print is None:
  524. raise RuntimeError("pywin32 not installed. Run: pip install pywin32")
  525. handle = win32print.OpenPrinter(dest)
  526. try:
  527. job = win32print.StartDocPrinter(handle, 1, ("FPSMS Label", None, "RAW"))
  528. win32print.StartPagePrinter(handle)
  529. win32print.WritePrinter(handle, raw_bytes)
  530. win32print.EndPagePrinter(handle)
  531. win32print.EndDocPrinter(handle)
  532. finally:
  533. win32print.ClosePrinter(handle)
  534. return
  535. # Fallback: serial COM port
  536. if serial is None:
  537. raise RuntimeError("pyserial not installed. Run: pip install pyserial")
  538. ser = serial.Serial(dest, timeout=5)
  539. try:
  540. ser.write(raw_bytes)
  541. finally:
  542. ser.close()
  543. def load_laser_last_count() -> tuple[int, Optional[str]]:
  544. """Load last batch count and date from laser counter file. Returns (count, date_str)."""
  545. if not os.path.exists(LASER_COUNTER_FILE):
  546. return 0, None
  547. try:
  548. with open(LASER_COUNTER_FILE, "r", encoding="utf-8") as f:
  549. lines = f.read().strip().splitlines()
  550. if len(lines) >= 2:
  551. return int(lines[1].strip()), lines[0].strip()
  552. except Exception:
  553. pass
  554. return 0, None
  555. def save_laser_last_count(date_str: str, count: int) -> None:
  556. """Save laser batch count and date to file."""
  557. try:
  558. with open(LASER_COUNTER_FILE, "w", encoding="utf-8") as f:
  559. f.write(f"{date_str}\n{count}")
  560. except Exception:
  561. pass
  562. LASER_PUSH_INTERVAL = 2 # seconds between pushes (like sample script)
  563. # Click row with 激光機 selected: send this many times, delay between sends (not after last).
  564. LASER_ROW_SEND_COUNT = 3
  565. LASER_ROW_SEND_DELAY_SEC = 3
  566. def laser_push_loop(
  567. ip: str,
  568. port: int,
  569. stop_event: threading.Event,
  570. root: tk.Tk,
  571. on_error: Callable[[str], None],
  572. ) -> None:
  573. """
  574. Run in a background thread: persistent connection to EZCAD, push B{yymmdd}{count:03d};;
  575. every LASER_PUSH_INTERVAL seconds. Resets count each new day. Uses counter file.
  576. """
  577. conn = None
  578. push_count, last_saved_date = load_laser_last_count()
  579. while not stop_event.is_set():
  580. try:
  581. if conn is None:
  582. conn = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  583. conn.settimeout(0.4)
  584. conn.connect((ip, port))
  585. now = datetime.now()
  586. today_str = now.strftime("%y%m%d")
  587. if last_saved_date != today_str:
  588. push_count = 1
  589. last_saved_date = today_str
  590. batch = f"B{today_str}{push_count:03d}"
  591. reply = f"{batch};;"
  592. conn.sendall(reply.encode("utf-8"))
  593. save_laser_last_count(today_str, push_count)
  594. rlist, _, _ = select.select([conn], [], [], 0.4)
  595. if rlist:
  596. data = conn.recv(4096)
  597. if not data:
  598. conn.close()
  599. conn = None
  600. push_count += 1
  601. for _ in range(int(LASER_PUSH_INTERVAL * 2)):
  602. if stop_event.is_set():
  603. break
  604. time.sleep(0.5)
  605. except socket.timeout:
  606. pass
  607. except Exception as e:
  608. if conn:
  609. try:
  610. conn.close()
  611. except Exception:
  612. pass
  613. conn = None
  614. try:
  615. root.after(0, lambda msg=str(e): on_error(msg))
  616. except Exception:
  617. pass
  618. for _ in range(6):
  619. if stop_event.is_set():
  620. break
  621. time.sleep(0.5)
  622. if conn:
  623. try:
  624. conn.close()
  625. except Exception:
  626. pass
  627. def send_job_to_laser(
  628. conn_ref: list,
  629. ip: str,
  630. port: int,
  631. item_id: Optional[int],
  632. stock_in_line_id: Optional[int],
  633. item_code: str,
  634. item_name: str,
  635. ) -> tuple[bool, str]:
  636. """
  637. Send to laser using `;` separated 3 params:
  638. {"itemID": itemId, "stockInLineId": stockInLineId} ; itemCode ; itemName ;;
  639. conn_ref: [socket or None] - reused across calls; closed only when switching printer.
  640. When both item_id and stock_in_line_id present, sends JSON first param; else fallback: 0;item_code;item_name;;
  641. Returns (success, message).
  642. """
  643. code_str = (item_code or "").strip().replace(";", ",")
  644. name_str = (item_name or "").strip().replace(";", ",")
  645. if item_id is not None and stock_in_line_id is not None:
  646. # Use compact JSON so device-side parser doesn't get spaces.
  647. json_part = json.dumps(
  648. {"itemID": item_id, "stockInLineId": stock_in_line_id},
  649. separators=(",", ":"),
  650. )
  651. reply = f"{json_part};{code_str};{name_str};;"
  652. else:
  653. reply = f"0;{code_str};{name_str};;"
  654. conn = conn_ref[0]
  655. try:
  656. if conn is None:
  657. conn = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  658. conn.settimeout(3.0)
  659. conn.connect((ip, port))
  660. conn_ref[0] = conn
  661. conn.settimeout(3.0)
  662. conn.sendall(reply.encode("utf-8"))
  663. conn.settimeout(0.5)
  664. try:
  665. data = conn.recv(4096)
  666. if data:
  667. ack = data.decode("utf-8", errors="ignore").strip().lower()
  668. if "receive" in ack and "invalid" not in ack:
  669. return True, f"已送出激光機:{reply}(已確認)"
  670. except socket.timeout:
  671. pass
  672. return True, f"已送出激光機:{reply}"
  673. except (ConnectionRefusedError, socket.timeout, OSError) as e:
  674. if conn_ref[0] is not None:
  675. try:
  676. conn_ref[0].close()
  677. except Exception:
  678. pass
  679. conn_ref[0] = None
  680. if isinstance(e, ConnectionRefusedError):
  681. return False, f"無法連線至 {ip}:{port},請確認激光機已開機且 IP 正確。"
  682. if isinstance(e, socket.timeout):
  683. return False, f"連線逾時 ({ip}:{port}),請檢查網路與連接埠。"
  684. return False, f"激光機送出失敗:{e}"
  685. def send_job_to_laser_with_retry(
  686. conn_ref: list,
  687. ip: str,
  688. port: int,
  689. item_id: Optional[int],
  690. stock_in_line_id: Optional[int],
  691. item_code: str,
  692. item_name: str,
  693. ) -> tuple[bool, str]:
  694. """Send job to laser; on failure, retry once. Returns (success, message)."""
  695. ok, msg = send_job_to_laser(conn_ref, ip, port, item_id, stock_in_line_id, item_code, item_name)
  696. if ok:
  697. return True, msg
  698. ok2, msg2 = send_job_to_laser(conn_ref, ip, port, item_id, stock_in_line_id, item_code, item_name)
  699. return ok2, msg2
  700. def run_laser_row_send_thread(
  701. root: tk.Tk,
  702. laser_conn_ref: list,
  703. laser_busy_ref: list,
  704. ip: str,
  705. port: int,
  706. item_id: Optional[int],
  707. stock_in_line_id: Optional[int],
  708. item_code: str,
  709. item_name: str,
  710. set_status_message: Callable[[str, bool], None],
  711. base_url: Optional[str] = None,
  712. job_order_id: Optional[int] = None,
  713. on_recorded: Optional[Callable[[], None]] = None,
  714. ) -> None:
  715. """
  716. On row click with 激光機: send LASER_ROW_SEND_COUNT times with LASER_ROW_SEND_DELAY_SEC between sends.
  717. UI updates on main thread; work runs in background so the window does not freeze.
  718. After success, POST LASER qty to API when job_order_id and base_url are set.
  719. """
  720. if laser_busy_ref[0]:
  721. return
  722. laser_busy_ref[0] = True
  723. def worker() -> None:
  724. try:
  725. n = LASER_ROW_SEND_COUNT
  726. for i in range(n):
  727. ok, msg = send_job_to_laser_with_retry(
  728. laser_conn_ref,
  729. ip,
  730. port,
  731. item_id,
  732. stock_in_line_id,
  733. item_code,
  734. item_name,
  735. )
  736. if not ok:
  737. root.after(
  738. 0,
  739. lambda m=msg: messagebox.showwarning("激光機", m),
  740. )
  741. return
  742. if i < n - 1:
  743. time.sleep(LASER_ROW_SEND_DELAY_SEC)
  744. posted = False
  745. if base_url and job_order_id is not None:
  746. try:
  747. submit_job_order_print_submit(base_url, int(job_order_id), n, "LASER")
  748. posted = True
  749. except requests.RequestException as ex:
  750. root.after(
  751. 0,
  752. lambda err=str(ex): messagebox.showwarning(
  753. "激光機",
  754. f"已發送,但伺服器記錄失敗:{err}",
  755. ),
  756. )
  757. root.after(
  758. 0,
  759. lambda: set_status_message("已發送", is_error=False),
  760. )
  761. if on_recorded is not None and posted:
  762. root.after(0, on_recorded)
  763. except Exception as e:
  764. root.after(
  765. 0,
  766. lambda err=str(e): messagebox.showwarning("激光機", f"送出失敗:{err}"),
  767. )
  768. finally:
  769. laser_busy_ref[0] = False
  770. threading.Thread(target=worker, daemon=True).start()
  771. def _printed_qty_int(raw) -> int:
  772. """Parse API printed qty field (may be float JSON) to int."""
  773. try:
  774. return int(float(raw)) if raw is not None else 0
  775. except (TypeError, ValueError):
  776. return 0
  777. def _filter_job_orders_by_search(data: list, needle: str) -> list:
  778. """Substring match on item code, job order code, item name, lot (case-insensitive)."""
  779. n = needle.strip().lower()
  780. if not n:
  781. return data
  782. out: list = []
  783. for jo in data:
  784. parts = [
  785. str(jo.get("itemCode") or ""),
  786. str(jo.get("code") or ""),
  787. str(jo.get("itemName") or ""),
  788. str(jo.get("lotNo") or ""),
  789. ]
  790. if any(n in p.lower() for p in parts):
  791. out.append(jo)
  792. return out
  793. def format_qty(val) -> str:
  794. """Format quantity: integer without .0, with thousand separator."""
  795. if val is None:
  796. return "—"
  797. try:
  798. n = float(val)
  799. if n == int(n):
  800. return f"{int(n):,}"
  801. return f"{n:,.2f}".rstrip("0").rstrip(".")
  802. except (TypeError, ValueError):
  803. return str(val)
  804. def batch_no(year: int, job_order_id: int) -> str:
  805. """Batch no.: B + 4-digit year + jobOrderId zero-padded to 6 digits."""
  806. return f"B{year}{job_order_id:06d}"
  807. def get_font(size: int = FONT_SIZE, bold: bool = False) -> tuple:
  808. try:
  809. return (FONT_FAMILY, size, "bold" if bold else "normal")
  810. except Exception:
  811. return ("TkDefaultFont", size, "bold" if bold else "normal")
  812. def fetch_job_orders(base_url: str, plan_start: date) -> list:
  813. """Call GET /py/job-orders and return the JSON list."""
  814. url = f"{base_url.rstrip('/')}/py/job-orders"
  815. params = {"planStart": plan_start.isoformat()}
  816. resp = requests.get(url, params=params, timeout=30)
  817. resp.raise_for_status()
  818. return resp.json()
  819. def submit_job_order_print_submit(
  820. base_url: str,
  821. job_order_id: int,
  822. qty: int,
  823. print_channel: str = "LABEL",
  824. ) -> None:
  825. """POST /py/job-order-print-submit — one row per submit for DB wastage/stock tracking."""
  826. url = f"{base_url.rstrip('/')}/py/job-order-print-submit"
  827. resp = requests.post(
  828. url,
  829. json={"jobOrderId": job_order_id, "qty": qty, "printChannel": print_channel},
  830. timeout=30,
  831. )
  832. resp.raise_for_status()
  833. def set_row_highlight(row_frame: tk.Frame, selected: bool) -> None:
  834. """Set row and all nested Frame/Label children to selected or normal background."""
  835. bg = BG_ROW_SELECTED if selected else BG_ROW
  836. def _paint(w: tk.Misc) -> None:
  837. if isinstance(w, (tk.Frame, tk.Label)):
  838. w.configure(bg=bg)
  839. for c in w.winfo_children():
  840. _paint(c)
  841. _paint(row_frame)
  842. def on_job_order_click(jo: dict, batch: str) -> None:
  843. """Show message and highlight row (keeps printing to selected printer)."""
  844. item_code = jo.get("itemCode") or "—"
  845. item_name = jo.get("itemName") or "—"
  846. messagebox.showinfo(
  847. "工單",
  848. f'已點選:批次 {batch}\n品號 {item_code} {item_name}',
  849. )
  850. def ask_label_count(parent: tk.Tk) -> Optional[int]:
  851. """
  852. When printer is 標簽機, ask how many labels to print:
  853. optional direct qty in text field, +50/+10/+5/+1, 重置, then 確認送出.
  854. Returns count (>= 1), or None if cancelled.
  855. """
  856. result: list[Optional[int]] = [None]
  857. qty_var = tk.StringVar(value="0")
  858. win = tk.Toplevel(parent)
  859. win.title("標簽印數")
  860. win.geometry("580x280")
  861. win.transient(parent)
  862. win.grab_set()
  863. win.configure(bg=BG_TOP)
  864. ttk.Label(win, text="印多少個?", font=get_font(FONT_SIZE)).pack(pady=(12, 4))
  865. entry_row = tk.Frame(win, bg=BG_TOP)
  866. entry_row.pack(pady=8)
  867. tk.Label(entry_row, text="需求數量:", font=get_font(FONT_SIZE), bg=BG_TOP).pack(side=tk.LEFT, padx=(0, 6))
  868. qty_entry = tk.Entry(
  869. entry_row,
  870. textvariable=qty_var,
  871. width=12,
  872. font=get_font(FONT_SIZE),
  873. bg="white",
  874. justify=tk.RIGHT,
  875. )
  876. qty_entry.pack(side=tk.LEFT, padx=4)
  877. def current_qty() -> int:
  878. s = (qty_var.get() or "").strip().replace(",", "")
  879. if not s:
  880. return 0
  881. try:
  882. return max(0, int(s))
  883. except ValueError:
  884. return 0
  885. def reset_qty() -> None:
  886. qty_var.set("0")
  887. ttk.Button(entry_row, text="重置", command=reset_qty, width=8).pack(side=tk.LEFT, padx=8)
  888. def add(n: int) -> None:
  889. qty_var.set(str(current_qty() + n))
  890. def confirm() -> None:
  891. q = current_qty()
  892. if q < 1:
  893. messagebox.showwarning("標簽機", "請輸入需求數量或按 +50、+10、+5、+1。", parent=win)
  894. return
  895. result[0] = q
  896. win.destroy()
  897. btn_row1 = tk.Frame(win, bg=BG_TOP)
  898. btn_row1.pack(pady=8)
  899. for label, value in [("+50", 50), ("+10", 10), ("+5", 5), ("+1", 1)]:
  900. def make_add(v: int):
  901. return lambda: add(v)
  902. ttk.Button(btn_row1, text=label, command=make_add(value), width=8).pack(side=tk.LEFT, padx=4)
  903. ttk.Button(win, text="確認送出", command=confirm, width=14).pack(pady=12)
  904. qty_entry.bind("<Return>", lambda e: confirm())
  905. win.protocol("WM_DELETE_WINDOW", win.destroy)
  906. win.wait_window()
  907. return result[0]
  908. def ask_bag_count(parent: tk.Tk) -> Optional[int]:
  909. """
  910. When printer is 打袋機 DataFlex, ask how many bags:
  911. optional direct qty in text field, +50/+10/+5/+1, 重置, then 確認送出.
  912. Returns count (>= 1), or None if cancelled.
  913. """
  914. result: list[Optional[int]] = [None]
  915. qty_var = tk.StringVar(value="0")
  916. win = tk.Toplevel(parent)
  917. win.title("打袋列印數量")
  918. win.geometry("580x280")
  919. win.transient(parent)
  920. win.grab_set()
  921. win.configure(bg=BG_TOP)
  922. ttk.Label(win, text="列印多少個袋?", font=get_font(FONT_SIZE)).pack(pady=(12, 4))
  923. entry_row = tk.Frame(win, bg=BG_TOP)
  924. entry_row.pack(pady=8)
  925. tk.Label(entry_row, text="需求數量:", font=get_font(FONT_SIZE), bg=BG_TOP).pack(side=tk.LEFT, padx=(0, 6))
  926. qty_entry = tk.Entry(
  927. entry_row,
  928. textvariable=qty_var,
  929. width=12,
  930. font=get_font(FONT_SIZE),
  931. bg="white",
  932. justify=tk.RIGHT,
  933. )
  934. qty_entry.pack(side=tk.LEFT, padx=4)
  935. def current_qty() -> int:
  936. s = (qty_var.get() or "").strip().replace(",", "")
  937. if not s:
  938. return 0
  939. try:
  940. return max(0, int(s))
  941. except ValueError:
  942. return 0
  943. def reset_qty() -> None:
  944. qty_var.set("0")
  945. ttk.Button(entry_row, text="重置", command=reset_qty, width=8).pack(side=tk.LEFT, padx=8)
  946. def add(n: int) -> None:
  947. qty_var.set(str(current_qty() + n))
  948. def confirm() -> None:
  949. q = current_qty()
  950. if q < 1:
  951. messagebox.showwarning("打袋機", "請輸入需求數量或按 +50、+10、+5、+1。", parent=win)
  952. return
  953. result[0] = q
  954. win.destroy()
  955. btn_row1 = tk.Frame(win, bg=BG_TOP)
  956. btn_row1.pack(pady=8)
  957. for label, value in [("+50", 50), ("+10", 10), ("+5", 5), ("+1", 1)]:
  958. def make_add(v: int):
  959. return lambda: add(v)
  960. ttk.Button(btn_row1, text=label, command=make_add(value), width=8).pack(side=tk.LEFT, padx=4)
  961. ttk.Button(win, text="確認送出", command=confirm, width=14).pack(pady=12)
  962. qty_entry.bind("<Return>", lambda e: confirm())
  963. win.protocol("WM_DELETE_WINDOW", win.destroy)
  964. win.wait_window()
  965. return result[0]
  966. def main() -> None:
  967. settings = load_settings()
  968. base_url_ref = [build_base_url(settings["api_ip"], settings["api_port"])]
  969. root = tk.Tk()
  970. root.title("FP-MTMS Bag v1.1 打袋機")
  971. root.geometry("1120x960")
  972. root.minsize(480, 360)
  973. root.configure(bg=BG_ROOT)
  974. # Style: larger font for aged users; light blue theme
  975. style = ttk.Style()
  976. try:
  977. style.configure(".", font=get_font(FONT_SIZE), background=BG_TOP)
  978. style.configure("TButton", font=get_font(FONT_SIZE_BUTTONS), background=BG_TOP)
  979. style.configure("TLabel", font=get_font(FONT_SIZE), background=BG_TOP)
  980. style.configure("TEntry", font=get_font(FONT_SIZE))
  981. style.configure("TFrame", background=BG_TOP)
  982. except tk.TclError:
  983. pass
  984. # Status bar at top: connection state (no popup on error)
  985. status_frame = tk.Frame(root, bg=BG_STATUS_ERROR, padx=12, pady=6)
  986. status_frame.pack(fill=tk.X)
  987. status_lbl = tk.Label(
  988. status_frame,
  989. text="連接不到服務器",
  990. font=get_font(FONT_SIZE_BUTTONS),
  991. bg=BG_STATUS_ERROR,
  992. fg=FG_STATUS_ERROR,
  993. anchor=tk.CENTER,
  994. )
  995. status_lbl.pack(fill=tk.X)
  996. def set_status_ok():
  997. status_frame.configure(bg=BG_STATUS_OK)
  998. status_lbl.configure(text="連接正常", bg=BG_STATUS_OK, fg=FG_STATUS_OK)
  999. def set_status_error():
  1000. status_frame.configure(bg=BG_STATUS_ERROR)
  1001. status_lbl.configure(text="連接不到服務器", bg=BG_STATUS_ERROR, fg=FG_STATUS_ERROR)
  1002. def set_status_message(msg: str, is_error: bool = False) -> None:
  1003. """Show a message on the status bar."""
  1004. if is_error:
  1005. status_frame.configure(bg=BG_STATUS_ERROR)
  1006. status_lbl.configure(text=msg, bg=BG_STATUS_ERROR, fg=FG_STATUS_ERROR)
  1007. else:
  1008. status_frame.configure(bg=BG_STATUS_OK)
  1009. status_lbl.configure(text=msg, bg=BG_STATUS_OK, fg=FG_STATUS_OK)
  1010. # Laser: keep connection open for repeated sends; close when switching away
  1011. laser_conn_ref: list = [None]
  1012. laser_send_busy_ref: list = [False]
  1013. # Top: left [前一天] [date] [後一天] | right [printer dropdown]
  1014. top = tk.Frame(root, padx=12, pady=12, bg=BG_TOP)
  1015. top.pack(fill=tk.X)
  1016. date_var = tk.StringVar(value=date.today().isoformat())
  1017. printer_options = ["打袋機 DataFlex", "標簽機", "激光機"]
  1018. printer_var = tk.StringVar(value=printer_options[0])
  1019. def go_prev_day() -> None:
  1020. try:
  1021. d = date.fromisoformat(date_var.get().strip())
  1022. date_var.set((d - timedelta(days=1)).isoformat())
  1023. load_job_orders(from_user_date_change=True)
  1024. except ValueError:
  1025. date_var.set(date.today().isoformat())
  1026. load_job_orders(from_user_date_change=True)
  1027. def go_next_day() -> None:
  1028. try:
  1029. d = date.fromisoformat(date_var.get().strip())
  1030. date_var.set((d + timedelta(days=1)).isoformat())
  1031. load_job_orders(from_user_date_change=True)
  1032. except ValueError:
  1033. date_var.set(date.today().isoformat())
  1034. load_job_orders(from_user_date_change=True)
  1035. # 前一天 (previous day) with left arrow icon
  1036. btn_prev = ttk.Button(top, text="◀ 前一天", command=go_prev_day)
  1037. btn_prev.pack(side=tk.LEFT, padx=(0, 8))
  1038. # Date field (no "日期:" label); shorter width
  1039. date_entry = tk.Entry(
  1040. top,
  1041. textvariable=date_var,
  1042. font=get_font(FONT_SIZE),
  1043. width=10,
  1044. bg="white",
  1045. )
  1046. date_entry.pack(side=tk.LEFT, padx=(0, 8), ipady=4)
  1047. # 後一天 (next day) with right arrow icon
  1048. btn_next = ttk.Button(top, text="後一天 ▶", command=go_next_day)
  1049. btn_next.pack(side=tk.LEFT, padx=(0, 8))
  1050. # Top right: Setup button + printer selection
  1051. right_frame = tk.Frame(top, bg=BG_TOP)
  1052. right_frame.pack(side=tk.RIGHT)
  1053. ttk.Button(right_frame, text="設定", command=lambda: open_setup_window(root, settings, base_url_ref)).pack(
  1054. side=tk.LEFT, padx=(0, 12)
  1055. )
  1056. # 列印機 label: green when printer connected, red when not (checked periodically)
  1057. printer_status_lbl = tk.Label(
  1058. right_frame,
  1059. text="列印機:",
  1060. font=get_font(FONT_SIZE),
  1061. bg=BG_STATUS_ERROR,
  1062. fg="black",
  1063. padx=6,
  1064. pady=2,
  1065. )
  1066. printer_status_lbl.pack(side=tk.LEFT, padx=(0, 4))
  1067. printer_combo = ttk.Combobox(
  1068. right_frame,
  1069. textvariable=printer_var,
  1070. values=printer_options,
  1071. state="readonly",
  1072. width=14,
  1073. font=get_font(FONT_SIZE),
  1074. )
  1075. printer_combo.pack(side=tk.LEFT)
  1076. printer_after_ref = [None]
  1077. def set_printer_status_ok():
  1078. printer_status_lbl.configure(bg=BG_STATUS_OK, fg=FG_STATUS_OK)
  1079. def set_printer_status_error():
  1080. printer_status_lbl.configure(bg=BG_STATUS_ERROR, fg=FG_STATUS_ERROR)
  1081. def check_printer() -> None:
  1082. if printer_after_ref[0] is not None:
  1083. root.after_cancel(printer_after_ref[0])
  1084. printer_after_ref[0] = None
  1085. ok = try_printer_connection(printer_var.get(), settings)
  1086. if ok:
  1087. set_printer_status_ok()
  1088. printer_after_ref[0] = root.after(PRINTER_CHECK_MS, check_printer)
  1089. else:
  1090. set_printer_status_error()
  1091. printer_after_ref[0] = root.after(PRINTER_RETRY_MS, check_printer)
  1092. def on_printer_selection_changed(*args) -> None:
  1093. check_printer()
  1094. if printer_var.get() != "激光機":
  1095. if laser_conn_ref[0] is not None:
  1096. try:
  1097. laser_conn_ref[0].close()
  1098. except Exception:
  1099. pass
  1100. laser_conn_ref[0] = None
  1101. printer_var.trace_add("write", on_printer_selection_changed)
  1102. def open_setup_window(parent_win: tk.Tk, sett: dict, base_url_ref_list: list) -> None:
  1103. """Modal setup: API IP/port, 打袋機/激光機 IP+port, 標簽機 COM port."""
  1104. d = tk.Toplevel(parent_win)
  1105. d.title("設定")
  1106. d.geometry("440x520")
  1107. d.transient(parent_win)
  1108. d.grab_set()
  1109. d.configure(bg=BG_TOP)
  1110. f = tk.Frame(d, padx=16, pady=16, bg=BG_TOP)
  1111. f.pack(fill=tk.BOTH, expand=True)
  1112. grid_row = [0] # use list so inner function can update
  1113. def _ensure_dot_in_entry(entry: tk.Entry) -> None:
  1114. """Allow typing dot (.) in Entry when IME or layout blocks it (e.g. 192.168.17.27)."""
  1115. def on_key(event):
  1116. if event.keysym in ("period", "decimal"):
  1117. pos = entry.index(tk.INSERT)
  1118. entry.insert(tk.INSERT, ".")
  1119. return "break"
  1120. entry.bind("<KeyPress>", on_key)
  1121. def add_section(label_text: str, key_ip: str | None, key_port: str | None, key_single: str | None):
  1122. out = []
  1123. ttk.Label(f, text=label_text, font=get_font(FONT_SIZE_BUTTONS)).grid(
  1124. row=grid_row[0], column=0, columnspan=2, sticky=tk.W, pady=(8, 2)
  1125. )
  1126. grid_row[0] += 1
  1127. if key_single:
  1128. ttk.Label(
  1129. f,
  1130. text="列印機名稱 (Windows):",
  1131. ).grid(
  1132. row=grid_row[0],
  1133. column=0,
  1134. sticky=tk.W,
  1135. pady=2,
  1136. )
  1137. var = tk.StringVar(value=sett.get(key_single, ""))
  1138. e = tk.Entry(f, textvariable=var, width=22, font=get_font(FONT_SIZE), bg="white")
  1139. e.grid(row=grid_row[0], column=1, sticky=tk.W, pady=2)
  1140. _ensure_dot_in_entry(e)
  1141. grid_row[0] += 1
  1142. return [(key_single, var)]
  1143. if key_ip:
  1144. ttk.Label(f, text="IP:").grid(row=grid_row[0], column=0, sticky=tk.W, pady=2)
  1145. var_ip = tk.StringVar(value=sett.get(key_ip, ""))
  1146. e_ip = tk.Entry(f, textvariable=var_ip, width=22, font=get_font(FONT_SIZE), bg="white")
  1147. e_ip.grid(row=grid_row[0], column=1, sticky=tk.W, pady=2)
  1148. _ensure_dot_in_entry(e_ip)
  1149. grid_row[0] += 1
  1150. out.append((key_ip, var_ip))
  1151. if key_port:
  1152. ttk.Label(f, text="Port:").grid(row=grid_row[0], column=0, sticky=tk.W, pady=2)
  1153. var_port = tk.StringVar(value=sett.get(key_port, ""))
  1154. e_port = tk.Entry(f, textvariable=var_port, width=12, font=get_font(FONT_SIZE), bg="white")
  1155. e_port.grid(row=grid_row[0], column=1, sticky=tk.W, pady=2)
  1156. _ensure_dot_in_entry(e_port)
  1157. grid_row[0] += 1
  1158. out.append((key_port, var_port))
  1159. return out
  1160. all_vars = []
  1161. all_vars.extend(add_section("API 伺服器", "api_ip", "api_port", None))
  1162. all_vars.extend(add_section("打袋機 DataFlex", "dabag_ip", "dabag_port", None))
  1163. all_vars.extend(add_section("激光機", "laser_ip", "laser_port", None))
  1164. all_vars.extend(add_section("標簽機 (USB)", None, None, "label_com"))
  1165. def on_save():
  1166. for key, var in all_vars:
  1167. sett[key] = var.get().strip()
  1168. save_settings(sett)
  1169. base_url_ref_list[0] = build_base_url(sett["api_ip"], sett["api_port"])
  1170. d.destroy()
  1171. btn_f = tk.Frame(d, bg=BG_TOP)
  1172. btn_f.pack(pady=12)
  1173. ttk.Button(btn_f, text="儲存", command=on_save).pack(side=tk.LEFT, padx=4)
  1174. ttk.Button(btn_f, text="取消", command=d.destroy).pack(side=tk.LEFT, padx=4)
  1175. d.wait_window()
  1176. job_orders_frame = tk.Frame(root, bg=BG_LIST)
  1177. job_orders_frame.pack(fill=tk.BOTH, expand=True, padx=12, pady=12)
  1178. search_var = tk.StringVar()
  1179. search_frame = tk.Frame(job_orders_frame, bg=BG_LIST)
  1180. search_frame.pack(fill=tk.X, pady=(0, 6))
  1181. tk.Label(
  1182. search_frame,
  1183. text="搜尋品號/工單/批號:",
  1184. font=get_font(FONT_SIZE_QTY),
  1185. bg=BG_LIST,
  1186. fg="black",
  1187. ).pack(side=tk.LEFT, padx=(0, 6))
  1188. search_entry = tk.Entry(
  1189. search_frame,
  1190. textvariable=search_var,
  1191. width=32,
  1192. font=get_font(FONT_SIZE_QTY),
  1193. bg="white",
  1194. )
  1195. search_entry.pack(side=tk.LEFT, fill=tk.X, expand=True, padx=(0, 8))
  1196. # Scrollable area for buttons
  1197. canvas = tk.Canvas(job_orders_frame, highlightthickness=0, bg=BG_LIST)
  1198. scrollbar = ttk.Scrollbar(job_orders_frame, orient=tk.VERTICAL, command=canvas.yview)
  1199. inner = tk.Frame(canvas, bg=BG_LIST)
  1200. win_id = canvas.create_window((0, 0), window=inner, anchor=tk.NW)
  1201. canvas.configure(yscrollcommand=scrollbar.set)
  1202. def _on_inner_configure(event):
  1203. canvas.configure(scrollregion=canvas.bbox("all"))
  1204. def _on_canvas_configure(event):
  1205. canvas.itemconfig(win_id, width=event.width)
  1206. inner.bind("<Configure>", _on_inner_configure)
  1207. canvas.bind("<Configure>", _on_canvas_configure)
  1208. # Mouse wheel: default Tk scroll speed (one unit per notch)
  1209. def _on_mousewheel(event):
  1210. if getattr(event, "delta", None) is not None:
  1211. canvas.yview_scroll(int(-1 * (event.delta / 120)), "units")
  1212. elif event.num == 5:
  1213. canvas.yview_scroll(1, "units")
  1214. elif event.num == 4:
  1215. canvas.yview_scroll(-1, "units")
  1216. canvas.bind("<MouseWheel>", _on_mousewheel)
  1217. inner.bind("<MouseWheel>", _on_mousewheel)
  1218. canvas.bind("<Button-4>", _on_mousewheel)
  1219. canvas.bind("<Button-5>", _on_mousewheel)
  1220. inner.bind("<Button-4>", _on_mousewheel)
  1221. inner.bind("<Button-5>", _on_mousewheel)
  1222. scrollbar.pack(side=tk.RIGHT, fill=tk.Y)
  1223. canvas.pack(side=tk.LEFT, fill=tk.BOTH, expand=True)
  1224. # Track which row is highlighted (selected for printing) and which job id
  1225. selected_row_holder = [None] # [tk.Frame | None]
  1226. selected_jo_id_ref = [None] # [int | None] job order id for selection preservation
  1227. last_data_ref = [None] # [list | None] last successful fetch for current date
  1228. last_plan_start_ref = [date.today()] # plan date for the current list (search filter uses same)
  1229. after_id_ref = [None] # [str | None] root.after id to cancel retry/refresh
  1230. def _data_equal(a: Optional[list], b: Optional[list]) -> bool:
  1231. if a is None or b is None:
  1232. return a is b
  1233. if len(a) != len(b):
  1234. return False
  1235. for x, y in zip(a, b):
  1236. if x.get("id") != y.get("id"):
  1237. return False
  1238. for k in ("bagPrintedQty", "labelPrintedQty", "laserPrintedQty"):
  1239. if x.get(k) != y.get(k):
  1240. return False
  1241. return True
  1242. def _build_list_from_data(data: list, plan_start: date, preserve_selection: bool) -> None:
  1243. selected_row_holder[0] = None
  1244. year = plan_start.year
  1245. selected_id = selected_jo_id_ref[0] if preserve_selection else None
  1246. found_row = None
  1247. for jo in data:
  1248. jo_id = jo.get("id")
  1249. raw_batch = batch_no(year, jo_id) if jo_id is not None else "—"
  1250. lot_no_val = jo.get("lotNo")
  1251. batch = (lot_no_val or "—").strip() if lot_no_val else "—"
  1252. jo_no_display = (jo.get("code") or "").strip()
  1253. if not jo_no_display and jo_id is not None:
  1254. jo_no_display = raw_batch
  1255. elif not jo_no_display:
  1256. jo_no_display = "—"
  1257. # Line 1: job order no.; line 2: 需求 + 已印(袋/標/激)on one row for compact scrolling
  1258. head_line = f"工單:{jo_no_display}"
  1259. item_code = jo.get("itemCode") or "—"
  1260. item_name = jo.get("itemName") or "—"
  1261. req_qty = jo.get("reqQty")
  1262. qty_str = format_qty(req_qty)
  1263. bag_pq = _printed_qty_int(jo.get("bagPrintedQty"))
  1264. label_pq = _printed_qty_int(jo.get("labelPrintedQty"))
  1265. laser_pq = _printed_qty_int(jo.get("laserPrintedQty"))
  1266. meta_line = (
  1267. f"需求:{qty_str} "
  1268. f"已印 袋{bag_pq:,} 標{label_pq:,} 激{laser_pq:,}"
  1269. )
  1270. # Columns: fixed-width left | fixed-width 品號 | 品名 (expand)
  1271. row = tk.Frame(
  1272. inner,
  1273. bg=BG_ROW,
  1274. relief=tk.RAISED,
  1275. bd=2,
  1276. cursor="hand2",
  1277. padx=10,
  1278. pady=LIST_ROW_IPADY,
  1279. )
  1280. row.pack(fill=tk.X, pady=LIST_ROW_PADY)
  1281. left = tk.Frame(row, bg=BG_ROW, width=LEFT_COL_WIDTH_PX)
  1282. left.pack_propagate(False)
  1283. left.pack(side=tk.LEFT, anchor=tk.NW, fill=tk.Y)
  1284. batch_lbl = tk.Label(
  1285. left,
  1286. text=head_line,
  1287. font=get_font(FONT_SIZE_BUTTONS),
  1288. bg=BG_ROW,
  1289. fg="black",
  1290. )
  1291. batch_lbl.pack(anchor=tk.W)
  1292. meta_lbl = tk.Label(
  1293. left,
  1294. text=meta_line,
  1295. font=get_font(FONT_SIZE_META),
  1296. bg=BG_ROW,
  1297. fg="#222222",
  1298. anchor=tk.W,
  1299. justify=tk.LEFT,
  1300. wraplength=LEFT_COL_WIDTH_PX - 8,
  1301. )
  1302. meta_lbl.pack(anchor=tk.W)
  1303. code_col = tk.Frame(row, bg=BG_ROW, width=CODE_COL_WIDTH_PX)
  1304. code_col.pack_propagate(False)
  1305. code_col.pack(side=tk.LEFT, anchor=tk.NW, fill=tk.Y, padx=(6, 2))
  1306. code_lbl = tk.Label(
  1307. code_col,
  1308. text=item_code,
  1309. font=get_font(FONT_SIZE_ITEM_CODE),
  1310. bg=BG_ROW,
  1311. fg="black",
  1312. wraplength=ITEM_CODE_WRAP,
  1313. justify=tk.LEFT,
  1314. anchor=tk.NW,
  1315. )
  1316. code_lbl.pack(anchor=tk.NW)
  1317. name_col = tk.Frame(row, bg=BG_ROW)
  1318. name_col.pack(side=tk.LEFT, fill=tk.BOTH, expand=True, anchor=tk.NW)
  1319. name_lbl = tk.Label(
  1320. name_col,
  1321. text=item_name or "—",
  1322. font=get_font(FONT_SIZE_ITEM_NAME),
  1323. bg=BG_ROW,
  1324. fg="black",
  1325. wraplength=ITEM_NAME_WRAP,
  1326. justify=tk.LEFT,
  1327. anchor=tk.NW,
  1328. )
  1329. name_lbl.pack(anchor=tk.NW)
  1330. def _on_click(e, j=jo, b=batch, r=row):
  1331. if selected_row_holder[0] is not None:
  1332. set_row_highlight(selected_row_holder[0], False)
  1333. set_row_highlight(r, True)
  1334. selected_row_holder[0] = r
  1335. selected_jo_id_ref[0] = j.get("id")
  1336. if printer_var.get() == "打袋機 DataFlex":
  1337. ip = (settings.get("dabag_ip") or "").strip()
  1338. port_str = (settings.get("dabag_port") or "3008").strip()
  1339. try:
  1340. port = int(port_str)
  1341. except ValueError:
  1342. port = 3008
  1343. if not ip:
  1344. messagebox.showerror("打袋機", "請在設定中填寫打袋機 DataFlex 的 IP。")
  1345. else:
  1346. count = ask_bag_count(root)
  1347. if count is not None:
  1348. item_code = j.get("itemCode") or "—"
  1349. item_name = j.get("itemName") or "—"
  1350. item_id = j.get("itemId")
  1351. stock_in_line_id = j.get("stockInLineId")
  1352. lot_no = j.get("lotNo")
  1353. zpl = generate_zpl_dataflex(
  1354. b,
  1355. item_code,
  1356. item_name,
  1357. item_id=item_id,
  1358. stock_in_line_id=stock_in_line_id,
  1359. lot_no=lot_no,
  1360. )
  1361. label_text = (lot_no or b).strip()
  1362. n = count
  1363. try:
  1364. for i in range(n):
  1365. send_zpl_to_dataflex(ip, port, zpl)
  1366. if i < n - 1:
  1367. time.sleep(2)
  1368. set_status_message(f"已送出列印:批次 {label_text} x {n} 張", is_error=False)
  1369. jo_id = j.get("id")
  1370. if jo_id is not None:
  1371. try:
  1372. submit_job_order_print_submit(
  1373. base_url_ref[0], int(jo_id), n, "DATAFLEX"
  1374. )
  1375. load_job_orders(from_user_date_change=False)
  1376. except requests.RequestException as ex:
  1377. messagebox.showwarning(
  1378. "打袋機",
  1379. f"已送出 {n} 張,但伺服器記錄失敗:{ex}",
  1380. )
  1381. except ConnectionRefusedError:
  1382. set_status_message(f"無法連線至 {ip}:{port},請確認印表機已開機且 IP 正確。", is_error=True)
  1383. except socket.timeout:
  1384. set_status_message(f"連線逾時 ({ip}:{port}),請檢查網路與連接埠。", is_error=True)
  1385. except OSError as err:
  1386. set_status_message(f"列印失敗:{err}", is_error=True)
  1387. elif printer_var.get() == "標簽機":
  1388. com = (settings.get("label_com") or "").strip()
  1389. if not com:
  1390. messagebox.showerror("標簽機", "請在設定中填寫標簽機名稱 (例如:TSC TTP-246M Pro)。")
  1391. else:
  1392. count = ask_label_count(root)
  1393. if count is not None:
  1394. item_code = j.get("itemCode") or "—"
  1395. item_name = j.get("itemName") or "—"
  1396. item_id = j.get("itemId")
  1397. stock_in_line_id = j.get("stockInLineId")
  1398. lot_no = j.get("lotNo")
  1399. n = count
  1400. try:
  1401. # Always render to image (Chinese OK), then send as ZPL graphic (^GFA).
  1402. # This is more reliable than Windows GDI and works for both Windows printer name and COM.
  1403. if not _HAS_PIL_QR:
  1404. raise RuntimeError("請先安裝 Pillow + qrcode(pip install Pillow qrcode[pil])。")
  1405. label_img = render_label_to_image(
  1406. b, item_code, item_name,
  1407. item_id=item_id, stock_in_line_id=stock_in_line_id,
  1408. lot_no=lot_no,
  1409. )
  1410. zpl_img = _image_to_zpl_gfa(label_img)
  1411. for i in range(n):
  1412. send_zpl_to_label_printer(com, zpl_img)
  1413. if i < n - 1:
  1414. time.sleep(0.5)
  1415. jo_id = j.get("id")
  1416. if jo_id is not None:
  1417. try:
  1418. submit_job_order_print_submit(
  1419. base_url_ref[0], int(jo_id), n, "LABEL"
  1420. )
  1421. load_job_orders(from_user_date_change=False)
  1422. messagebox.showinfo(
  1423. "標籤機",
  1424. f"已送出列印:{n} 張標籤(已記錄)",
  1425. )
  1426. except requests.RequestException as ex:
  1427. messagebox.showwarning(
  1428. "標籤機",
  1429. f"標籤已列印 {n} 張,但伺服器記錄失敗:{ex}",
  1430. )
  1431. else:
  1432. messagebox.showwarning(
  1433. "標籤機",
  1434. f"已送出列印:{n} 張標籤(無工單 id,無法寫入伺服器記錄)",
  1435. )
  1436. except Exception as err:
  1437. messagebox.showerror("標簽機", f"列印失敗:{err}")
  1438. elif printer_var.get() == "激光機":
  1439. ip = (settings.get("laser_ip") or "").strip()
  1440. port_str = (settings.get("laser_port") or "45678").strip()
  1441. try:
  1442. port = int(port_str)
  1443. except ValueError:
  1444. port = 45678
  1445. if not ip:
  1446. set_status_message("請在設定中填寫激光機的 IP。", is_error=True)
  1447. else:
  1448. item_id = j.get("itemId")
  1449. stock_in_line_id = j.get("stockInLineId")
  1450. item_code_val = j.get("itemCode") or ""
  1451. item_name_val = j.get("itemName") or ""
  1452. run_laser_row_send_thread(
  1453. root=root,
  1454. laser_conn_ref=laser_conn_ref,
  1455. laser_busy_ref=laser_send_busy_ref,
  1456. ip=ip,
  1457. port=port,
  1458. item_id=item_id,
  1459. stock_in_line_id=stock_in_line_id,
  1460. item_code=item_code_val,
  1461. item_name=item_name_val,
  1462. set_status_message=set_status_message,
  1463. base_url=base_url_ref[0],
  1464. job_order_id=j.get("id"),
  1465. on_recorded=lambda: load_job_orders(from_user_date_change=False),
  1466. )
  1467. for w in (
  1468. row,
  1469. left,
  1470. batch_lbl,
  1471. meta_lbl,
  1472. code_col,
  1473. code_lbl,
  1474. name_col,
  1475. name_lbl,
  1476. ):
  1477. w.bind("<Button-1>", _on_click)
  1478. w.bind("<MouseWheel>", _on_mousewheel)
  1479. w.bind("<Button-4>", _on_mousewheel)
  1480. w.bind("<Button-5>", _on_mousewheel)
  1481. if preserve_selection and selected_id is not None and jo.get("id") == selected_id:
  1482. found_row = row
  1483. if found_row is not None:
  1484. set_row_highlight(found_row, True)
  1485. selected_row_holder[0] = found_row
  1486. def refresh_visible_list() -> None:
  1487. """Re-apply search filter to last fetched rows without hitting the API."""
  1488. raw = last_data_ref[0]
  1489. if raw is None:
  1490. return
  1491. ps = last_plan_start_ref[0]
  1492. needle = search_var.get().strip()
  1493. shown = _filter_job_orders_by_search(raw, needle) if needle else raw
  1494. for w in inner.winfo_children():
  1495. w.destroy()
  1496. _build_list_from_data(shown, ps, preserve_selection=True)
  1497. search_entry.bind("<KeyRelease>", lambda e: refresh_visible_list())
  1498. def load_job_orders(from_user_date_change: bool = False) -> None:
  1499. if after_id_ref[0] is not None:
  1500. root.after_cancel(after_id_ref[0])
  1501. after_id_ref[0] = None
  1502. date_str = date_var.get().strip()
  1503. try:
  1504. plan_start = date.fromisoformat(date_str)
  1505. except ValueError:
  1506. messagebox.showerror("日期錯誤", f"請使用 yyyy-MM-dd 格式。目前:{date_str}")
  1507. return
  1508. if from_user_date_change:
  1509. selected_row_holder[0] = None
  1510. selected_jo_id_ref[0] = None
  1511. try:
  1512. data = fetch_job_orders(base_url_ref[0], plan_start)
  1513. except requests.RequestException:
  1514. set_status_error()
  1515. after_id_ref[0] = root.after(RETRY_MS, lambda: load_job_orders(from_user_date_change=False))
  1516. return
  1517. set_status_ok()
  1518. old_data = last_data_ref[0]
  1519. last_data_ref[0] = data
  1520. last_plan_start_ref[0] = plan_start
  1521. data_changed = not _data_equal(old_data, data)
  1522. if data_changed or from_user_date_change:
  1523. # Rebuild list: clear and rebuild from current data (last_data_ref already updated)
  1524. for w in inner.winfo_children():
  1525. w.destroy()
  1526. preserve = not from_user_date_change
  1527. needle = search_var.get().strip()
  1528. shown = _filter_job_orders_by_search(data, needle) if needle else data
  1529. _build_list_from_data(shown, plan_start, preserve_selection=preserve)
  1530. if from_user_date_change:
  1531. canvas.yview_moveto(0)
  1532. after_id_ref[0] = root.after(REFRESH_MS, lambda: load_job_orders(from_user_date_change=False))
  1533. # Load default (today) on start; then start printer connection check
  1534. root.after(100, lambda: load_job_orders(from_user_date_change=True))
  1535. root.after(300, check_printer)
  1536. root.mainloop()
  1537. if __name__ == "__main__":
  1538. main()