meerk40t 0.9.3001__py2.py3-none-any.whl → 0.9.7020__py2.py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (446) hide show
  1. meerk40t/__init__.py +1 -1
  2. meerk40t/balormk/balor_params.py +167 -167
  3. meerk40t/balormk/clone_loader.py +457 -457
  4. meerk40t/balormk/controller.py +1566 -1512
  5. meerk40t/balormk/cylindermod.py +64 -0
  6. meerk40t/balormk/device.py +966 -1959
  7. meerk40t/balormk/driver.py +778 -591
  8. meerk40t/balormk/galvo_commands.py +1194 -0
  9. meerk40t/balormk/gui/balorconfig.py +237 -111
  10. meerk40t/balormk/gui/balorcontroller.py +191 -184
  11. meerk40t/balormk/gui/baloroperationproperties.py +116 -115
  12. meerk40t/balormk/gui/corscene.py +845 -0
  13. meerk40t/balormk/gui/gui.py +179 -147
  14. meerk40t/balormk/livelightjob.py +466 -382
  15. meerk40t/balormk/mock_connection.py +131 -109
  16. meerk40t/balormk/plugin.py +133 -135
  17. meerk40t/balormk/usb_connection.py +306 -301
  18. meerk40t/camera/__init__.py +1 -1
  19. meerk40t/camera/camera.py +514 -397
  20. meerk40t/camera/gui/camerapanel.py +1241 -1095
  21. meerk40t/camera/gui/gui.py +58 -58
  22. meerk40t/camera/plugin.py +441 -399
  23. meerk40t/ch341/__init__.py +27 -27
  24. meerk40t/ch341/ch341device.py +628 -628
  25. meerk40t/ch341/libusb.py +595 -589
  26. meerk40t/ch341/mock.py +171 -171
  27. meerk40t/ch341/windriver.py +157 -157
  28. meerk40t/constants.py +13 -0
  29. meerk40t/core/__init__.py +1 -1
  30. meerk40t/core/bindalias.py +550 -539
  31. meerk40t/core/core.py +47 -47
  32. meerk40t/core/cutcode/cubiccut.py +73 -73
  33. meerk40t/core/cutcode/cutcode.py +315 -312
  34. meerk40t/core/cutcode/cutgroup.py +141 -137
  35. meerk40t/core/cutcode/cutobject.py +192 -185
  36. meerk40t/core/cutcode/dwellcut.py +37 -37
  37. meerk40t/core/cutcode/gotocut.py +29 -29
  38. meerk40t/core/cutcode/homecut.py +29 -29
  39. meerk40t/core/cutcode/inputcut.py +34 -34
  40. meerk40t/core/cutcode/linecut.py +33 -33
  41. meerk40t/core/cutcode/outputcut.py +34 -34
  42. meerk40t/core/cutcode/plotcut.py +335 -335
  43. meerk40t/core/cutcode/quadcut.py +61 -61
  44. meerk40t/core/cutcode/rastercut.py +168 -148
  45. meerk40t/core/cutcode/waitcut.py +34 -34
  46. meerk40t/core/cutplan.py +1843 -1316
  47. meerk40t/core/drivers.py +330 -329
  48. meerk40t/core/elements/align.py +801 -669
  49. meerk40t/core/elements/branches.py +1858 -1507
  50. meerk40t/core/elements/clipboard.py +229 -219
  51. meerk40t/core/elements/element_treeops.py +4595 -2837
  52. meerk40t/core/elements/element_types.py +125 -105
  53. meerk40t/core/elements/elements.py +4315 -3617
  54. meerk40t/core/elements/files.py +117 -64
  55. meerk40t/core/elements/geometry.py +473 -224
  56. meerk40t/core/elements/grid.py +467 -316
  57. meerk40t/core/elements/materials.py +158 -94
  58. meerk40t/core/elements/notes.py +50 -38
  59. meerk40t/core/elements/offset_clpr.py +934 -912
  60. meerk40t/core/elements/offset_mk.py +963 -955
  61. meerk40t/core/elements/penbox.py +339 -267
  62. meerk40t/core/elements/placements.py +300 -83
  63. meerk40t/core/elements/render.py +785 -687
  64. meerk40t/core/elements/shapes.py +2618 -2092
  65. meerk40t/core/elements/testcases.py +105 -0
  66. meerk40t/core/elements/trace.py +651 -563
  67. meerk40t/core/elements/tree_commands.py +415 -409
  68. meerk40t/core/elements/undo_redo.py +116 -58
  69. meerk40t/core/elements/wordlist.py +319 -200
  70. meerk40t/core/exceptions.py +9 -9
  71. meerk40t/core/laserjob.py +220 -220
  72. meerk40t/core/logging.py +63 -63
  73. meerk40t/core/node/blobnode.py +83 -86
  74. meerk40t/core/node/bootstrap.py +105 -103
  75. meerk40t/core/node/branch_elems.py +40 -31
  76. meerk40t/core/node/branch_ops.py +45 -38
  77. meerk40t/core/node/branch_regmark.py +48 -41
  78. meerk40t/core/node/cutnode.py +29 -32
  79. meerk40t/core/node/effect_hatch.py +375 -257
  80. meerk40t/core/node/effect_warp.py +398 -0
  81. meerk40t/core/node/effect_wobble.py +441 -309
  82. meerk40t/core/node/elem_ellipse.py +404 -309
  83. meerk40t/core/node/elem_image.py +1082 -801
  84. meerk40t/core/node/elem_line.py +358 -292
  85. meerk40t/core/node/elem_path.py +259 -201
  86. meerk40t/core/node/elem_point.py +129 -102
  87. meerk40t/core/node/elem_polyline.py +310 -246
  88. meerk40t/core/node/elem_rect.py +376 -286
  89. meerk40t/core/node/elem_text.py +445 -418
  90. meerk40t/core/node/filenode.py +59 -40
  91. meerk40t/core/node/groupnode.py +138 -74
  92. meerk40t/core/node/image_processed.py +777 -766
  93. meerk40t/core/node/image_raster.py +156 -113
  94. meerk40t/core/node/layernode.py +31 -31
  95. meerk40t/core/node/mixins.py +135 -107
  96. meerk40t/core/node/node.py +1427 -1304
  97. meerk40t/core/node/nutils.py +117 -114
  98. meerk40t/core/node/op_cut.py +463 -335
  99. meerk40t/core/node/op_dots.py +296 -251
  100. meerk40t/core/node/op_engrave.py +414 -311
  101. meerk40t/core/node/op_image.py +755 -369
  102. meerk40t/core/node/op_raster.py +787 -522
  103. meerk40t/core/node/place_current.py +37 -40
  104. meerk40t/core/node/place_point.py +329 -126
  105. meerk40t/core/node/refnode.py +58 -47
  106. meerk40t/core/node/rootnode.py +225 -219
  107. meerk40t/core/node/util_console.py +48 -48
  108. meerk40t/core/node/util_goto.py +84 -65
  109. meerk40t/core/node/util_home.py +61 -61
  110. meerk40t/core/node/util_input.py +102 -102
  111. meerk40t/core/node/util_output.py +102 -102
  112. meerk40t/core/node/util_wait.py +65 -65
  113. meerk40t/core/parameters.py +709 -707
  114. meerk40t/core/planner.py +875 -785
  115. meerk40t/core/plotplanner.py +656 -652
  116. meerk40t/core/space.py +120 -113
  117. meerk40t/core/spoolers.py +706 -705
  118. meerk40t/core/svg_io.py +1836 -1549
  119. meerk40t/core/treeop.py +534 -445
  120. meerk40t/core/undos.py +278 -124
  121. meerk40t/core/units.py +784 -680
  122. meerk40t/core/view.py +393 -322
  123. meerk40t/core/webhelp.py +62 -62
  124. meerk40t/core/wordlist.py +513 -504
  125. meerk40t/cylinder/cylinder.py +247 -0
  126. meerk40t/cylinder/gui/cylindersettings.py +41 -0
  127. meerk40t/cylinder/gui/gui.py +24 -0
  128. meerk40t/device/__init__.py +1 -1
  129. meerk40t/device/basedevice.py +322 -123
  130. meerk40t/device/devicechoices.py +50 -0
  131. meerk40t/device/dummydevice.py +163 -128
  132. meerk40t/device/gui/defaultactions.py +618 -602
  133. meerk40t/device/gui/effectspanel.py +114 -0
  134. meerk40t/device/gui/formatterpanel.py +253 -290
  135. meerk40t/device/gui/warningpanel.py +337 -260
  136. meerk40t/device/mixins.py +13 -13
  137. meerk40t/dxf/__init__.py +1 -1
  138. meerk40t/dxf/dxf_io.py +766 -554
  139. meerk40t/dxf/plugin.py +47 -35
  140. meerk40t/external_plugins.py +79 -79
  141. meerk40t/external_plugins_build.py +28 -28
  142. meerk40t/extra/cag.py +112 -116
  143. meerk40t/extra/coolant.py +403 -0
  144. meerk40t/extra/encode_detect.py +204 -0
  145. meerk40t/extra/ezd.py +1165 -1165
  146. meerk40t/extra/hershey.py +834 -340
  147. meerk40t/extra/imageactions.py +322 -316
  148. meerk40t/extra/inkscape.py +628 -622
  149. meerk40t/extra/lbrn.py +424 -424
  150. meerk40t/extra/outerworld.py +283 -0
  151. meerk40t/extra/param_functions.py +1542 -1556
  152. meerk40t/extra/potrace.py +257 -253
  153. meerk40t/extra/serial_exchange.py +118 -0
  154. meerk40t/extra/updater.py +602 -453
  155. meerk40t/extra/vectrace.py +147 -146
  156. meerk40t/extra/winsleep.py +83 -83
  157. meerk40t/extra/xcs_reader.py +597 -0
  158. meerk40t/fill/fills.py +781 -335
  159. meerk40t/fill/patternfill.py +1061 -1061
  160. meerk40t/fill/patterns.py +614 -567
  161. meerk40t/grbl/control.py +87 -87
  162. meerk40t/grbl/controller.py +990 -903
  163. meerk40t/grbl/device.py +1084 -768
  164. meerk40t/grbl/driver.py +989 -771
  165. meerk40t/grbl/emulator.py +532 -497
  166. meerk40t/grbl/gcodejob.py +783 -767
  167. meerk40t/grbl/gui/grblconfiguration.py +373 -298
  168. meerk40t/grbl/gui/grblcontroller.py +485 -271
  169. meerk40t/grbl/gui/grblhardwareconfig.py +269 -153
  170. meerk40t/grbl/gui/grbloperationconfig.py +105 -0
  171. meerk40t/grbl/gui/gui.py +147 -116
  172. meerk40t/grbl/interpreter.py +44 -44
  173. meerk40t/grbl/loader.py +22 -22
  174. meerk40t/grbl/mock_connection.py +56 -56
  175. meerk40t/grbl/plugin.py +294 -264
  176. meerk40t/grbl/serial_connection.py +93 -88
  177. meerk40t/grbl/tcp_connection.py +81 -79
  178. meerk40t/grbl/ws_connection.py +112 -0
  179. meerk40t/gui/__init__.py +1 -1
  180. meerk40t/gui/about.py +2042 -296
  181. meerk40t/gui/alignment.py +1644 -1608
  182. meerk40t/gui/autoexec.py +199 -0
  183. meerk40t/gui/basicops.py +791 -670
  184. meerk40t/gui/bufferview.py +77 -71
  185. meerk40t/gui/busy.py +232 -133
  186. meerk40t/gui/choicepropertypanel.py +1662 -1469
  187. meerk40t/gui/consolepanel.py +706 -542
  188. meerk40t/gui/devicepanel.py +687 -581
  189. meerk40t/gui/dialogoptions.py +110 -107
  190. meerk40t/gui/executejob.py +316 -306
  191. meerk40t/gui/fonts.py +90 -90
  192. meerk40t/gui/functionwrapper.py +252 -0
  193. meerk40t/gui/gui_mixins.py +729 -0
  194. meerk40t/gui/guicolors.py +205 -182
  195. meerk40t/gui/help_assets/help_assets.py +218 -201
  196. meerk40t/gui/helper.py +154 -0
  197. meerk40t/gui/hersheymanager.py +1440 -846
  198. meerk40t/gui/icons.py +3422 -2747
  199. meerk40t/gui/imagesplitter.py +555 -508
  200. meerk40t/gui/keymap.py +354 -344
  201. meerk40t/gui/laserpanel.py +897 -806
  202. meerk40t/gui/laserrender.py +1470 -1232
  203. meerk40t/gui/lasertoolpanel.py +805 -793
  204. meerk40t/gui/magnetoptions.py +436 -0
  205. meerk40t/gui/materialmanager.py +2944 -0
  206. meerk40t/gui/materialtest.py +1722 -1694
  207. meerk40t/gui/mkdebug.py +646 -359
  208. meerk40t/gui/mwindow.py +163 -140
  209. meerk40t/gui/navigationpanels.py +2605 -2467
  210. meerk40t/gui/notes.py +143 -142
  211. meerk40t/gui/opassignment.py +414 -410
  212. meerk40t/gui/operation_info.py +310 -299
  213. meerk40t/gui/plugin.py +500 -328
  214. meerk40t/gui/position.py +714 -669
  215. meerk40t/gui/preferences.py +901 -650
  216. meerk40t/gui/propertypanels/attributes.py +1461 -1131
  217. meerk40t/gui/propertypanels/blobproperty.py +117 -114
  218. meerk40t/gui/propertypanels/consoleproperty.py +83 -80
  219. meerk40t/gui/propertypanels/gotoproperty.py +77 -0
  220. meerk40t/gui/propertypanels/groupproperties.py +223 -217
  221. meerk40t/gui/propertypanels/hatchproperty.py +489 -469
  222. meerk40t/gui/propertypanels/imageproperty.py +2244 -1384
  223. meerk40t/gui/propertypanels/inputproperty.py +59 -58
  224. meerk40t/gui/propertypanels/opbranchproperties.py +82 -80
  225. meerk40t/gui/propertypanels/operationpropertymain.py +1890 -1638
  226. meerk40t/gui/propertypanels/outputproperty.py +59 -58
  227. meerk40t/gui/propertypanels/pathproperty.py +389 -380
  228. meerk40t/gui/propertypanels/placementproperty.py +1214 -383
  229. meerk40t/gui/propertypanels/pointproperty.py +140 -136
  230. meerk40t/gui/propertypanels/propertywindow.py +313 -181
  231. meerk40t/gui/propertypanels/rasterwizardpanels.py +996 -912
  232. meerk40t/gui/propertypanels/regbranchproperties.py +76 -0
  233. meerk40t/gui/propertypanels/textproperty.py +770 -755
  234. meerk40t/gui/propertypanels/waitproperty.py +56 -55
  235. meerk40t/gui/propertypanels/warpproperty.py +121 -0
  236. meerk40t/gui/propertypanels/wobbleproperty.py +255 -204
  237. meerk40t/gui/ribbon.py +2471 -2210
  238. meerk40t/gui/scene/scene.py +1100 -1051
  239. meerk40t/gui/scene/sceneconst.py +22 -22
  240. meerk40t/gui/scene/scenepanel.py +439 -349
  241. meerk40t/gui/scene/scenespacewidget.py +365 -365
  242. meerk40t/gui/scene/widget.py +518 -505
  243. meerk40t/gui/scenewidgets/affinemover.py +215 -215
  244. meerk40t/gui/scenewidgets/attractionwidget.py +315 -309
  245. meerk40t/gui/scenewidgets/bedwidget.py +120 -97
  246. meerk40t/gui/scenewidgets/elementswidget.py +137 -107
  247. meerk40t/gui/scenewidgets/gridwidget.py +785 -745
  248. meerk40t/gui/scenewidgets/guidewidget.py +765 -765
  249. meerk40t/gui/scenewidgets/laserpathwidget.py +66 -66
  250. meerk40t/gui/scenewidgets/machineoriginwidget.py +86 -86
  251. meerk40t/gui/scenewidgets/nodeselector.py +28 -28
  252. meerk40t/gui/scenewidgets/rectselectwidget.py +592 -346
  253. meerk40t/gui/scenewidgets/relocatewidget.py +33 -33
  254. meerk40t/gui/scenewidgets/reticlewidget.py +83 -83
  255. meerk40t/gui/scenewidgets/selectionwidget.py +2958 -2756
  256. meerk40t/gui/simpleui.py +362 -333
  257. meerk40t/gui/simulation.py +2451 -2094
  258. meerk40t/gui/snapoptions.py +208 -203
  259. meerk40t/gui/spoolerpanel.py +1227 -1180
  260. meerk40t/gui/statusbarwidgets/defaultoperations.py +480 -353
  261. meerk40t/gui/statusbarwidgets/infowidget.py +520 -483
  262. meerk40t/gui/statusbarwidgets/opassignwidget.py +356 -355
  263. meerk40t/gui/statusbarwidgets/selectionwidget.py +172 -171
  264. meerk40t/gui/statusbarwidgets/shapepropwidget.py +754 -236
  265. meerk40t/gui/statusbarwidgets/statusbar.py +272 -260
  266. meerk40t/gui/statusbarwidgets/statusbarwidget.py +268 -270
  267. meerk40t/gui/statusbarwidgets/strokewidget.py +267 -251
  268. meerk40t/gui/themes.py +200 -78
  269. meerk40t/gui/tips.py +590 -0
  270. meerk40t/gui/toolwidgets/circlebrush.py +35 -35
  271. meerk40t/gui/toolwidgets/toolcircle.py +248 -242
  272. meerk40t/gui/toolwidgets/toolcontainer.py +82 -77
  273. meerk40t/gui/toolwidgets/tooldraw.py +97 -90
  274. meerk40t/gui/toolwidgets/toolellipse.py +219 -212
  275. meerk40t/gui/toolwidgets/toolimagecut.py +25 -132
  276. meerk40t/gui/toolwidgets/toolline.py +39 -144
  277. meerk40t/gui/toolwidgets/toollinetext.py +79 -236
  278. meerk40t/gui/toolwidgets/toollinetext_inline.py +296 -0
  279. meerk40t/gui/toolwidgets/toolmeasure.py +163 -216
  280. meerk40t/gui/toolwidgets/toolnodeedit.py +2088 -2074
  281. meerk40t/gui/toolwidgets/toolnodemove.py +92 -94
  282. meerk40t/gui/toolwidgets/toolparameter.py +754 -668
  283. meerk40t/gui/toolwidgets/toolplacement.py +108 -108
  284. meerk40t/gui/toolwidgets/toolpoint.py +68 -59
  285. meerk40t/gui/toolwidgets/toolpointlistbuilder.py +294 -0
  286. meerk40t/gui/toolwidgets/toolpointmove.py +183 -0
  287. meerk40t/gui/toolwidgets/toolpolygon.py +288 -403
  288. meerk40t/gui/toolwidgets/toolpolyline.py +38 -196
  289. meerk40t/gui/toolwidgets/toolrect.py +211 -207
  290. meerk40t/gui/toolwidgets/toolrelocate.py +72 -72
  291. meerk40t/gui/toolwidgets/toolribbon.py +598 -113
  292. meerk40t/gui/toolwidgets/tooltabedit.py +546 -0
  293. meerk40t/gui/toolwidgets/tooltext.py +98 -89
  294. meerk40t/gui/toolwidgets/toolvector.py +213 -204
  295. meerk40t/gui/toolwidgets/toolwidget.py +39 -39
  296. meerk40t/gui/usbconnect.py +98 -91
  297. meerk40t/gui/utilitywidgets/buttonwidget.py +18 -18
  298. meerk40t/gui/utilitywidgets/checkboxwidget.py +90 -90
  299. meerk40t/gui/utilitywidgets/controlwidget.py +14 -14
  300. meerk40t/gui/utilitywidgets/cyclocycloidwidget.py +343 -340
  301. meerk40t/gui/utilitywidgets/debugwidgets.py +148 -0
  302. meerk40t/gui/utilitywidgets/handlewidget.py +27 -27
  303. meerk40t/gui/utilitywidgets/harmonograph.py +450 -447
  304. meerk40t/gui/utilitywidgets/openclosewidget.py +40 -40
  305. meerk40t/gui/utilitywidgets/rotationwidget.py +54 -54
  306. meerk40t/gui/utilitywidgets/scalewidget.py +75 -75
  307. meerk40t/gui/utilitywidgets/seekbarwidget.py +183 -183
  308. meerk40t/gui/utilitywidgets/togglewidget.py +142 -142
  309. meerk40t/gui/utilitywidgets/toolbarwidget.py +8 -8
  310. meerk40t/gui/wordlisteditor.py +985 -931
  311. meerk40t/gui/wxmeerk40t.py +1447 -1169
  312. meerk40t/gui/wxmmain.py +5644 -4112
  313. meerk40t/gui/wxmribbon.py +1591 -1076
  314. meerk40t/gui/wxmscene.py +1631 -1453
  315. meerk40t/gui/wxmtree.py +2416 -2089
  316. meerk40t/gui/wxutils.py +1769 -1099
  317. meerk40t/gui/zmatrix.py +102 -102
  318. meerk40t/image/__init__.py +1 -1
  319. meerk40t/image/dither.py +429 -0
  320. meerk40t/image/imagetools.py +2793 -2269
  321. meerk40t/internal_plugins.py +150 -130
  322. meerk40t/kernel/__init__.py +63 -12
  323. meerk40t/kernel/channel.py +259 -212
  324. meerk40t/kernel/context.py +538 -538
  325. meerk40t/kernel/exceptions.py +41 -41
  326. meerk40t/kernel/functions.py +463 -414
  327. meerk40t/kernel/jobs.py +100 -100
  328. meerk40t/kernel/kernel.py +3828 -3571
  329. meerk40t/kernel/lifecycles.py +71 -71
  330. meerk40t/kernel/module.py +49 -49
  331. meerk40t/kernel/service.py +147 -147
  332. meerk40t/kernel/settings.py +383 -343
  333. meerk40t/lihuiyu/controller.py +883 -876
  334. meerk40t/lihuiyu/device.py +1181 -1069
  335. meerk40t/lihuiyu/driver.py +1466 -1372
  336. meerk40t/lihuiyu/gui/gui.py +127 -106
  337. meerk40t/lihuiyu/gui/lhyaccelgui.py +377 -363
  338. meerk40t/lihuiyu/gui/lhycontrollergui.py +741 -651
  339. meerk40t/lihuiyu/gui/lhydrivergui.py +470 -446
  340. meerk40t/lihuiyu/gui/lhyoperationproperties.py +238 -237
  341. meerk40t/lihuiyu/gui/tcpcontroller.py +226 -190
  342. meerk40t/lihuiyu/interpreter.py +53 -53
  343. meerk40t/lihuiyu/laserspeed.py +450 -450
  344. meerk40t/lihuiyu/loader.py +90 -90
  345. meerk40t/lihuiyu/parser.py +404 -404
  346. meerk40t/lihuiyu/plugin.py +101 -102
  347. meerk40t/lihuiyu/tcp_connection.py +111 -109
  348. meerk40t/main.py +231 -165
  349. meerk40t/moshi/builder.py +788 -781
  350. meerk40t/moshi/controller.py +505 -499
  351. meerk40t/moshi/device.py +495 -442
  352. meerk40t/moshi/driver.py +862 -696
  353. meerk40t/moshi/gui/gui.py +78 -76
  354. meerk40t/moshi/gui/moshicontrollergui.py +538 -522
  355. meerk40t/moshi/gui/moshidrivergui.py +87 -75
  356. meerk40t/moshi/plugin.py +43 -43
  357. meerk40t/network/console_server.py +140 -57
  358. meerk40t/network/kernelserver.py +10 -9
  359. meerk40t/network/tcp_server.py +142 -140
  360. meerk40t/network/udp_server.py +103 -77
  361. meerk40t/network/web_server.py +404 -0
  362. meerk40t/newly/controller.py +1158 -1144
  363. meerk40t/newly/device.py +874 -732
  364. meerk40t/newly/driver.py +540 -412
  365. meerk40t/newly/gui/gui.py +219 -188
  366. meerk40t/newly/gui/newlyconfig.py +116 -101
  367. meerk40t/newly/gui/newlycontroller.py +193 -186
  368. meerk40t/newly/gui/operationproperties.py +51 -51
  369. meerk40t/newly/mock_connection.py +82 -82
  370. meerk40t/newly/newly_params.py +56 -56
  371. meerk40t/newly/plugin.py +1214 -1246
  372. meerk40t/newly/usb_connection.py +322 -322
  373. meerk40t/rotary/gui/gui.py +52 -46
  374. meerk40t/rotary/gui/rotarysettings.py +240 -232
  375. meerk40t/rotary/rotary.py +202 -98
  376. meerk40t/ruida/control.py +291 -91
  377. meerk40t/ruida/controller.py +138 -1088
  378. meerk40t/ruida/device.py +676 -231
  379. meerk40t/ruida/driver.py +534 -472
  380. meerk40t/ruida/emulator.py +1494 -1491
  381. meerk40t/ruida/exceptions.py +4 -4
  382. meerk40t/ruida/gui/gui.py +71 -76
  383. meerk40t/ruida/gui/ruidaconfig.py +239 -72
  384. meerk40t/ruida/gui/ruidacontroller.py +187 -184
  385. meerk40t/ruida/gui/ruidaoperationproperties.py +48 -47
  386. meerk40t/ruida/loader.py +54 -52
  387. meerk40t/ruida/mock_connection.py +57 -109
  388. meerk40t/ruida/plugin.py +124 -87
  389. meerk40t/ruida/rdjob.py +2084 -945
  390. meerk40t/ruida/serial_connection.py +116 -0
  391. meerk40t/ruida/tcp_connection.py +146 -0
  392. meerk40t/ruida/udp_connection.py +73 -0
  393. meerk40t/svgelements.py +9671 -9669
  394. meerk40t/tools/driver_to_path.py +584 -579
  395. meerk40t/tools/geomstr.py +5583 -4680
  396. meerk40t/tools/jhfparser.py +357 -292
  397. meerk40t/tools/kerftest.py +904 -890
  398. meerk40t/tools/livinghinges.py +1168 -1033
  399. meerk40t/tools/pathtools.py +987 -949
  400. meerk40t/tools/pmatrix.py +234 -0
  401. meerk40t/tools/pointfinder.py +942 -942
  402. meerk40t/tools/polybool.py +941 -940
  403. meerk40t/tools/rasterplotter.py +1660 -547
  404. meerk40t/tools/shxparser.py +1047 -901
  405. meerk40t/tools/ttfparser.py +726 -446
  406. meerk40t/tools/zinglplotter.py +595 -593
  407. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7020.dist-info}/LICENSE +21 -21
  408. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7020.dist-info}/METADATA +150 -139
  409. meerk40t-0.9.7020.dist-info/RECORD +446 -0
  410. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7020.dist-info}/WHEEL +1 -1
  411. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7020.dist-info}/top_level.txt +0 -1
  412. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7020.dist-info}/zip-safe +1 -1
  413. meerk40t/balormk/elementlightjob.py +0 -159
  414. meerk40t-0.9.3001.dist-info/RECORD +0 -437
  415. test/bootstrap.py +0 -63
  416. test/test_cli.py +0 -12
  417. test/test_core_cutcode.py +0 -418
  418. test/test_core_elements.py +0 -144
  419. test/test_core_plotplanner.py +0 -397
  420. test/test_core_viewports.py +0 -312
  421. test/test_drivers_grbl.py +0 -108
  422. test/test_drivers_lihuiyu.py +0 -443
  423. test/test_drivers_newly.py +0 -113
  424. test/test_element_degenerate_points.py +0 -43
  425. test/test_elements_classify.py +0 -97
  426. test/test_elements_penbox.py +0 -22
  427. test/test_file_svg.py +0 -176
  428. test/test_fill.py +0 -155
  429. test/test_geomstr.py +0 -1523
  430. test/test_geomstr_nodes.py +0 -18
  431. test/test_imagetools_actualize.py +0 -306
  432. test/test_imagetools_wizard.py +0 -258
  433. test/test_kernel.py +0 -200
  434. test/test_laser_speeds.py +0 -3303
  435. test/test_length.py +0 -57
  436. test/test_lifecycle.py +0 -66
  437. test/test_operations.py +0 -251
  438. test/test_operations_hatch.py +0 -57
  439. test/test_ruida.py +0 -19
  440. test/test_spooler.py +0 -22
  441. test/test_tools_rasterplotter.py +0 -29
  442. test/test_wobble.py +0 -133
  443. test/test_zingl.py +0 -124
  444. {test → meerk40t/cylinder}/__init__.py +0 -0
  445. /meerk40t/{core/element_commands.py → cylinder/gui/__init__.py} +0 -0
  446. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7020.dist-info}/entry_points.txt +0 -0
@@ -0,0 +1,1194 @@
1
+ import os
2
+ import re
3
+ import struct
4
+ import time
5
+
6
+ from usb.core import NoBackendError
7
+ from meerk40t.balormk.driver import BalorDriver
8
+ from meerk40t.balormk.livelightjob import LiveLightJob
9
+ from meerk40t.core.laserjob import LaserJob
10
+ from meerk40t.kernel import CommandSyntaxError
11
+ from meerk40t.tools.geomstr import Geomstr
12
+
13
+
14
+ def plugin(service, lifecycle):
15
+ if lifecycle == "service":
16
+ return "provider/device/balor"
17
+ if lifecycle != "added":
18
+ return
19
+
20
+ _ = service._
21
+
22
+ ########################
23
+ # LIGHT JOBS COMMANDS
24
+ ########################
25
+
26
+ @service.console_option(
27
+ "travel_speed", "t", type=float, help="Set the travel speed."
28
+ )
29
+ @service.console_option(
30
+ "jump_delay",
31
+ "d",
32
+ type=float,
33
+ default=200.0,
34
+ help="Sets the jump delay for light travel moves",
35
+ )
36
+ @service.console_option(
37
+ "simulation_speed",
38
+ "m",
39
+ type=float,
40
+ help="sets the simulation speed for this operation",
41
+ )
42
+ @service.console_option(
43
+ "quantization",
44
+ "Q",
45
+ type=int,
46
+ default=500,
47
+ help="Number of line segments to break this path into",
48
+ )
49
+ @service.console_command(
50
+ ("light", "light-simulate"),
51
+ input_type="geometry",
52
+ help=_("runs light on events."),
53
+ )
54
+ def light(
55
+ command,
56
+ channel,
57
+ _,
58
+ travel_speed=None,
59
+ jump_delay=200,
60
+ simulation_speed=None,
61
+ quantization=500,
62
+ data=None,
63
+ **kwgs,
64
+ ):
65
+ """
66
+ Creates a shape based light job for use with the Galvo driver
67
+ """
68
+ if data is None:
69
+ channel("Nothing sent")
70
+ return
71
+ service.job = LiveLightJob(
72
+ service,
73
+ mode="geometry",
74
+ geometry=data,
75
+ travel_speed=travel_speed,
76
+ jump_delay=jump_delay,
77
+ quantization=quantization,
78
+ listen=False,
79
+ )
80
+ if command != "light":
81
+ service.job.set_travel_speed(simulation_speed)
82
+ service.spooler.send(service.job)
83
+
84
+ @service.console_command("select-light", help=_("Execute selection light idle job"))
85
+ def select_light(**kwargs):
86
+ """
87
+ Start a live bounds job.
88
+ """
89
+ # Live Bounds Job.
90
+ if service.job is not None:
91
+ service.job.stop()
92
+ service.job = LiveLightJob(service, mode="bounds")
93
+ service.spooler.send(service.job)
94
+
95
+ @service.console_command("full-light", help=_("Execute full light idle job"))
96
+ def full_light(**kwargs):
97
+ if service.job is not None:
98
+ service.job.stop()
99
+ service.job = LiveLightJob(service)
100
+ service.spooler.send(service.job)
101
+
102
+ # @service.console_command(
103
+ # "regmark-light", help=_("Execute regmark live light idle job")
104
+ # )
105
+ # def reg_light(**kwargs):
106
+ # if service.job is not None:
107
+ # service.job.stop()
108
+ # service.job = LiveLightJob(service, mode="regmarks")
109
+ # service.spooler.send(service.job)
110
+
111
+ @service.console_command("hull-light", help=_("Execute convex hull light idle job"))
112
+ def hull_light(**kwargs):
113
+ if service.job is not None:
114
+ service.job.stop()
115
+ service.job = LiveLightJob(service, mode="hull")
116
+ service.spooler.send(service.job)
117
+
118
+ @service.console_command(
119
+ "stop",
120
+ help=_("stops the idle running job"),
121
+ )
122
+ def stoplight(command, channel, _, data=None, remainder=None, **kwgs):
123
+ if service.job is None:
124
+ channel("No job is currently set")
125
+ return
126
+ channel("Stopping idle job")
127
+ service.job.stop()
128
+
129
+ @service.console_option(
130
+ "count",
131
+ "c",
132
+ default=256,
133
+ type=int,
134
+ help="Number of instances of boxes to draw.",
135
+ )
136
+ @service.console_command(
137
+ "box",
138
+ help=_("outline the current selected elements"),
139
+ output_type="geometry",
140
+ )
141
+ def shapes_selected(
142
+ command, channel, _, count=256, data=None, args=tuple(), **kwargs
143
+ ):
144
+ """
145
+ Draws an outline of the current shape.
146
+ """
147
+ bounds = service.elements.selected_area()
148
+ if bounds is None:
149
+ channel(_("Nothing Selected"))
150
+ return
151
+ xmin, ymin, xmax, ymax = bounds
152
+ channel(_("Element bounds: {bounds}").format(bounds=str(bounds)))
153
+ geometry = Geomstr.rect(xmin, ymin, xmax - xmin, ymax - ymin)
154
+ if count > 1:
155
+ geometry.copies(count)
156
+ return "geometry", geometry
157
+
158
+ @service.console_command(
159
+ "hull",
160
+ help=_("convex hull of the current selected elements"),
161
+ input_type=(None, "elements"),
162
+ output_type="geometry",
163
+ )
164
+ def shapes_hull(channel, _, data=None, **kwargs):
165
+ """
166
+ Draws an outline of the current shape.
167
+ """
168
+ if data is None:
169
+ data = list(service.elements.elems(emphasized=True))
170
+ g = Geomstr()
171
+ for e in data:
172
+ if hasattr(e, "as_image"):
173
+ bounds = e.bounds
174
+ g.append(
175
+ Geomstr.rect(
176
+ bounds[0],
177
+ bounds[1],
178
+ bounds[2] - bounds[0],
179
+ bounds[3] - bounds[1],
180
+ )
181
+ )
182
+ elif e.type == "elem text":
183
+ continue # We can't outline text.
184
+ else:
185
+ g.append(e.as_geometry())
186
+ hull = Geomstr.hull(g)
187
+ if len(hull) == 0:
188
+ channel(_("No elements bounds to trace."))
189
+ return
190
+ return "geometry", hull
191
+
192
+ def ant_points(points, steps):
193
+ points = list(points)
194
+ movement = 1 + int(steps / 10)
195
+ forward_steps = steps + movement
196
+ pos = 0
197
+ size = len(points)
198
+ cycles = int(size / movement) + 1
199
+ for cycle in range(cycles):
200
+ for f in range(pos, pos + forward_steps, 1):
201
+ index = f % size
202
+ point = points[index]
203
+ yield point
204
+ pos += forward_steps
205
+ for f in range(pos, pos - steps, -1):
206
+ index = f % size
207
+ point = points[index]
208
+ yield point
209
+ pos -= steps
210
+
211
+ @service.console_option(
212
+ "quantization",
213
+ "q",
214
+ default=50,
215
+ type=int,
216
+ help="Number of segments to break each path into.",
217
+ )
218
+ @service.console_command(
219
+ "ants",
220
+ help=_("Marching ants of the given element path."),
221
+ input_type=(None, "elements"),
222
+ output_type="geometry",
223
+ )
224
+ def element_ants(command, channel, _, data=None, quantization=50, **kwargs):
225
+ """
226
+ Draws an outline of the current shape.
227
+ """
228
+ if data is None:
229
+ data = list(service.elements.elems(emphasized=True))
230
+ geom = Geomstr()
231
+ for e in data:
232
+ try:
233
+ path = e.as_geometry()
234
+ except AttributeError:
235
+ continue
236
+ ants = list(
237
+ ant_points(
238
+ path.as_equal_interpolated_points(distance=quantization),
239
+ int(quantization / 2),
240
+ )
241
+ )
242
+ geom.polyline(ants)
243
+ geom.end()
244
+ return "geometry", geom
245
+
246
+ ########################
247
+ # LASER CONTROL COMMANDS
248
+ ########################
249
+
250
+ @service.console_command(
251
+ "estop",
252
+ help=_("stops the current job, deletes the spooler"),
253
+ input_type=None,
254
+ )
255
+ def estop(command, channel, _, data=None, remainder=None, **kwgs):
256
+ channel("Stopping Job")
257
+ if service.job is not None:
258
+ service.job.stop()
259
+ service.spooler.clear_queue()
260
+ service.driver.set_abort()
261
+ try:
262
+ channel("Resetting controller.")
263
+ service.driver.reset()
264
+ service.signal("pause")
265
+ except ConnectionRefusedError:
266
+ pass
267
+
268
+ @service.console_command(
269
+ "pause",
270
+ help=_("Pauses the currently running job"),
271
+ )
272
+ def pause(command, channel, _, data=None, remainder=None, **kwgs):
273
+ if service.driver.paused:
274
+ channel("Resuming current job")
275
+ else:
276
+ channel("Pausing current job")
277
+ try:
278
+ service.driver.pause()
279
+ except ConnectionRefusedError:
280
+ channel(_("Could not contact Galvo laser."))
281
+ service.signal("pause")
282
+
283
+ @service.console_command(
284
+ "resume",
285
+ help=_("Resume the currently running job"),
286
+ )
287
+ def resume(command, channel, _, data=None, remainder=None, **kwgs):
288
+ channel("Resume the current job")
289
+ try:
290
+ service.driver.resume()
291
+ except ConnectionRefusedError:
292
+ channel(_("Could not contact Galvo laser."))
293
+ service.signal("pause")
294
+
295
+ @service.console_option(
296
+ "idonotlovemyhouse",
297
+ type=bool,
298
+ action="store_true",
299
+ help=_("override one second laser fire pulse duration"),
300
+ )
301
+ @service.console_argument("time", type=float, help=_("laser fire pulse duration"))
302
+ @service.console_command(
303
+ "pulse",
304
+ help=_("pulse <time>: Pulse the laser in place."),
305
+ )
306
+ def pulse(command, channel, _, time=None, idonotlovemyhouse=False, **kwargs):
307
+ if time is None:
308
+ channel(_("Must specify a pulse time in milliseconds."))
309
+ return
310
+ if time > 1000.0:
311
+ channel(
312
+ _('"{time}ms" exceeds 1 second limit to fire a standing laser.').format(
313
+ time=time
314
+ )
315
+ )
316
+ try:
317
+ if not idonotlovemyhouse:
318
+ return
319
+ except IndexError:
320
+ return
321
+ if service.spooler.is_idle:
322
+ service.spooler.command("pulse", time)
323
+ channel(_("Pulse laser for {time} milliseconds").format(time=time))
324
+ else:
325
+ channel(_("Pulse laser failed: Busy"))
326
+ return
327
+
328
+ ########################
329
+ # USB COMMANDS
330
+ ########################
331
+
332
+ @service.console_command(
333
+ "usb_connect",
334
+ help=_("connect usb"),
335
+ )
336
+ def usb_connect(command, channel, _, data=None, remainder=None, **kwgs):
337
+ service.spooler.command("connect", priority=1)
338
+
339
+ @service.console_command(
340
+ "usb_disconnect",
341
+ help=_("connect usb"),
342
+ )
343
+ def usb_disconnect(command, channel, _, data=None, remainder=None, **kwgs):
344
+ service.spooler.command("disconnect", priority=1)
345
+
346
+ @service.console_command("usb_abort", help=_("Stops USB retries"))
347
+ def usb_abort(command, channel, _, **kwargs):
348
+ service.spooler.command("abort_retry", priority=1)
349
+
350
+ ########################
351
+ # PROJECT IO COMMANDS
352
+ ########################
353
+
354
+ @service.console_argument("filename", type=str)
355
+ @service.console_command("save_job", help=_("save job export"), input_type="plan")
356
+ def galvo_save(channel, _, filename, data=None, **kwargs):
357
+ if filename is None:
358
+ raise CommandSyntaxError
359
+ try:
360
+ with open(filename, "w") as f:
361
+ driver = BalorDriver(service, force_mock=True)
362
+ job = LaserJob(filename, list(data.plan), driver=driver)
363
+ from meerk40t.balormk.controller import list_command_lookup
364
+
365
+ def write(index, cmd):
366
+ cmds = [
367
+ struct.unpack("<6H", cmd[i : i + 12])
368
+ for i in range(0, len(cmd), 12)
369
+ ]
370
+ for v in cmds:
371
+ if v[0] >= 0x8000:
372
+ f.write(
373
+ f"{list_command_lookup.get(v[0], f'{v[0]:04x}').ljust(20)} "
374
+ f"{v[1]:04x} {v[2]:04x} {v[3]:04x} {v[4]:04x} {v[5]:04x}\n"
375
+ )
376
+ if v[0] == 0x8002:
377
+ break
378
+ try:
379
+ driver.connection.connect_if_needed()
380
+ except (ConnectionRefusedError, NoBackendError):
381
+ channel("Could not connect to Galvo")
382
+ return
383
+ driver.connection.connection.write = write
384
+ job.execute()
385
+
386
+ except (PermissionError, OSError):
387
+ channel(_("Could not save: {filename}").format(filename=filename))
388
+
389
+ @service.console_option(
390
+ "default",
391
+ "d",
392
+ help=_("Allow default list commands to persist within the raw command"),
393
+ type=bool,
394
+ action="store_true",
395
+ )
396
+ @service.console_option(
397
+ "raw",
398
+ "r",
399
+ help=_("Data is explicitly little-ended hex from a data capture"),
400
+ type=bool,
401
+ action="store_true",
402
+ )
403
+ @service.console_option(
404
+ "binary_in",
405
+ "b",
406
+ help=_("Read data is explicitly in binary"),
407
+ type=bool,
408
+ action="store_true",
409
+ )
410
+ @service.console_option(
411
+ "binary_out",
412
+ "B",
413
+ help=_("Write data should be explicitly in binary"),
414
+ type=bool,
415
+ action="store_true",
416
+ )
417
+ @service.console_option(
418
+ "short",
419
+ "s",
420
+ help=_("Export data is assumed short command only"),
421
+ type=bool,
422
+ action="store_true",
423
+ )
424
+ @service.console_option(
425
+ "hard",
426
+ "h",
427
+ help=_("Do not send regular list protocol commands"),
428
+ type=bool,
429
+ action="store_true",
430
+ )
431
+ @service.console_option(
432
+ "trim",
433
+ "t",
434
+ help=_("Trim the first number of characters"),
435
+ type=int,
436
+ )
437
+ @service.console_option(
438
+ "input", "i", type=str, default=None, help="input data for given file"
439
+ )
440
+ @service.console_option(
441
+ "output", "o", type=str, default=None, help="output data to given file"
442
+ )
443
+ @service.console_command(
444
+ "raw",
445
+ help=_("sends raw galvo list command exactly as composed"),
446
+ )
447
+ def galvo_raw(
448
+ channel,
449
+ _,
450
+ default=False,
451
+ raw=False,
452
+ binary_in=False,
453
+ binary_out=False,
454
+ short=False,
455
+ hard=False,
456
+ trim=0,
457
+ input=None,
458
+ output=None,
459
+ remainder=None,
460
+ **kwgs,
461
+ ):
462
+ """
463
+ Raw for galvo performs raw actions and sends these commands directly to the laser.
464
+ There are methods for reading and writing raw info from files in order to send that
465
+ data. You can also use shorthand commands.
466
+ """
467
+ from meerk40t.balormk.controller import (
468
+ list_command_lookup,
469
+ single_command_lookup,
470
+ )
471
+
472
+ # Establish reverse lookup for string commands to binary command.
473
+ reverse_lookup = {}
474
+ for k in list_command_lookup:
475
+ command_string = list_command_lookup[k]
476
+ reverse_lookup[command_string] = k
477
+ reverse_lookup[command_string.lower()[4:]] = k
478
+
479
+ for k in single_command_lookup:
480
+ command_string = single_command_lookup[k]
481
+ reverse_lookup[command_string] = k
482
+ reverse_lookup[command_string.lower()] = k
483
+
484
+ if remainder is None and input is None:
485
+ # "raw" was typed without any data or input file, so we list the permitted commands
486
+ channel("Permitted List Commands:")
487
+ for k in list_command_lookup:
488
+ command_string = list_command_lookup[k]
489
+ channel(f"{command_string.lower()[4:]} aka {k:04x}")
490
+ channel("----------------------------")
491
+
492
+ channel("Permitted Short Commands:")
493
+ for k in single_command_lookup:
494
+ command_string = single_command_lookup[k]
495
+ channel(f"{command_string.lower()} aka {k:04x}")
496
+ return
497
+
498
+ if input is not None:
499
+ # We were given an input file. We load that data, in either binary plain text.
500
+ from os.path import exists
501
+
502
+ if exists(input):
503
+ channel(f"Loading data from: {input}")
504
+ try:
505
+ if binary_in:
506
+ with open(input, "br") as f:
507
+ remainder = f.read().hex()
508
+ else:
509
+ with open(input) as f:
510
+ remainder = f.read()
511
+ except OSError:
512
+ channel("File could not be read.")
513
+ else:
514
+ channel(f"The file at {os.path.realpath(input)} does not exist.")
515
+ return
516
+
517
+ cmds = None
518
+ if raw or binary_in:
519
+ # Our data is 6 values int16le
520
+ if trim:
521
+ # Used to cut off raw header data
522
+ remainder = remainder[trim:]
523
+ try:
524
+ cmds = [
525
+ struct.unpack("<6H", bytearray.fromhex(remainder[i : i + 24]))
526
+ for i in range(0, len(remainder), 24)
527
+ ]
528
+ cmds = [
529
+ f"{v[0]:04x} {v[1]:04x} {v[2]:04x} {v[3]:04x} {v[4]:04x} {v[5]:04x}"
530
+ for v in cmds
531
+ ]
532
+ except (struct.error, ValueError) as e:
533
+ channel(f"Data was declared raw but could not parse because '{e}'")
534
+
535
+ if cmds is None:
536
+ cmds = list(re.split(r"[,\n\r]", remainder))
537
+
538
+ raw_commands = list()
539
+
540
+ # Compile commands.
541
+ for cmd_i, cmd in enumerate(cmds):
542
+ cmd = cmd.strip()
543
+ if not cmd:
544
+ continue
545
+
546
+ values = [0] * 6
547
+ byte_i = 0
548
+ split_bytes = [b for b in cmd.split(" ") if b.strip()]
549
+ if len(split_bytes) > 6:
550
+ channel(
551
+ f"Invalid command line {cmd_i}: {split_bytes} has more than six entries."
552
+ )
553
+ return
554
+ for b in split_bytes:
555
+ v = None
556
+ convert = reverse_lookup.get(b)
557
+ if convert is not None:
558
+ v = int(convert)
559
+ else:
560
+ try:
561
+ p = struct.unpack(">H", bytearray.fromhex(b))
562
+ v = p[0]
563
+ except (ValueError, struct.error):
564
+ pass
565
+ if not isinstance(v, int):
566
+ channel(f'Compile error. Line #{cmd_i+1} value "{b}"')
567
+ return
568
+ values[byte_i] = v
569
+ byte_i += 1
570
+ raw_commands.append(values)
571
+
572
+ if output is not None:
573
+ # Output to file
574
+ channel(f"Writing data to: {output}")
575
+ try:
576
+ if binary_out:
577
+ with open(output, "wb") as f:
578
+ for v in raw_commands:
579
+ b_data = struct.pack("<6H", *v)
580
+ f.write(b_data)
581
+ else:
582
+ lines = []
583
+ for v in raw_commands:
584
+ lines.append(
585
+ f"{list_command_lookup.get(v[0], f'{v[0]:04x}').ljust(20)} "
586
+ f"{v[1]:04x} {v[2]:04x} {v[3]:04x} {v[4]:04x} {v[5]:04x}\n"
587
+ )
588
+ with open(output, "w") as f:
589
+ f.writelines(lines)
590
+ except OSError:
591
+ channel("File could not be written.")
592
+ return # If we output to file, we do not output to device.
593
+
594
+ # OUTPUT TO DEVICE
595
+ if hard:
596
+ # Hard raw mode, disable any control values being sent.
597
+ service.driver.connection.raw_mode()
598
+ if not default:
599
+ service.driver.connection.raw_clear()
600
+ for v in raw_commands:
601
+ command = v[0]
602
+ if command >= 0x8000:
603
+ service.driver.connection._list_write(*v)
604
+ else:
605
+ service.driver.connection._list_end()
606
+ service.driver.connection._command(*v)
607
+ return
608
+
609
+ if short:
610
+ # Short mode only sending pure shorts.
611
+ for v in raw_commands:
612
+ service.driver.connection.raw_write(*v)
613
+ return
614
+
615
+ # Hybrid mode. Sending list and short commands using the right mode changes.
616
+ service.driver.connection.rapid_mode()
617
+ service.driver.connection.program_mode()
618
+ if not default:
619
+ service.driver.connection.raw_clear()
620
+ for v in raw_commands:
621
+ command = v[0]
622
+ if command >= 0x8000:
623
+ service.driver.connection.program_mode()
624
+ service.driver.connection._list_write(*v)
625
+ else:
626
+ service.driver.connection.rapid_mode()
627
+ service.driver.connection._command(*v)
628
+ service.driver.connection.rapid_mode()
629
+
630
+ ########################
631
+ # CONTROLLER COMMANDS
632
+ ########################
633
+
634
+ @service.console_argument("x", type=float, default=0.0)
635
+ @service.console_argument("y", type=float, default=0.0)
636
+ @service.console_command(
637
+ "goto",
638
+ help=_("send laser a goto command"),
639
+ )
640
+ def galvo_goto(command, channel, _, x=None, y=None, remainder=None, **kwgs):
641
+ if x is not None and y is not None:
642
+ rx = int(0x8000 + x) & 0xFFFF
643
+ ry = int(0x8000 + y) & 0xFFFF
644
+ service.driver.connection.set_xy(rx, ry)
645
+
646
+ @service.console_option("minspeed", "n", type=int, default=100)
647
+ @service.console_option("maxspeed", "x", type=int, default=5000)
648
+ @service.console_option("acc_time", "a", type=int, default=100)
649
+ @service.console_argument("position", type=int, default=0)
650
+ @service.console_command(
651
+ "rotary_to",
652
+ help=_("Send laser rotary command info."),
653
+ all_arguments_required=True,
654
+ )
655
+ def galvo_rotary(
656
+ command, channel, _, position, minspeed, maxspeed, acc_time, **kwgs
657
+ ):
658
+ service.driver.connection.set_axis_motion_param(
659
+ minspeed & 0xFFFF, maxspeed & 0xFFFF
660
+ )
661
+ service.driver.connection.set_axis_origin_param(acc_time) # Unsure why 100.
662
+ pos = position if position >= 0 else -position + 0x80000000
663
+ p1 = (pos >> 16) & 0xFFFF
664
+ p0 = pos & 0xFFFF
665
+ service.driver.connection.move_axis_to(p0, p1)
666
+ service.driver.connection.wait_axis()
667
+
668
+ @service.console_option("minspeed", "n", type=int, default=100)
669
+ @service.console_option("maxspeed", "x", type=int, default=5000)
670
+ @service.console_option("acc_time", "a", type=int, default=100)
671
+ @service.console_argument(
672
+ "delta_rotary", type=int, default=0, help="relative amount"
673
+ )
674
+ @service.console_command(
675
+ "rotary_relative",
676
+ help=_("Advance the rotary by the given amount"),
677
+ all_arguments_required=True,
678
+ )
679
+ def galvo_rotary_advance(
680
+ command, channel, _, delta_rotary, minspeed, maxspeed, acc_time, **kwgs
681
+ ):
682
+ pos_args = service.driver.connection.get_axis_pos()
683
+ current = pos_args[1] | pos_args[2] << 16
684
+ if current > 0x80000000:
685
+ current = -current + 0x80000000
686
+ position = current + delta_rotary
687
+
688
+ service.driver.connection.set_axis_motion_param(
689
+ minspeed & 0xFFFF, maxspeed & 0xFFFF
690
+ )
691
+ service.driver.connection.set_axis_origin_param(acc_time) # Unsure why 100.
692
+ pos = position if position >= 0 else -position + 0x80000000
693
+ p1 = (pos >> 16) & 0xFFFF
694
+ p0 = pos & 0xFFFF
695
+ service.driver.connection.move_axis_to(p0, p1)
696
+ service.driver.connection.wait_axis()
697
+
698
+ @service.console_option("axis_index", "i", type=int, default=0)
699
+ @service.console_command(
700
+ "rotary_pos",
701
+ help=_("Check the rotary position"),
702
+ )
703
+ def galvo_rotary_pos(command, channel, _, axis_index=0, **kwgs):
704
+ pos_args = service.driver.connection.get_axis_pos(axis_index)
705
+ if pos_args is None:
706
+ channel("Not connected, cannot get axis pos.")
707
+ return
708
+ current = pos_args[1] | pos_args[2] << 16
709
+ if current > 0x80000000:
710
+ current = -current + 0x80000000
711
+ channel(f"Rotary Position: {current}")
712
+
713
+ @service.console_argument("off", type=str)
714
+ @service.console_command(
715
+ "red",
716
+ help=_("Turns redlight on/off"),
717
+ )
718
+ def galvo_on(command, channel, _, off=None, remainder=None, **kwgs):
719
+ try:
720
+ if off == "off":
721
+ service.driver.connection.light_off()
722
+ service.driver.connection.write_port()
723
+ service.redlight_preferred = False
724
+ channel("Turning off redlight.")
725
+ service.signal("red_dot", False)
726
+ else:
727
+ service.driver.connection.light_on()
728
+ service.driver.connection.write_port()
729
+ channel("Turning on redlight.")
730
+ service.redlight_preferred = True
731
+ service.signal("red_dot", True)
732
+ except ConnectionRefusedError:
733
+ service.signal(
734
+ "warning",
735
+ _("Connection was aborted. Manual connection required."),
736
+ _("Not Connected"),
737
+ )
738
+ channel("Could not alter redlight. Connection is aborted.")
739
+
740
+ @service.console_argument(
741
+ "filename", type=str, default=None, help="filename or none"
742
+ )
743
+ @service.console_option(
744
+ "default", "d", type=bool, action="store_true", help="restore to default"
745
+ )
746
+ @service.console_command(
747
+ "force_correction",
748
+ help=_("Resets the galvo laser"),
749
+ )
750
+ def force_correction(
751
+ command, channel, _, filename=None, default=False, remainder=None, **kwgs
752
+ ):
753
+ if default:
754
+ filename = service.corfile
755
+ channel(f"Using default corfile: {filename}")
756
+ if filename is None:
757
+ service.driver.connection.write_correction_file(None)
758
+ channel("Force set corrections to blank.")
759
+ else:
760
+ from os.path import exists
761
+
762
+ if exists(filename):
763
+ channel(f"Force set corrections: {filename}")
764
+ service.driver.connection.write_correction_file(filename)
765
+ else:
766
+ channel(f"The file at {os.path.realpath(filename)} does not exist.")
767
+
768
+ @service.console_command(
769
+ "softreboot",
770
+ help=_("Resets the galvo laser"),
771
+ )
772
+ def galvo_reset(command, channel, _, remainder=None, **kwgs):
773
+ service.driver.connection.init_laser()
774
+ channel(f"Soft reboot: {service.label}")
775
+
776
+ @service.console_option(
777
+ "duration", "d", type=float, help=_("time to set/unset the port")
778
+ )
779
+ @service.console_argument("off", type=str)
780
+ @service.console_argument("bit", type=int)
781
+ @service.console_command(
782
+ "port",
783
+ help=_("Turns port on or off, eg. port off 8"),
784
+ all_arguments_required=True,
785
+ )
786
+ def galvo_port(command, channel, _, off, bit=None, duration=None, **kwgs):
787
+ off = off == "off"
788
+ if off:
789
+ service.driver.connection.port_off(bit)
790
+ service.driver.connection.write_port()
791
+ channel(f"Turning off bit {bit}")
792
+ else:
793
+ service.driver.connection.port_on(bit)
794
+ service.driver.connection.write_port()
795
+ channel(f"Turning on bit {bit}")
796
+ if duration is not None:
797
+ if off:
798
+ service(f".timer 1 {duration} port on {bit}")
799
+ else:
800
+ service(f".timer 1 {duration} port off {bit}")
801
+
802
+ @service.console_command(
803
+ "status",
804
+ help=_("Sends status check"),
805
+ )
806
+ def galvo_status(command, channel, _, remainder=None, **kwgs):
807
+ reply = service.driver.connection.get_version()
808
+ if reply is None:
809
+ channel("Not connected, cannot get serial number.")
810
+ return
811
+ channel(f"Command replied: {reply}")
812
+ for index, b in enumerate(reply):
813
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
814
+
815
+ @service.console_command(
816
+ "lstatus",
817
+ help=_("Checks the list status."),
818
+ )
819
+ def galvo_liststatus(command, channel, _, remainder=None, **kwgs):
820
+ reply = service.driver.connection.get_list_status()
821
+ if reply is None:
822
+ channel("Not connected, cannot get serial number.")
823
+ return
824
+ channel(f"Command replied: {reply}")
825
+ for index, b in enumerate(reply):
826
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
827
+
828
+ @service.console_command(
829
+ "mark_time",
830
+ help=_("Checks the Mark Time."),
831
+ )
832
+ def galvo_mark_time(command, channel, _, remainder=None, **kwgs):
833
+ reply = service.driver.connection.get_mark_time()
834
+ if reply is None:
835
+ channel("Not connected, cannot get mark time.")
836
+ return
837
+ channel(f"Command replied: {reply}")
838
+ for index, b in enumerate(reply):
839
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
840
+
841
+ @service.console_command(
842
+ "mark_count",
843
+ help=_("Checks the Mark Count."),
844
+ )
845
+ def galvo_mark_count(command, channel, _, remainder=None, **kwgs):
846
+ reply = service.driver.connection.get_mark_count()
847
+ if reply is None:
848
+ channel("Not connected, cannot get mark count.")
849
+ return
850
+ channel(f"Command replied: {reply}")
851
+ for index, b in enumerate(reply):
852
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
853
+
854
+ @service.console_command(
855
+ "axis_pos",
856
+ help=_("Checks the Axis Position."),
857
+ )
858
+ def galvo_axis_pos(command, channel, _, remainder=None, **kwgs):
859
+ reply = service.driver.connection.get_axis_pos()
860
+ if reply is None:
861
+ channel("Not connected, cannot get axis position.")
862
+ return
863
+ channel(f"Command replied: {reply}")
864
+ for index, b in enumerate(reply):
865
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
866
+
867
+ @service.console_command(
868
+ "user_data",
869
+ help=_("Checks the User Data."),
870
+ )
871
+ def galvo_user_data(command, channel, _, remainder=None, **kwgs):
872
+ reply = service.driver.connection.get_user_data()
873
+ if reply is None:
874
+ channel("Not connected, cannot get user data.")
875
+ return
876
+ channel(f"Command replied: {reply}")
877
+ for index, b in enumerate(reply):
878
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
879
+
880
+ @service.console_command(
881
+ "position_xy",
882
+ help=_("Checks the Position XY"),
883
+ )
884
+ def galvo_position_xy(command, channel, _, remainder=None, **kwgs):
885
+ reply = service.driver.connection.get_position_xy()
886
+ if reply is None:
887
+ channel("Not connected, cannot get position xy.")
888
+ return
889
+ channel(f"Command replied: {reply}")
890
+ for index, b in enumerate(reply):
891
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
892
+
893
+ @service.console_command(
894
+ "fly_speed",
895
+ help=_("Checks the Fly Speed."),
896
+ )
897
+ def galvo_fly_speed(command, channel, _, remainder=None, **kwgs):
898
+ reply = service.driver.connection.get_fly_speed()
899
+ if reply is None:
900
+ channel("Not connected, cannot get fly speed.")
901
+ return
902
+ channel(f"Command replied: {reply}")
903
+ for index, b in enumerate(reply):
904
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
905
+
906
+ @service.console_command(
907
+ "fly_wait_count",
908
+ help=_("Checks the fiber config extend"),
909
+ )
910
+ def galvo_fly_wait_count(command, channel, _, remainder=None, **kwgs):
911
+ reply = service.driver.connection.get_fly_wait_count()
912
+ if reply is None:
913
+ channel("Not connected, cannot get fly weight count.")
914
+ return
915
+ channel(f"Command replied: {reply}")
916
+ for index, b in enumerate(reply):
917
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
918
+
919
+ @service.console_command(
920
+ "fiber_st_mo_ap",
921
+ help=_("Checks the fiber st mo ap"),
922
+ )
923
+ def galvo_fiber_st_mo_ap(command, channel, _, remainder=None, **kwgs):
924
+ reply = service.driver.connection.get_fiber_st_mo_ap()
925
+ if reply is None:
926
+ channel("Not connected, cannot get fiber_st_mo_ap.")
927
+ return
928
+ channel(f"Command replied: {reply}")
929
+ for index, b in enumerate(reply):
930
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
931
+
932
+ def from_binary(p: str):
933
+ if p.startswith("0b"):
934
+ p = p[2:]
935
+ for c in p:
936
+ if c not in ("0", "1", "x", "X"):
937
+ raise ValueError("Not valid binary")
938
+ return p.lower()
939
+
940
+ @service.console_argument(
941
+ "input",
942
+ help=_("input binary to wait for. Use 'x' for any bit."),
943
+ type=from_binary,
944
+ nargs="*",
945
+ )
946
+ @service.console_option(
947
+ "debug", "d", action="store_true", type=bool, help="debug output"
948
+ )
949
+ @service.console_command("wait_for_input", all_arguments_required=True, hidden=True)
950
+ def wait_for_input(channel, input, debug=False, **kwargs):
951
+ """
952
+ Wait for input is intended as a spooler command. It will halt the calling thread (spooler thread) until the
953
+ matching input is matched. Unimportant bits or bytes can be denoted with `x` for example:
954
+ `wait_for_input x x x 1xxxx` would wait for a 1 on the 5th bit of the 4th word.
955
+
956
+ Omitted values are assumed to be unimportant.
957
+ """
958
+ input_unmatched = True
959
+ while input_unmatched:
960
+ reply = service.driver.connection.read_port()
961
+ input_unmatched = False
962
+ word = 0
963
+ for a, b in zip(reply, input):
964
+ a = bin(a)
965
+ if debug:
966
+ channel(f"input check: {a} match {b} in word #{word}")
967
+ word += 1
968
+ for i in range(-1, -len(a), -1):
969
+ try:
970
+ ac = a[i]
971
+ bc = b[i]
972
+ except IndexError:
973
+ # Assume remaining bits are no-care.
974
+ break
975
+ if bc in "x":
976
+ # This is a no-care bit.
977
+ continue
978
+ if ac != bc:
979
+ if debug:
980
+ channel(f"Fail at {~i} because {ac} != {bc}")
981
+ # We care, and they weren't equal
982
+ time.sleep(0.1)
983
+ input_unmatched = True
984
+ break
985
+ if not input_unmatched:
986
+ if debug:
987
+ channel("Input matched.")
988
+ return # We exited
989
+
990
+ @service.console_command(
991
+ "read_port",
992
+ help=_("Checks the read_port"),
993
+ )
994
+ def galvo_read_port(command, channel, _, remainder=None, **kwgs):
995
+ reply = service.driver.connection.read_port()
996
+ if reply is None:
997
+ channel("Not connected, cannot get read port.")
998
+ return
999
+ channel(f"Command replied: {reply}")
1000
+ for index, b in enumerate(reply):
1001
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1002
+
1003
+ @service.console_command(
1004
+ "input_port",
1005
+ help=_("Checks the input_port"),
1006
+ )
1007
+ def galvo_input_port(command, channel, _, remainder=None, **kwgs):
1008
+ reply = service.driver.connection.get_input_port()
1009
+ if reply is None:
1010
+ channel("Not connected, cannot get input port.")
1011
+ return
1012
+ channel(f"Command replied: {reply}")
1013
+ for index, b in enumerate(reply):
1014
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1015
+
1016
+ @service.console_command(
1017
+ "clear_lock_input_port",
1018
+ help=_("clear the input_port"),
1019
+ )
1020
+ def galvo_clear_input_port(command, channel, _, remainder=None, **kwgs):
1021
+ reply = service.driver.connection.clear_lock_input_port()
1022
+ if reply is None:
1023
+ channel("Not connected, cannot get input port.")
1024
+ return
1025
+ channel(f"Command replied: {reply}")
1026
+ for index, b in enumerate(reply):
1027
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1028
+
1029
+ @service.console_command(
1030
+ "enable_lock_input_port",
1031
+ help=_("clear the input_port"),
1032
+ )
1033
+ def galvo_enable_lock_input_port(command, channel, _, remainder=None, **kwgs):
1034
+ reply = service.driver.connection.enable_lock_input_port()
1035
+ if reply is None:
1036
+ channel("Not connected, cannot get input port.")
1037
+ return
1038
+ channel(f"Command replied: {reply}")
1039
+ for index, b in enumerate(reply):
1040
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1041
+
1042
+ @service.console_command(
1043
+ "disable_lock_input_port",
1044
+ help=_("clear the input_port"),
1045
+ )
1046
+ def galvo_disable_lock_input_port(command, channel, _, remainder=None, **kwgs):
1047
+ reply = service.driver.connection.disable_lock_input_port()
1048
+ if reply is None:
1049
+ channel("Not connected, cannot get input port.")
1050
+ return
1051
+ channel(f"Command replied: {reply}")
1052
+ for index, b in enumerate(reply):
1053
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1054
+
1055
+ @service.console_command(
1056
+ "fiber_config_extend",
1057
+ help=_("Checks the fiber config extend"),
1058
+ )
1059
+ def galvo_fiber_config_extend(command, channel, _, remainder=None, **kwgs):
1060
+ reply = service.driver.connection.get_fiber_config_extend()
1061
+ if reply is None:
1062
+ channel("Not connected, cannot get fiber config extend.")
1063
+ return
1064
+ channel(f"Command replied: {reply}")
1065
+ for index, b in enumerate(reply):
1066
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1067
+
1068
+ @service.console_command(
1069
+ "serial_number",
1070
+ help=_("Checks the serial number."),
1071
+ )
1072
+ def galvo_serial(command, channel, _, remainder=None, **kwgs):
1073
+ reply = service.driver.connection.get_serial_number()
1074
+ if reply is None:
1075
+ channel("Not connected, cannot get serial number.")
1076
+ return
1077
+
1078
+ channel(f"Command replied: {reply}")
1079
+ for index, b in enumerate(reply):
1080
+ channel(f"Bit {index}: 0x{b:04x} 0b{b:016b}")
1081
+
1082
+ @service.console_argument("filename", type=str, default=None)
1083
+ @service.console_command(
1084
+ "correction",
1085
+ help=_("set the correction file"),
1086
+ )
1087
+ def set_corfile(command, channel, _, filename=None, remainder=None, **kwgs):
1088
+ if filename is None:
1089
+ file = service.corfile
1090
+ if file is None:
1091
+ channel("No correction file set.")
1092
+ else:
1093
+ channel(f"Correction file is set to: {service.corfile}")
1094
+ from os.path import exists
1095
+
1096
+ if exists(file):
1097
+ channel("Correction file exists!")
1098
+ else:
1099
+ channel("WARNING: Correction file does not exist.")
1100
+ else:
1101
+ from os.path import exists
1102
+
1103
+ if exists(filename):
1104
+ service.corfile = filename
1105
+ service.signal("corfile", filename)
1106
+ else:
1107
+ channel(f"The file at {os.path.realpath(filename)} does not exist.")
1108
+ channel("Correction file was not set.")
1109
+
1110
+ @service.console_command(
1111
+ "position",
1112
+ help=_("give the position of the selection box in galvos"),
1113
+ )
1114
+ def galvo_pos(command, channel, _, data=None, args=tuple(), **kwargs):
1115
+ """
1116
+ Draws an outline of the current shape.
1117
+ """
1118
+ bounds = service.elements.selected_area()
1119
+ if bounds is None:
1120
+ channel(_("Nothing Selected"))
1121
+ return
1122
+ x0, y0 = service.view.position(bounds[0], bounds[1])
1123
+ x1, y1 = service.view.position(bounds[2], bounds[3])
1124
+ channel(
1125
+ f"Top,Right: ({x0:.02f}, {y0:.02f}). Lower, Left: ({x1:.02f},{y1:.02f})"
1126
+ )
1127
+
1128
+ @service.console_argument("lens_size", type=str, default=None)
1129
+ @service.console_command(
1130
+ "lens",
1131
+ help=_("set the lens size"),
1132
+ )
1133
+ def galvo_lens(
1134
+ command, channel, _, data=None, lens_size=None, args=tuple(), **kwargs
1135
+ ):
1136
+ """
1137
+ Sets lens size.
1138
+ """
1139
+ if lens_size is None:
1140
+ raise SyntaxError
1141
+ service.lens_size = lens_size
1142
+ service.width = lens_size
1143
+ service.height = lens_size
1144
+ service.signal("lens_size", (service.lens_size, service.lens_size))
1145
+ channel(f"Set Bed Size : ({service.lens_size}, {service.lens_size}).")
1146
+
1147
+ @service.console_argument("filename", type=str)
1148
+ @service.console_command(
1149
+ "clone_init",
1150
+ help=_("Initializes a galvo clone board from specified file."),
1151
+ )
1152
+ def codes_update(channel, filename, **kwargs):
1153
+ import platform
1154
+
1155
+ from meerk40t.balormk.clone_loader import load_sys
1156
+
1157
+ kernel = service.kernel
1158
+
1159
+ service.setting(str, "clone_sys", "chunks")
1160
+ if filename is not None:
1161
+ service.clone_sys = filename
1162
+ if service.clone_sys == "chunks":
1163
+ from meerk40t.balormk.clone_loader import load_chunks
1164
+
1165
+ load_chunks(channel=channel)
1166
+ return
1167
+
1168
+ # Check for file in local directory
1169
+ p = service.clone_sys
1170
+ if os.path.exists(p):
1171
+ load_sys(p, channel=channel)
1172
+ return
1173
+
1174
+ # Check for file in the meerk40t directory (safe_path)
1175
+ directory = kernel.os_information["WORKDIR"]
1176
+ p = os.path.join(directory, service.clone_sys)
1177
+ if os.path.exists(p):
1178
+ load_sys(p, channel=channel)
1179
+ return
1180
+
1181
+ if platform.system() != "Windows":
1182
+ return
1183
+
1184
+ # In windows, check the system32/drivers directory.
1185
+ system32 = os.path.join(
1186
+ os.environ["SystemRoot"],
1187
+ "SysNative" if platform.architecture()[0] == "32bit" else "System32",
1188
+ )
1189
+ p = os.path.join(system32, "drivers", service.clone_sys)
1190
+ if os.path.exists(p):
1191
+ load_sys(p, channel=channel)
1192
+ return
1193
+
1194
+ channel(f"{service.clone_sys} file was not found.")