admin_program.py 80 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095
  1. import abc
  2. import tkinter as tk
  3. import tkinter.ttk as ttk
  4. from tkinter.filedialog import askdirectory, askopenfilename
  5. import matplotlib
  6. from matplotlib import font_manager
  7. from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg, NavigationToolbar2Tk
  8. from matplotlib.axes import Axes
  9. import numpy as np
  10. from matplotlib.figure import Figure
  11. from tool.color import random_color
  12. from tool.type_ import *
  13. from tool.tk import make_font, set_tk_disable_from_list
  14. from tool.login import create_uid
  15. from conf import Config
  16. from . import admin
  17. from . import admin_event as tk_event
  18. from sql import DBBit
  19. from sql.user import find_user_by_name
  20. from core.garbage import GarbageType
  21. class AdminProgram(metaclass=abc.ABCMeta):
  22. def __init__(self, station: "admin.AdminStation", win: Union[tk.Frame, tk.Toplevel, tk.Tk], color: str, title: str):
  23. self.station = station
  24. self.win = win
  25. self.color = color
  26. self.frame = tk.Frame(self.win)
  27. self.frame['bg'] = color
  28. self.program_title = title
  29. @abc.abstractmethod
  30. def set_disable(self):
  31. ...
  32. @abc.abstractmethod
  33. def reset_disable(self):
  34. ...
  35. @abc.abstractmethod
  36. def conf_gui(self, n: int = 1):
  37. ...
  38. def to_program(self):
  39. pass
  40. def get_program_frame(self) -> Tuple[str, tk.Frame]:
  41. return self.program_title, self.frame
  42. class WelcomeProgram(AdminProgram):
  43. def __init__(self, station, win, color):
  44. super().__init__(station, win, color, "欢迎页")
  45. self.title = tk.Label(self.frame)
  46. self.info = tk.Label(self.frame)
  47. self.__conf_font()
  48. def __conf_font(self, n: int = 1):
  49. self.title_font_size = int(25 * n)
  50. self.info_font_size = int(14 * n)
  51. def conf_gui(self, n: int = 1):
  52. self.__conf_font(n)
  53. title_font = make_font(size=self.title_font_size, weight="bold")
  54. info_font = make_font(size=self.info_font_size)
  55. self.title['font'] = title_font
  56. self.title['bg'] = self.color
  57. self.title['text'] = '欢迎使用 HGSSystem 管理员系统\n[帮助]'
  58. self.info['bg'] = self.color
  59. self.info['font'] = info_font
  60. self.info['anchor'] = 'nw'
  61. self.info['justify'] = 'left'
  62. self.info['text'] = (f'''
  63. HGSSystem 管理者界面:
  64. 1) 点击菜单按钮进入子菜单或程序
  65. 2) 创建 菜单包含创建类的程序
  66. 3) 删除 菜单包含删除类的程序
  67. 4) 搜索 菜单包含数据分析类的程序
  68. 5) 更新 菜单包含数据更新类的程序
  69. 6) 当离开操作系统时请退出登录以确保安全
  70. 7) 只能使用具有管理员权限的账号登陆系统
  71. 8) 只有admin用户可以完成危险操作(例如删除所有垃圾袋数据)
  72. 程序的运行:
  73. 1) 在菜单中选中程序后,根据程序界面提示完成操作
  74. 2) 操作过程通常会显示进度条,除非任务执行迅速
  75. 3) 结果通常会被反馈, 且不会自动消失
  76. 系统登录:
  77. 1) 仅Manager用户可以登录
  78. 关于彩蛋: 存在
  79. '''.strip())
  80. self.title.place(relx=0.1, rely=0.0, relwidth=0.8, relheight=0.2)
  81. self.info.place(relx=0.05, rely=0.21, relwidth=0.90, relheight=0.75)
  82. def set_disable(self):
  83. pass
  84. def reset_disable(self):
  85. pass
  86. class AboutProgram(AdminProgram):
  87. def __init__(self, station, win, color):
  88. super().__init__(station, win, color, "关于")
  89. self.title = tk.Label(self.frame)
  90. self.info = tk.Label(self.frame)
  91. self.__conf_font()
  92. def __conf_font(self, n: int = 1):
  93. self.title_font_size = int(25 * n)
  94. self.info_font_size = int(14 * n)
  95. def conf_gui(self, n: int = 1):
  96. self.__conf_font(n)
  97. title_font = make_font(size=self.title_font_size, weight="bold")
  98. info_font = make_font(size=self.info_font_size)
  99. self.title['font'] = title_font
  100. self.title['bg'] = self.color
  101. self.title['text'] = '关于 HGSSystem 管理员系统'
  102. self.info['bg'] = self.color
  103. self.info['font'] = info_font
  104. self.info['anchor'] = 'nw'
  105. self.info['justify'] = 'left'
  106. self.info['text'] = Config.about_info
  107. self.title.place(relx=0.1, rely=0.0, relwidth=0.8, relheight=0.2)
  108. self.info.place(relx=0.05, rely=0.21, relwidth=0.90, relheight=0.75)
  109. def set_disable(self):
  110. pass
  111. def reset_disable(self):
  112. pass
  113. class CreateUserProgramBase(AdminProgram):
  114. def __init__(self, station, win, color, title: str):
  115. super().__init__(station, win, color, title)
  116. self.enter_frame = tk.Frame(self.frame)
  117. self.title: List[tk.Label] = [tk.Label(self.enter_frame) for _ in range(3)]
  118. self.enter: List[tk.Entry] = [tk.Entry(self.enter_frame) for _ in range(3)]
  119. self.var: List[tk.Variable] = [tk.StringVar() for _ in range(3)]
  120. self.btn: List[tk.Button] = [tk.Button(self.frame) for _ in range(2)]
  121. self._conf("#FA8072", False) # 默认颜色
  122. self.__conf_font()
  123. def _conf(self, bg_color, is_manager: bool):
  124. self.bg_color = bg_color
  125. self.is_manager = is_manager
  126. return self
  127. def __conf_font(self, n: int = 1):
  128. self.title_font_size = int(16 * n)
  129. self.btn_font_size = int(14 * n)
  130. def conf_gui(self, n: int = 1):
  131. self.__conf_font(n)
  132. title_font = make_font(size=self.title_font_size)
  133. btn_font = make_font(size=self.btn_font_size)
  134. self.enter_frame['bg'] = self.bg_color
  135. self.enter_frame['bd'] = 5
  136. self.enter_frame['relief'] = "ridge"
  137. self.enter_frame.place(relx=0.2, rely=0.3, relwidth=0.6, relheight=0.30)
  138. height = 0.1
  139. for lb, text, enter, var in zip(self.title, ["用户名:", "用户密码:", "手机号:"], self.enter, self.var):
  140. lb['font'] = title_font
  141. lb['text'] = text
  142. lb['bg'] = self.bg_color
  143. lb['anchor'] = 'e'
  144. enter['font'] = title_font
  145. enter['textvariable'] = var
  146. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.17)
  147. enter.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.17)
  148. height += 0.30
  149. for btn, text, x, func in zip(self.btn,
  150. ["创建用户", "获取用户ID"],
  151. [0.2, 0.6],
  152. [lambda: self.create_by_name(), lambda: self.get_uid()]):
  153. btn['font'] = btn_font
  154. btn['text'] = text
  155. btn['bg'] = Config.tk_btn_bg
  156. btn['command'] = func
  157. btn.place(relx=x, rely=0.7, relwidth=0.2, relheight=0.08)
  158. def __get_info(self) -> Optional[Tuple[uname_t, passwd_t, str]]:
  159. name: uname_t = self.var[0].get()
  160. passwd: passwd_t = self.var[1].get()
  161. phone: str = self.var[2].get()
  162. if len(name) == 0 or len(passwd) == 0 or len(phone) != 11:
  163. self.station.show_msg("用户创建失败", "请再次尝试, 输入用户名, 用户密码和11位手机号")
  164. return None
  165. return name, passwd, phone
  166. def create_by_name(self):
  167. res = self.__get_info()
  168. if res is None:
  169. return
  170. name, passwd, phone = res
  171. event = tk_event.CreateUserEvent(self.station).start(name, passwd, phone, self.is_manager)
  172. self.station.push_event(event)
  173. def get_uid(self):
  174. res = self.__get_info()
  175. if res is None:
  176. return
  177. name, passwd, phone = res
  178. uid = create_uid(name, passwd, phone)
  179. self.station.show_msg("获取用户ID", f"用户名: {name}\n用户ID: {uid}")
  180. def set_disable(self):
  181. set_tk_disable_from_list(self.btn)
  182. set_tk_disable_from_list(self.enter)
  183. def reset_disable(self):
  184. set_tk_disable_from_list(self.btn, flat='normal')
  185. set_tk_disable_from_list(self.enter, flat='normal')
  186. class CreateNormalUserProgram(CreateUserProgramBase):
  187. def __init__(self, station, win, color):
  188. super(CreateNormalUserProgram, self).__init__(station, win, color, "创建普通用户")
  189. class CreateManagerUserProgram(CreateUserProgramBase):
  190. def __init__(self, station, win, color):
  191. super(CreateManagerUserProgram, self).__init__(station, win, color, "创建管理员")
  192. self._conf("#4b5cc4", True)
  193. class CreateAutoNormalUserProgram(AdminProgram):
  194. def __init__(self, station, win, color):
  195. super().__init__(station, win, color, "创建自动用户")
  196. self.enter_frame = tk.Frame(self.frame)
  197. self.title: tk.Label = tk.Label(self.enter_frame)
  198. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  199. self.var: tk.Variable = tk.StringVar()
  200. self.btn: tk.Button = tk.Button(self.frame) # create(生成用户) try(计算uid)
  201. self.__conf_font()
  202. def __conf_font(self, n: int = 1):
  203. self.title_font_size = int(16 * n)
  204. self.btn_font_size = int(14 * n)
  205. def conf_gui(self, n: int = 1):
  206. self.__conf_font(n)
  207. title_font = make_font(size=self.title_font_size)
  208. btn_font = make_font(size=self.btn_font_size)
  209. self.enter_frame['bg'] = "#bce672"
  210. self.enter_frame['bd'] = 5
  211. self.enter_frame['relief'] = "ridge"
  212. self.enter_frame.place(relx=0.2, rely=0.3, relwidth=0.6, relheight=0.12)
  213. self.title['font'] = title_font
  214. self.title['text'] = "手机号:"
  215. self.title['bg'] = "#bce672"
  216. self.title['anchor'] = 'e'
  217. self.enter['font'] = title_font
  218. self.enter['textvariable'] = self.var
  219. self.title.place(relx=0.02, rely=0.25, relwidth=0.25, relheight=0.50)
  220. self.enter.place(relx=0.30, rely=0.25, relwidth=0.60, relheight=0.50)
  221. self.btn['font'] = btn_font
  222. self.btn['text'] = "创建用户"
  223. self.btn['bg'] = Config.tk_btn_bg
  224. self.btn['command'] = lambda: self.create_user()
  225. self.btn.place(relx=0.4, rely=0.7, relwidth=0.2, relheight=0.08)
  226. def create_user(self):
  227. phone = self.var.get()
  228. if len(phone) != 11:
  229. self.station.show_msg("UserInfoError", "Please, enter Phone(11)")
  230. event = tk_event.CreateUserEvent(self.station).start(None, None, phone, False)
  231. self.station.push_event(event)
  232. def set_disable(self):
  233. self.btn['state'] = 'disable'
  234. self.enter['state'] = 'disable'
  235. def reset_disable(self):
  236. self.btn['state'] = 'normal'
  237. self.enter['state'] = 'normal'
  238. class CreateGarbageProgram(AdminProgram):
  239. def __init__(self, station, win, color):
  240. super().__init__(station, win, color, "创建垃圾袋")
  241. self.enter_frame = tk.Frame(self.frame)
  242. self.title: List[tk.Label] = [tk.Label(self.enter_frame), tk.Label(self.enter_frame)]
  243. self.enter: List[tk.Entry] = [tk.Entry(self.enter_frame), tk.Entry(self.enter_frame)]
  244. self.var: List[tk.Variable] = [tk.StringVar(), tk.StringVar()]
  245. self.create_btn: tk.Button = tk.Button(self.frame)
  246. self.file_btn: tk.Button = tk.Button(self.frame)
  247. self.__conf_font()
  248. def __conf_font(self, n: int = 1):
  249. self.title_font_size = int(16 * n)
  250. self.btn_font_size = int(14 * n)
  251. def conf_gui(self, n: int = 1):
  252. self.__conf_font(n)
  253. title_font = make_font(size=self.title_font_size)
  254. btn_font = make_font(size=self.btn_font_size)
  255. self.enter_frame['bg'] = "#b69968"
  256. self.enter_frame['bd'] = 5
  257. self.enter_frame['relief'] = "ridge"
  258. self.enter_frame.place(relx=0.2, rely=0.3, relwidth=0.6, relheight=0.17)
  259. height = 0.1
  260. for lb, text, enter, var in zip(self.title, ["数量:", "导出位置:"], self.enter, self.var):
  261. lb['font'] = title_font
  262. lb['text'] = text
  263. lb['bg'] = "#b69968"
  264. lb['anchor'] = 'e'
  265. enter['font'] = title_font
  266. enter['textvariable'] = var
  267. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.35)
  268. enter.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.35)
  269. height += 0.43
  270. for btn, text, x, func in zip([self.create_btn, self.file_btn],
  271. ["创建垃圾袋", "选择目录"],
  272. [0.2, 0.6],
  273. [lambda: self.create_garbage(), lambda: self.choose_file()]):
  274. btn['font'] = btn_font
  275. btn['text'] = text
  276. btn['bg'] = Config.tk_btn_bg
  277. btn['command'] = func
  278. btn.place(relx=x, rely=0.7, relwidth=0.2, relheight=0.08)
  279. def choose_file(self):
  280. path = askdirectory(title='选择二维码导出位置')
  281. self.var[1].set(path)
  282. def create_garbage(self):
  283. try:
  284. count = int(self.var[0].get())
  285. if count <= 0:
  286. raise ValueError
  287. except (ValueError, TypeError):
  288. self.station.show_msg("类型错误", "数量必须为大于0的数字")
  289. else:
  290. path = self.var[1].get()
  291. if len(path) == 0:
  292. path = None
  293. event = tk_event.CreateGarbageEvent(self.station).start(path, count)
  294. self.station.push_event(event)
  295. def set_disable(self):
  296. self.create_btn['state'] = 'disable'
  297. self.file_btn['state'] = 'disable'
  298. set_tk_disable_from_list(self.enter)
  299. def reset_disable(self):
  300. self.create_btn['state'] = 'normal'
  301. self.file_btn['state'] = 'normal'
  302. set_tk_disable_from_list(self.enter, flat='normal')
  303. class ExportProgramBase(AdminProgram):
  304. def __init__(self, station, win, color, title: str):
  305. super().__init__(station, win, color, title)
  306. self.gid_frame = tk.Frame(self.frame)
  307. self.gid_title: List[tk.Label] = [tk.Label(self.gid_frame), tk.Label(self.gid_frame)]
  308. self.gid_enter: List[tk.Entry] = [tk.Entry(self.gid_frame), tk.Entry(self.gid_frame)]
  309. self.gid_var: List[tk.Variable] = [tk.StringVar(), tk.StringVar()]
  310. self.where_frame = tk.Frame(self.frame)
  311. self.where_title: List[tk.Label] = [tk.Label(self.where_frame), tk.Label(self.where_frame)]
  312. self.where_enter: List[tk.Entry] = [tk.Entry(self.where_frame), tk.Entry(self.where_frame)]
  313. self.where_var: List[tk.Variable] = [tk.StringVar(), tk.StringVar()]
  314. self.create_btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  315. self.file_btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  316. self._conf("", [], [], [])
  317. self.__conf_font()
  318. def _conf(self, bg_color: str, title_id, title_where, title_command):
  319. self.bg_color = bg_color
  320. self.title_id = title_id
  321. self.title_where = title_where
  322. self.title_command = title_command
  323. def __conf_font(self, n: int = 1):
  324. self.title_font_size = int(16 * n)
  325. self.btn_font_size = int(14 * n)
  326. def conf_gui(self, n: int = 1):
  327. self.__conf_font(n)
  328. title_font = make_font(size=self.title_font_size)
  329. btn_font = make_font(size=self.btn_font_size)
  330. self.where_frame['bg'] = self.bg_color
  331. self.where_frame['bd'] = 5
  332. self.where_frame['relief'] = "ridge"
  333. self.where_frame.place(relx=0.2, rely=0.2, relwidth=0.6, relheight=0.17)
  334. self.gid_frame['bg'] = self.bg_color
  335. self.gid_frame['bd'] = 5
  336. self.gid_frame['relief'] = "ridge"
  337. self.gid_frame.place(relx=0.2, rely=0.6, relwidth=0.6, relheight=0.17)
  338. height = 0.1
  339. for lb, text, enter, var, lb_w, text_w, enter_w, var_w in zip(
  340. self.gid_title, self.title_id, self.gid_enter, self.gid_var,
  341. self.where_title, self.title_where, self.where_enter, self.where_var):
  342. lb['font'] = title_font
  343. lb['text'] = text
  344. lb['bg'] = self.bg_color
  345. lb['anchor'] = 'e'
  346. lb_w['font'] = title_font
  347. lb_w['text'] = text_w
  348. lb_w['bg'] = self.bg_color
  349. lb_w['anchor'] = 'e'
  350. enter['textvariable'] = var
  351. enter['font'] = title_font
  352. enter_w['textvariable'] = var_w
  353. enter_w['font'] = title_font
  354. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.35)
  355. enter.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.35)
  356. lb_w.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.35)
  357. enter_w.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.35)
  358. height += 0.43
  359. for btn, text in zip(self.create_btn + self.file_btn, self.title_command):
  360. btn['font'] = btn_font
  361. btn['text'] = text
  362. btn['bg'] = Config.tk_btn_bg
  363. self.create_btn[1]['command'] = self.export_where
  364. self.create_btn[0]['command'] = self.export_id
  365. self.create_btn[1].place(relx=0.2, rely=0.39, relwidth=0.25, relheight=0.08)
  366. self.create_btn[0].place(relx=0.2, rely=0.79, relwidth=0.25, relheight=0.08)
  367. self.file_btn[1]['command'] = self.choose_file_where
  368. self.file_btn[0]['command'] = self.choose_file_id
  369. self.file_btn[1].place(relx=0.6, rely=0.39, relwidth=0.2, relheight=0.08)
  370. self.file_btn[0].place(relx=0.6, rely=0.79, relwidth=0.2, relheight=0.08)
  371. def choose_file_id(self):
  372. path = askdirectory(title='选择二维码导出位置')
  373. self.gid_var[1].set(path)
  374. def choose_file_where(self):
  375. path = askdirectory(title='选择二维码导出位置')
  376. self.where_var[1].set(path)
  377. def export_id(self):
  378. ...
  379. def export_where(self):
  380. ...
  381. def set_disable(self):
  382. set_tk_disable_from_list(self.gid_enter)
  383. set_tk_disable_from_list(self.create_btn)
  384. set_tk_disable_from_list(self.file_btn)
  385. def reset_disable(self):
  386. set_tk_disable_from_list(self.gid_enter, flat='normal')
  387. set_tk_disable_from_list(self.create_btn, flat='normal')
  388. set_tk_disable_from_list(self.file_btn, flat='normal')
  389. class ExportGarbageProgram(ExportProgramBase):
  390. def __init__(self, station, win, color):
  391. super().__init__(station, win, color, "导出垃圾袋二维码")
  392. self._conf("#afdfe4", ["垃圾袋ID:", "导出位置:"], ["条件:", "导出位置:"],
  393. ["根据垃圾袋ID导出", "根据条件导出", "选择目录", "选择目录"])
  394. def export_id(self):
  395. gid = self.gid_var[0].get()
  396. path = self.gid_var[1].get()
  397. event = tk_event.ExportGarbageByIDEvent(self.station).start(path, gid)
  398. self.station.push_event(event)
  399. def export_where(self):
  400. where = self.where_var[0].get()
  401. path = self.where_var[1].get()
  402. event = tk_event.ExportGarbageAdvancedEvent(self.station).start(path, where)
  403. self.station.push_event(event)
  404. class ExportUserProgram(ExportProgramBase):
  405. def __init__(self, station, win, color):
  406. super().__init__(station, win, color, "导出用户二维码")
  407. self._conf("#f69c9f", ["用户ID:", "导出位置:"], ["条件:", "导出位置:"],
  408. ["根据用户ID导出", "根据条件导出", "选择目录", "选择目录"])
  409. def export_id(self):
  410. uid = self.gid_var[0].get()
  411. path = self.gid_var[1].get()
  412. event = tk_event.ExportUserByIDEvent(self.station).start(path, uid)
  413. self.station.push_event(event)
  414. def export_where(self):
  415. where = self.where_var[0].get()
  416. path = self.where_var[1].get()
  417. event = tk_event.ExportUserAdvancedEvent(self.station).start(path, where)
  418. self.station.push_event(event)
  419. class CreateUserFromCSVProgram(AdminProgram):
  420. def __init__(self, station, win, color):
  421. super().__init__(station, win, color, "从CSV导入用户")
  422. self.auto_frame = tk.Frame(self.frame)
  423. self.auto_title: tk.Label = tk.Label(self.auto_frame)
  424. self.auto_enter: tk.Entry = tk.Entry(self.auto_frame)
  425. self.auto_var: tk.Variable = tk.StringVar()
  426. self.enter_frame = tk.Frame(self.frame)
  427. self.path_title: tk.Label = tk.Label(self.enter_frame)
  428. self.path_enter: tk.Entry = tk.Entry(self.enter_frame)
  429. self.path_var: tk.Variable = tk.StringVar()
  430. self.create_btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  431. self.file_btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  432. self.__conf_font()
  433. def __conf_font(self, n: int = 1):
  434. self.title_font_size = int(16 * n)
  435. self.btn_font_size = int(14 * n)
  436. def conf_gui(self, n: int = 1):
  437. self.__conf_font(n)
  438. title_font = make_font(size=self.title_font_size)
  439. btn_font = make_font(size=self.btn_font_size)
  440. self.enter_frame['bg'] = "#EEE8AA"
  441. self.enter_frame['bd'] = 5
  442. self.enter_frame['relief'] = "ridge"
  443. self.enter_frame.place(relx=0.2, rely=0.2, relwidth=0.6, relheight=0.12)
  444. self.auto_frame['bg'] = "#EEE8AA"
  445. self.auto_frame['bd'] = 5
  446. self.auto_frame['relief'] = "ridge"
  447. self.auto_frame.place(relx=0.2, rely=0.6, relwidth=0.6, relheight=0.12)
  448. self.auto_title['font'] = title_font
  449. self.auto_title['text'] = "CSV文件:"
  450. self.auto_title['bg'] = "#EEE8AA"
  451. self.auto_title['anchor'] = 'e'
  452. self.path_title['font'] = title_font
  453. self.path_title['text'] = "CSV文件:"
  454. self.path_title['bg'] = "#EEE8AA"
  455. self.path_title['anchor'] = 'e'
  456. self.auto_enter['textvariable'] = self.auto_var
  457. self.auto_enter['font'] = title_font
  458. self.path_enter['textvariable'] = self.path_var
  459. self.path_enter['font'] = title_font
  460. self.auto_title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  461. self.auto_enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  462. self.path_title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  463. self.path_enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  464. for btn, text in zip(self.create_btn + self.file_btn,
  465. ["创建用户", "创建自动用户", "选择CSV", "选择CSV"]):
  466. btn['font'] = btn_font
  467. btn['text'] = text
  468. btn['bg'] = Config.tk_btn_bg
  469. self.create_btn[0]['command'] = self.create
  470. self.create_btn[1]['command'] = self.create_auto
  471. self.create_btn[0].place(relx=0.2, rely=0.34, relwidth=0.25, relheight=0.08)
  472. self.create_btn[1].place(relx=0.2, rely=0.74, relwidth=0.25, relheight=0.08)
  473. self.file_btn[0]['command'] = self.choose_file
  474. self.file_btn[1]['command'] = self.choose_file_auto
  475. self.file_btn[0].place(relx=0.6, rely=0.34, relwidth=0.2, relheight=0.08)
  476. self.file_btn[1].place(relx=0.6, rely=0.74, relwidth=0.2, relheight=0.08)
  477. def choose_file_auto(self):
  478. path = askopenfilename(title='选择CSV文件', filetypes=[("CSV", ".csv")])
  479. self.auto_var.set(path)
  480. def choose_file(self):
  481. path = askopenfilename(title='选择CSV文件', filetypes=[("CSV", ".csv")])
  482. self.path_var.set(path)
  483. def create_auto(self):
  484. path = self.auto_var.get()
  485. event = tk_event.CreateAutoUserFromCSVEvent(self.station).start(path)
  486. self.station.push_event(event)
  487. def create(self):
  488. path = self.path_var.get()
  489. event = tk_event.CreateUserFromCSVEvent(self.station).start(path)
  490. self.station.push_event(event)
  491. def set_disable(self):
  492. self.auto_enter['state'] = 'disable'
  493. self.path_enter['state'] = 'disable'
  494. set_tk_disable_from_list(self.create_btn)
  495. set_tk_disable_from_list(self.file_btn)
  496. def reset_disable(self):
  497. self.auto_enter['state'] = 'normal'
  498. self.path_enter['state'] = 'normal'
  499. set_tk_disable_from_list(self.create_btn, flat='normal')
  500. set_tk_disable_from_list(self.file_btn, flat='normal')
  501. class DeleteUserProgram(AdminProgram):
  502. def __init__(self, station, win, color):
  503. super().__init__(station, win, color, "删除用户")
  504. self.uid_frame = tk.Frame(self.frame)
  505. self.uid_title: tk.Label = tk.Label(self.uid_frame)
  506. self.uid_enter: tk.Entry = tk.Entry(self.uid_frame)
  507. self.uid_var: tk.Variable = tk.StringVar()
  508. self.name_frame = tk.Frame(self.frame)
  509. self.name_title: List[tk.Label] = [tk.Label(self.name_frame) for _ in range(2)]
  510. self.name_enter: List[tk.Entry] = [tk.Entry(self.name_frame) for _ in range(2)]
  511. self.name_var: List[tk.Variable] = [tk.StringVar() for _ in range(2)]
  512. self.btn: List[tk.Button] = [tk.Button(self.frame) for _ in range(2)] # uid-del, name-passwd-del
  513. self.__conf_font()
  514. def __conf_font(self, n: int = 1):
  515. self.title_font_size = int(16 * n)
  516. self.btn_font_size = int(14 * n)
  517. def conf_gui(self, n: int = 1):
  518. self.__conf_font(n)
  519. title_font = make_font(size=self.title_font_size)
  520. btn_font = make_font(size=self.btn_font_size)
  521. self.uid_frame['bg'] = "#FA8072"
  522. self.uid_frame['bd'] = 5
  523. self.uid_frame['relief'] = "ridge"
  524. self.uid_frame.place(relx=0.2, rely=0.20, relwidth=0.6, relheight=0.10)
  525. self.name_frame['bg'] = "#FA8072"
  526. self.name_frame['bd'] = 5
  527. self.name_frame['relief'] = "ridge"
  528. self.name_frame.place(relx=0.2, rely=0.48, relwidth=0.6, relheight=0.25)
  529. height = 0.17
  530. for lb, text, enter, var in zip(self.name_title, ["用户名:", "密码:"], self.name_enter, self.name_var):
  531. lb['font'] = title_font
  532. lb['text'] = text
  533. lb['bg'] = "#FA8072"
  534. lb['anchor'] = 'e'
  535. enter['font'] = title_font
  536. enter['textvariable'] = var
  537. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.20)
  538. enter.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.20)
  539. height += 0.45
  540. self.uid_title['font'] = title_font
  541. self.uid_title['text'] = "用户ID:"
  542. self.uid_title['bg'] = "#FA8072"
  543. self.uid_title['anchor'] = 'e'
  544. self.uid_enter['font'] = title_font
  545. self.uid_enter['textvariable'] = self.uid_var
  546. self.uid_title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  547. self.uid_enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  548. for btn, text, func in zip(self.btn,
  549. ["通过用户ID删除", "通过用户名删除"],
  550. [lambda: self.del_by_uid(), lambda: self.del_by_name()]):
  551. btn['font'] = btn_font
  552. btn['text'] = text
  553. btn['bg'] = Config.tk_btn_bg
  554. btn['command'] = func
  555. self.btn[0].place(relx=0.6, rely=0.32, relwidth=0.2, relheight=0.08)
  556. self.btn[1].place(relx=0.6, rely=0.75, relwidth=0.2, relheight=0.08)
  557. def del_by_uid(self):
  558. uid = self.uid_var.get()
  559. if len(uid) != 32:
  560. self.station.show_warning("用户ID错误", "用户ID必须为32位")
  561. return
  562. event = tk_event.DelUserEvent(self.station).start(uid)
  563. self.station.push_event(event)
  564. def del_by_name(self):
  565. name = self.name_var[0].get()
  566. passwd = self.name_var[1].get()
  567. if len(name) == 0 or len(passwd) == 0:
  568. self.station.show_warning("用户名或密码错误", "请输入用户名和密码")
  569. return
  570. uid = create_uid(name, passwd)
  571. event = tk_event.DelUserEvent(self.station).start(uid)
  572. self.station.push_event(event)
  573. def set_disable(self):
  574. set_tk_disable_from_list(self.btn)
  575. set_tk_disable_from_list(self.name_enter)
  576. self.uid_enter['state'] = 'disable'
  577. def reset_disable(self):
  578. set_tk_disable_from_list(self.btn, flat='normal')
  579. set_tk_disable_from_list(self.name_enter, flat='normal')
  580. self.uid_enter['state'] = 'normal'
  581. class DeleteUsersProgram(AdminProgram):
  582. def __init__(self, station, win, color):
  583. super().__init__(station, win, color, "删除多个用户")
  584. self.enter_frame = tk.Frame(self.frame)
  585. self.title: tk.Label = tk.Label(self.enter_frame)
  586. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  587. self.var: tk.Variable = tk.StringVar()
  588. self.btn: List[tk.Button] = [tk.Button(self.frame) for _ in range(2)] # del, scan
  589. self.__conf_font()
  590. def __conf_font(self, n: int = 1):
  591. self.title_font_size = int(16 * n)
  592. self.btn_font_size = int(14 * n)
  593. def conf_gui(self, n: int = 1):
  594. self.__conf_font(n)
  595. title_font = make_font(size=self.title_font_size)
  596. btn_font = make_font(size=self.btn_font_size)
  597. self.enter_frame['bg'] = "#48c0a3"
  598. self.enter_frame['bd'] = 5
  599. self.enter_frame['relief'] = "ridge"
  600. self.enter_frame.place(relx=0.2, rely=0.30, relwidth=0.6, relheight=0.10)
  601. self.title['font'] = title_font
  602. self.title['text'] = "条件:"
  603. self.title['anchor'] = 'e'
  604. self.title['bg'] = "#48c0a3"
  605. self.enter['font'] = title_font
  606. self.enter['textvariable'] = self.var
  607. self.title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  608. self.enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  609. for btn, text, x, func in zip(self.btn,
  610. ["删除", "扫描"],
  611. [0.2, 0.6],
  612. [lambda: self.delete_user(), lambda: self.scan_user()]):
  613. btn['font'] = btn_font
  614. btn['text'] = text
  615. btn['bg'] = Config.tk_btn_bg
  616. btn['command'] = func
  617. btn.place(relx=x, rely=0.6, relwidth=0.2, relheight=0.08)
  618. def delete_user(self):
  619. where = self.var.get()
  620. if len(where) == 0:
  621. self.station.show_warning("条件错误", "条件必须为正确的SQL语句")
  622. return
  623. event = tk_event.DelUserFromWhereEvent(self.station).start(where)
  624. self.station.push_event(event)
  625. def scan_user(self):
  626. where = self.var.get()
  627. if len(where) == 0:
  628. self.station.show_warning("条件错误", "条件必须为正确的SQL语句")
  629. return
  630. event = tk_event.DelUserFromWhereScanEvent(self.station).start(where)
  631. self.station.push_event(event)
  632. def set_disable(self):
  633. set_tk_disable_from_list(self.btn)
  634. self.enter['state'] = 'disable'
  635. def reset_disable(self):
  636. set_tk_disable_from_list(self.btn, flat='normal')
  637. self.enter['state'] = 'normal'
  638. class DeleteGarbageProgramBase(AdminProgram):
  639. def __init__(self, station, win, color, title: str):
  640. super().__init__(station, win, color, title)
  641. self.enter_frame = tk.Frame(self.frame)
  642. self.title: tk.Label = tk.Label(self.enter_frame)
  643. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  644. self.var: tk.Variable = tk.StringVar()
  645. self.int_var: tk.Variable = tk.IntVar()
  646. self.int_var.set(0)
  647. self.radio: List[tk.Radiobutton] = [tk.Radiobutton(self.frame) for _ in range(4)]
  648. self.btn: tk.Button = tk.Button(self.frame)
  649. self.__conf_font()
  650. self._conf()
  651. def _conf(self, title: str = "垃圾袋ID:", color: str = "#b69968", support_del_all: bool = True):
  652. self.frame_title = title
  653. self.frame_color = color
  654. self.support_del_all = support_del_all
  655. def __conf_font(self, n: int = 1):
  656. self.title_font_size = int(16 * n)
  657. self.btn_font_size = int(14 * n)
  658. def conf_gui(self, n: int = 1):
  659. self.__conf_font(n)
  660. title_font = make_font(size=self.title_font_size)
  661. btn_font = make_font(size=self.btn_font_size)
  662. self.enter_frame['bg'] = self.frame_color
  663. self.enter_frame['bd'] = 5
  664. self.enter_frame['relief'] = "ridge"
  665. self.enter_frame.place(relx=0.2, rely=0.30, relwidth=0.6, relheight=0.10)
  666. self.title['font'] = title_font
  667. self.title['text'] = self.frame_title
  668. self.title['bg'] = self.frame_color
  669. self.title['anchor'] = 'e'
  670. self.enter['font'] = title_font
  671. self.enter['textvariable'] = self.var
  672. self.title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  673. self.enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  674. for i in range(4):
  675. radio = self.radio[i]
  676. radio['font'] = btn_font
  677. radio['text'] = ['均可', '仅未使用', '仅待检测', '仅已检测'][i]
  678. radio['bg'] = self.color
  679. radio['value'] = i
  680. radio['variable'] = self.int_var
  681. radio['anchor'] = 'w'
  682. if not self.support_del_all:
  683. self.int_var.set(1)
  684. self.radio[0]['state'] = 'disable'
  685. self.radio[0].place(relx=0.20, rely=0.43, relwidth=0.20, relheight=0.1)
  686. self.radio[1].place(relx=0.60, rely=0.43, relwidth=0.20, relheight=0.1)
  687. self.radio[2].place(relx=0.20, rely=0.55, relwidth=0.20, relheight=0.1)
  688. self.radio[3].place(relx=0.60, rely=0.55, relwidth=0.20, relheight=0.1)
  689. self.btn['font'] = btn_font
  690. self.btn['text'] = '删除'
  691. self.btn['bg'] = Config.tk_btn_bg
  692. self.btn['command'] = lambda: self.delete_garbage()
  693. self.btn.place(relx=0.4, rely=0.68, relwidth=0.2, relheight=0.08)
  694. def delete_garbage(self):
  695. ...
  696. def set_disable(self):
  697. self.enter['state'] = 'disable'
  698. self.btn['state'] = 'disable'
  699. def reset_disable(self):
  700. self.enter['state'] = 'normal'
  701. self.btn['state'] = 'normal'
  702. class DeleteGarbageProgram(DeleteGarbageProgramBase):
  703. def __init__(self, station, win, color):
  704. super(DeleteGarbageProgram, self).__init__(station, win, color, "删除垃圾袋")
  705. def delete_garbage(self):
  706. where = self.int_var.get()
  707. assert where in [0, 1, 2, 3]
  708. gid = self.var.get()
  709. if len(gid) == 0:
  710. self.station.show_warning("垃圾袋ID错误", "请输入正确的垃圾袋ID")
  711. return
  712. event = tk_event.DelGarbageEvent(self.station).start(gid, where)
  713. self.station.push_event(event)
  714. class DeleteGarbageMoreProgram(DeleteGarbageProgramBase):
  715. def __init__(self, station, win, color):
  716. super(DeleteGarbageMoreProgram, self).__init__(station, win, color, "删除多个垃圾袋")
  717. self.scan_btn = tk.Button(self.frame)
  718. self._conf("条件:", "#f58f98", False)
  719. def conf_gui(self, n: int = 1):
  720. super(DeleteGarbageMoreProgram, self).conf_gui(n)
  721. self.btn.place_forget()
  722. self.btn.place(relx=0.2, rely=0.68, relwidth=0.2, relheight=0.08)
  723. self.scan_btn['font'] = make_font(size=self.btn_font_size)
  724. self.scan_btn['text'] = '扫描'
  725. self.scan_btn['bg'] = Config.tk_btn_bg
  726. self.scan_btn['command'] = self.scan_garbage
  727. self.scan_btn.place(relx=0.6, rely=0.68, relwidth=0.2, relheight=0.08)
  728. def set_disable(self):
  729. super(DeleteGarbageMoreProgram, self).set_disable()
  730. self.scan_btn['state'] = 'disable'
  731. def reset_disable(self):
  732. super(DeleteGarbageMoreProgram, self).reset_disable()
  733. self.scan_btn['state'] = 'normal'
  734. def delete_garbage(self):
  735. where = self.int_var.get()
  736. assert where in [1, 2, 3]
  737. where_sql = self.var.get()
  738. if len(where_sql) == 0:
  739. self.station.show_warning("条件错误", "条件必须为正确的SQL语句")
  740. return
  741. event = tk_event.DelGarbageWhereEvent(self.station).start(where, where_sql)
  742. self.station.push_event(event)
  743. def scan_garbage(self):
  744. where = self.int_var.get()
  745. assert where in [1, 2, 3]
  746. where_sql = self.var.get()
  747. if len(where_sql) == 0:
  748. self.station.show_warning("条件错误", "条件必须为正确的SQL语句")
  749. return
  750. event = tk_event.DelGarbageWhereScanEvent(self.station).start(where, where_sql)
  751. self.station.push_event(event)
  752. class DeleteAllGarbageProgram(AdminProgram):
  753. def __init__(self, station, win, color):
  754. super().__init__(station, win, color, "删除所有垃圾袋")
  755. self.dangerous: tk.Label = tk.Label(self.frame)
  756. self.enter_frame = tk.Frame(self.frame)
  757. self.title: tk.Label = tk.Label(self.enter_frame)
  758. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  759. self.var: tk.Variable = tk.StringVar()
  760. self.btn: List[tk.Button] = [tk.Button(self.frame) for _ in range(2)] # del, scan
  761. self.__conf_font()
  762. def __conf_font(self, n: int = 1):
  763. self.danger_font_size = int(20 * n)
  764. self.title_font_size = int(16 * n)
  765. self.btn_font_size = int(14 * n)
  766. def conf_gui(self, n: int = 1):
  767. self.__conf_font(n)
  768. danger_font = make_font(size=self.danger_font_size, weight="bold", underline=1)
  769. title_font = make_font(size=self.title_font_size)
  770. btn_font = make_font(size=self.btn_font_size)
  771. danger_btn_font = make_font(size=self.btn_font_size, weight="bold", overstrike=1)
  772. self.dangerous['bg'] = self.color
  773. self.dangerous['font'] = danger_font
  774. self.dangerous['fg'] = "#f20c00"
  775. self.dangerous['text'] = ("确定要从数据库删除所有垃圾袋吗?\n"
  776. "请输入[admin]用户的密码再继续操作.\n"
  777. "只有[admin]用户具有该操作的权限.\n"
  778. "这是相当危险的操作.\n"
  779. "操作后数据库可能无法恢复原数据.\n"
  780. "SuperHuan和程序的缔造者不会对\n"
  781. "此操作负责.\n"
  782. "删库跑路可不是一件好事.\n"
  783. "请遵守当地法律法规.")
  784. self.dangerous.place(relx=0.05, rely=0.03, relwidth=0.9, relheight=0.53)
  785. self.enter_frame['bg'] = "#f20c00"
  786. self.enter_frame['bd'] = 5
  787. self.enter_frame['relief'] = "ridge"
  788. self.enter_frame.place(relx=0.2, rely=0.60, relwidth=0.6, relheight=0.10)
  789. self.title['font'] = title_font
  790. self.title['text'] = "密码:"
  791. self.title['bg'] = "#f20c00"
  792. self.title['anchor'] = 'e'
  793. self.enter['font'] = title_font
  794. self.enter['textvariable'] = self.var
  795. self.title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  796. self.enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  797. for btn, text, x in zip(self.btn, ["删除", "扫描"], [0.2, 0.6]):
  798. btn['text'] = text
  799. btn.place(relx=x, rely=0.78, relwidth=0.2, relheight=0.08)
  800. self.btn[0]['font'] = danger_btn_font
  801. self.btn[0]['bg'] = "#f20c00"
  802. self.btn[0]['command'] = lambda: self.delete_garbage()
  803. self.btn[1]['font'] = btn_font
  804. self.btn[1]['bg'] = Config.tk_btn_bg
  805. self.btn[1]['command'] = lambda: self.scan_garbage()
  806. def scan_garbage(self):
  807. event = tk_event.DelAllGarbageScanEvent(self.station) # 不需要start
  808. self.station.push_event(event)
  809. def delete_garbage(self):
  810. passwd = self.var.get()
  811. if len(passwd) == 0:
  812. self.station.show_warning("密码错误", "请输入正确的[admin]用户密码")
  813. user = find_user_by_name('admin', passwd, self.station.get_db())
  814. if user is None or not user.is_manager():
  815. self.station.show_warning("密码错误", "请输入正确的[admin]用户密码")
  816. return
  817. event = tk_event.DelAllGarbageEvent(self.station) # 不需要start
  818. self.station.push_event(event)
  819. def set_disable(self):
  820. set_tk_disable_from_list(self.btn)
  821. self.enter['state'] = 'disable'
  822. def reset_disable(self):
  823. set_tk_disable_from_list(self.btn, flat='normal')
  824. self.enter['state'] = 'normal'
  825. class SearchProgramBase(AdminProgram, metaclass=abc.ABCMeta):
  826. def __init__(self, station, win, color, title: str):
  827. super().__init__(station, win, color, title)
  828. self.view_frame = tk.Frame(self.frame)
  829. self.view = ttk.Treeview(self.view_frame)
  830. self.y_scroll = tk.Scrollbar(self.view_frame)
  831. self.x_scroll = tk.Scrollbar(self.view_frame)
  832. def conf_view_gui(self, columns: list, relx, rely, relwidth, relheight,
  833. x_scroll=0.05, y_scroll=0.02, color: str = "#FA8072"):
  834. self.view_frame['bg'] = color
  835. self.view_frame['bd'] = 2
  836. self.view_frame['relief'] = "ridge"
  837. self.view_frame.place(relx=relx, rely=rely, relwidth=relwidth, relheight=relheight)
  838. self.view['columns'] = columns
  839. self.view['show'] = 'headings'
  840. self.view['selectmode'] = 'none'
  841. for i in columns:
  842. self.view.column(i, anchor="c")
  843. self.view.heading(i, text=i)
  844. self.y_scroll['orient'] = 'vertical'
  845. self.y_scroll['command'] = self.view.yview
  846. self.view['yscrollcommand'] = self.y_scroll.set
  847. self.x_scroll['orient'] = 'horizontal'
  848. self.x_scroll['command'] = self.view.xview
  849. self.view['xscrollcommand'] = self.x_scroll.set
  850. self.view.place(relx=0.0, rely=0.0, relwidth=1 - y_scroll, relheight=1 - x_scroll)
  851. self.y_scroll.place(relx=0.98, rely=0.0, relwidth=y_scroll, relheight=1.0)
  852. self.x_scroll.place(relx=0.0, rely=1 - x_scroll, relwidth=1 - y_scroll, relheight=x_scroll)
  853. class SearchUserProgram(SearchProgramBase):
  854. def __init__(self, station, win, color):
  855. super().__init__(station, win, color, "搜索用户")
  856. self.enter_frame = tk.Frame(self.frame)
  857. self.title: List[tk.Label] = [tk.Label(self.enter_frame) for _ in range(3)]
  858. self.enter: List[tk.Entry] = [tk.Entry(self.enter_frame) for _ in range(3)]
  859. self.var: List[tk.Variable] = [tk.StringVar() for _ in range(3)]
  860. self.check: List[Tuple[tk.Checkbutton, tk.Variable]] = [(tk.Checkbutton(self.enter_frame), tk.IntVar())
  861. for _ in range(3)]
  862. self.btn: tk.Button = tk.Button(self.frame)
  863. self._columns = ["UserID", "Name", "Phone", "Score", "Reputation", "IsManager"]
  864. self._columns_ch = ["用户ID[UserID]", "用户名[Name]", "手机号[Phone]",
  865. "积分[Score]", "垃圾分类信用[Reputation]", "是否管理员[IsManager]"]
  866. self.__conf_font()
  867. def __conf_font(self, n: int = 1):
  868. self.title_font_size = int(16 * n)
  869. self.btn_font_size = int(14 * n)
  870. def conf_gui(self, n: int = 1):
  871. self.__conf_font(n)
  872. title_font = make_font(size=self.title_font_size)
  873. btn_font = make_font(size=self.btn_font_size)
  874. self.enter_frame['bg'] = "#FA8072"
  875. self.enter_frame['bd'] = 5
  876. self.enter_frame['relief'] = "ridge"
  877. self.enter_frame.place(relx=0.2, rely=0.0, relwidth=0.6, relheight=0.30)
  878. height = 0.1
  879. for lb, text, enter, var, check in zip(self.title,
  880. ["用户ID:", "用户名:", "手机号:"],
  881. self.enter, self.var, self.check):
  882. lb['font'] = title_font
  883. lb['text'] = text
  884. lb['bg'] = "#FA8072"
  885. lb['anchor'] = 'e'
  886. enter['font'] = title_font
  887. enter['textvariable'] = var
  888. check[0]['font'] = title_font
  889. check[0]['text'] = ''
  890. check[0]['bg'] = "#FA8072"
  891. check[0]['variable'] = check[1]
  892. check[1].set(1)
  893. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.17)
  894. enter.place(relx=0.35, rely=height, relwidth=0.55, relheight=0.17)
  895. check[0].place(relx=0.92, rely=height, relwidth=0.04, relheight=0.17)
  896. height += 0.30
  897. self.btn['font'] = btn_font
  898. self.btn['text'] = "搜索"
  899. self.btn['bg'] = Config.tk_btn_bg
  900. self.btn['command'] = self.search_user
  901. self.btn.place(relx=0.4, rely=0.9, relwidth=0.2, relheight=0.08)
  902. self.conf_view_gui(self._columns_ch, relx=0.05, rely=0.32, relwidth=0.9, relheight=0.55)
  903. def search_user(self):
  904. use_uid = self.check[0][1].get()
  905. use_name = self.check[1][1].get()
  906. use_phone = self.check[2][1].get()
  907. uid = None
  908. name = None
  909. phone = None
  910. if use_uid:
  911. uid = self.var[0].get()
  912. if len(uid) == 0:
  913. uid = None
  914. if use_name:
  915. name = self.var[1].get()
  916. if len(name) == 0:
  917. name = None
  918. if use_phone:
  919. phone = self.var[2].get()
  920. if len(phone) == 0:
  921. phone = None
  922. event = tk_event.SearchUserEvent(self.station).start(self._columns, uid, name, phone, self)
  923. self.station.push_event(event)
  924. def set_disable(self):
  925. self.btn['state'] = 'disable'
  926. set_tk_disable_from_list(self.enter)
  927. def reset_disable(self):
  928. self.btn['state'] = 'normal'
  929. set_tk_disable_from_list(self.enter, flat='normal')
  930. class SearchAdvancedProgramBase(SearchProgramBase, metaclass=abc.ABCMeta):
  931. def __init__(self, station, win, color, title: str):
  932. super().__init__(station, win, color, title)
  933. self.enter_frame = tk.Frame(self.frame)
  934. self.title: tk.Label = tk.Label(self.enter_frame)
  935. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  936. self.var: tk.Variable = tk.StringVar()
  937. self.btn: tk.Button = tk.Button(self.frame)
  938. self._conf([], [], "#FA8072") # 默认颜色
  939. self.__conf_font()
  940. def _conf(self, columns: list, columns_ch: list, bg_color):
  941. self.bg_color = bg_color
  942. self._columns = columns
  943. self._columns_ch = columns_ch
  944. return self
  945. def __conf_font(self, n: int = 1):
  946. self.title_font_size = int(16 * n)
  947. self.btn_font_size = int(14 * n)
  948. def conf_gui(self, n: int = 1):
  949. self.__conf_font(n)
  950. title_font = make_font(size=self.title_font_size)
  951. btn_font = make_font(size=self.btn_font_size)
  952. self.enter_frame['bg'] = self.bg_color
  953. self.enter_frame['bd'] = 5
  954. self.enter_frame['relief'] = "ridge"
  955. self.enter_frame.place(relx=0.2, rely=0.00, relwidth=0.6, relheight=0.10)
  956. self.title['font'] = title_font
  957. self.title['bg'] = self.bg_color
  958. self.title['text'] = "条件:"
  959. self.title['anchor'] = 'e'
  960. self.enter['font'] = title_font
  961. self.enter['textvariable'] = self.var
  962. self.title.place(relx=0.01, rely=0.25, relwidth=0.30, relheight=0.50)
  963. self.enter.place(relx=0.35, rely=0.25, relwidth=0.60, relheight=0.50)
  964. self.btn['text'] = "搜索"
  965. self.btn['font'] = btn_font
  966. self.btn['bg'] = Config.tk_btn_bg
  967. self.btn['command'] = self.search
  968. self.btn.place(relx=0.4, rely=0.9, relwidth=0.2, relheight=0.08)
  969. self.conf_view_gui(self._columns_ch, relx=0.05, rely=0.12, relwidth=0.9, relheight=0.76)
  970. def search(self):
  971. ...
  972. def set_disable(self):
  973. self.btn['state'] = 'disable'
  974. self.enter['state'] = 'disable'
  975. def reset_disable(self):
  976. self.btn['state'] = 'normal'
  977. self.enter['state'] = 'normal'
  978. class SearchUserAdvancedProgram(SearchAdvancedProgramBase):
  979. def __init__(self, station, win, color):
  980. super(SearchUserAdvancedProgram, self).__init__(station, win, color, "高级搜索-用户")
  981. columns = ["UserID", "Name", "Phone", "Score", "Reputation", "IsManager"]
  982. columns_ch = ["用户ID[UserID]", "用户名[Name]", "手机号[Phone]",
  983. "积分[Score]", "垃圾分类信用[Reputation]", "是否管理员[IsManager]"]
  984. self._conf(columns, columns_ch, '#48c0a3')
  985. def search(self):
  986. where = self.var.get()
  987. event = tk_event.SearchUserAdvancedEvent(self.station).start(self._columns, where, self)
  988. self.station.push_event(event)
  989. class SearchGarbageProgram(SearchProgramBase):
  990. def __init__(self, station, win, color):
  991. super().__init__(station, win, color, "搜索垃圾袋")
  992. self.enter_frame = tk.Frame(self.frame)
  993. self.title: List[tk.Label] = [tk.Label(self.enter_frame) for _ in range(8)]
  994. self.enter: List[tk.Entry] = [tk.Entry(self.enter_frame) for _ in range(8)]
  995. self.var: List[tk.Variable] = [tk.StringVar() for _ in range(8)]
  996. self.check: List[Tuple[tk.Checkbutton, tk.Variable]] = [(tk.Checkbutton(self.enter_frame), tk.IntVar())
  997. for _ in range(8)]
  998. self._columns = ["GarbageID", "UserID", "CheckerID", "CreateTime", "UseTime", "Location", "GarbageType",
  999. "CheckResult"]
  1000. self._columns_zh = ["垃圾袋ID[GarbageID]", "使用者ID[UserID]", "检测者ID[CheckerID]", "创建时间[CreateTime]",
  1001. "使用时间[UseTime]", "使用地点[Location]", "垃圾类型[GarbageType]", "检测结果[CheckResult]"]
  1002. self.btn: tk.Button = tk.Button(self.frame)
  1003. self.__conf_font()
  1004. def __conf_font(self, n: int = 1):
  1005. self.title_font_size = int(16 * n)
  1006. self.btn_font_size = int(14 * n)
  1007. def conf_gui(self, n: int = 1):
  1008. self.__conf_font(n)
  1009. title_font = make_font(size=self.title_font_size)
  1010. btn_font = make_font(size=self.btn_font_size)
  1011. self.enter_frame['bg'] = "#7bbfea"
  1012. self.enter_frame['bd'] = 5
  1013. self.enter_frame['relief'] = "ridge"
  1014. self.enter_frame.place(relx=0.2, rely=0.0, relwidth=0.6, relheight=0.47)
  1015. height = 0.02
  1016. for lb, text, enter, var, check in zip(self.title,
  1017. ["垃圾袋ID:", "使用者ID:", "检查者ID:", "创建时间:", "使用时间:",
  1018. "使用地点:", "垃圾类型:", "检测结果:"],
  1019. self.enter, self.var, self.check):
  1020. lb['font'] = title_font
  1021. lb['text'] = text
  1022. lb['bg'] = "#7bbfea"
  1023. lb['anchor'] = 'e'
  1024. enter['font'] = title_font
  1025. enter['textvariable'] = var
  1026. check[0]['font'] = title_font
  1027. check[0]['bg'] = "#7bbfea"
  1028. check[0]['text'] = ''
  1029. check[0]['variable'] = check[1]
  1030. check[1].set(1)
  1031. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.10)
  1032. enter.place(relx=0.35, rely=height, relwidth=0.55, relheight=0.10)
  1033. check[0].place(relx=0.92, rely=height, relwidth=0.04, relheight=0.10)
  1034. height += 0.121
  1035. self.btn['font'] = btn_font
  1036. self.btn['bg'] = Config.tk_btn_bg
  1037. self.btn['text'] = "Search"
  1038. self.btn['command'] = self.search_user
  1039. self.btn.place(relx=0.4, rely=0.9, relwidth=0.2, relheight=0.08)
  1040. self.conf_view_gui(self._columns_zh, relx=0.05, rely=0.49, relwidth=0.9, relheight=0.38, x_scroll=0.07)
  1041. def search_user(self):
  1042. keys = ["gid", "uid", "cuid", "create_time", "use_time", "loc", "type_", "check"]
  1043. key_values = {}
  1044. for i, key in enumerate(keys):
  1045. ck = self.check[i][1].get()
  1046. if ck:
  1047. res = self.enter[i].get()
  1048. if len(res) > 0:
  1049. key_values[key] = res
  1050. continue
  1051. key_values[key] = None
  1052. event = tk_event.SearchGarbageEvent(self.station).start(self._columns, key_values, self)
  1053. self.station.push_event(event)
  1054. def set_disable(self):
  1055. self.btn['state'] = 'disable'
  1056. set_tk_disable_from_list(self.enter)
  1057. def reset_disable(self):
  1058. self.btn['state'] = 'normal'
  1059. set_tk_disable_from_list(self.enter, flat='normal')
  1060. class SearchGarbageAdvancedProgram(SearchAdvancedProgramBase):
  1061. def __init__(self, station, win, color):
  1062. super(SearchGarbageAdvancedProgram, self).__init__(station, win, color, "高级搜索-垃圾袋")
  1063. columns = ["GarbageID", "UserID", "CheckerID", "CreateTime", "UseTime", "Location", "GarbageType",
  1064. "CheckResult"]
  1065. columns_zh = ["垃圾袋ID[GarbageID]", "使用者ID[UserID]", "检测者ID[CheckerID]", "创建时间[CreateTime]",
  1066. "使用时间[UseTime]", "使用地点[Location]", "垃圾类型[GarbageType]", "检测结果[CheckResult]"]
  1067. self._conf(columns, columns_zh, '#d1923f')
  1068. def search(self):
  1069. where = self.var.get()
  1070. event = tk_event.SearchGarbageAdvancedEvent(self.station).start(self._columns, where, self)
  1071. self.station.push_event(event)
  1072. class SearchAdvancedProgram(SearchAdvancedProgramBase):
  1073. def __init__(self, station, win, color):
  1074. super(SearchAdvancedProgram, self).__init__(station, win, color, "高级搜索")
  1075. columns = ["GarbageID", "UserID", "UserName", "UserPhone", "UserScore",
  1076. "UserReputation", "CheckerID", "CheckerName", "CheckerPhone",
  1077. "CreateTime", "UseTime", "Location", "GarbageType", "CheckResult"]
  1078. columns_zh = ["垃圾袋ID[GarbageID]", "使用者ID[UserID]", "使用者名[UserName]", "使用者手机号[UserPhone]",
  1079. "使用者积分[UserScore]", "使用者垃圾分类信用[UserReputation]", "检测者ID[CheckerID]",
  1080. "检测这名[CheckerName]", "检测者手机号[CheckerPhone]", "创建时间[CreateTime]", "使用时间[UseTime]",
  1081. "使用地点[Location]", "垃圾类型[GarbageType]", "检测结果[CheckResult]"]
  1082. self._conf(columns, columns_zh, '#426ab3')
  1083. def search(self):
  1084. where = self.var.get()
  1085. event = tk_event.SearchAdvancedEvent(self.station).start(self._columns, where, self)
  1086. self.station.push_event(event)
  1087. class UpdateUserProgramBase(AdminProgram):
  1088. def __init__(self, station, win, color, title: str):
  1089. super().__init__(station, win, color, title)
  1090. self.enter_frame = tk.Frame(self.frame)
  1091. self.title: List[tk.Label] = [tk.Label(self.enter_frame) for _ in range(2)]
  1092. self.enter: List[tk.Entry] = [tk.Entry(self.enter_frame) for _ in range(2)]
  1093. self.var: List[tk.Variable] = [tk.StringVar() for _ in range(2)]
  1094. self.where_frame = tk.Frame(self.frame)
  1095. self.where_title: List[tk.Label] = [tk.Label(self.where_frame) for _ in range(2)]
  1096. self.where_enter: List[tk.Entry] = [tk.Entry(self.where_frame) for _ in range(2)]
  1097. self.where_var: List[tk.Variable] = [tk.StringVar() for _ in range(2)]
  1098. self.btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  1099. self._conf(["", ""], "#FA8072")
  1100. self.__conf_font()
  1101. def _conf(self, title: List[str], bg_color: str):
  1102. self.bg_color = bg_color
  1103. self.bg_color_where = bg_color
  1104. self.enter_title = title
  1105. def __conf_font(self, n: int = 1):
  1106. self.title_font_size = int(16 * n)
  1107. self.btn_font_size = int(14 * n)
  1108. def conf_gui(self, n: int = 1):
  1109. self.__conf_font(n)
  1110. title_font = make_font(size=self.title_font_size)
  1111. btn_font = make_font(size=self.btn_font_size)
  1112. self.where_frame['bg'] = self.bg_color_where
  1113. self.where_frame['bd'] = 5
  1114. self.where_frame['relief'] = "ridge"
  1115. self.where_frame.place(relx=0.2, rely=0.20, relwidth=0.6, relheight=0.17)
  1116. self.enter_frame['bg'] = self.bg_color
  1117. self.enter_frame['bd'] = 5
  1118. self.enter_frame['relief'] = "ridge"
  1119. self.enter_frame.place(relx=0.2, rely=0.58, relwidth=0.6, relheight=0.17)
  1120. height = 0.1
  1121. for lb, text, enter, var, lb_w, text_w, enter_w, var_w in (
  1122. zip(self.title, self.enter_title, self.enter, self.var,
  1123. self.where_title, ["条件:", self.enter_title[1]], self.where_enter, self.where_var)):
  1124. lb['font'] = title_font
  1125. lb['text'] = text
  1126. lb['bg'] = self.bg_color
  1127. lb['anchor'] = 'e'
  1128. lb_w['font'] = title_font
  1129. lb_w['text'] = text_w
  1130. lb_w['bg'] = self.bg_color_where
  1131. lb_w['anchor'] = 'e'
  1132. enter['font'] = title_font
  1133. enter['textvariable'] = var
  1134. enter_w['font'] = title_font
  1135. enter_w['textvariable'] = var_w
  1136. lb.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.35)
  1137. enter.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.35)
  1138. lb_w.place(relx=0.01, rely=height, relwidth=0.30, relheight=0.35)
  1139. enter_w.place(relx=0.35, rely=height, relwidth=0.60, relheight=0.35)
  1140. height += 0.43
  1141. for btn, text, func in zip(self.btn,
  1142. ["通过条件更新", "通过用户ID更新"],
  1143. [self.update_by_where, self.update_by_uid]):
  1144. btn['font'] = btn_font
  1145. btn['text'] = text
  1146. btn['bg'] = Config.tk_btn_bg
  1147. btn['command'] = func
  1148. self.btn[0].place(relx=0.55, rely=0.40, relwidth=0.25, relheight=0.08)
  1149. self.btn[1].place(relx=0.55, rely=0.78, relwidth=0.25, relheight=0.08)
  1150. def update_by_uid(self):
  1151. ...
  1152. def update_by_where(self):
  1153. ...
  1154. def set_disable(self):
  1155. set_tk_disable_from_list(self.btn)
  1156. set_tk_disable_from_list(self.enter)
  1157. def reset_disable(self):
  1158. set_tk_disable_from_list(self.btn, flat='normal')
  1159. set_tk_disable_from_list(self.enter, flat='normal')
  1160. class UpdateUserScoreBase(UpdateUserProgramBase):
  1161. def __init__(self, station, win, color):
  1162. super(UpdateUserScoreBase, self).__init__(station, win, color, "更新用户-积分")
  1163. self._conf(["用户ID:", "积分:"], "#afdfe4")
  1164. def update_by_uid(self):
  1165. uid = self.enter[0].get()
  1166. score = int(self.enter[1].get())
  1167. event = tk_event.UpdateUserScoreEvent(self.station).start(score, f"UserID='{uid}'")
  1168. self.station.push_event(event)
  1169. def update_by_where(self):
  1170. where = self.where_enter[0].get()
  1171. score = int(self.where_enter[1].get())
  1172. event = tk_event.UpdateUserScoreEvent(self.station).start(score, where)
  1173. self.station.push_event(event)
  1174. class UpdateUserReputationBase(UpdateUserProgramBase):
  1175. def __init__(self, station, win, color):
  1176. super(UpdateUserReputationBase, self).__init__(station, win, color, "更新用户-垃圾分类信用")
  1177. self._conf(["用户ID:", "垃圾分类信用:"], "#f8aba6")
  1178. def update_by_uid(self):
  1179. uid = self.enter[0].get()
  1180. reputation = int(self.enter[1].get())
  1181. event = tk_event.UpdateUserReputationEvent(self.station).start(reputation, f"UserID='{uid}'")
  1182. self.station.push_event(event)
  1183. def update_by_where(self):
  1184. where = self.where_enter[0].get()
  1185. reputation = int(self.where_enter[1].get())
  1186. event = tk_event.UpdateUserReputationEvent(self.station).start(reputation, where)
  1187. self.station.push_event(event)
  1188. class UpdateGarbageTypeProgram(AdminProgram):
  1189. def __init__(self, station, win, color):
  1190. super().__init__(station, win, color, "更新垃圾袋-垃圾类型")
  1191. self.enter_frame = tk.Frame(self.frame)
  1192. self.title: tk.Label = tk.Label(self.enter_frame)
  1193. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  1194. self.type: List[tk.Radiobutton] = [tk.Radiobutton(self.frame) for _ in range(4)]
  1195. self.var: List[tk.Variable] = [tk.StringVar, tk.IntVar()]
  1196. self.where_frame = tk.Frame(self.frame)
  1197. self.where_title: tk.Label = tk.Label(self.where_frame)
  1198. self.where_enter: tk.Entry = tk.Entry(self.where_frame)
  1199. self.where_type: List[tk.Radiobutton] = [tk.Radiobutton(self.frame) for _ in range(4)]
  1200. self.where_var: List[tk.Variable] = [tk.StringVar, tk.IntVar()]
  1201. self.btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  1202. self.__conf_font()
  1203. def __conf_font(self, n: int = 1):
  1204. self.title_font_size = int(16 * n)
  1205. self.btn_font_size = int(14 * n)
  1206. def conf_gui(self, n: int = 1):
  1207. self.__conf_font(n)
  1208. title_font = make_font(size=self.title_font_size)
  1209. btn_font = make_font(size=self.btn_font_size)
  1210. self.where_frame['bg'] = "#fdb933"
  1211. self.where_frame['bd'] = 5
  1212. self.where_frame['relief'] = "ridge"
  1213. self.where_frame.place(relx=0.2, rely=0.20, relwidth=0.6, relheight=0.10)
  1214. self.enter_frame['bg'] = "#fdb933"
  1215. self.enter_frame['bd'] = 5
  1216. self.enter_frame['relief'] = "ridge"
  1217. self.enter_frame.place(relx=0.2, rely=0.60, relwidth=0.6, relheight=0.10)
  1218. for lb, enter, radios, var, y, text in zip([self.title, self.where_title],
  1219. [self.enter, self.where_enter],
  1220. [self.type, self.where_type],
  1221. [self.var, self.where_var],
  1222. [0.32, 0.72],
  1223. ["垃圾袋ID:", "条件:"]):
  1224. lb['font'] = title_font
  1225. lb['text'] = text
  1226. lb['bg'] = "#fdb933"
  1227. lb['anchor'] = 'e'
  1228. enter['font'] = title_font
  1229. enter['textvariable'] = var[0]
  1230. for i, radio in enumerate(radios):
  1231. radio['font'] = btn_font
  1232. radio['bg'] = self.color
  1233. radio['text'] = GarbageType.GarbageTypeStrList_ch[i + 1]
  1234. radio['value'] = i + 1
  1235. radio['variable'] = var[1]
  1236. radio['anchor'] = 'w'
  1237. var[1].set(1)
  1238. radios[0].place(relx=0.20, rely=y + 0.00, relwidth=0.20, relheight=0.04)
  1239. radios[1].place(relx=0.60, rely=y + 0.00, relwidth=0.20, relheight=0.04)
  1240. radios[2].place(relx=0.20, rely=y + 0.05, relwidth=0.20, relheight=0.04)
  1241. radios[3].place(relx=0.60, rely=y + 0.05, relwidth=0.20, relheight=0.04)
  1242. lb.place(relx=0.02, rely=0.2, relwidth=0.25, relheight=0.48)
  1243. enter.place(relx=0.30, rely=0.2, relwidth=0.60, relheight=0.48)
  1244. for btn, text, func in zip(self.btn,
  1245. ["通过条件更新", "通过垃圾袋ID更新"],
  1246. [self.update_by_where, self.update_by_gid]):
  1247. btn['font'] = btn_font
  1248. btn['text'] = text
  1249. btn['bg'] = Config.tk_btn_bg
  1250. btn['command'] = func
  1251. self.btn[0].place(relx=0.55, rely=0.43, relwidth=0.25, relheight=0.08)
  1252. self.btn[1].place(relx=0.55, rely=0.83, relwidth=0.25, relheight=0.08)
  1253. def update_by_gid(self):
  1254. gid = self.enter.get()
  1255. type_ = self.var[1].get()
  1256. event = tk_event.UpdateGarbageTypeEvent(self.station).start(type_, f"GarbageID={gid}")
  1257. self.station.push_event(event)
  1258. def update_by_where(self):
  1259. where = self.where_enter.get()
  1260. type_ = self.where_var[1].get()
  1261. event = tk_event.UpdateGarbageTypeEvent(self.station).start(type_, where)
  1262. self.station.push_event(event)
  1263. def set_disable(self):
  1264. set_tk_disable_from_list(self.btn)
  1265. self.enter['state'] = 'disable'
  1266. self.where_enter['state'] = 'normal'
  1267. def reset_disable(self):
  1268. set_tk_disable_from_list(self.btn, flat='normal')
  1269. self.enter['state'] = 'normal'
  1270. self.where_enter['state'] = 'normal'
  1271. class UpdateGarbageCheckResultProgram(AdminProgram):
  1272. def __init__(self, station, win, color):
  1273. super().__init__(station, win, color, "更新垃圾袋-检测结果")
  1274. self.enter_frame = tk.Frame(self.frame)
  1275. self.title: tk.Label = tk.Label(self.enter_frame)
  1276. self.enter: tk.Entry = tk.Entry(self.enter_frame)
  1277. self.type: List[tk.Radiobutton] = [tk.Radiobutton(self.frame) for _ in range(2)]
  1278. self.var: List[tk.Variable] = [tk.StringVar, tk.IntVar()]
  1279. self.where_frame = tk.Frame(self.frame)
  1280. self.where_title: tk.Label = tk.Label(self.where_frame)
  1281. self.where_enter: tk.Entry = tk.Entry(self.where_frame)
  1282. self.where_type: List[tk.Radiobutton] = [tk.Radiobutton(self.frame) for _ in range(2)]
  1283. self.where_var: List[tk.Variable] = [tk.StringVar, tk.IntVar()]
  1284. self.btn: List[tk.Button] = [tk.Button(self.frame), tk.Button(self.frame)]
  1285. self.__conf_font()
  1286. def __conf_font(self, n: int = 1):
  1287. self.title_font_size = int(16 * n)
  1288. self.btn_font_size = int(14 * n)
  1289. def conf_gui(self, n: int = 1):
  1290. self.__conf_font(n)
  1291. title_font = make_font(size=self.title_font_size)
  1292. btn_font = make_font(size=self.btn_font_size)
  1293. self.where_frame['bg'] = "#abc88b"
  1294. self.where_frame['bd'] = 5
  1295. self.where_frame['relief'] = "ridge"
  1296. self.where_frame.place(relx=0.2, rely=0.20, relwidth=0.6, relheight=0.10)
  1297. self.enter_frame['bg'] = "#abc88b"
  1298. self.enter_frame['bd'] = 5
  1299. self.enter_frame['relief'] = "ridge"
  1300. self.enter_frame.place(relx=0.2, rely=0.60, relwidth=0.6, relheight=0.10)
  1301. for lb, enter, radios, var, y, text in zip([self.title, self.where_title],
  1302. [self.enter, self.where_enter],
  1303. [self.type, self.where_type],
  1304. [self.var, self.where_var],
  1305. [0.32, 0.72],
  1306. ["垃圾袋ID:", "条件:"]):
  1307. lb['font'] = title_font
  1308. lb['text'] = text
  1309. lb['bg'] = "#abc88b"
  1310. lb['anchor'] = 'e'
  1311. enter['font'] = title_font
  1312. enter['textvariable'] = var[0]
  1313. for i, radio in enumerate(radios):
  1314. radio['font'] = btn_font
  1315. radio['bg'] = self.color
  1316. radio['text'] = ["投放错误", "投放正确"][i]
  1317. radio['value'] = i
  1318. radio['variable'] = var[1]
  1319. radio['anchor'] = 'w'
  1320. var[1].set(1)
  1321. radios[0].place(relx=0.20, rely=y + 0.00, relwidth=0.20, relheight=0.04)
  1322. radios[1].place(relx=0.60, rely=y + 0.00, relwidth=0.20, relheight=0.04)
  1323. lb.place(relx=0.02, rely=0.2, relwidth=0.25, relheight=0.48)
  1324. enter.place(relx=0.30, rely=0.2, relwidth=0.60, relheight=0.48)
  1325. for btn, text, func in zip(self.btn,
  1326. ["通过条件更新", "通过垃圾袋ID更新"],
  1327. [self.update_by_where, self.update_by_gid]):
  1328. btn['font'] = btn_font
  1329. btn['bg'] = Config.tk_btn_bg
  1330. btn['text'] = text
  1331. btn['command'] = func
  1332. self.btn[0].place(relx=0.55, rely=0.38, relwidth=0.25, relheight=0.08)
  1333. self.btn[1].place(relx=0.55, rely=0.78, relwidth=0.25, relheight=0.08)
  1334. def update_by_gid(self):
  1335. gid = self.enter.get()
  1336. check = (self.var[1].get() == 1)
  1337. event = tk_event.UpdateGarbageCheckEvent(self.station).start(check, f"GarbageID={gid}")
  1338. self.station.push_event(event)
  1339. def update_by_where(self):
  1340. where = self.where_enter.get()
  1341. check = (self.where_var[1].get() == 1)
  1342. event = tk_event.UpdateGarbageCheckEvent(self.station).start(check, where)
  1343. self.station.push_event(event)
  1344. def set_disable(self):
  1345. set_tk_disable_from_list(self.btn)
  1346. self.enter['state'] = 'disable'
  1347. self.where_enter['state'] = 'normal'
  1348. def reset_disable(self):
  1349. set_tk_disable_from_list(self.btn, flat='normal')
  1350. self.enter['state'] = 'normal'
  1351. self.where_enter['state'] = 'normal'
  1352. class StatisticsTimeBaseProgram(AdminProgram):
  1353. def __init__(self, station, win, color, title: str):
  1354. super().__init__(station, win, color, title)
  1355. self.figure_frame = tk.Frame(self.frame)
  1356. self.figure = Figure(dpi=100)
  1357. self.plt: Axes = self.figure.add_subplot(111) # 添加子图:1行1列第1个
  1358. self.font = font_manager.FontProperties(fname=Config.font_d["noto"])
  1359. self.canvas = FigureCanvasTkAgg(self.figure, master=self.figure_frame)
  1360. self.canvas_tk = self.canvas.get_tk_widget()
  1361. self.toolbar = NavigationToolbar2Tk(self.canvas, self.figure_frame)
  1362. self.color_frame = tk.Frame(self.frame)
  1363. self.show_list_tk = tk.Listbox(self.color_frame)
  1364. self.show_list_scroll = tk.Scrollbar(self.color_frame)
  1365. self.hide_list_tk = tk.Listbox(self.color_frame)
  1366. self.hide_list_scroll = tk.Scrollbar(self.color_frame)
  1367. self.btn_show = tk.Button(self.color_frame)
  1368. self.btn_hide = tk.Button(self.color_frame)
  1369. self.color_show_dict = {}
  1370. self.color_hide_dict = {}
  1371. self.export_btn = tk.Button(self.frame)
  1372. self.refresh_btn = tk.Button(self.frame)
  1373. self.reset_btn = tk.Button(self.frame)
  1374. self.reverse_btn = tk.Button(self.frame)
  1375. self.legend_show = tk.Checkbutton(self.frame), tk.IntVar()
  1376. self._conf("#abc88b")
  1377. self.__conf_font()
  1378. def _conf(self, bg_color):
  1379. self.bg_color = bg_color
  1380. def __conf_font(self, n: int = 1):
  1381. self.btn_font_size = int(14 * n)
  1382. self.little_btn_font_size = int(12 * n)
  1383. def to_program(self):
  1384. self.refresh()
  1385. def update_listbox(self):
  1386. self.show_list_tk.delete(0, tk.END) # 清空列表
  1387. self.hide_list_tk.delete(0, tk.END) # 清空列表
  1388. for i in self.color_show_dict:
  1389. self.show_list_tk.insert(tk.END, i)
  1390. self.show_list_tk.itemconfig(tk.END,
  1391. selectbackground=self.color_show_dict[i],
  1392. bg=self.color_show_dict[i],
  1393. selectforeground='#FFFFFF',
  1394. fg='#000000')
  1395. for i in self.color_hide_dict:
  1396. self.hide_list_tk.insert(tk.END, i)
  1397. self.hide_list_tk.itemconfig(tk.END,
  1398. selectbackground=self.color_hide_dict[i],
  1399. bg=self.color_hide_dict[i],
  1400. selectforeground='#FFFFFF',
  1401. fg='#000000')
  1402. def check_show(self, res: str):
  1403. color = self.color_show_dict.get(res)
  1404. if color is not None:
  1405. return color
  1406. color = self.color_hide_dict.get(res)
  1407. if color is not None:
  1408. return None
  1409. color = random_color()
  1410. self.color_show_dict[res] = color
  1411. return color
  1412. def hide(self):
  1413. i = self.show_list_tk.curselection()
  1414. if len(i) == 0:
  1415. return
  1416. res = self.show_list_tk.get(i[0])
  1417. self.hide_(res)
  1418. self.update_listbox()
  1419. def show(self):
  1420. i = self.hide_list_tk.curselection()
  1421. if len(i) == 0:
  1422. return
  1423. res = self.hide_list_tk.get(i[0])
  1424. self.show_(res)
  1425. self.update_listbox()
  1426. def hide_(self, res):
  1427. color = self.color_show_dict.get(res)
  1428. if color is not None:
  1429. del self.color_show_dict[res]
  1430. self.color_hide_dict[res] = color
  1431. def show_(self, res):
  1432. color = self.color_hide_dict.get(res)
  1433. if color is not None:
  1434. del self.color_hide_dict[res]
  1435. self.color_show_dict[res] = color
  1436. def conf_gui(self, n: int = 1):
  1437. self.__conf_font(n)
  1438. btn_font = make_font(size=self.btn_font_size)
  1439. little_btn_font = make_font(size=self.little_btn_font_size)
  1440. self.color_frame['bg'] = self.bg_color
  1441. self.color_frame['bd'] = 5
  1442. self.color_frame['relief'] = "ridge"
  1443. self.show_list_tk.place(relx=0, rely=0, relwidth=0.90, relheight=0.475)
  1444. self.show_list_scroll.place(relx=0.90, rely=0, relwidth=0.10, relheight=0.475)
  1445. self.show_list_scroll['orient'] = 'vertical'
  1446. self.show_list_scroll['command'] = self.show_list_tk.yview
  1447. self.show_list_tk['yscrollcommand'] = self.show_list_scroll.set
  1448. self.show_list_tk['activestyle'] = tk.NONE
  1449. self.hide_list_tk.place(relx=0, rely=0.525, relwidth=0.90, relheight=0.475)
  1450. self.hide_list_scroll.place(relx=0.90, rely=0.525, relwidth=0.10, relheight=0.475)
  1451. self.hide_list_scroll['orient'] = 'vertical'
  1452. self.hide_list_scroll['command'] = self.hide_list_tk.yview
  1453. self.hide_list_tk['yscrollcommand'] = self.hide_list_scroll.set
  1454. self.hide_list_tk['activestyle'] = tk.NONE
  1455. for btn, text, func, x in zip([self.btn_show, self.btn_hide],
  1456. ["Show", "Hide"],
  1457. [self.show, self.hide],
  1458. [0.00, 0.50]):
  1459. btn['font'] = little_btn_font
  1460. btn['bg'] = Config.tk_btn_bg
  1461. btn['text'] = text
  1462. btn['command'] = func
  1463. btn.place(relx=x, rely=0.475, relwidth=0.50, relheight=0.05)
  1464. self.color_frame.place(relx=0.01, rely=0.02, relwidth=0.18, relheight=0.88)
  1465. self.figure_frame['bg'] = self.bg_color
  1466. self.figure_frame['bd'] = 5
  1467. self.figure_frame['relief'] = "ridge"
  1468. self.figure_frame.place(relx=0.21, rely=0.02, relwidth=0.79, relheight=0.88)
  1469. self.canvas_tk.place(relx=0, rely=0, relwidth=1.0, relheight=0.9)
  1470. self.toolbar.place(relx=0, rely=0.9, relwidth=1.0, relheight=0.1)
  1471. for btn, text, func, x in zip([self.reset_btn, self.reverse_btn, self.refresh_btn, self.export_btn],
  1472. ["复位选择", "反转选择", "刷新数据", "导出数据"],
  1473. [self.reset, self.reverse, self.refresh, self.export],
  1474. [0.37, 0.53, 0.69, 0.85]):
  1475. btn['font'] = btn_font
  1476. btn['bg'] = Config.tk_btn_bg
  1477. btn['text'] = text
  1478. btn['command'] = func
  1479. btn.place(relx=x, rely=0.91, relwidth=0.15, relheight=0.08)
  1480. self.legend_show[0]['font'] = btn_font
  1481. self.legend_show[0]['bg'] = self.color
  1482. self.legend_show[0]['text'] = "显示图例"
  1483. self.legend_show[0]['variable'] = self.legend_show[1]
  1484. self.legend_show[0].place(relx=0.21, rely=0.91, relwidth=0.15, relheight=0.08)
  1485. def export(self):
  1486. ...
  1487. def refresh(self):
  1488. self.plt.cla()
  1489. def reset(self):
  1490. self.color_show_dict.update(self.color_hide_dict)
  1491. self.color_hide_dict = {}
  1492. self.update_listbox()
  1493. def reverse(self):
  1494. tmp = self.color_show_dict
  1495. self.color_show_dict = self.color_hide_dict
  1496. self.color_hide_dict = tmp
  1497. self.update_listbox()
  1498. def show_result(self, res: Dict[str, any]):
  1499. bottom = np.zeros(24)
  1500. label_num = [i for i in range(24)]
  1501. label_str = [f"{i}" for i in range(24)]
  1502. res_type_lst: List = res['res_type']
  1503. for res_type in res_type_lst:
  1504. res_count: Tuple[str] = res[res_type]
  1505. if len(res_count) != 0:
  1506. color = self.check_show(res_type)
  1507. if color is None:
  1508. continue
  1509. y = [0 for _ in range(24)]
  1510. for i in res_count:
  1511. y[int(i[0])] = int(i[1])
  1512. self.color_show_dict[res_type] = color
  1513. self.plt.bar(label_num, y,
  1514. color=color,
  1515. align="center",
  1516. bottom=bottom,
  1517. tick_label=label_str,
  1518. label=res_type)
  1519. bottom += np.array(y)
  1520. if self.legend_show[1].get() == 1:
  1521. self.plt.legend(loc="upper left", prop=self.font)
  1522. self.canvas.draw()
  1523. self.toolbar.update()
  1524. self.update_listbox()
  1525. def set_disable(self):
  1526. self.export_btn['state'] = 'disable'
  1527. def reset_disable(self):
  1528. self.export_btn['state'] = 'normal'
  1529. class StatisticsTimeLocProgram(StatisticsTimeBaseProgram):
  1530. def __init__(self, station, win, color):
  1531. super().__init__(station, win, color, "时段分析-按投放区域")
  1532. self._conf("#abc88b")
  1533. def refresh(self):
  1534. super().refresh()
  1535. event = tk_event.CountThrowTimeEvent(self.station).start(["Location"], lambda i: i[2], self)
  1536. self.station.push_event(event)
  1537. class StatisticsTimeTypeProgram(StatisticsTimeBaseProgram):
  1538. def __init__(self, station, win, color):
  1539. super().__init__(station, win, color, "时段分析-按投放类型")
  1540. self._conf("#abc88b")
  1541. self.color_show_dict[GarbageType.GarbageTypeStrList[1]] = "#00BFFF"
  1542. self.color_show_dict[GarbageType.GarbageTypeStrList[2]] = "#32CD32"
  1543. self.color_show_dict[GarbageType.GarbageTypeStrList[3]] = "#DC143C"
  1544. self.color_show_dict[GarbageType.GarbageTypeStrList[4]] = "#A9A9A9"
  1545. def refresh(self):
  1546. super().refresh()
  1547. event = tk_event.CountThrowTimeEvent(self.station).start(["GarbageType"], self.get_name, self)
  1548. self.station.push_event(event)
  1549. @staticmethod
  1550. def get_name(i: Tuple):
  1551. data: bytes = i[2]
  1552. return GarbageType.GarbageTypeStrList[int(data.decode('utf-8'))]
  1553. class StatisticsTimeTypeLocProgram(StatisticsTimeBaseProgram):
  1554. def __init__(self, station, win, color):
  1555. super().__init__(station, win, color, "时段分析-按投放类型和区域")
  1556. self._conf("#abc88b")
  1557. def refresh(self):
  1558. super().refresh()
  1559. event = tk_event.CountThrowTimeEvent(self.station).start(["GarbageType", "Location"], self.get_name, self)
  1560. self.station.push_event(event)
  1561. @staticmethod
  1562. def get_name(i: Tuple):
  1563. data: bytes = i[2]
  1564. return f"{GarbageType.GarbageTypeStrList[int(data.decode('utf-8'))]}-{i[3]}"
  1565. class StatisticsTimeCheckResultProgram(StatisticsTimeBaseProgram):
  1566. def __init__(self, station, win, color):
  1567. super().__init__(station, win, color, "时段分析-按检查结果")
  1568. self._conf("#abc88b")
  1569. self.color_show_dict['Pass'] = "#00BFFF"
  1570. self.color_show_dict['Fail'] = "#DC143C"
  1571. def refresh(self):
  1572. super().refresh()
  1573. event = tk_event.CountThrowTimeEvent(self.station).start(["CheckResult"], self.get_name, self)
  1574. self.station.push_event(event)
  1575. @staticmethod
  1576. def get_name(i: Tuple):
  1577. data: bytes = i[2]
  1578. return 'Pass' if data == DBBit.BIT_1 else 'Fail'
  1579. class StatisticsTimeCheckResultAndTypeProgram(StatisticsTimeBaseProgram):
  1580. def __init__(self, station, win, color):
  1581. super().__init__(station, win, color, "时段分析-按检查结果和类型")
  1582. self._conf("#abc88b")
  1583. def refresh(self):
  1584. super().refresh()
  1585. event = tk_event.CountThrowTimeEvent(self.station).start(["CheckResult", "GarbageType"], self.get_name, self)
  1586. self.station.push_event(event)
  1587. @staticmethod
  1588. def get_name(i: Tuple):
  1589. data_1: bytes = i[2]
  1590. data_2: bytes = i[3]
  1591. return ((f'Pass' if data_1 == DBBit.BIT_1 else 'Fail') +
  1592. f'-{GarbageType.GarbageTypeStrList[int(data_2.decode("utf-8"))]}')
  1593. class StatisticsTimeCheckResultAndLocProgram(StatisticsTimeBaseProgram):
  1594. def __init__(self, station, win, color):
  1595. super().__init__(station, win, color, "时段分析-按检查结果和区域")
  1596. self._conf("#abc88b")
  1597. def refresh(self):
  1598. super().refresh()
  1599. event = tk_event.CountThrowTimeEvent(self.station).start(["CheckResult", "Location"], self.get_name, self)
  1600. self.station.push_event(event)
  1601. @staticmethod
  1602. def get_name(i: Tuple):
  1603. data_1: bytes = i[2]
  1604. return (f'Pass' if data_1 == DBBit.BIT_1 else 'Fail') + f"-{i[3]}"
  1605. class StatisticsTimeDetailProgram(StatisticsTimeBaseProgram):
  1606. def __init__(self, station, win, color):
  1607. super().__init__(station, win, color, "时段分析-详细分类")
  1608. self._conf("#abc88b")
  1609. def refresh(self):
  1610. super().refresh()
  1611. event = tk_event.CountThrowTimeEvent(self.station)
  1612. event.start(["CheckResult", "GarbageType", "Location"], self.get_name, self)
  1613. self.station.push_event(event)
  1614. @staticmethod
  1615. def get_name(i: Tuple):
  1616. data_1: bytes = i[2]
  1617. data_2: bytes = i[3]
  1618. return ((f'Pass' if data_1 == DBBit.BIT_1 else 'Fail') +
  1619. f'-{GarbageType.GarbageTypeStrList[int(data_2.decode("utf-8"))]}' + f'-{i[4]}')
  1620. all_program = [WelcomeProgram, CreateNormalUserProgram, CreateManagerUserProgram, CreateAutoNormalUserProgram,
  1621. CreateGarbageProgram, DeleteUserProgram, DeleteUsersProgram, DeleteGarbageProgram,
  1622. DeleteGarbageMoreProgram, DeleteAllGarbageProgram, SearchUserProgram, SearchUserAdvancedProgram,
  1623. SearchGarbageProgram, SearchGarbageAdvancedProgram, SearchAdvancedProgram, UpdateUserScoreBase,
  1624. UpdateUserReputationBase, UpdateGarbageTypeProgram, UpdateGarbageCheckResultProgram,
  1625. ExportGarbageProgram, ExportUserProgram, CreateUserFromCSVProgram, AboutProgram,
  1626. StatisticsTimeLocProgram, StatisticsTimeTypeProgram, StatisticsTimeTypeLocProgram,
  1627. StatisticsTimeCheckResultProgram, StatisticsTimeCheckResultAndTypeProgram,
  1628. StatisticsTimeCheckResultAndLocProgram, StatisticsTimeDetailProgram]