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
meerk40t/core/wordlist.py CHANGED
@@ -1,504 +1,513 @@
1
- """
2
- Base wordlist class that holds some wordlist logic. Most of the interactions with wordlists are done in the
3
- elements service.
4
- """
5
-
6
- import csv
7
- import json
8
- import os
9
- import re
10
- from copy import copy
11
- from datetime import datetime
12
-
13
-
14
- class Wordlist:
15
- """
16
- The Wordlist class provides some logic to hold, update and maintain a set of
17
- variables for text-fields (and later on for other stuff) to allow for
18
- on-the-fly recalculation / repopulation
19
- """
20
-
21
- def __init__(self, versionstr, directory=None):
22
- # The content-dictionary contains an array per entry
23
- # index 0 indicates the type:
24
- # 0 (static) text entry
25
- # 1 text entry array coming from a csv file
26
- # 2 is a numeric counter
27
- # index 1 indicates the position of the current array (always 2 for type 0 and 2)
28
- # index 2 and onwards contain the actual data
29
- self.content = {
30
- "version": [0, 2, versionstr],
31
- "date": [0, 2, self.wordlist_datestr()],
32
- "time": [0, 2, self.wordlist_timestr()],
33
- "op_device": [0, 2, "<device>"],
34
- "op_speed": [0, 2, "<speed>"],
35
- "op_power": [0, 2, "<power>"],
36
- "op_passes": [0, 2, "<passes>"],
37
- "op_dpi": [0, 2, "<dpi>"],
38
- }
39
- self.prohibited = (
40
- "version",
41
- "date",
42
- "time",
43
- "op_device",
44
- "op_speed",
45
- "op_power",
46
- "op_passes",
47
- "op_dpi",
48
- )
49
- self._stack = list()
50
- self.transaction_open = False
51
- self.content_backup = {}
52
- if directory is None:
53
- directory = os.getcwd()
54
- self.default_filename = os.path.join(directory, "wordlist.json")
55
- self.load_data(self.default_filename)
56
-
57
- def add(self, key, value, wtype=None):
58
- self.add_value(key, value, wtype)
59
-
60
- def fetch(self, key):
61
- result = self.fetch_value(key, None)
62
- return result
63
-
64
- def fetch_value(self, skey, idx):
65
- skey = skey.lower()
66
- result = None
67
- try:
68
- wordlist = self.content[skey]
69
- except KeyError:
70
- return None
71
- if skey == "date":
72
- return self.wordlist_datestr(None)
73
- elif skey == "time":
74
- return self.wordlist_timestr(None)
75
- # print (f"Retrieve {wordlist} for {skey}")
76
- if idx is None: # Default
77
- idx = wordlist[1]
78
-
79
- if idx <= len(wordlist):
80
- try:
81
- result = wordlist[idx]
82
- except IndexError:
83
- result = None
84
- return result
85
-
86
- def add_value(self, skey, value, wtype=None):
87
- skey = skey.lower()
88
- if skey not in self.content:
89
- if wtype is None:
90
- wtype = 0
91
- self.content[skey] = [
92
- wtype,
93
- 2,
94
- ] # incomplete, as it will be appended right after this
95
- self.content[skey].append(value)
96
-
97
- def delete_value(self, skey, idx):
98
- skey = skey.lower()
99
- if not skey in self.content:
100
- return
101
- if idx is None or idx < 0:
102
- return
103
-
104
- # Zerobased outside + 2 inside
105
- idx += 2
106
- if idx >= len(self.content[skey]):
107
- return
108
- self.content[skey].pop(idx)
109
-
110
- def move_all_indices(self, delta):
111
- for wkey in self.content:
112
- wordlist = self.content[wkey]
113
- if wkey in self.prohibited:
114
- continue
115
- if wordlist[0] in (0, 1): # Text or csv
116
- last_index = len(wordlist) - 1
117
- # Zero-based outside, +2 inside
118
- newidx = min(wordlist[1] + delta, last_index)
119
- if newidx < 2:
120
- newidx = 2
121
- wordlist[1] = newidx
122
- elif wordlist[0] == 2: # Counter-type
123
- value = wordlist[2]
124
- try:
125
- value = int(value) + delta
126
- except ValueError:
127
- value = 0
128
- if value < 0:
129
- value = 0
130
- wordlist[2] = value
131
-
132
- def set_value(self, skey, value, idx=None, wtype=None):
133
- # Special treatment:
134
- # Index = None - use current
135
- # Index < 0 append
136
- skey = skey.lower()
137
- if not skey in self.content:
138
- # hasn't been there, so establish it
139
- if wtype is None:
140
- wtype = 0
141
- self.content[skey] = [wtype, 2, value]
142
- else:
143
- if idx is None:
144
- # use current position
145
- idx = self.content[skey][1]
146
- try:
147
- idx = int(idx)
148
- except ValueError:
149
- idx = 0
150
- elif idx < 0:
151
- # append
152
- self.content[skey].append(value)
153
- else: # Zerobased outside + 2 inside
154
- idx += 2
155
-
156
- if idx >= len(self.content[skey]):
157
- idx = len(self.content[skey]) - 1
158
- self.content[skey][idx] = value
159
-
160
- def set_index(self, skey, idx, wtype=None):
161
- skey = skey.lower()
162
-
163
- if isinstance(idx, str):
164
- relative = idx.startswith("+") or idx.startswith("-")
165
- index = int(idx)
166
- else:
167
- relative = False
168
- index = idx
169
- wordlists = []
170
-
171
- if skey == "@all": # Set it for all fields from a csv file
172
- wordlists.extend(self.content)
173
- else:
174
- wordlists.extend(list(skey.split(",")))
175
- for wkey in wordlists:
176
- wordlist = self.content[wkey]
177
- if (
178
- wordlist[0] in (0, 1) and wkey not in self.prohibited
179
- ): # Variable Wordlist type.
180
- last_index = len(wordlist) - 1
181
- # Zero-based outside, +2 inside
182
- if relative:
183
- self.content[wkey][1] = min(wordlist[1] + index, last_index)
184
- else:
185
- self.content[wkey][1] = min(index + 2, last_index)
186
-
187
- def reset(self, skey=None):
188
- # Resets position
189
- skey = skey.lower()
190
- if skey is None:
191
- for skey in self.content:
192
- self.content[skey][1] = 2
193
- else:
194
- self.content[skey][1] = 2
195
-
196
- def translate(self, pattern, increment=True):
197
- if pattern is None:
198
- return ""
199
- result = str(pattern)
200
- brackets = re.compile(r"\{[^}]+\}")
201
- for bracketed_key in brackets.findall(result):
202
- # print(f"Key found: {bracketed_key}")
203
- key = bracketed_key[1:-1].lower().strip()
204
- # Let's check whether we have a modifier at the end: #<num>
205
- # if key.endswith("++"):
206
- # autoincrement = True
207
- # key = key[:-2].strip()
208
- # else:
209
- # autoincrement = False
210
- autoincrement = False
211
-
212
- reset = False
213
- relative = 0
214
- pos = key.find("#")
215
- if pos > 0: # Needs to be after first character
216
- # Process offset modification.
217
- index_string = key[pos + 1 :]
218
- key = key[:pos].strip()
219
-
220
- if not index_string.startswith("+") and not index_string.startswith(
221
- "-"
222
- ):
223
- # We have a #<index> value without + or -, specific index value from 0
224
- reset = True
225
- try:
226
- # This covers +x, -x, x
227
- relative = int(index_string)
228
- except ValueError:
229
- relative = 0
230
-
231
- # And now date and time...
232
- if key == "date":
233
- # Do we have a format str?
234
- sformat = None
235
- if key in self.content:
236
- value = self.fetch_value(key, 2)
237
- if value is not None and isinstance(value, str) and len(value) > 0:
238
- if "%" in value:
239
- # Seems to be a format string, so let's try it...
240
- sformat = value
241
- value = self.wordlist_datestr(sformat)
242
- elif key == "time":
243
- # Do we have a format str?
244
- sformat = None
245
- if key in self.content:
246
- value = self.fetch_value(key, 2)
247
- if value is not None and isinstance(value, str) and len(value) > 0:
248
- if "%" in value:
249
- # Seems to be a format string, so let's try it...
250
- sformat = value
251
- value = self.wordlist_timestr(sformat)
252
- elif key.startswith("date@"):
253
- # Original cASEs, vkey is already lowered...
254
- sformat = bracketed_key[6:-1]
255
- value = self.wordlist_datestr(sformat)
256
- elif key.startswith("time@"):
257
- # Original cASEs, vkey is already lowered...
258
- sformat = bracketed_key[6:-1]
259
- value = self.wordlist_timestr(sformat)
260
- else:
261
- # Must be a wordlist type.
262
- if key not in self.content:
263
- # This is not a wordlist name.
264
- continue
265
- wordlist = self.content[key]
266
-
267
- if wordlist[0] == 2: # Counter-type
268
- # Counter index is the value.
269
- value = wordlist[2] if not reset else 0
270
- try:
271
- value = int(value)
272
- except ValueError:
273
- value = 0
274
- value += relative
275
- if autoincrement and increment:
276
- # autoincrement of counter means value + 1
277
- wordlist[2] = value + 1
278
- else:
279
- # This is a variable wordlist.
280
- current_index = wordlist[1] if not reset else 2 # 2 as 2 based
281
- current_index += relative
282
- value = self.fetch_value(key, current_index)
283
- if autoincrement and increment:
284
- # Index set to current index + 1
285
- wordlist[1] = current_index + 1
286
-
287
- if value is not None:
288
- result = result.replace(bracketed_key, str(value))
289
-
290
- return result
291
-
292
- @staticmethod
293
- def wordlist_datestr(format=None):
294
- time = datetime.now()
295
- if format is None:
296
- format = "%x"
297
- try:
298
- result = time.strftime(format)
299
- except:
300
- result = "invalid"
301
- return result
302
-
303
- @staticmethod
304
- def wordlist_timestr(format=None):
305
- time = datetime.now()
306
- if format is None:
307
- format = "%X"
308
- try:
309
- result = time.strftime(format)
310
- except ValueError:
311
- result = "invalid"
312
- return result
313
-
314
- def get_variable_list(self):
315
- choices = []
316
- for skey in self.content:
317
- value = self.fetch(skey)
318
- choices.append(f"{skey} ({value})")
319
- return choices
320
-
321
- def begin_transaction(self):
322
- # We want to store all our values
323
- if not self.transaction_open:
324
- self.content_backup = {}
325
- for key in self.content:
326
- item = copy(self.content[key])
327
- self.content_backup[key] = item
328
- self.transaction_open = True
329
-
330
- def rollback_transaction(self):
331
- if self.transaction_open:
332
- self.content = {}
333
- for key in self.content_backup:
334
- item = copy(self.content_backup[key])
335
- self.content[key] = item
336
- self.transaction_open = False
337
- self.content_backup = {}
338
-
339
- def commit_transaction(self):
340
- if self.transaction_open:
341
- self.transaction_open = False
342
- self.content_backup = {}
343
-
344
- def load_data(self, filename):
345
- if filename is None:
346
- filename = self.default_filename
347
- try:
348
- with open(filename) as f:
349
- self.content = json.load(f)
350
- except (json.JSONDecodeError, PermissionError, OSError, FileNotFoundError):
351
- pass
352
- self.transaction_open = False
353
-
354
- def save_data(self, filename):
355
- if filename is None:
356
- filename = self.default_filename
357
- with open(filename, "w") as f:
358
- json.dump(self.content, f)
359
- self.transaction_open = False
360
-
361
- def delete(self, skey):
362
- try:
363
- self.content.pop(skey)
364
- except KeyError:
365
- pass
366
-
367
- def rename_key(self, oldkey, newkey):
368
- oldkey = oldkey.lower()
369
- newkey = newkey.lower()
370
- if oldkey in self.prohibited:
371
- return False
372
- if oldkey == newkey:
373
- return True
374
- if newkey in self.content:
375
- return False
376
- try:
377
- self.content[newkey] = self.content[oldkey]
378
- self.delete(oldkey)
379
- except:
380
- return False
381
- return True
382
-
383
- def empty_csv(self):
384
- # remove all traces of the previous csv file
385
- names = []
386
- for skey in self.content:
387
- if self.content[skey][0] == 1: # csv
388
- names.append(skey)
389
- for skey in names:
390
- self.delete(skey)
391
-
392
- def load_csv_file(self, filename, force_header=None):
393
- self.empty_csv()
394
- headers = []
395
- try:
396
- with open(filename, newline="") as csvfile:
397
- buffer = csvfile.read(1024)
398
- if force_header is None:
399
- has_header = csv.Sniffer().has_header(buffer)
400
- else:
401
- has_header = force_header
402
- # print (f"Header={has_header}, Force={force_header}")
403
- dialect = csv.Sniffer().sniff(buffer)
404
- csvfile.seek(0)
405
- reader = csv.reader(csvfile, dialect)
406
- headers = next(reader)
407
- if not has_header:
408
- # Use Line as Data and set some default names
409
- for idx, entry in enumerate(headers):
410
- skey = f"Column_{idx + 1}"
411
- self.set_value(skey=skey, value=entry, idx=-1, wtype=1)
412
- headers[idx] = skey.lower()
413
- ct = 1
414
- else:
415
- ct = 0
416
- for row in reader:
417
- for idx, entry in enumerate(row):
418
- skey = headers[idx].lower()
419
- # Append...
420
- self.set_value(skey=skey, value=entry, idx=-1, wtype=1)
421
- ct += 1
422
- except (csv.Error, PermissionError, OSError, FileNotFoundError):
423
- ct = 0
424
- headers = []
425
- colcount = len(headers)
426
- return ct, colcount, headers
427
-
428
- def wordlist_delta(self, orgtext, increase):
429
- newtext = str(orgtext)
430
- toreplace = []
431
- # list of tuples, (index found, old, new )
432
- # Lets gather the {} first...
433
- brackets = re.compile(r"\{[^}]+\}")
434
- for bracketed_key in brackets.findall(str(orgtext)):
435
- # print(f"Key found: {bracketed_key}")
436
- newpattern = ""
437
- key = bracketed_key[1:-1].lower().strip()
438
- relative = 0
439
- pos = key.find("#")
440
- if pos > 0: # Needs to be after first character
441
- # Process offset modification.
442
- index_string = key[pos + 1 :]
443
- key = key[:pos].strip()
444
-
445
- if not index_string.startswith("+") and not index_string.startswith(
446
- "-"
447
- ):
448
- # We have a #<index> value without + or -, specific index value from 0
449
- # no need to do something
450
- continue
451
- try:
452
- # This covers +x, -x, x
453
- relative = int(index_string)
454
- except ValueError:
455
- relative = 0
456
- else:
457
- # it's the unmodified key...
458
- if key.startswith("time@"):
459
- key = "time"
460
- elif key.startswith("date@"):
461
- key = "date"
462
- if key not in self.content:
463
- continue
464
- if key in self.prohibited:
465
- continue
466
- newindex = relative + increase
467
- if newindex > 0:
468
- newpattern = f"{{{key}#+{newindex}}}"
469
- elif newindex < 0:
470
- newpattern = f"{{{key}#{newindex}}}"
471
- else:
472
- # 0
473
- newpattern = f"{{{key}}}"
474
- if newpattern != bracketed_key:
475
- item = [relative, bracketed_key, newpattern]
476
- toreplace.append(item)
477
-
478
- # Then sort the list according to the direction,
479
- # as we don't want to replace the same pattern again and again
480
- if increase >= 0:
481
- toreplace.sort(key=lambda n: n[0])
482
- else:
483
- toreplace.sort(reverse=True, key=lambda n: n[0])
484
- for item in toreplace:
485
- newtext = newtext.replace(item[1], item[2])
486
- return newtext
487
-
488
- def push(self):
489
- """Stores the current content on the stack"""
490
- copied_content = {}
491
- for key, entry in self.content.items():
492
- copied_content[key] = copy(entry)
493
- self._stack.append(copied_content)
494
- # print (f"push was called, when name was: '{self.content['name']}'")
495
-
496
- def pop(self):
497
- """Restores the last added stack entry"""
498
- if len(self._stack) > 0:
499
- copied_content = self._stack[-1]
500
- self._stack.pop(-1)
501
- self.content = {}
502
- for key, entry in copied_content.items():
503
- self.content[key] = copy(entry)
504
- # print (f"pop was called, name now '{self.content['name']}'")
1
+ """
2
+ Base wordlist class that holds some wordlist logic. Most of the interactions with wordlists are done in the
3
+ elements service.
4
+ """
5
+
6
+ import csv
7
+ import json
8
+ import os
9
+ import re
10
+ from copy import copy
11
+ from datetime import datetime
12
+ from ..extra.encode_detect import EncodingDetectFile
13
+
14
+
15
+ class Wordlist:
16
+ """
17
+ The Wordlist class provides some logic to hold, update and maintain a set of
18
+ variables for text-fields (and later on for other stuff) to allow for
19
+ on-the-fly recalculation / repopulation
20
+ """
21
+
22
+ def __init__(self, versionstr, directory=None):
23
+ # The content-dictionary contains an array per entry
24
+ # index 0 indicates the type:
25
+ # 0 (static) text entry
26
+ # 1 text entry array coming from a csv file
27
+ # 2 is a numeric counter
28
+ # index 1 indicates the position of the current array (always 2 for type 0 and 2)
29
+ # index 2 and onwards contain the actual data
30
+ self.content = {
31
+ "version": [0, 2, versionstr],
32
+ "date": [0, 2, self.wordlist_datestr()],
33
+ "time": [0, 2, self.wordlist_timestr()],
34
+ "op_device": [0, 2, "<device>"],
35
+ "op_speed": [0, 2, "<speed>"],
36
+ "op_power": [0, 2, "<power>"],
37
+ "op_passes": [0, 2, "<passes>"],
38
+ "op_dpi": [0, 2, "<dpi>"],
39
+ }
40
+ self.prohibited = (
41
+ "version",
42
+ "date",
43
+ "time",
44
+ "op_device",
45
+ "op_speed",
46
+ "op_power",
47
+ "op_passes",
48
+ "op_dpi",
49
+ )
50
+ self._stack = list()
51
+ self.transaction_open = False
52
+ self.content_backup = {}
53
+ if directory is None:
54
+ directory = os.getcwd()
55
+ self.default_filename = os.path.join(directory, "wordlist.json")
56
+ self.load_data(self.default_filename)
57
+
58
+ def add(self, key, value, wtype=None):
59
+ self.add_value(key, value, wtype)
60
+
61
+ def fetch(self, key):
62
+ result = self.fetch_value(key, None)
63
+ return result
64
+
65
+ def fetch_value(self, skey, idx):
66
+ skey = skey.lower()
67
+ result = None
68
+ try:
69
+ wordlist = self.content[skey]
70
+ except KeyError:
71
+ return None
72
+ if skey == "date":
73
+ return self.wordlist_datestr(None)
74
+ elif skey == "time":
75
+ return self.wordlist_timestr(None)
76
+ # print (f"Retrieve {wordlist} for {skey}")
77
+ if idx is None: # Default
78
+ idx = wordlist[1]
79
+
80
+ if idx <= len(wordlist):
81
+ try:
82
+ result = wordlist[idx]
83
+ except IndexError:
84
+ result = None
85
+ return result
86
+
87
+ def add_value(self, skey, value, wtype=None):
88
+ skey = skey.lower()
89
+ if skey not in self.content:
90
+ if wtype is None:
91
+ wtype = 0
92
+ self.content[skey] = [
93
+ wtype,
94
+ 2,
95
+ ] # incomplete, as it will be appended right after this
96
+ self.content[skey].append(value)
97
+
98
+ def delete_value(self, skey, idx):
99
+ skey = skey.lower()
100
+ if not skey in self.content:
101
+ return
102
+ if idx is None or idx < 0:
103
+ return
104
+
105
+ # Zerobased outside + 2 inside
106
+ idx += 2
107
+ if idx >= len(self.content[skey]):
108
+ return
109
+ self.content[skey].pop(idx)
110
+
111
+ def move_all_indices(self, delta):
112
+ for wkey in self.content:
113
+ wordlist = self.content[wkey]
114
+ if wkey in self.prohibited:
115
+ continue
116
+ if wordlist[0] in (0, 1): # Text or csv
117
+ last_index = len(wordlist) - 1
118
+ # Zero-based outside, +2 inside
119
+ newidx = min(wordlist[1] + delta, last_index)
120
+ if newidx < 2:
121
+ newidx = 2
122
+ wordlist[1] = newidx
123
+ elif wordlist[0] == 2: # Counter-type
124
+ value = wordlist[2]
125
+ try:
126
+ value = int(value) + delta
127
+ except ValueError:
128
+ value = 0
129
+ if value < 0:
130
+ value = 0
131
+ wordlist[2] = value
132
+
133
+ def set_value(self, skey, value, idx=None, wtype=None):
134
+ # Special treatment:
135
+ # Index = None - use current
136
+ # Index < 0 append
137
+ skey = skey.lower()
138
+ if not skey in self.content:
139
+ # hasn't been there, so establish it
140
+ if wtype is None:
141
+ wtype = 0
142
+ self.content[skey] = [wtype, 2, value]
143
+ else:
144
+ if idx is None:
145
+ # use current position
146
+ idx = self.content[skey][1]
147
+ try:
148
+ idx = int(idx)
149
+ except ValueError:
150
+ idx = 0
151
+ elif idx < 0:
152
+ # append
153
+ self.content[skey].append(value)
154
+ else: # Zerobased outside + 2 inside
155
+ idx += 2
156
+
157
+ if idx >= len(self.content[skey]):
158
+ idx = len(self.content[skey]) - 1
159
+ self.content[skey][idx] = value
160
+
161
+ def set_index(self, skey, idx, wtype=None):
162
+ skey = skey.lower()
163
+
164
+ if isinstance(idx, str):
165
+ relative = idx.startswith("+") or idx.startswith("-")
166
+ index = int(idx)
167
+ else:
168
+ relative = False
169
+ index = idx
170
+ wordlists = []
171
+
172
+ if skey == "@all": # Set it for all fields from a csv file
173
+ wordlists.extend(self.content)
174
+ else:
175
+ wordlists.extend(list(skey.split(",")))
176
+ for wkey in wordlists:
177
+ if wkey not in self.content:
178
+ continue
179
+ wordlist = self.content[wkey]
180
+ if (
181
+ wordlist[0] in (0, 1) and wkey not in self.prohibited
182
+ ): # Variable Wordlist type.
183
+ last_index = len(wordlist) - 1
184
+ # Zero-based outside, +2 inside
185
+ if relative:
186
+ self.content[wkey][1] = min(wordlist[1] + index, last_index)
187
+ else:
188
+ self.content[wkey][1] = min(index + 2, last_index)
189
+
190
+ def reset(self, skey=None):
191
+ # Resets position
192
+ skey = skey.lower()
193
+ if skey is None:
194
+ for skey in self.content:
195
+ self.content[skey][1] = 2
196
+ else:
197
+ self.content[skey][1] = 2
198
+
199
+ def translate(self, pattern, increment=True):
200
+ if pattern is None:
201
+ return ""
202
+ result = str(pattern)
203
+ brackets = re.compile(r"\{[^}]+\}")
204
+ for bracketed_key in brackets.findall(result):
205
+ # print(f"Key found: {bracketed_key}")
206
+ key = bracketed_key[1:-1].lower().strip()
207
+ # Let's check whether we have a modifier at the end: #<num>
208
+ # if key.endswith("++"):
209
+ # autoincrement = True
210
+ # key = key[:-2].strip()
211
+ # else:
212
+ # autoincrement = False
213
+ autoincrement = False
214
+
215
+ reset = False
216
+ relative = 0
217
+ pos = key.find("#")
218
+ if pos > 0: # Needs to be after first character
219
+ # Process offset modification.
220
+ index_string = key[pos + 1 :]
221
+ key = key[:pos].strip()
222
+
223
+ if not index_string.startswith("+") and not index_string.startswith(
224
+ "-"
225
+ ):
226
+ # We have a #<index> value without + or -, specific index value from 0
227
+ reset = True
228
+ try:
229
+ # This covers +x, -x, x
230
+ relative = int(index_string)
231
+ except ValueError:
232
+ relative = 0
233
+
234
+ # And now date and time...
235
+ if key == "date":
236
+ # Do we have a format str?
237
+ sformat = None
238
+ if key in self.content:
239
+ value = self.fetch_value(key, 2)
240
+ if value is not None and isinstance(value, str) and len(value) > 0:
241
+ if "%" in value:
242
+ # Seems to be a format string, so let's try it...
243
+ sformat = value
244
+ value = self.wordlist_datestr(sformat)
245
+ elif key == "time":
246
+ # Do we have a format str?
247
+ sformat = None
248
+ if key in self.content:
249
+ value = self.fetch_value(key, 2)
250
+ if value is not None and isinstance(value, str) and len(value) > 0:
251
+ if "%" in value:
252
+ # Seems to be a format string, so let's try it...
253
+ sformat = value
254
+ value = self.wordlist_timestr(sformat)
255
+ elif key.startswith("date@"):
256
+ # Original cASEs, vkey is already lowered...
257
+ sformat = bracketed_key[6:-1]
258
+ value = self.wordlist_datestr(sformat)
259
+ elif key.startswith("time@"):
260
+ # Original cASEs, vkey is already lowered...
261
+ sformat = bracketed_key[6:-1]
262
+ value = self.wordlist_timestr(sformat)
263
+ else:
264
+ # Must be a wordlist type.
265
+ if key not in self.content:
266
+ # This is not a wordlist name.
267
+ continue
268
+ wordlist = self.content[key]
269
+
270
+ if wordlist[0] == 2: # Counter-type
271
+ # Counter index is the value.
272
+ value = wordlist[2] if not reset else 0
273
+ try:
274
+ value = int(value)
275
+ except ValueError:
276
+ value = 0
277
+ value += relative
278
+ if autoincrement and increment:
279
+ # autoincrement of counter means value + 1
280
+ wordlist[2] = value + 1
281
+ else:
282
+ # This is a variable wordlist.
283
+ current_index = wordlist[1] if not reset else 2 # 2 as 2 based
284
+ current_index += relative
285
+ value = self.fetch_value(key, current_index)
286
+ if autoincrement and increment:
287
+ # Index set to current index + 1
288
+ wordlist[1] = current_index + 1
289
+
290
+ if value is not None:
291
+ result = result.replace(bracketed_key, str(value))
292
+
293
+ return result
294
+
295
+ @staticmethod
296
+ def wordlist_datestr(format=None):
297
+ time = datetime.now()
298
+ if format is None:
299
+ format = "%x"
300
+ try:
301
+ result = time.strftime(format)
302
+ except:
303
+ result = "invalid"
304
+ return result
305
+
306
+ @staticmethod
307
+ def wordlist_timestr(format=None):
308
+ time = datetime.now()
309
+ if format is None:
310
+ format = "%X"
311
+ try:
312
+ result = time.strftime(format)
313
+ except ValueError:
314
+ result = "invalid"
315
+ return result
316
+
317
+ def get_variable_list(self):
318
+ choices = []
319
+ for skey in self.content:
320
+ value = self.fetch(skey)
321
+ choices.append(f"{skey} ({value})")
322
+ return choices
323
+
324
+ def begin_transaction(self):
325
+ # We want to store all our values
326
+ if not self.transaction_open:
327
+ self.content_backup = {}
328
+ for key in self.content:
329
+ item = copy(self.content[key])
330
+ self.content_backup[key] = item
331
+ self.transaction_open = True
332
+
333
+ def rollback_transaction(self):
334
+ if self.transaction_open:
335
+ self.content = {}
336
+ for key in self.content_backup:
337
+ item = copy(self.content_backup[key])
338
+ self.content[key] = item
339
+ self.transaction_open = False
340
+ self.content_backup = {}
341
+
342
+ def commit_transaction(self):
343
+ if self.transaction_open:
344
+ self.transaction_open = False
345
+ self.content_backup = {}
346
+
347
+ def load_data(self, filename):
348
+ if filename is None:
349
+ filename = self.default_filename
350
+ try:
351
+ with open(filename) as f:
352
+ self.content = json.load(f)
353
+ except (json.JSONDecodeError, PermissionError, OSError, FileNotFoundError):
354
+ pass
355
+ self.transaction_open = False
356
+
357
+ def save_data(self, filename):
358
+ if filename is None:
359
+ filename = self.default_filename
360
+ with open(filename, "w") as f:
361
+ json.dump(self.content, f)
362
+ self.transaction_open = False
363
+
364
+ def delete(self, skey):
365
+ try:
366
+ self.content.pop(skey)
367
+ except KeyError:
368
+ pass
369
+
370
+ def rename_key(self, oldkey, newkey):
371
+ oldkey = oldkey.lower()
372
+ newkey = newkey.lower()
373
+ if oldkey in self.prohibited:
374
+ return False
375
+ if oldkey == newkey:
376
+ return True
377
+ if newkey in self.content:
378
+ return False
379
+ try:
380
+ self.content[newkey] = self.content[oldkey]
381
+ self.delete(oldkey)
382
+ except:
383
+ return False
384
+ return True
385
+
386
+ def empty_csv(self):
387
+ # remove all traces of the previous csv file
388
+ names = []
389
+ for skey in self.content:
390
+ if self.content[skey][0] == 1: # csv
391
+ names.append(skey)
392
+ for skey in names:
393
+ self.delete(skey)
394
+
395
+ def load_csv_file(self, filename, force_header=None):
396
+ self.empty_csv()
397
+ headers = []
398
+ decoder = EncodingDetectFile()
399
+ result = decoder.load(filename)
400
+ if result:
401
+ encoding, bom_marker, file_content = result
402
+
403
+ try:
404
+ with open(filename, newline="", encoding=encoding) as csvfile:
405
+ buffer = csvfile.read(1024)
406
+ if force_header is None:
407
+ has_header = csv.Sniffer().has_header(buffer)
408
+ else:
409
+ has_header = force_header
410
+ # print (f"Header={has_header}, Force={force_header}")
411
+ dialect = csv.Sniffer().sniff(buffer)
412
+ csvfile.seek(0)
413
+ reader = csv.reader(csvfile, dialect)
414
+ headers = next(reader)
415
+ if not has_header:
416
+ # Use Line as Data and set some default names
417
+ for idx, entry in enumerate(headers):
418
+ skey = f"Column_{idx + 1}"
419
+ self.set_value(skey=skey, value=entry, idx=-1, wtype=1)
420
+ headers[idx] = skey.lower()
421
+ ct = 1
422
+ else:
423
+ ct = 0
424
+ for row in reader:
425
+ for idx, entry in enumerate(row):
426
+ skey = headers[idx].lower()
427
+ if skey.startswith("\\ufeff"):
428
+ skey = skey[7:]
429
+ # Append...
430
+ self.set_value(skey=skey, value=entry, idx=-1, wtype=1)
431
+ ct += 1
432
+ except (csv.Error, PermissionError, OSError, FileNotFoundError) as e:
433
+ ct = 0
434
+ headers = []
435
+ colcount = len(headers)
436
+ return ct, colcount, headers
437
+
438
+ def wordlist_delta(self, orgtext, increase):
439
+ newtext = str(orgtext)
440
+ toreplace = []
441
+ # list of tuples, (index found, old, new )
442
+ # Let's gather the {} first...
443
+ brackets = re.compile(r"\{[^}]+\}")
444
+ for bracketed_key in brackets.findall(str(orgtext)):
445
+ key = bracketed_key[1:-1].lower().strip()
446
+ relative = 0
447
+ pos = key.find("#")
448
+ if pos > 0:
449
+ # Needs to be after first character
450
+ # Process offset modification.
451
+ index_string = key[pos + 1 :]
452
+ key = key[:pos].strip()
453
+
454
+ if not index_string.startswith("+") and not index_string.startswith(
455
+ "-"
456
+ ):
457
+ # We have a #<index> value without + or -, specific index value from 0
458
+ # no need to do something
459
+ continue
460
+ try:
461
+ # This covers +x, -x, x
462
+ relative = int(index_string)
463
+ except ValueError:
464
+ relative = 0
465
+ else:
466
+ # it's the unmodified key...
467
+ if key.startswith("time@"):
468
+ key = "time"
469
+ elif key.startswith("date@"):
470
+ key = "date"
471
+ if key not in self.content:
472
+ continue
473
+ if key in self.prohibited:
474
+ continue
475
+ newindex = relative + increase
476
+ if newindex > 0:
477
+ newpattern = f"{{{key}#+{newindex}}}"
478
+ elif newindex < 0:
479
+ newpattern = f"{{{key}#{newindex}}}"
480
+ else:
481
+ # 0
482
+ newpattern = f"{{{key}}}"
483
+ if newpattern != bracketed_key:
484
+ item = [relative, bracketed_key, newpattern]
485
+ toreplace.append(item)
486
+
487
+ # Then sort the list according to the direction,
488
+ # as we don't want to replace the same pattern again and again
489
+ if increase >= 0:
490
+ toreplace.sort(key=lambda n: n[0])
491
+ else:
492
+ toreplace.sort(reverse=True, key=lambda n: n[0])
493
+ for item in toreplace:
494
+ newtext = newtext.replace(item[1], item[2])
495
+ return newtext
496
+
497
+ def push(self):
498
+ """Stores the current content on the stack"""
499
+ copied_content = {}
500
+ for key, entry in self.content.items():
501
+ copied_content[key] = copy(entry)
502
+ self._stack.append(copied_content)
503
+ # print (f"push was called, when name was: '{self.content['name']}'")
504
+
505
+ def pop(self):
506
+ """Restores the last added stack entry"""
507
+ if len(self._stack) > 0:
508
+ copied_content = self._stack[-1]
509
+ self._stack.pop(-1)
510
+ self.content = {}
511
+ for key, entry in copied_content.items():
512
+ self.content[key] = copy(entry)
513
+ # print (f"pop was called, name now '{self.content['name']}'")