Hello.py 19 KB

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