ranking.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. import abc
  2. import tkinter as tk
  3. import tkinter.font as font
  4. from PIL import Image, ImageTk
  5. import conf
  6. from tool.type_ import *
  7. from sql.db import DB
  8. class RankingStationException(Exception):
  9. ...
  10. class RankingError(RankingStationException):
  11. ...
  12. class RankingPageError(RankingStationException):
  13. ...
  14. class RankingStationBase(metaclass=abc.ABCMeta):
  15. def __init__(self, db: DB):
  16. self._db = db
  17. self.rank = [[]]
  18. self.rank_index = 0
  19. self.rank_count = 1
  20. self.offset = 0
  21. self.limit_n = 2
  22. self.auto: bool = False
  23. self.auto_to_next: bool = True # auto的移动方向
  24. self.auto_time: int = 5000 # 5s
  25. def get_rank(self, offset: int = 0) -> Tuple[bool, list]:
  26. limit = self.rank_count * self.limit_n
  27. offset = self.offset + limit * offset # offset为0表示不移动, 1表示向前, -1表示向后
  28. cur = self._db.search((f"SELECT uid, name, score, reputation "
  29. f"FROM user "
  30. f"WHERE manager = 0 "
  31. f"ORDER BY reputation DESC, score DESC "
  32. f"LIMIT {limit} OFFSET {offset};"))
  33. if cur is None or cur.rowcount == 0:
  34. return False, []
  35. self.offset = offset
  36. rank_list = list(cur.fetchall())
  37. rank = [[]]
  38. for i, r in enumerate(rank_list):
  39. if len(rank[-1]) == self.rank_count:
  40. rank.append([])
  41. color = None
  42. if self.offset + i == 0:
  43. color = "#eaff56"
  44. elif self.offset + i == 1:
  45. color = "#ffa631"
  46. elif self.offset + i == 2:
  47. color = "#ff7500"
  48. rank[-1].append((self.offset + i + 1, r[1], r[0], r[2], r[3], color))
  49. return True, rank
  50. def rank_page_to_next(self):
  51. if self.rank_index == len(self.rank) - 1:
  52. self.set_next_btn(True) # 当 rank_index为最后一项时, 该函数不应该被调用(除非数据库被外部改动)
  53. return
  54. self.rank_index += 1
  55. if self.rank_index == len(self.rank) - 1: # 最后一项
  56. if len(self.rank[self.rank_index]) == self.rank_count:
  57. res, rank = self.get_rank(1) # 向前移动一个offset
  58. if res:
  59. self.rank = self.rank[self.rank_index], *rank # 调整rank的内容
  60. self.rank_index = 0
  61. else:
  62. self.set_next_btn(True)
  63. else: # 如果最后一个表格没有填满, 直接判定无next
  64. self.set_next_btn(True)
  65. self.show_rank(self.rank[self.rank_index])
  66. self.set_prev_btn(False)
  67. if self.auto:
  68. self.set_run_after_now(self.auto_time, self.update_rank_auto)
  69. def rank_page_to_prev(self):
  70. if self.rank_index == 0: # 当 rank_index为最后一项时, 该函数不应该被调用(除非数据库被外部改动)
  71. self.set_prev_btn(True)
  72. return
  73. self.rank_index -= 1
  74. if self.rank_index == 0: # 回到第一项
  75. res, rank = self.get_rank(-1) # 向后移动一个offset
  76. if res:
  77. self.rank = *rank, self.rank[self.rank_index] # 调整rank的内容
  78. self.rank_index = 0
  79. else:
  80. self.set_prev_btn(True)
  81. self.show_rank(self.rank[self.rank_index])
  82. self.set_next_btn(False)
  83. if self.auto:
  84. self.set_run_after_now(self.auto_time, self.update_rank_auto)
  85. def show_rank_first(self):
  86. self.rank_index = 0
  87. self.offset = 0
  88. self.rank_count = self.rank_count
  89. res, self.rank = self.get_rank(0)
  90. self.show_rank(self.rank[0])
  91. self.set_prev_btn(True)
  92. if len(self.rank) == 1:
  93. self.set_next_btn(True)
  94. else:
  95. self.set_next_btn(False)
  96. def rank_auto(self, auto):
  97. if auto:
  98. self.auto = True
  99. self.disable_btn()
  100. self.set_run_after_now(self.auto_time, self.update_rank_auto) # 注册自动函数
  101. else:
  102. self.auto = False
  103. self.able_btn()
  104. def update_rank_auto(self):
  105. if not self.auto:
  106. return
  107. if (self.auto_to_next and self.is_able_next() or
  108. not self.auto_to_next and not self.is_able_prev() and self.is_able_next()):
  109. self.rank_page_to_next()
  110. self.auto_to_next = True
  111. elif (not self.auto_to_next and self.is_able_prev() or
  112. self.auto_to_next and not self.is_able_next() and self.is_able_prev()):
  113. self.rank_page_to_prev()
  114. self.auto_to_next = False
  115. else:
  116. return # 无法动弹
  117. @abc.abstractmethod
  118. def show_rank(self, rank_info: List[Tuple[int, uname_t, uid_t, score_t, score_t, Optional[str]]]):
  119. ...
  120. @abc.abstractmethod
  121. def set_next_btn(self, disable: False):
  122. ...
  123. @abc.abstractmethod
  124. def set_prev_btn(self, disable: False):
  125. ...
  126. @abc.abstractmethod
  127. def disable_btn(self):
  128. ...
  129. @abc.abstractmethod
  130. def able_btn(self):
  131. ...
  132. @abc.abstractmethod
  133. def is_able_prev(self):
  134. ...
  135. @abc.abstractmethod
  136. def is_able_next(self):
  137. ...
  138. @abc.abstractmethod
  139. def set_run_after_now(self, ms, func, *args):
  140. ...
  141. @abc.abstractmethod
  142. def mainloop(self):
  143. ...
  144. class RankingStation(RankingStationBase):
  145. def __init__(self, db: DB):
  146. self.init_after_run_list: List[Tuple[int, Callable, Tuple]] = []
  147. super(RankingStation, self).__init__(db)
  148. self.window = tk.Tk()
  149. self._sys_height = self.window.winfo_screenheight()
  150. self._sys_width = self.window.winfo_screenwidth()
  151. self.height = int(self._sys_height * (2 / 3))
  152. self.width = int(self._sys_width * (1 / 3))
  153. self._full_screen = False
  154. self.__conf_windows()
  155. self.next_btn: bool = True # 表示开关是否启用
  156. self.prev_btn: bool = True
  157. self.__conf_font_size()
  158. self.__creat_tk()
  159. self.__conf_tk()
  160. self.show_rank_first()
  161. def __creat_tk(self):
  162. self.rank_frame = tk.Frame(self.window)
  163. self.rank_title = tk.Label(self.rank_frame)
  164. self.rank_title_var = tk.StringVar()
  165. self.rank_count = 7
  166. self.rank_label = [tk.Label(self.rank_frame) for _ in range(self.rank_count)]
  167. self.rank_y_height: List[Tuple[float, float]] = [] # rank 标签的y坐标信息
  168. self.rank_var = [tk.StringVar() for _ in range(self.rank_count)]
  169. self.rank_btn = [tk.Button(self.rank_frame) for _ in range(3)] # prev, auto, next
  170. def __conf_font_size(self, n: Union[int, float] = 1):
  171. self._rank_font_title_size = int(24 * n)
  172. self._rank_font_size = int(16 * n)
  173. self._rank_font_btn_size = int(20 * n)
  174. def __conf_tk(self):
  175. self.__conf_windows_bg()
  176. self.__conf_rank()
  177. def __conf_windows(self):
  178. self.window.title('HGSSystem: Ranking')
  179. self.window.geometry(f'{self.width}x{self.height}')
  180. self.window['bg'] = "#F0FFF0" # 蜜瓜绿
  181. self.window.resizable(False, False)
  182. self.bg_img = None
  183. self.bg_lb = tk.Label(self.window)
  184. self.bg_lb['bg'] = "#F0FFF0" # 蜜瓜绿
  185. self.bg_lb.place(relx=0, rely=0, relwidth=1, relheight=1)
  186. self.window.bind("<F11>", lambda _: self.__switch_full_screen())
  187. def __conf_windows_bg(self):
  188. img = Image.open(conf.pic_d['rank_bg']).resize((self.width, self.height))
  189. self.bg_img = ImageTk.PhotoImage(img)
  190. self.bg_lb['image'] = self.bg_img
  191. self.bg_lb.place(relx=0, rely=0, relwidth=1, relheight=1)
  192. def __switch_full_screen(self):
  193. self._full_screen = not self._full_screen
  194. self.window.attributes("-fullscreen", self._full_screen)
  195. width = self._sys_width * (1 / 3)
  196. height = self._sys_height * (2 / 3)
  197. self.width = self.window.winfo_width()
  198. self.height = self.window.winfo_height()
  199. n = min((self.height / height), (self.width / width)) # 因为横和纵不是平均放大, 因此取倍数小的
  200. self.__conf_font_size(n)
  201. self.__conf_tk()
  202. self.show_rank_first()
  203. def __conf_rank(self):
  204. title_font = self.__make_font(size=self._rank_font_title_size, weight="bold")
  205. info_font = self.__make_font(size=self._rank_font_size)
  206. btn_font = self.__make_font(size=self._rank_font_btn_size)
  207. height = self.height * 0.95
  208. width = height * (3 / 4)
  209. # 宽度过大
  210. if width >= self.width:
  211. width = self.width * 0.95
  212. height = width * (4 / 3) / self.width
  213. relwidth = width / self.width
  214. relheight = height / self.height
  215. relx = (1 - relwidth) / 2
  216. rely = (1 - relheight) / 2
  217. self.rank_frame['relief'] = "ridge"
  218. self.rank_frame['bd'] = 5
  219. self.rank_frame['bg'] = "#F5F5DC"
  220. self.rank_frame.place(relx=relx, rely=rely, relwidth=relwidth, relheight=relheight)
  221. self.rank_title['font'] = title_font
  222. self.rank_title['bg'] = "#F5F5DC"
  223. self.rank_title['textvariable'] = self.rank_title_var
  224. self.rank_title.place(relx=0.02, rely=0.00, relwidth=0.96, relheight=0.1)
  225. self.rank_title_var.set("Ranking.loading...")
  226. """
  227. 标签所拥有的总高度为0.82
  228. 标签数为c
  229. 间隔数为c-1
  230. 标签高度为x
  231. 间隔高度为y
  232. 规定 5y = x
  233. 所以: cx + (c - 1)y = 0.82
  234. 得到: 5cx + (c - 1)y = (5c + c - 1)y = 0.82
  235. 因此: x = 0.82 / (5c + c - 1)
  236. """
  237. height_s = (0.82 / (self.rank_count * 6 - 1)) # 标签的高度
  238. height_l = height_s * 5 # 间隔的高度
  239. height = 0.10 # 从 0.10 到 0.82
  240. for i in range(self.rank_count):
  241. self.rank_label[i]['font'] = info_font
  242. self.rank_label[i]['bg'] = "#F5FFFA"
  243. self.rank_label[i]['textvariable'] = self.rank_var[i]
  244. self.rank_label[i]['relief'] = "ridge"
  245. self.rank_label[i]['bd'] = 2
  246. self.rank_var[i].set("")
  247. self.rank_y_height.append((height, height_l))
  248. height += height_l + height_s
  249. for btn, text, x in zip(self.rank_btn,
  250. ("prev", "manual" if self.auto else "auto", "next"), (0.050, 0.375, 0.700)):
  251. btn['font'] = btn_font
  252. btn['text'] = text
  253. btn['bg'] = "#00CED1"
  254. btn.place(relx=x, rely=0.93, relwidth=0.25, relheight=0.06)
  255. self.rank_btn[0]['command'] = lambda: self.rank_page_to_prev()
  256. self.rank_btn[1]['command'] = lambda: self.rank_auto(True)
  257. self.rank_btn[2]['command'] = lambda: self.rank_page_to_next()
  258. def set_run_after_now(self, ms, func, *args):
  259. self.window.after(ms, func, *args)
  260. def __set_rank_info(self, rank_info: List[Tuple[int, uname_t, uid_t, score_t, score_t, Optional[str]]]):
  261. if len(rank_info) > self.rank_count:
  262. rank_info = rank_info[:self.rank_count]
  263. for lb in self.rank_label: # 隐藏全部标签
  264. lb.place_forget()
  265. for i, info in enumerate(rank_info):
  266. no, name, uid, score, eval_, color = info
  267. self.rank_var[i].set(f"NO.{no} {name}\nUID: {uid[0:conf.ranking_tk_show_uid_len]}\n"
  268. f"Score: {score} Reputation: {eval_}")
  269. if color is None:
  270. self.rank_label[i]['bg'] = "#F5FFFA"
  271. else:
  272. self.rank_label[i]['bg'] = color
  273. rely = self.rank_y_height[i][0]
  274. relheight = self.rank_y_height[i][1]
  275. self.rank_label[i].place(relx=0.04, rely=rely, relwidth=0.92, relheight=relheight)
  276. @staticmethod
  277. def __make_font(family: str = 'noto', **kwargs):
  278. return font.Font(family=conf.font_d[family], **kwargs)
  279. def show_rank(self, rank_info: List[Tuple[int, uname_t, uid_t, score_t, score_t, Optional[str]]]):
  280. self.__set_rank_info(rank_info)
  281. self.rank_title_var.set("Ranking")
  282. def rank_auto(self, auto):
  283. super(RankingStation, self).rank_auto(auto)
  284. if auto:
  285. self.rank_btn[1]['command'] = lambda: self.rank_auto(False)
  286. self.rank_btn[1]['text'] = 'manual'
  287. else:
  288. self.rank_btn[1]['command'] = lambda: self.rank_auto(True)
  289. self.rank_btn[1]['text'] = 'auto'
  290. def set_next_btn(self, disable: False):
  291. if disable or self.auto: # auto 模式令btn失效
  292. self.rank_btn[2]['state'] = 'disable'
  293. else:
  294. self.rank_btn[2]['state'] = 'normal'
  295. self.next_btn = not disable
  296. def set_prev_btn(self, disable: False):
  297. if disable or self.auto:
  298. self.rank_btn[0]['state'] = 'disable'
  299. else:
  300. self.rank_btn[0]['state'] = 'normal'
  301. self.prev_btn = not disable
  302. def disable_btn(self):
  303. self.rank_btn[0]['state'] = 'disable'
  304. self.rank_btn[2]['state'] = 'disable'
  305. def able_btn(self):
  306. if self.prev_btn:
  307. self.rank_btn[0]['state'] = 'normal'
  308. if self.next_btn:
  309. self.rank_btn[2]['state'] = 'normal'
  310. def is_able_prev(self):
  311. return self.prev_btn
  312. def is_able_next(self):
  313. return self.next_btn
  314. def mainloop(self):
  315. self.window.mainloop()
  316. if __name__ == '__main__':
  317. mysql_db = DB()
  318. station = RankingStation(mysql_db)
  319. station.mainloop()