Hello.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722
  1. from multiprocessing import Process, Queue, freeze_support
  2. import threading
  3. from _tkinter import TclError
  4. import tkinter
  5. from tkinter import ttk
  6. import tkinter.font as tkfont
  7. from PIL import ImageTk, Image
  8. import time
  9. import os
  10. import tkinter.messagebox
  11. import webbrowser
  12. import random
  13. from newtkinter import DragWindow, center_windows
  14. img = None
  15. SCREEN = None
  16. queue_screen = None
  17. draftboard_start = None
  18. datascience_start = None
  19. functionmapping_start = None
  20. functionfactory_start = None
  21. algebraicfactory_start = None
  22. machinelearner_start = None
  23. git_start = None
  24. crawlef_start = None
  25. title_color = '#F0FFFF'
  26. button_color = '#FFFFFF'
  27. button_cursor = 'tcross'
  28. pic_list = os.listdir(f'{os.getcwd()}{os.sep}Pic')
  29. del pic_list[pic_list.index('favicon.ico')]
  30. pic_list = random.sample(pic_list, 10)
  31. class QueueController:
  32. def __init__(self):
  33. self.in_dict = {}
  34. self.out_dict = {}
  35. self.var_dict = {}
  36. self.queue_list = []
  37. self.var_from = {}
  38. self.update_var = lambda x, y: None
  39. self.update_queue = lambda x: None
  40. self.run = False
  41. self.stop_str = "__--$$stop_process$$--__"
  42. def can_stop(self):
  43. return len(self.out_dict) == 0
  44. def __call__(self, *args, **kwargs):
  45. self.run = True
  46. def done():
  47. while self.run:
  48. stop_pid = []
  49. old_var = list(self.var_dict.keys())
  50. for out in self.out_dict:
  51. output: Queue = self.out_dict[out]
  52. if output.empty():
  53. continue
  54. dict_index = f'var_{len(self.var_dict)}'
  55. get_out = output.get()
  56. if get_out == self.stop_str:
  57. stop_pid.append(out)
  58. else:
  59. self.var_dict[dict_index] = get_out
  60. self.var_from[dict_index] = out
  61. if old_var != list(self.var_dict.keys()):
  62. self.update_var(self.var_dict, self.var_from)
  63. if stop_pid:
  64. for i in stop_pid:
  65. del self.in_dict[i]
  66. del self.out_dict[i]
  67. self.queue_list = list(self.in_dict.keys())
  68. self.update_queue(self.queue_list.copy())
  69. t = threading.Thread(target=done)
  70. t.setDaemon(True)
  71. t.start()
  72. return self
  73. def stop(self):
  74. self.run = False
  75. def add_queue(self, inqueue, outqueue, name):
  76. self.stop()
  77. self.in_dict[name] = inqueue
  78. self.out_dict[name] = outqueue
  79. self.queue_list = list(self.in_dict.keys())
  80. self.update_queue(self.queue_list.copy())
  81. self.update_var(self.var_dict, self.var_from)
  82. def init(self, update_var, update_queue):
  83. self.update_var = update_var
  84. self.update_queue = update_queue
  85. self.update_queue(list(self.in_dict.keys()))
  86. self.update_var(self.var_dict, self.var_from)
  87. def put(self, value: str, index):
  88. name_space = self.var_dict.copy()
  89. name_space.update(globals())
  90. in_queue = self.in_dict[self.queue_list[index]]
  91. if value.startswith('put_var '):
  92. var_name = value[7:]
  93. in_queue.put(self.var_dict.get(var_name))
  94. elif value.startswith('put_eval '):
  95. in_queue.put(eval(value[8:]), name_space)
  96. elif value.startswith('put_file ') and value.startswith('.py'):
  97. try:
  98. with open(value[4:], 'r') as f:
  99. code_file = f.read()
  100. new_name_space = name_space
  101. exec(code_file, new_name_space)
  102. dict_index = f'var_{len(self.var_dict)}'
  103. in_queue.put(list(new_name_space.keys()))
  104. self.var_dict[dict_index] = new_name_space.copy()
  105. self.var_from[dict_index] = 'self'
  106. except BaseException as e:
  107. in_queue.put(str(e))
  108. else:
  109. in_queue.put(value)
  110. queue_controller = QueueController()
  111. def progress_bar(func):
  112. def make_bar(*agrs, **kwargs):
  113. SCREEN.update()
  114. in_queue: Queue
  115. out_queue: Queue
  116. in_queue, out_queue = func(*agrs, **kwargs)
  117. pid = out_queue.get()
  118. name = func.__name__
  119. queue_controller.add_queue(in_queue, out_queue, f'{name}_{pid}')
  120. progress_screen = tkinter.Toplevel()
  121. progress_screen.title('系统持续加载中...')
  122. progress_screen.geometry("+10+10") # 设置所在位置
  123. progress = ttk.Progressbar(
  124. progress_screen, orient="horizontal", length=300, mode="determinate"
  125. )
  126. progress.pack()
  127. progress_screen.resizable(width=False, height=False)
  128. progress["maximum"] = 10
  129. progress["value"] = 0
  130. i = 0
  131. a = 10
  132. while out_queue.empty():
  133. i += 1
  134. a += 1
  135. try:
  136. progress["value"] = i
  137. progress["maximum"] = a
  138. progress_screen.update()
  139. except TclError:
  140. pass
  141. SCREEN.update()
  142. time.sleep(0.015)
  143. try:
  144. out_queue.get()
  145. t = 0.3
  146. for di in range(10):
  147. t -= 0.03
  148. progress_screen.title(f'检查({round(t,3)})...')
  149. progress["value"] = i + di
  150. progress_screen.update()
  151. time.sleep(0.03)
  152. progress_screen.destroy()
  153. except TclError:
  154. pass
  155. queue_controller()
  156. return make_bar
  157. def draftboard_main(in_queue, out_queue):
  158. out_queue.put(str(os.getpid()))
  159. from draftboard import draw_main
  160. out_queue.put('start')
  161. # 不需要等待
  162. draw_main(in_queue, out_queue)
  163. @progress_bar
  164. def draftboard_run():
  165. in_queue = Queue(10)
  166. out_queue = Queue(10)
  167. Process(target=draftboard_main, args=(in_queue, out_queue)).start()
  168. return in_queue, out_queue
  169. def datascience_main(in_queue, out_queue):
  170. out_queue.put(str(os.getpid()))
  171. from datascience import machine_learning
  172. out_queue.put('start')
  173. time.sleep(0.5)
  174. machine_learning(in_queue, out_queue)
  175. @progress_bar
  176. def datascience_run():
  177. in_queue = Queue(10)
  178. out_queue = Queue(10)
  179. Process(target=datascience_main, args=(in_queue, out_queue)).start()
  180. return in_queue, out_queue
  181. def functionmapping_main(in_queue, out_queue):
  182. out_queue.put(str(os.getpid()))
  183. from funcsystem.map import function_mapping
  184. out_queue.put('start')
  185. time.sleep(0.5)
  186. function_mapping(in_queue, out_queue)
  187. @progress_bar
  188. def functionmapping_run():
  189. in_queue = Queue(10)
  190. out_queue = Queue(10)
  191. Process(target=functionmapping_main, args=(in_queue, out_queue)).start()
  192. return in_queue, out_queue
  193. def functionfactory_main(in_queue, out_queue):
  194. out_queue.put(str(os.getpid()))
  195. from funcsystem.factory import function_factory_main
  196. out_queue.put('start')
  197. time.sleep(0.5)
  198. function_factory_main(in_queue, out_queue)
  199. @progress_bar
  200. def functionfactory_run():
  201. in_queue = Queue(10)
  202. out_queue = Queue(10)
  203. Process(target=functionfactory_main, args=(in_queue, out_queue)).start()
  204. return in_queue, out_queue
  205. def algebraicfactory_main(in_queue, out_queue):
  206. out_queue.put(str(os.getpid()))
  207. from algebraicfactory import algebraic_factory_main
  208. out_queue.put('start')
  209. time.sleep(0.5)
  210. algebraic_factory_main(in_queue, out_queue)
  211. @progress_bar
  212. def algebraicfactory_run():
  213. in_queue = Queue(10)
  214. out_queue = Queue(10)
  215. Process(target=algebraicfactory_main, args=(in_queue, out_queue)).start()
  216. return in_queue, out_queue
  217. def machinelearner_main(in_queue, out_queue):
  218. out_queue.put(str(os.getpid()))
  219. from machinelearning import machine_learning
  220. out_queue.put('start')
  221. time.sleep(0.5)
  222. machine_learning(in_queue, out_queue)
  223. @progress_bar
  224. def machinelearner_run():
  225. in_queue = Queue(10)
  226. out_queue = Queue(10)
  227. Process(target=machinelearner_main, args=(in_queue, out_queue)).start()
  228. return in_queue, out_queue
  229. def git_main(in_queue, out_queue):
  230. out_queue.put(str(os.getpid()))
  231. from gitrepo import git_main
  232. out_queue.put('start')
  233. time.sleep(0.5)
  234. git_main(in_queue, out_queue)
  235. @progress_bar
  236. def git_run():
  237. in_queue = Queue(10)
  238. out_queue = Queue(10)
  239. Process(target=git_main, args=(in_queue, out_queue)).start()
  240. return in_queue, out_queue
  241. def crawler_main(in_queue, out_queue):
  242. out_queue.put(str(os.getpid()))
  243. from crawler import crawler_main
  244. out_queue.put('start')
  245. time.sleep(0.5)
  246. crawler_main(in_queue, out_queue)
  247. @progress_bar
  248. def crawlef_run():
  249. in_queue = Queue(10)
  250. out_queue = Queue(10)
  251. Process(target=crawler_main, args=(in_queue, out_queue)).start()
  252. return in_queue, out_queue
  253. def system_main(in_queue, out_queue):
  254. out_queue.put(str(os.getpid()))
  255. from system.gui import system_main
  256. out_queue.put('start')
  257. time.sleep(0.5)
  258. system_main(in_queue, out_queue)
  259. @progress_bar
  260. def system_run(): # 不需要进度条
  261. in_queue = Queue(10)
  262. out_queue = Queue(10)
  263. Process(target=system_main, args=(in_queue, out_queue)).start()
  264. return in_queue, out_queue
  265. def queuer():
  266. global title_color, button_color, button_cursor, queue_screen
  267. try:
  268. queue_screen.destroy()
  269. except (AttributeError, TclError):
  270. pass
  271. queue_screen = tkinter.Toplevel()
  272. queue_screen.title('通信管理器')
  273. queue_screen.resizable(width=False, height=False)
  274. queue_screen.geometry(f'+30+30')
  275. font = ("黑体", 11) # 设置字体
  276. def sent():
  277. nonlocal sent_text, queue_box
  278. value = sent_text.get()
  279. try:
  280. index = queue_box.curselection()[0]
  281. except IndexError:
  282. return
  283. queue_controller.put(value, index)
  284. width_b = 20
  285. height_b = 2
  286. a_x = 0
  287. a_y = 0
  288. sent_text = tkinter.Entry(queue_screen, width=width_b * 2)
  289. sent_text.grid(column=a_x, row=a_y, columnspan=2, sticky=tkinter.E + tkinter.W + tkinter.S + tkinter.N)
  290. tkinter.Button(queue_screen, bg=button_color, text='发送', command=sent, font=font, width=10, height=height_b)\
  291. .grid(column=a_x+2, row=a_y, sticky=tkinter.E + tkinter.W + tkinter.S + tkinter.N)
  292. a_y += 1
  293. queue_box = tkinter.Listbox(queue_screen, height=height_b * 8)
  294. queue_box.grid(column=a_x, row=a_y, columnspan=3, rowspan=8, sticky=tkinter.E + tkinter.W + tkinter.S + tkinter.N)
  295. a_x += 3
  296. a_y = 0
  297. var_box = tkinter.Listbox(queue_screen, width=width_b * 3, height=height_b * 9)
  298. var_box.grid(column=a_x, row=a_y, columnspan=3, rowspan=9, sticky=tkinter.E + tkinter.W + tkinter.S + tkinter.N)
  299. def update_queue_box(queue_list):
  300. try:
  301. queue_box.delete(0, tkinter.END)
  302. queue_box.insert(0, *queue_list)
  303. except TclError:
  304. pass
  305. def update_var_box(var_dict, var_from):
  306. var = []
  307. for name in var_dict:
  308. var.append(f'{name}[{var_from[name]}] : {var_dict[name]}')
  309. try:
  310. var_box.delete(0, tkinter.END)
  311. var_box.insert(0, *var)
  312. except TclError:
  313. pass
  314. queue_controller.init(update_var_box, update_queue_box)
  315. def to_website():
  316. SCREEN.update()
  317. t = threading.Thread(target=webbrowser.open, args=('https://cotan.songzh.website/',))
  318. t.start()
  319. def close():
  320. global SCREEN
  321. if not queue_controller.can_stop():
  322. tkinter.messagebox.showinfo('操作不被允许', '请先关闭其他模块。')
  323. else:
  324. SCREEN.destroy()
  325. def cotan_main():
  326. global SCREEN, title_color, button_color, button_cursor, img
  327. SCREEN = DragWindow(width=1200, height=800)
  328. font1 = tkfont.Font(family='Comic Sans MS', size=20, weight=tkfont.BOLD)
  329. font2 = tkfont.Font(family='Comic Sans MS', size=16, weight=tkfont.BOLD)
  330. font3 = tkfont.Font(family='Comic Sans MS', size=10)
  331. SCREEN.title('')
  332. SCREEN.resizable(width=False, height=False)
  333. SCREEN.geometry(f'1200x800+30+30')
  334. center_windows(SCREEN, 1200, 800)
  335. # 渲染白色
  336. frame = tkinter.Frame(SCREEN, width=1200, height=800, bg='#FFFFFF')
  337. frame.pack()
  338. # 图片
  339. canvas = tkinter.Canvas(
  340. frame,
  341. bd=0,
  342. width=1000,
  343. height=800,
  344. highlightthickness=0)
  345. pic = pic_list[int(str(time.time()).split()[0][-1])]
  346. bg_image = ImageTk.PhotoImage(Image.open(f'Pic{os.sep}{pic}'))
  347. canvas.create_image(400, 400, image=bg_image)
  348. canvas.grid(column=1, row=0, sticky=tkinter.S + tkinter.N, rowspan=20)
  349. SCREEN.iconbitmap(bitmap=f'Pic{os.sep}favicon.ico', default=f'Pic{os.sep}favicon.ico')
  350. # 标题
  351. tkinter.Label(
  352. frame,
  353. text='CoTan~科学计算系统',
  354. width=20,
  355. bg='#FFFFFF',
  356. font=font1).grid(
  357. column=0,
  358. row=0,
  359. sticky=tkinter.N) # 设置说明
  360. tkinter.Label(
  361. frame,
  362. text='CoTan工具',
  363. bg=title_color,
  364. font=font2).grid(
  365. column=0,
  366. row=1,
  367. sticky=tkinter.W +
  368. tkinter.E)
  369. tkinter.Button(
  370. frame,
  371. text='CoTan草稿板',
  372. cursor=button_cursor,
  373. height=2,
  374. font=font3,
  375. bg=button_color,
  376. command=draftboard_run,
  377. activebackground=title_color,
  378. bd=0,
  379. justify=tkinter.LEFT).grid(
  380. column=0,
  381. row=2,
  382. sticky=tkinter.N +
  383. tkinter.E +
  384. tkinter.W)
  385. tkinter.Button(
  386. frame,
  387. text='自动化网页',
  388. cursor=button_cursor,
  389. command=crawlef_run,
  390. height=2,
  391. font=font3,
  392. bg=button_color,
  393. activebackground=title_color,
  394. bd=0,
  395. justify=tkinter.LEFT).grid(
  396. column=0,
  397. row=3,
  398. sticky=tkinter.N +
  399. tkinter.E +
  400. tkinter.W)
  401. tkinter.Button(
  402. frame,
  403. text='Git仓库控制器',
  404. cursor=button_cursor,
  405. command=git_run,
  406. height=1,
  407. font=font3,
  408. bg=button_color,
  409. activebackground=title_color,
  410. bd=0,
  411. justify=tkinter.LEFT).grid(
  412. column=0,
  413. row=4,
  414. sticky=tkinter.N +
  415. tkinter.E +
  416. tkinter.W)
  417. tkinter.Button(
  418. frame,
  419. text='CoTan社区',
  420. cursor=button_cursor,
  421. command=to_website,
  422. height=1,
  423. font=font3,
  424. bg=button_color,
  425. activebackground=title_color,
  426. bd=0,
  427. justify=tkinter.LEFT).grid(
  428. column=0,
  429. row=5,
  430. sticky=tkinter.N +
  431. tkinter.E +
  432. tkinter.W)
  433. title_color = '#FFFAFA'
  434. tkinter.Label(
  435. frame,
  436. text='数学系统',
  437. bg=title_color,
  438. font=font2).grid(
  439. column=0,
  440. row=6,
  441. sticky=tkinter.W +
  442. tkinter.E)
  443. tkinter.Button(
  444. frame,
  445. text='代数工厂',
  446. cursor=button_cursor,
  447. command=algebraicfactory_run,
  448. height=2,
  449. font=font3,
  450. bg=button_color,
  451. activebackground=title_color,
  452. bd=0,
  453. justify=tkinter.LEFT).grid(
  454. column=0,
  455. row=7,
  456. sticky=tkinter.N +
  457. tkinter.E +
  458. tkinter.W)
  459. tkinter.Button(
  460. frame,
  461. text='机器学习',
  462. cursor=button_cursor,
  463. command=machinelearner_run,
  464. height=2,
  465. font=font3,
  466. bg=button_color,
  467. activebackground=title_color,
  468. bd=0,
  469. justify=tkinter.LEFT).grid(
  470. column=0,
  471. row=8,
  472. sticky=tkinter.N +
  473. tkinter.E +
  474. tkinter.W)
  475. tkinter.Button(
  476. frame,
  477. text='数据科学',
  478. cursor=button_cursor,
  479. command=datascience_run,
  480. height=2,
  481. font=font3,
  482. bg=button_color,
  483. activebackground=title_color,
  484. bd=0,
  485. justify=tkinter.LEFT).grid(
  486. column=0,
  487. row=9,
  488. sticky=tkinter.N +
  489. tkinter.E +
  490. tkinter.W)
  491. tkinter.Button(
  492. frame,
  493. text='函数工厂',
  494. cursor=button_cursor,
  495. command=functionfactory_run,
  496. height=1,
  497. font=font3,
  498. bg=button_color,
  499. activebackground=title_color,
  500. bd=0,
  501. justify=tkinter.LEFT).grid(
  502. column=0,
  503. row=10,
  504. sticky=tkinter.N +
  505. tkinter.E +
  506. tkinter.W)
  507. tkinter.Button(
  508. frame,
  509. text='函数实验室',
  510. cursor=button_cursor,
  511. command=functionmapping_run,
  512. height=1,
  513. font=font3,
  514. bg=button_color,
  515. activebackground=title_color,
  516. bd=0,
  517. justify=tkinter.LEFT).grid(
  518. column=0,
  519. row=11,
  520. sticky=tkinter.N +
  521. tkinter.E +
  522. tkinter.W)
  523. title_color = '#F5FFFA'
  524. tkinter.Label(
  525. frame,
  526. text='物化系统',
  527. bg=title_color,
  528. font=font2).grid(
  529. column=0,
  530. row=12,
  531. sticky=tkinter.W +
  532. tkinter.E)
  533. tkinter.Button(
  534. frame,
  535. text='几何车间',
  536. cursor=button_cursor,
  537. height=2,
  538. font=font3,
  539. bg=button_color,
  540. activebackground=title_color,
  541. bd=0,
  542. justify=tkinter.LEFT).grid(
  543. column=0,
  544. row=13,
  545. sticky=tkinter.N +
  546. tkinter.E +
  547. tkinter.W)
  548. tkinter.Button(
  549. frame,
  550. text='物理车间',
  551. cursor=button_cursor,
  552. height=2,
  553. font=font3,
  554. bg=button_color,
  555. activebackground=title_color,
  556. bd=0,
  557. justify=tkinter.LEFT).grid(
  558. column=0,
  559. row=14,
  560. sticky=tkinter.N +
  561. tkinter.E +
  562. tkinter.W)
  563. tkinter.Button(
  564. frame,
  565. text='化学车间',
  566. cursor=button_cursor,
  567. height=1,
  568. font=font3,
  569. bg=button_color,
  570. activebackground=title_color,
  571. bd=0,
  572. justify=tkinter.LEFT).grid(
  573. column=0,
  574. row=15,
  575. sticky=tkinter.N +
  576. tkinter.E +
  577. tkinter.W)
  578. tkinter.Button(
  579. frame,
  580. text='实验室管理',
  581. cursor=button_cursor,
  582. height=1,
  583. font=font3,
  584. bg=button_color,
  585. activebackground=title_color,
  586. bd=0,
  587. justify=tkinter.LEFT).grid(
  588. column=0,
  589. row=16,
  590. sticky=tkinter.N +
  591. tkinter.E +
  592. tkinter.W)
  593. title_color = '#F8F8FF'
  594. tkinter.Label(
  595. frame,
  596. text='其他工具',
  597. bg=title_color,
  598. font=font2).grid(
  599. column=0,
  600. row=17,
  601. sticky=tkinter.W +
  602. tkinter.E)
  603. tkinter.Button(
  604. frame,
  605. text='系统管理',
  606. cursor=button_cursor,
  607. command=system_run,
  608. height=1,
  609. font=font3,
  610. bg=button_color,
  611. activebackground=title_color,
  612. bd=0,
  613. justify=tkinter.LEFT).grid(
  614. column=0,
  615. row=18,
  616. sticky=tkinter.N +
  617. tkinter.E +
  618. tkinter.W)
  619. tkinter.Button(
  620. frame,
  621. text='通信管理器',
  622. cursor=button_cursor,
  623. height=1,
  624. font=font3,
  625. bg=button_color,
  626. command=queuer,
  627. activebackground=title_color,
  628. bd=0,
  629. justify=tkinter.LEFT).grid(
  630. column=0,
  631. row=19,
  632. sticky=tkinter.N +
  633. tkinter.E +
  634. tkinter.W)
  635. tkinter.Label(
  636. frame,
  637. text='',
  638. bg='#FFFFFF',
  639. font=font2,
  640. height=5).grid(
  641. column=0,
  642. row=20,
  643. sticky=tkinter.W +
  644. tkinter.E)
  645. # canvas.create_text(450, 740, text='Welcome to CoTan', font=font4, fill='#FFFFE0')
  646. SCREEN.protocol("WM_DELETE_WINDOW", close)
  647. SCREEN.mainloop()
  648. if __name__ == "__main__":
  649. freeze_support()
  650. cotan_main()