Hello.py 19 KB

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