meerk40t 0.9.3001__py2.py3-none-any.whl → 0.9.7010__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 (445) 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 +1195 -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 +1844 -1507
  50. meerk40t/core/elements/clipboard.py +229 -219
  51. meerk40t/core/elements/element_treeops.py +4561 -2837
  52. meerk40t/core/elements/element_types.py +125 -105
  53. meerk40t/core/elements/elements.py +4329 -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 +933 -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/trace.py +651 -563
  66. meerk40t/core/elements/tree_commands.py +415 -409
  67. meerk40t/core/elements/undo_redo.py +116 -58
  68. meerk40t/core/elements/wordlist.py +319 -200
  69. meerk40t/core/exceptions.py +9 -9
  70. meerk40t/core/laserjob.py +220 -220
  71. meerk40t/core/logging.py +63 -63
  72. meerk40t/core/node/blobnode.py +83 -86
  73. meerk40t/core/node/bootstrap.py +105 -103
  74. meerk40t/core/node/branch_elems.py +40 -31
  75. meerk40t/core/node/branch_ops.py +45 -38
  76. meerk40t/core/node/branch_regmark.py +48 -41
  77. meerk40t/core/node/cutnode.py +29 -32
  78. meerk40t/core/node/effect_hatch.py +375 -257
  79. meerk40t/core/node/effect_warp.py +398 -0
  80. meerk40t/core/node/effect_wobble.py +441 -309
  81. meerk40t/core/node/elem_ellipse.py +404 -309
  82. meerk40t/core/node/elem_image.py +1082 -801
  83. meerk40t/core/node/elem_line.py +358 -292
  84. meerk40t/core/node/elem_path.py +259 -201
  85. meerk40t/core/node/elem_point.py +129 -102
  86. meerk40t/core/node/elem_polyline.py +310 -246
  87. meerk40t/core/node/elem_rect.py +376 -286
  88. meerk40t/core/node/elem_text.py +445 -418
  89. meerk40t/core/node/filenode.py +59 -40
  90. meerk40t/core/node/groupnode.py +138 -74
  91. meerk40t/core/node/image_processed.py +777 -766
  92. meerk40t/core/node/image_raster.py +156 -113
  93. meerk40t/core/node/layernode.py +31 -31
  94. meerk40t/core/node/mixins.py +135 -107
  95. meerk40t/core/node/node.py +1427 -1304
  96. meerk40t/core/node/nutils.py +117 -114
  97. meerk40t/core/node/op_cut.py +462 -335
  98. meerk40t/core/node/op_dots.py +296 -251
  99. meerk40t/core/node/op_engrave.py +414 -311
  100. meerk40t/core/node/op_image.py +755 -369
  101. meerk40t/core/node/op_raster.py +787 -522
  102. meerk40t/core/node/place_current.py +37 -40
  103. meerk40t/core/node/place_point.py +329 -126
  104. meerk40t/core/node/refnode.py +58 -47
  105. meerk40t/core/node/rootnode.py +225 -219
  106. meerk40t/core/node/util_console.py +48 -48
  107. meerk40t/core/node/util_goto.py +84 -65
  108. meerk40t/core/node/util_home.py +61 -61
  109. meerk40t/core/node/util_input.py +102 -102
  110. meerk40t/core/node/util_output.py +102 -102
  111. meerk40t/core/node/util_wait.py +65 -65
  112. meerk40t/core/parameters.py +709 -707
  113. meerk40t/core/planner.py +875 -785
  114. meerk40t/core/plotplanner.py +656 -652
  115. meerk40t/core/space.py +120 -113
  116. meerk40t/core/spoolers.py +706 -705
  117. meerk40t/core/svg_io.py +1836 -1549
  118. meerk40t/core/treeop.py +534 -445
  119. meerk40t/core/undos.py +278 -124
  120. meerk40t/core/units.py +784 -680
  121. meerk40t/core/view.py +393 -322
  122. meerk40t/core/webhelp.py +62 -62
  123. meerk40t/core/wordlist.py +513 -504
  124. meerk40t/cylinder/cylinder.py +247 -0
  125. meerk40t/cylinder/gui/cylindersettings.py +41 -0
  126. meerk40t/cylinder/gui/gui.py +24 -0
  127. meerk40t/device/__init__.py +1 -1
  128. meerk40t/device/basedevice.py +322 -123
  129. meerk40t/device/devicechoices.py +50 -0
  130. meerk40t/device/dummydevice.py +163 -128
  131. meerk40t/device/gui/defaultactions.py +618 -602
  132. meerk40t/device/gui/effectspanel.py +114 -0
  133. meerk40t/device/gui/formatterpanel.py +253 -290
  134. meerk40t/device/gui/warningpanel.py +337 -260
  135. meerk40t/device/mixins.py +13 -13
  136. meerk40t/dxf/__init__.py +1 -1
  137. meerk40t/dxf/dxf_io.py +766 -554
  138. meerk40t/dxf/plugin.py +47 -35
  139. meerk40t/external_plugins.py +79 -79
  140. meerk40t/external_plugins_build.py +28 -28
  141. meerk40t/extra/cag.py +112 -116
  142. meerk40t/extra/coolant.py +403 -0
  143. meerk40t/extra/encode_detect.py +198 -0
  144. meerk40t/extra/ezd.py +1165 -1165
  145. meerk40t/extra/hershey.py +835 -340
  146. meerk40t/extra/imageactions.py +322 -316
  147. meerk40t/extra/inkscape.py +630 -622
  148. meerk40t/extra/lbrn.py +424 -424
  149. meerk40t/extra/outerworld.py +284 -0
  150. meerk40t/extra/param_functions.py +1542 -1556
  151. meerk40t/extra/potrace.py +257 -253
  152. meerk40t/extra/serial_exchange.py +118 -0
  153. meerk40t/extra/updater.py +602 -453
  154. meerk40t/extra/vectrace.py +147 -146
  155. meerk40t/extra/winsleep.py +83 -83
  156. meerk40t/extra/xcs_reader.py +597 -0
  157. meerk40t/fill/fills.py +781 -335
  158. meerk40t/fill/patternfill.py +1061 -1061
  159. meerk40t/fill/patterns.py +614 -567
  160. meerk40t/grbl/control.py +87 -87
  161. meerk40t/grbl/controller.py +990 -903
  162. meerk40t/grbl/device.py +1081 -768
  163. meerk40t/grbl/driver.py +989 -771
  164. meerk40t/grbl/emulator.py +532 -497
  165. meerk40t/grbl/gcodejob.py +783 -767
  166. meerk40t/grbl/gui/grblconfiguration.py +373 -298
  167. meerk40t/grbl/gui/grblcontroller.py +485 -271
  168. meerk40t/grbl/gui/grblhardwareconfig.py +269 -153
  169. meerk40t/grbl/gui/grbloperationconfig.py +105 -0
  170. meerk40t/grbl/gui/gui.py +147 -116
  171. meerk40t/grbl/interpreter.py +44 -44
  172. meerk40t/grbl/loader.py +22 -22
  173. meerk40t/grbl/mock_connection.py +56 -56
  174. meerk40t/grbl/plugin.py +294 -264
  175. meerk40t/grbl/serial_connection.py +93 -88
  176. meerk40t/grbl/tcp_connection.py +81 -79
  177. meerk40t/grbl/ws_connection.py +112 -0
  178. meerk40t/gui/__init__.py +1 -1
  179. meerk40t/gui/about.py +2042 -296
  180. meerk40t/gui/alignment.py +1644 -1608
  181. meerk40t/gui/autoexec.py +199 -0
  182. meerk40t/gui/basicops.py +791 -670
  183. meerk40t/gui/bufferview.py +77 -71
  184. meerk40t/gui/busy.py +170 -133
  185. meerk40t/gui/choicepropertypanel.py +1673 -1469
  186. meerk40t/gui/consolepanel.py +706 -542
  187. meerk40t/gui/devicepanel.py +687 -581
  188. meerk40t/gui/dialogoptions.py +110 -107
  189. meerk40t/gui/executejob.py +316 -306
  190. meerk40t/gui/fonts.py +90 -90
  191. meerk40t/gui/functionwrapper.py +252 -0
  192. meerk40t/gui/gui_mixins.py +729 -0
  193. meerk40t/gui/guicolors.py +205 -182
  194. meerk40t/gui/help_assets/help_assets.py +218 -201
  195. meerk40t/gui/helper.py +154 -0
  196. meerk40t/gui/hersheymanager.py +1430 -846
  197. meerk40t/gui/icons.py +3422 -2747
  198. meerk40t/gui/imagesplitter.py +555 -508
  199. meerk40t/gui/keymap.py +354 -344
  200. meerk40t/gui/laserpanel.py +892 -806
  201. meerk40t/gui/laserrender.py +1470 -1232
  202. meerk40t/gui/lasertoolpanel.py +805 -793
  203. meerk40t/gui/magnetoptions.py +436 -0
  204. meerk40t/gui/materialmanager.py +2917 -0
  205. meerk40t/gui/materialtest.py +1722 -1694
  206. meerk40t/gui/mkdebug.py +646 -359
  207. meerk40t/gui/mwindow.py +163 -140
  208. meerk40t/gui/navigationpanels.py +2605 -2467
  209. meerk40t/gui/notes.py +143 -142
  210. meerk40t/gui/opassignment.py +414 -410
  211. meerk40t/gui/operation_info.py +310 -299
  212. meerk40t/gui/plugin.py +494 -328
  213. meerk40t/gui/position.py +714 -669
  214. meerk40t/gui/preferences.py +901 -650
  215. meerk40t/gui/propertypanels/attributes.py +1461 -1131
  216. meerk40t/gui/propertypanels/blobproperty.py +117 -114
  217. meerk40t/gui/propertypanels/consoleproperty.py +83 -80
  218. meerk40t/gui/propertypanels/gotoproperty.py +77 -0
  219. meerk40t/gui/propertypanels/groupproperties.py +223 -217
  220. meerk40t/gui/propertypanels/hatchproperty.py +489 -469
  221. meerk40t/gui/propertypanels/imageproperty.py +2244 -1384
  222. meerk40t/gui/propertypanels/inputproperty.py +59 -58
  223. meerk40t/gui/propertypanels/opbranchproperties.py +82 -80
  224. meerk40t/gui/propertypanels/operationpropertymain.py +1890 -1638
  225. meerk40t/gui/propertypanels/outputproperty.py +59 -58
  226. meerk40t/gui/propertypanels/pathproperty.py +389 -380
  227. meerk40t/gui/propertypanels/placementproperty.py +1214 -383
  228. meerk40t/gui/propertypanels/pointproperty.py +140 -136
  229. meerk40t/gui/propertypanels/propertywindow.py +313 -181
  230. meerk40t/gui/propertypanels/rasterwizardpanels.py +996 -912
  231. meerk40t/gui/propertypanels/regbranchproperties.py +76 -0
  232. meerk40t/gui/propertypanels/textproperty.py +770 -755
  233. meerk40t/gui/propertypanels/waitproperty.py +56 -55
  234. meerk40t/gui/propertypanels/warpproperty.py +121 -0
  235. meerk40t/gui/propertypanels/wobbleproperty.py +255 -204
  236. meerk40t/gui/ribbon.py +2468 -2210
  237. meerk40t/gui/scene/scene.py +1100 -1051
  238. meerk40t/gui/scene/sceneconst.py +22 -22
  239. meerk40t/gui/scene/scenepanel.py +439 -349
  240. meerk40t/gui/scene/scenespacewidget.py +365 -365
  241. meerk40t/gui/scene/widget.py +518 -505
  242. meerk40t/gui/scenewidgets/affinemover.py +215 -215
  243. meerk40t/gui/scenewidgets/attractionwidget.py +315 -309
  244. meerk40t/gui/scenewidgets/bedwidget.py +120 -97
  245. meerk40t/gui/scenewidgets/elementswidget.py +137 -107
  246. meerk40t/gui/scenewidgets/gridwidget.py +785 -745
  247. meerk40t/gui/scenewidgets/guidewidget.py +765 -765
  248. meerk40t/gui/scenewidgets/laserpathwidget.py +66 -66
  249. meerk40t/gui/scenewidgets/machineoriginwidget.py +86 -86
  250. meerk40t/gui/scenewidgets/nodeselector.py +28 -28
  251. meerk40t/gui/scenewidgets/rectselectwidget.py +589 -346
  252. meerk40t/gui/scenewidgets/relocatewidget.py +33 -33
  253. meerk40t/gui/scenewidgets/reticlewidget.py +83 -83
  254. meerk40t/gui/scenewidgets/selectionwidget.py +2952 -2756
  255. meerk40t/gui/simpleui.py +357 -333
  256. meerk40t/gui/simulation.py +2431 -2094
  257. meerk40t/gui/snapoptions.py +208 -203
  258. meerk40t/gui/spoolerpanel.py +1227 -1180
  259. meerk40t/gui/statusbarwidgets/defaultoperations.py +480 -353
  260. meerk40t/gui/statusbarwidgets/infowidget.py +520 -483
  261. meerk40t/gui/statusbarwidgets/opassignwidget.py +356 -355
  262. meerk40t/gui/statusbarwidgets/selectionwidget.py +172 -171
  263. meerk40t/gui/statusbarwidgets/shapepropwidget.py +754 -236
  264. meerk40t/gui/statusbarwidgets/statusbar.py +272 -260
  265. meerk40t/gui/statusbarwidgets/statusbarwidget.py +268 -270
  266. meerk40t/gui/statusbarwidgets/strokewidget.py +267 -251
  267. meerk40t/gui/themes.py +200 -78
  268. meerk40t/gui/tips.py +591 -0
  269. meerk40t/gui/toolwidgets/circlebrush.py +35 -35
  270. meerk40t/gui/toolwidgets/toolcircle.py +248 -242
  271. meerk40t/gui/toolwidgets/toolcontainer.py +82 -77
  272. meerk40t/gui/toolwidgets/tooldraw.py +97 -90
  273. meerk40t/gui/toolwidgets/toolellipse.py +219 -212
  274. meerk40t/gui/toolwidgets/toolimagecut.py +25 -132
  275. meerk40t/gui/toolwidgets/toolline.py +39 -144
  276. meerk40t/gui/toolwidgets/toollinetext.py +79 -236
  277. meerk40t/gui/toolwidgets/toollinetext_inline.py +296 -0
  278. meerk40t/gui/toolwidgets/toolmeasure.py +160 -216
  279. meerk40t/gui/toolwidgets/toolnodeedit.py +2088 -2074
  280. meerk40t/gui/toolwidgets/toolnodemove.py +92 -94
  281. meerk40t/gui/toolwidgets/toolparameter.py +754 -668
  282. meerk40t/gui/toolwidgets/toolplacement.py +108 -108
  283. meerk40t/gui/toolwidgets/toolpoint.py +68 -59
  284. meerk40t/gui/toolwidgets/toolpointlistbuilder.py +294 -0
  285. meerk40t/gui/toolwidgets/toolpointmove.py +183 -0
  286. meerk40t/gui/toolwidgets/toolpolygon.py +288 -403
  287. meerk40t/gui/toolwidgets/toolpolyline.py +38 -196
  288. meerk40t/gui/toolwidgets/toolrect.py +211 -207
  289. meerk40t/gui/toolwidgets/toolrelocate.py +72 -72
  290. meerk40t/gui/toolwidgets/toolribbon.py +598 -113
  291. meerk40t/gui/toolwidgets/tooltabedit.py +546 -0
  292. meerk40t/gui/toolwidgets/tooltext.py +98 -89
  293. meerk40t/gui/toolwidgets/toolvector.py +213 -204
  294. meerk40t/gui/toolwidgets/toolwidget.py +39 -39
  295. meerk40t/gui/usbconnect.py +98 -91
  296. meerk40t/gui/utilitywidgets/buttonwidget.py +18 -18
  297. meerk40t/gui/utilitywidgets/checkboxwidget.py +90 -90
  298. meerk40t/gui/utilitywidgets/controlwidget.py +14 -14
  299. meerk40t/gui/utilitywidgets/cyclocycloidwidget.py +343 -340
  300. meerk40t/gui/utilitywidgets/debugwidgets.py +148 -0
  301. meerk40t/gui/utilitywidgets/handlewidget.py +27 -27
  302. meerk40t/gui/utilitywidgets/harmonograph.py +450 -447
  303. meerk40t/gui/utilitywidgets/openclosewidget.py +40 -40
  304. meerk40t/gui/utilitywidgets/rotationwidget.py +54 -54
  305. meerk40t/gui/utilitywidgets/scalewidget.py +75 -75
  306. meerk40t/gui/utilitywidgets/seekbarwidget.py +183 -183
  307. meerk40t/gui/utilitywidgets/togglewidget.py +142 -142
  308. meerk40t/gui/utilitywidgets/toolbarwidget.py +8 -8
  309. meerk40t/gui/wordlisteditor.py +985 -931
  310. meerk40t/gui/wxmeerk40t.py +1444 -1169
  311. meerk40t/gui/wxmmain.py +5578 -4112
  312. meerk40t/gui/wxmribbon.py +1591 -1076
  313. meerk40t/gui/wxmscene.py +1635 -1453
  314. meerk40t/gui/wxmtree.py +2410 -2089
  315. meerk40t/gui/wxutils.py +1769 -1099
  316. meerk40t/gui/zmatrix.py +102 -102
  317. meerk40t/image/__init__.py +1 -1
  318. meerk40t/image/dither.py +429 -0
  319. meerk40t/image/imagetools.py +2778 -2269
  320. meerk40t/internal_plugins.py +150 -130
  321. meerk40t/kernel/__init__.py +63 -12
  322. meerk40t/kernel/channel.py +259 -212
  323. meerk40t/kernel/context.py +538 -538
  324. meerk40t/kernel/exceptions.py +41 -41
  325. meerk40t/kernel/functions.py +463 -414
  326. meerk40t/kernel/jobs.py +100 -100
  327. meerk40t/kernel/kernel.py +3809 -3571
  328. meerk40t/kernel/lifecycles.py +71 -71
  329. meerk40t/kernel/module.py +49 -49
  330. meerk40t/kernel/service.py +147 -147
  331. meerk40t/kernel/settings.py +383 -343
  332. meerk40t/lihuiyu/controller.py +883 -876
  333. meerk40t/lihuiyu/device.py +1181 -1069
  334. meerk40t/lihuiyu/driver.py +1466 -1372
  335. meerk40t/lihuiyu/gui/gui.py +127 -106
  336. meerk40t/lihuiyu/gui/lhyaccelgui.py +377 -363
  337. meerk40t/lihuiyu/gui/lhycontrollergui.py +741 -651
  338. meerk40t/lihuiyu/gui/lhydrivergui.py +470 -446
  339. meerk40t/lihuiyu/gui/lhyoperationproperties.py +238 -237
  340. meerk40t/lihuiyu/gui/tcpcontroller.py +226 -190
  341. meerk40t/lihuiyu/interpreter.py +53 -53
  342. meerk40t/lihuiyu/laserspeed.py +450 -450
  343. meerk40t/lihuiyu/loader.py +90 -90
  344. meerk40t/lihuiyu/parser.py +404 -404
  345. meerk40t/lihuiyu/plugin.py +101 -102
  346. meerk40t/lihuiyu/tcp_connection.py +111 -109
  347. meerk40t/main.py +231 -165
  348. meerk40t/moshi/builder.py +788 -781
  349. meerk40t/moshi/controller.py +505 -499
  350. meerk40t/moshi/device.py +495 -442
  351. meerk40t/moshi/driver.py +862 -696
  352. meerk40t/moshi/gui/gui.py +78 -76
  353. meerk40t/moshi/gui/moshicontrollergui.py +538 -522
  354. meerk40t/moshi/gui/moshidrivergui.py +87 -75
  355. meerk40t/moshi/plugin.py +43 -43
  356. meerk40t/network/console_server.py +102 -57
  357. meerk40t/network/kernelserver.py +10 -9
  358. meerk40t/network/tcp_server.py +142 -140
  359. meerk40t/network/udp_server.py +103 -77
  360. meerk40t/network/web_server.py +390 -0
  361. meerk40t/newly/controller.py +1158 -1144
  362. meerk40t/newly/device.py +874 -732
  363. meerk40t/newly/driver.py +540 -412
  364. meerk40t/newly/gui/gui.py +219 -188
  365. meerk40t/newly/gui/newlyconfig.py +116 -101
  366. meerk40t/newly/gui/newlycontroller.py +193 -186
  367. meerk40t/newly/gui/operationproperties.py +51 -51
  368. meerk40t/newly/mock_connection.py +82 -82
  369. meerk40t/newly/newly_params.py +56 -56
  370. meerk40t/newly/plugin.py +1214 -1246
  371. meerk40t/newly/usb_connection.py +322 -322
  372. meerk40t/rotary/gui/gui.py +52 -46
  373. meerk40t/rotary/gui/rotarysettings.py +240 -232
  374. meerk40t/rotary/rotary.py +202 -98
  375. meerk40t/ruida/control.py +291 -91
  376. meerk40t/ruida/controller.py +138 -1088
  377. meerk40t/ruida/device.py +672 -231
  378. meerk40t/ruida/driver.py +534 -472
  379. meerk40t/ruida/emulator.py +1494 -1491
  380. meerk40t/ruida/exceptions.py +4 -4
  381. meerk40t/ruida/gui/gui.py +71 -76
  382. meerk40t/ruida/gui/ruidaconfig.py +239 -72
  383. meerk40t/ruida/gui/ruidacontroller.py +187 -184
  384. meerk40t/ruida/gui/ruidaoperationproperties.py +48 -47
  385. meerk40t/ruida/loader.py +54 -52
  386. meerk40t/ruida/mock_connection.py +57 -109
  387. meerk40t/ruida/plugin.py +124 -87
  388. meerk40t/ruida/rdjob.py +2084 -945
  389. meerk40t/ruida/serial_connection.py +116 -0
  390. meerk40t/ruida/tcp_connection.py +146 -0
  391. meerk40t/ruida/udp_connection.py +73 -0
  392. meerk40t/svgelements.py +9671 -9669
  393. meerk40t/tools/driver_to_path.py +584 -579
  394. meerk40t/tools/geomstr.py +5583 -4680
  395. meerk40t/tools/jhfparser.py +357 -292
  396. meerk40t/tools/kerftest.py +904 -890
  397. meerk40t/tools/livinghinges.py +1168 -1033
  398. meerk40t/tools/pathtools.py +987 -949
  399. meerk40t/tools/pmatrix.py +234 -0
  400. meerk40t/tools/pointfinder.py +942 -942
  401. meerk40t/tools/polybool.py +940 -940
  402. meerk40t/tools/rasterplotter.py +1660 -547
  403. meerk40t/tools/shxparser.py +989 -901
  404. meerk40t/tools/ttfparser.py +726 -446
  405. meerk40t/tools/zinglplotter.py +595 -593
  406. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7010.dist-info}/LICENSE +21 -21
  407. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7010.dist-info}/METADATA +150 -139
  408. meerk40t-0.9.7010.dist-info/RECORD +445 -0
  409. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7010.dist-info}/WHEEL +1 -1
  410. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7010.dist-info}/top_level.txt +0 -1
  411. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7010.dist-info}/zip-safe +1 -1
  412. meerk40t/balormk/elementlightjob.py +0 -159
  413. meerk40t-0.9.3001.dist-info/RECORD +0 -437
  414. test/bootstrap.py +0 -63
  415. test/test_cli.py +0 -12
  416. test/test_core_cutcode.py +0 -418
  417. test/test_core_elements.py +0 -144
  418. test/test_core_plotplanner.py +0 -397
  419. test/test_core_viewports.py +0 -312
  420. test/test_drivers_grbl.py +0 -108
  421. test/test_drivers_lihuiyu.py +0 -443
  422. test/test_drivers_newly.py +0 -113
  423. test/test_element_degenerate_points.py +0 -43
  424. test/test_elements_classify.py +0 -97
  425. test/test_elements_penbox.py +0 -22
  426. test/test_file_svg.py +0 -176
  427. test/test_fill.py +0 -155
  428. test/test_geomstr.py +0 -1523
  429. test/test_geomstr_nodes.py +0 -18
  430. test/test_imagetools_actualize.py +0 -306
  431. test/test_imagetools_wizard.py +0 -258
  432. test/test_kernel.py +0 -200
  433. test/test_laser_speeds.py +0 -3303
  434. test/test_length.py +0 -57
  435. test/test_lifecycle.py +0 -66
  436. test/test_operations.py +0 -251
  437. test/test_operations_hatch.py +0 -57
  438. test/test_ruida.py +0 -19
  439. test/test_spooler.py +0 -22
  440. test/test_tools_rasterplotter.py +0 -29
  441. test/test_wobble.py +0 -133
  442. test/test_zingl.py +0 -124
  443. {test → meerk40t/cylinder}/__init__.py +0 -0
  444. /meerk40t/{core/element_commands.py → cylinder/gui/__init__.py} +0 -0
  445. {meerk40t-0.9.3001.dist-info → meerk40t-0.9.7010.dist-info}/entry_points.txt +0 -0
meerk40t/extra/hershey.py CHANGED
@@ -1,340 +1,835 @@
1
- from glob import glob
2
- from os.path import basename, exists, join, realpath, splitext
3
-
4
- from meerk40t.core.node.elem_path import PathNode
5
- from meerk40t.core.units import UNITS_PER_PIXEL, Length
6
- from meerk40t.kernel import get_safe_path
7
- from meerk40t.svgelements import Arc, Color, Path
8
- from meerk40t.tools.jhfparser import JhfFont
9
- from meerk40t.tools.shxparser import ShxFont, ShxFontParseError
10
- from meerk40t.tools.ttfparser import TrueTypeFont
11
-
12
-
13
- class FontPath:
14
- def __init__(self):
15
- self.path = Path()
16
-
17
- def new_path(self):
18
- pass
19
-
20
- def move(self, x, y):
21
- self.path.move((x, -y))
22
-
23
- def line(self, x0, y0, x1, y1):
24
- self.path.line((x1, -y1))
25
-
26
- def quad(self, x0, y0, x1, y1, x2, y2):
27
- self.path.quad((x1, -y1), (x2, -y2))
28
-
29
- def cubic(self, x0, y0, x1, y1, x2, y2, x3, y3):
30
- self.path.cubic((x1, -y1), (x2, -y2), (x3, -y3))
31
-
32
- def close(self):
33
- self.path.closed()
34
-
35
- def arc(self, x0, y0, cx, cy, x1, y1):
36
- arc = Arc(start=(x0, -y0), control=(cx, -cy), end=(x1, -y1))
37
- self.path += arc
38
-
39
-
40
- def fonts_registered():
41
- registered_fonts = {
42
- "shx": ("Autocad", ShxFont),
43
- "jhf": ("Hershey", JhfFont),
44
- "ttf": ("TrueType", TrueTypeFont),
45
- }
46
- return registered_fonts
47
-
48
-
49
- def have_hershey_fonts(context):
50
- safe_dir = realpath(get_safe_path(context.kernel.name))
51
- context.setting(str, "font_directory", safe_dir)
52
- font_dir = context.font_directory
53
- registered_fonts = fonts_registered()
54
- for extension in registered_fonts:
55
- for p in glob(join(font_dir, "*." + extension.lower())):
56
- return True
57
- for p in glob(join(font_dir, "*." + extension.upper())):
58
- return True
59
- return False
60
-
61
-
62
- def validate_node(node):
63
- # After a svg load the attributes are still a string...
64
- if not hasattr(node, "mkfontsize"):
65
- return
66
- if isinstance(node.mkfontsize, str):
67
- try:
68
- value = float(node.mkfontsize)
69
- except ValueError:
70
- value = Length("20px")
71
- node.mkfontsize = value
72
- # if not hasattr(node, "mkcoordx"):
73
- # node.mkcoordx = 0
74
- # if not hasattr(node, "mkcoordy"):
75
- # node.mkcoordy = 0
76
- # if isinstance(node.mkcoordx, str):
77
- # try:
78
- # value = float(node.mkcoordx)
79
- # except ValueError:
80
- # value = 0
81
- # node.mkcoordx = value
82
- # if isinstance(node.mkcoordy, str):
83
- # try:
84
- # value = float(node.mkcoordy)
85
- # except ValueError:
86
- # value = 0
87
- # node.mkcoordy = value
88
-
89
-
90
- def update(context, node):
91
- # We need to check for the validity ourselves...
92
- if (
93
- hasattr(node, "mktext")
94
- and hasattr(node, "mkfont")
95
- and hasattr(node, "mkfontsize")
96
- ):
97
- update_linetext(context, node, node.mktext)
98
-
99
-
100
- def update_linetext(context, node, newtext):
101
- # print ("Update Linetext")
102
- if node is None:
103
- # print ("node is none, exit")
104
- return
105
- if not hasattr(node, "mkfont"):
106
- # print ("no font attr, exit")
107
- return
108
- if not hasattr(node, "mkfontsize"):
109
- # print ("no fontsize attr, exit")
110
- return
111
- oldtext = getattr(node, "_translated_text", "")
112
- registered_fonts = fonts_registered()
113
- fontname = node.mkfont
114
- fontsize = node.mkfontsize
115
- # old_color = node.stroke
116
- # old_strokewidth = node.stroke_width
117
- # old_strokescaled = node._stroke_scaled
118
- font_dir = getattr(context, "font_directory", "")
119
- font_path = join(font_dir, fontname)
120
- if not exists(font_path):
121
- # Fallback to meerk40t directory...
122
- safe_dir = realpath(get_safe_path(context.kernel.name))
123
- font_path = join(safe_dir, fontname)
124
- if not exists(font_path):
125
- return
126
- try:
127
- filename, file_extension = splitext(font_path)
128
- if len(file_extension) > 0:
129
- # Remove dot...
130
- file_extension = file_extension[1:].lower()
131
- item = registered_fonts[file_extension]
132
- fontclass = item[1]
133
- except (KeyError, IndexError):
134
- # channel(_("Unknown fonttype {ext}").format(ext=file_extension))
135
- # print ("unknown fonttype, exit")
136
- return
137
- # print("Nearly there, all fonts checked...")
138
- cfont = fontclass(font_path)
139
- path = FontPath()
140
- # print (f"Path={path}, text={remainder}, font-size={font_size}")
141
- horizontal = True
142
- mytext = context.elements.wordlist_translate(newtext)
143
- cfont.render(path, mytext, horizontal, float(fontsize))
144
- olda = node.path.transform.a
145
- oldb = node.path.transform.b
146
- oldc = node.path.transform.c
147
- oldd = node.path.transform.d
148
- olde = node.path.transform.e
149
- oldf = node.path.transform.f
150
- node.path = path.path
151
- node.path.transform.a = olda
152
- node.path.transform.b = oldb
153
- node.path.transform.c = oldc
154
- node.path.transform.d = oldd
155
- node.path.transform.e = olde
156
- node.path.transform.f = oldf
157
- # print (f"x={node.mkcoordx}, y={node.mkcoordy}")
158
- # node.path.transform = Matrix.translate(node.mkcoordx, node.mkcoordy)
159
- # print (f"Updated: from {oldtext} -> {mytext}")
160
- node.mktext = newtext
161
- node._translated_text = mytext
162
- node.altered()
163
-
164
-
165
- def create_linetext_node(context, x, y, text, font=None, font_size=None):
166
- registered_fonts = fonts_registered()
167
-
168
- if font_size is None:
169
- font_size = Length("20px")
170
-
171
- context.setting(str, "shx_preferred", None)
172
- safe_dir = realpath(get_safe_path(context.kernel.name))
173
- context.setting(str, "font_directory", safe_dir)
174
- font_dir = context.font_directory
175
- # Check whether the default is still valid
176
- if context.shx_preferred is not None and context.shx_preferred != "":
177
- font_path = join(font_dir, context.shx_preferred)
178
- if not exists(font_path):
179
- context.shx_preferred = None
180
- # Valid font?
181
- if font is not None and font != "":
182
- font_path = join(font_dir, font)
183
- if not exists(font_path):
184
- font = None
185
- if font is not None:
186
- context.shx_preferred = font
187
- else:
188
- if context.shx_preferred is not None:
189
- # print (f"Fallback to {context.shx_preferred}")
190
- font = context.shx_preferred
191
- # Still not valid?
192
- if font is None or font == "":
193
- font = ""
194
- # No preferred font set, let's try a couple of candidates...
195
- candidates = (
196
- "timesr.jhf",
197
- "romant.shx",
198
- "rowmans.jhf",
199
- "FUTURA.SHX",
200
- "arial.ttf",
201
- )
202
- for fname in candidates:
203
- fullfname = join(font_dir, fname)
204
- if exists(fullfname):
205
- # print (f"Taking font {fname} instead")
206
- font = fname
207
- context.shx_preferred = font
208
- break
209
- if font == "":
210
- # You know, I take anything at this point...
211
- for extension in registered_fonts:
212
- ext = "*." + extension
213
- if font == "":
214
- for p in glob(join(font_dir, ext.lower())):
215
- font = basename(p)
216
- # print (f"Fallback to first file found: {font}")
217
- context.shx_preferred = font
218
- break
219
- if font == "":
220
- for p in glob(join(font_dir, ext.upper())):
221
- font = basename(p)
222
- # print (f"Fallback to first file found: {font}")
223
- context.shx_preferred = font
224
- break
225
-
226
- if font is None or font == "":
227
- # print ("Font was empty")
228
- return None
229
- font_path = join(font_dir, font)
230
- try:
231
- filename, file_extension = splitext(font_path)
232
- if len(file_extension) > 0:
233
- # Remove dot...
234
- file_extension = file_extension[1:].lower()
235
- item = registered_fonts[file_extension]
236
- fontclass = item[1]
237
- except (KeyError, IndexError):
238
- # print(f"Unknown fonttype {file_extension}")
239
- return None
240
- horizontal = True
241
- try:
242
- cfont = fontclass(font_path)
243
- path = FontPath()
244
- # print (f"Path={path}, text={remainder}, font-size={font_size}")
245
- mytext = context.elements.wordlist_translate(text)
246
- cfont.render(path, mytext, horizontal, float(font_size))
247
- except ShxFontParseError as e:
248
- # print(f"FontParseError {e.args}")
249
- return
250
- # print (f"Pathlen={len(path.path)}")
251
- # if len(path.path) == 0:
252
- # print("Empty path...")
253
- # return None
254
-
255
- path_node = PathNode(
256
- path=path.path,
257
- stroke=Color("black"),
258
- )
259
- path_node.matrix.post_translate(x, y)
260
- path_node.mkfont = font
261
- path_node.mkfontsize = float(font_size)
262
- path_node.mktext = text
263
- path_node._translated_text = mytext
264
- path_node.mkcoordx = x
265
- path_node.mkcoordy = y
266
- path_node.stroke_width = UNITS_PER_PIXEL
267
-
268
- return path_node
269
-
270
-
271
- def plugin(kernel, lifecycle):
272
- if lifecycle == "register":
273
- _ = kernel.translation
274
- context = kernel.root
275
-
276
- # Register update routine for linetext
277
- kernel.register("path_updater/linetext", update)
278
-
279
- @context.console_option("font", "f", type=str, help=_("SHX font file."))
280
- @context.console_option(
281
- "font_size", "s", type=Length, default="20px", help=_("SHX font file.")
282
- )
283
- @context.console_command(
284
- "linetext", help=_("linetext <font> <font_size> <text>")
285
- )
286
- def linetext(
287
- command, channel, _, font=None, font_size=None, remainder=None, **kwargs
288
- ):
289
- def display_fonts():
290
- for extension, item in registered_fonts:
291
- desc = item[0]
292
- channel(
293
- _("{ftype} fonts in {path}:").format(ftype=desc, path=font_dir)
294
- )
295
- for p in glob(join(font_dir, "*." + extension.lower())):
296
- channel(p)
297
- for p in glob(join(font_dir, "*." + extension.upper())):
298
- channel(p)
299
- return
300
-
301
- registered_fonts = fonts_registered()
302
-
303
- context.setting(str, "shx_preferred", None)
304
- if font is not None:
305
- context.shx_preferred = font
306
- font = context.shx_preferred
307
-
308
- safe_dir = realpath(get_safe_path(context.kernel.name))
309
- context.setting(str, "font_directory", safe_dir)
310
- font_dir = context.font_directory
311
-
312
- if font is None:
313
- display_fonts()
314
- return
315
- font_path = join(font_dir, font)
316
- if not exists(font_path):
317
- channel(_("Font was not found at {path}").format(path=font_path))
318
- display_fonts()
319
- return
320
- if remainder is None:
321
- channel(_("No text to make a path with."))
322
- return
323
- x = 0
324
- y = float(font_size)
325
-
326
- try:
327
- font = ShxFont(font_path)
328
- path = FontPath()
329
- font.render(path, remainder, True, float(font_size))
330
- except ShxFontParseError as e:
331
- channel(f"{e.args}")
332
- return
333
- path_node = create_linetext_node(context, x, y, remainder, font, font_size)
334
- # path_node = PathNode(
335
- # path=path.path,
336
- # matrix=Matrix.translate(0, float(font_size)),
337
- # stroke=Color("black"),
338
- # )
339
- context.elements.elem_branch.add_node(path_node)
340
- context.signal("element_added", path_node)
1
+ import os
2
+ import platform
3
+ from functools import lru_cache
4
+ from glob import glob
5
+ from os.path import basename, exists, join, realpath, splitext
6
+
7
+ from meerk40t.core.node.elem_path import PathNode
8
+ from meerk40t.core.node.node import Fillrule, Linejoin
9
+ from meerk40t.core.units import UNITS_PER_INCH, Length
10
+ from meerk40t.kernel import get_safe_path
11
+ from meerk40t.tools.geomstr import BeamTable, Geomstr
12
+ from meerk40t.tools.jhfparser import JhfFont
13
+ from meerk40t.tools.shxparser import ShxFont, ShxFontParseError
14
+ from meerk40t.tools.ttfparser import TrueTypeFont, TTFParsingError
15
+
16
+ # import numpy as np
17
+
18
+
19
+ class FontPath:
20
+ def __init__(self, weld):
21
+ self.total_list = list()
22
+ self.total_geometry = Geomstr()
23
+ self.geom = Geomstr()
24
+ self._index = 0
25
+ self.start = None
26
+ self.weld = weld
27
+
28
+ def character_end(self):
29
+ if self.weld:
30
+ self.geom.as_interpolated_points()
31
+ c = Geomstr()
32
+ for sp in self.geom.as_subpaths():
33
+ for segs in sp.as_interpolated_segments(interpolate=10):
34
+ c.polyline(segs)
35
+ c.end()
36
+ c.flag_settings(flag=self._index)
37
+ self._index += 1
38
+ self.total_geometry.append(c)
39
+ else:
40
+ self.total_geometry.append(self.geom)
41
+ self.geom.clear()
42
+
43
+ @property
44
+ def geometry(self):
45
+ if not self.weld:
46
+ return self.total_geometry
47
+ bt = BeamTable(self.total_geometry.simplify())
48
+ union = bt.union(*list(range(self._index)))
49
+ # union.remove_0_length()
50
+ union.greedy_distance()
51
+ return union.simplify()
52
+
53
+ def new_path(self):
54
+ self.geom.end()
55
+
56
+ def move(self, x, y):
57
+ # self.geom.move((x, -y))
58
+ if self.start is not None:
59
+ self.geom.end()
60
+ self.start = x - 1j * y
61
+
62
+ def line(self, x0, y0, x1, y1):
63
+ # self.path.line((x1, -y1))
64
+ end = x1 - 1j * y1
65
+ self.geom.line(self.start, end)
66
+ self.start = end
67
+
68
+ def quad(self, x0, y0, x1, y1, x2, y2):
69
+ # self.path.quad((x1, -y1), (x2, -y2))
70
+ control = x1 - 1j * y1
71
+ end = x2 - 1j * y2
72
+ self.geom.quad(self.start, control, end)
73
+ self.start = end
74
+
75
+ def cubic(self, x0, y0, x1, y1, x2, y2, x3, y3):
76
+ # self.path.cubic((x1, -y1), (x2, -y2), (x3, -y3))
77
+ control0 = x1 - 1j * y1
78
+ control1 = x2 - 1j * y2
79
+ end = x3 - 1j * y3
80
+ self.geom.cubic(self.start, control0, control1, end)
81
+ self.start = end
82
+
83
+ def close(self):
84
+ self.geom.close()
85
+
86
+ def arc(self, x0, y0, cx, cy, x1, y1):
87
+ # arc = Arc(start=(x0, -y0), control=(cx, -cy), end=(x1, -y1))
88
+ # self.path += arc
89
+ control = cx - 1j * cy
90
+ end = x1 - 1j * y1
91
+ self.geom.arc(self.start, control, end)
92
+ self.start = end
93
+
94
+
95
+ class Meerk40tFonts:
96
+ """
97
+ Main Interface to access vector fonts.
98
+ Supported formats:
99
+ ttf - TrueType fonts
100
+ shx - AutoCad fonts
101
+ jhf - Hershey fonts
102
+ """
103
+
104
+ def __init__(self, context, **kwds):
105
+ self.context = context
106
+ self._available_fonts = None
107
+
108
+ @property
109
+ def fonts_registered(self):
110
+ fonts = {
111
+ "shx": ("Autocad", ShxFont, None),
112
+ "jhf": ("Hershey", JhfFont, None),
113
+ "ttf": ("TrueType", TrueTypeFont, TrueTypeFont.query_name),
114
+ }
115
+ return fonts
116
+
117
+ @property
118
+ def font_directory(self):
119
+ safe_dir = realpath(get_safe_path(self.context.kernel.name))
120
+ self.context.setting(str, "font_directory", safe_dir)
121
+ fontdir = self.context.font_directory
122
+ if not exists(fontdir):
123
+ # Fallback, something strange happened...
124
+ fontdir = safe_dir
125
+ self.context.font_directory = fontdir
126
+ return fontdir
127
+
128
+ @font_directory.setter
129
+ def font_directory(self, value):
130
+ if not exists(value):
131
+ # We cant allow a non-valid directory
132
+ value = realpath(get_safe_path(self.context.kernel.name))
133
+ self.context.setting(str, "font_directory", value)
134
+ self.context.font_directory = value
135
+ self._available_fonts = None
136
+
137
+ @property
138
+ def cache_file(self):
139
+ return join(self.font_directory, "fonts.cache")
140
+
141
+ def reset_cache(self):
142
+ fn = self.cache_file
143
+ try:
144
+ os.remove(fn)
145
+ except (OSError, FileNotFoundError, PermissionError):
146
+ pass
147
+ self._available_fonts = None
148
+ p = self.available_fonts()
149
+
150
+ def have_hershey_fonts(self):
151
+ p = self.available_fonts()
152
+ return len(p) > 0
153
+
154
+ @lru_cache(maxsize=512)
155
+ def get_font_information(self, full_file_name):
156
+ filename, file_extension = splitext(full_file_name)
157
+ if len(file_extension) == 0:
158
+ return None
159
+ # Remove dot...
160
+ file_extension = file_extension[1:].lower()
161
+ try:
162
+ item = self.fonts_registered[file_extension]
163
+ except KeyError:
164
+ return None
165
+ if item[2]:
166
+ return item[2](full_file_name)
167
+ return None
168
+
169
+ def _get_full_info(self, short):
170
+ s_lower = short.lower()
171
+ p = self.available_fonts()
172
+ for info in p:
173
+ # We don't care about capitalisation
174
+ f_lower = info[0].lower()
175
+ if f_lower.endswith(s_lower):
176
+ return info
177
+ return None
178
+
179
+ def is_system_font(self, short):
180
+ info = self._get_full_info(short)
181
+ if info:
182
+ return info[4]
183
+ return True
184
+
185
+ def face_to_full_name(self, short):
186
+ s_lower = short.lower()
187
+ p = self.available_fonts()
188
+ options = ("regular", "bold", "italic")
189
+ candidates = []
190
+ for info in p:
191
+ # We don't care about capitalisation
192
+ f_lower = info[1].lower()
193
+ # print (f"Comparing {s_lower} to {f_lower} ({info[1]}, {info[2]}, {info[3]})")
194
+ if f_lower == s_lower:
195
+ return info[0]
196
+ for idx, opt in enumerate(options):
197
+ if f"{s_lower} {opt}" == f_lower:
198
+ # print (f"Appending {idx} {f_lower}")
199
+ candidates.append((idx, info[0]))
200
+ if len(candidates):
201
+ candidates.sort(key=lambda e: e[0])
202
+ return candidates[0][1]
203
+ return None
204
+
205
+ def full_name(self, short):
206
+ info = self._get_full_info(short)
207
+ if info:
208
+ return info[0]
209
+ return None
210
+
211
+ def short_name(self, fullname):
212
+ return basename(fullname)
213
+
214
+ @lru_cache(maxsize=128)
215
+ def cached_fontclass(self, fontname):
216
+ registered_fonts = self.fonts_registered
217
+ if not exists(fontname):
218
+ return
219
+ try:
220
+ filename, file_extension = splitext(fontname)
221
+ if len(file_extension) > 0:
222
+ # Remove dot...
223
+ file_extension = file_extension[1:].lower()
224
+ item = registered_fonts[file_extension]
225
+ fontclass = item[1]
226
+ except (KeyError, IndexError):
227
+ # channel(_("Unknown fonttype {ext}").format(ext=file_extension))
228
+ # print ("unknown fonttype, exit")
229
+ return
230
+ # print("Nearly there, all fonts checked...")
231
+ try:
232
+ cfont = fontclass(fontname)
233
+ except (TTFParsingError, ShxFontParseError):
234
+ return None
235
+ return cfont
236
+
237
+ def validate_node(self, node):
238
+ # After a svg load the attributes are still a string...
239
+ if not hasattr(node, "mkfontsize"):
240
+ return
241
+ if isinstance(node.mkfontsize, str):
242
+ try:
243
+ value = float(node.mkfontsize)
244
+ except ValueError:
245
+ value = Length("20px")
246
+ node.mkfontsize = value
247
+ # if not hasattr(node, "mkcoordx"):
248
+ # node.mkcoordx = 0
249
+ # if not hasattr(node, "mkcoordy"):
250
+ # node.mkcoordy = 0
251
+ # if isinstance(node.mkcoordx, str):
252
+ # try:
253
+ # value = float(node.mkcoordx)
254
+ # except ValueError:
255
+ # value = 0
256
+ # node.mkcoordx = value
257
+ # if isinstance(node.mkcoordy, str):
258
+ # try:
259
+ # value = float(node.mkcoordy)
260
+ # except ValueError:
261
+ # value = 0
262
+ # node.mkcoordy = value
263
+
264
+ def update(self, context=None, node=None):
265
+ # We need to check for the validity ourselves...
266
+ if (
267
+ hasattr(node, "mktext")
268
+ and hasattr(node, "mkfont")
269
+ and hasattr(node, "mkfontsize")
270
+ ):
271
+ self.update_linetext(node, node.mktext)
272
+
273
+ def update_linetext(self, node, newtext):
274
+ # print ("Update Linetext")
275
+ if node is None:
276
+ # print ("node is none, exit")
277
+ return
278
+ if not hasattr(node, "mkfont"):
279
+ # print ("no font attr, exit")
280
+ return
281
+ if not hasattr(node, "mkfontsize"):
282
+ # print ("no fontsize attr, exit")
283
+ return
284
+ h_spacing = None
285
+ if hasattr(node, "mkfontspacing"):
286
+ try:
287
+ h_spacing = float(node.mkfontspacing)
288
+ except AttributeError:
289
+ pass
290
+ if h_spacing is None:
291
+ h_spacing = 1
292
+ v_spacing = None
293
+ if hasattr(node, "mklinegap"):
294
+ try:
295
+ v_spacing = float(node.mklinegap)
296
+ except AttributeError:
297
+ pass
298
+ if v_spacing is None:
299
+ v_spacing = 1.1
300
+
301
+ align = None
302
+ if hasattr(node, "mkalign"):
303
+ align = node.mkalign
304
+ if align is None or align not in ("start", "middle", "end"):
305
+ align = "start"
306
+
307
+ weld = None
308
+ if hasattr(node, "mkfontweld"):
309
+ try:
310
+ weld = bool(node.mkfontweld)
311
+ except ValueError:
312
+ pass
313
+ if weld is None:
314
+ weld = False
315
+ # from time import perf_counter
316
+ # _t0 = perf_counter()
317
+ # oldtext = getattr(node, "_translated_text", "")
318
+ fontname = node.mkfont
319
+ fontsize = node.mkfontsize
320
+ # old_color = node.stroke
321
+ # old_strokewidth = node.stroke_width
322
+ # old_strokescaled = node._stroke_scaled
323
+ fullfont = self.full_name(fontname)
324
+ if fullfont is None:
325
+ # This font does not exist in our environment
326
+ return
327
+ # Make sure any paths are removed
328
+ fontname = self.short_name(fontname)
329
+
330
+ cfont = self.cached_fontclass(fullfont)
331
+ if cfont is None:
332
+ # This font does not exist in our environment
333
+ return
334
+
335
+ # _t1 = perf_counter()
336
+
337
+ path = FontPath(weld)
338
+ # print (f"Path={path}, text={remainder}, font-size={font_size}")
339
+ horizontal = True
340
+ mytext = self.context.elements.wordlist_translate(newtext)
341
+ cfont.render(
342
+ path, mytext, horizontal, float(fontsize), h_spacing, v_spacing, align
343
+ )
344
+ if hasattr(cfont, "line_information"):
345
+ # Store the relative start / end positions of the text lines
346
+ # for any interested party...
347
+ node._line_information = cfont.line_information()
348
+ # _t2 = perf_counter()
349
+ olda = node.matrix.a
350
+ oldb = node.matrix.b
351
+ oldc = node.matrix.c
352
+ oldd = node.matrix.d
353
+ olde = node.matrix.e
354
+ oldf = node.matrix.f
355
+ node.geometry = path.geometry
356
+ node.matrix.a = olda
357
+ node.matrix.b = oldb
358
+ node.matrix.c = oldc
359
+ node.matrix.d = oldd
360
+ node.matrix.e = olde
361
+ node.matrix.f = oldf
362
+ # print (f"x={node.mkcoordx}, y={node.mkcoordy}")
363
+ # node.path.transform = Matrix.translate(node.mkcoordx, node.mkcoordy)
364
+ # print (f"Updated: from {oldtext} -> {mytext}")
365
+ node.mktext = newtext
366
+ node.mkfont = fontname
367
+ node._translated_text = mytext
368
+ # _t3 = perf_counter()
369
+ tlines = mytext.split("\n")
370
+ tlabel = f"Text: {tlines[0]}"
371
+ if node.label is None or node.label.startswith("Text: "):
372
+ node.label = tlabel
373
+
374
+ node.altered()
375
+ # _t4 = perf_counter()
376
+ # print (f"Readfont: {_t1 -_t0:.2f}s, render: {_t2 -_t1:.2f}s, path: {_t3 -_t2:.2f}s, alter: {_t4 -_t3:.2f}s, total={_t4 -_t0:.2f}s")
377
+
378
+ def _validate_font(self, font):
379
+ """
380
+ Check if the given font value is valid.
381
+
382
+ @param font:
383
+ @return:
384
+ """
385
+ if not font:
386
+ return False
387
+ # Let's check whether it's valid...
388
+ font_path = self.full_name(font)
389
+ if not font_path:
390
+ # Font isn't found as processed.
391
+ return False
392
+ try:
393
+ self.cached_fontclass(font_path)
394
+ except TTFParsingError:
395
+ # Font could not parse.
396
+ return False
397
+ return True
398
+
399
+ def _try_candidates(self):
400
+ # No preferred font set, let's try a couple of candidates...
401
+ candidates = (
402
+ "arial.ttf",
403
+ "opensans_regular.ttf",
404
+ "timesr.jhf",
405
+ "romant.shx",
406
+ "rowmans.jhf",
407
+ "FUTURA.SHX",
408
+ )
409
+ for fname in candidates:
410
+ if self._validate_font(fname):
411
+ return self.full_name(fname)
412
+ return None
413
+
414
+ def _try_available(self):
415
+ if not self.available_fonts():
416
+ return None
417
+ for i, font in enumerate(self._available_fonts):
418
+ candidate = font[0]
419
+ if self._validate_font(candidate):
420
+ return candidate
421
+ return None
422
+
423
+ def retrieve_font(self, font):
424
+ if not self._validate_font(font) and font is not None:
425
+ # Is the given font valid?
426
+ # It could still translate to a valid name
427
+ font_path = self.face_to_full_name(font)
428
+ if font_path:
429
+ font = self.short_name(font_path)
430
+ return font, font_path
431
+ font = None
432
+ if not font:
433
+ # No valid font, try last font.
434
+ font = self.context.last_font
435
+ if not self._validate_font(font):
436
+ font = None
437
+ if not font:
438
+ # Still not valid? Try preselected candidates.
439
+ font = self._try_candidates()
440
+ if not font:
441
+ # You know, I take anything at this point...
442
+ font = self._try_available()
443
+
444
+ if not font:
445
+ # No font could be located.
446
+ return None, None
447
+
448
+ # We have our valid font.
449
+ font_path = self.full_name(font)
450
+ font = self.short_name(font)
451
+ return font, font_path
452
+
453
+ def create_linetext_node(
454
+ self, x, y, text, font=None, font_size=None, font_spacing=1.0, align="start",
455
+ ):
456
+ if font_size is None:
457
+ font_size = Length("20px")
458
+ if font_spacing is None:
459
+ font_spacing = 1
460
+ self.context.setting(str, "last_font", "")
461
+ font, font_path = self.retrieve_font(font)
462
+
463
+ if not font:
464
+ # No font could be located.
465
+ return None
466
+
467
+ # We tried everything if there is a font, set it to last_font.
468
+ self.context.last_font = font
469
+
470
+ # We have our valid font.
471
+
472
+ horizontal = True
473
+ cfont = self.cached_fontclass(font_path)
474
+ weld = False
475
+
476
+ # Render the font.
477
+ try:
478
+ path = FontPath(weld)
479
+ # print (f"Path={path}, text={remainder}, font-size={font_size}")
480
+ mytext = self.context.elements.wordlist_translate(text)
481
+ cfont.render(path, mytext, horizontal=horizontal, font_size=float(font_size), h_spacing=font_spacing, align=align)
482
+ except (AttributeError, ShxFontParseError):
483
+ # Could not parse path.
484
+ pass
485
+
486
+ tlines = mytext.split("\n")
487
+ tlabel = "Text: {mktext}"
488
+ # Create the node.
489
+ path_node = PathNode(
490
+ geometry=path.geometry,
491
+ stroke=self.context.elements.default_stroke,
492
+ stroke_width=self.context.elements.default_strokewidth,
493
+ fill=self.context.elements.default_fill,
494
+ fillrule=Fillrule.FILLRULE_NONZERO,
495
+ linejoin=Linejoin.JOIN_BEVEL,
496
+ label=tlabel,
497
+ )
498
+ path_node.matrix.post_translate(x, y)
499
+ path_node.mkfont = font
500
+ path_node.mkfontsize = float(font_size)
501
+ path_node.mkfontspacing = float(font_spacing)
502
+ path_node.mkfontweld = weld
503
+ path_node.mkalign = align
504
+ path_node.mklinegap = 1.1
505
+ path_node.mktext = text
506
+ path_node._translated_text = mytext
507
+ path_node.mkcoordx = x
508
+ path_node.mkcoordy = y
509
+ if hasattr(cfont, "line_information"):
510
+ # Store the relative start / end positions of the text lines
511
+ # for any interested party...
512
+ path_node._line_information = cfont.line_information()
513
+
514
+ return path_node
515
+
516
+ def preview_file(self, fontfile):
517
+ def create_preview_image(fontfile, bmpfile, bitmap_format):
518
+ from math import isinf
519
+
520
+ simplefont = basename(fontfile)
521
+ pattern = "The quick brown fox..."
522
+ try:
523
+ node = self.create_linetext_node(
524
+ 0, 0, pattern, font=simplefont, font_size=Length("12pt")
525
+ )
526
+ except Exception:
527
+ # We may encounter an IndexError, a ValueError or an error thrown by struct
528
+ # The latter cannot be named? So a global except...
529
+ # print (f"Node creation failed: {e}")
530
+ return False
531
+ if node is None:
532
+ return False
533
+ if node.bounds is None:
534
+ return False
535
+ make_raster = self.context.elements.lookup("render-op/make_raster")
536
+ if make_raster is None:
537
+ return False
538
+ xmin, ymin, xmax, ymax = node.bounds
539
+ if isinf(xmin):
540
+ # No bounds for selected elements
541
+ return False
542
+ width = xmax - xmin
543
+ height = ymax - ymin
544
+ dpi = 150
545
+ dots_per_units = dpi / UNITS_PER_INCH
546
+ new_width = width * dots_per_units
547
+ new_height = height * dots_per_units
548
+ new_height = max(new_height, 1)
549
+ new_width = max(new_width, 1)
550
+ try:
551
+ bitmap = make_raster(
552
+ [node],
553
+ bounds=node.bounds,
554
+ width=new_width,
555
+ height=new_height,
556
+ bitmap=True,
557
+ )
558
+ except Exception:
559
+ # print (f"Raster failed: {e}")
560
+ # Invalid path or whatever...
561
+ return False
562
+ try:
563
+ bitmap.SaveFile(bmpfile, bitmap_format)
564
+ except (OSError, RuntimeError, PermissionError, FileNotFoundError):
565
+ # print (f"Save failed: {e}")
566
+ return False
567
+ return True
568
+
569
+ bitmap = None
570
+ try:
571
+ import wx
572
+ except ImportError:
573
+ return None
574
+ base, ext = splitext(basename(fontfile))
575
+ bmpfile = join(self.font_directory, base + ".png")
576
+ if not exists(bmpfile):
577
+ __ = create_preview_image(fontfile, bmpfile, wx.BITMAP_TYPE_PNG)
578
+ if exists(bmpfile):
579
+ bitmap = wx.Bitmap()
580
+ bitmap.LoadFile(bmpfile, wx.BITMAP_TYPE_PNG)
581
+ return bitmap
582
+
583
+ def available_fonts(self):
584
+ if self._available_fonts is not None:
585
+ return self._available_fonts
586
+
587
+ # Return a tuple of two values
588
+ # from time import perf_counter
589
+
590
+ _ = self.context.kernel.translation
591
+ # t0 = perf_counter()
592
+ self._available_fonts = []
593
+
594
+ cache = self.cache_file
595
+ if exists(cache):
596
+ try:
597
+ with open(cache, "r", encoding="utf-8") as f:
598
+ while True:
599
+ line = f.readline()
600
+ if not line:
601
+ break
602
+ line = line.strip()
603
+ parts = line.split("|")
604
+ if len(parts) > 4:
605
+ flag = False
606
+ if parts[4].lower in ("true", "1"):
607
+ flag = True
608
+ self._available_fonts.append(
609
+ (
610
+ parts[0],
611
+ parts[1],
612
+ parts[2],
613
+ parts[3],
614
+ flag,
615
+ )
616
+ )
617
+ except (OSError, FileNotFoundError, PermissionError):
618
+ self._available_fonts = []
619
+ if len(self._available_fonts):
620
+ # t1 = perf_counter()
621
+ # print (f"Cached, took {t1 - t0:.2f}sec")
622
+ return self._available_fonts
623
+
624
+ busy = self.context.kernel.busyinfo
625
+ busy.start(msg=_("Reading system fonts..."))
626
+ directories = []
627
+ directories.append(self.font_directory)
628
+ for d in self.context.system_font_directories:
629
+ directories.append(d)
630
+ # Walk through all folders recursively
631
+ found = dict()
632
+ font_types = self.fonts_registered
633
+ filelist = []
634
+ for idx, fontpath in enumerate(directories):
635
+ busy.change(msg=fontpath, keep=1)
636
+ busy.show()
637
+
638
+ systemfont = idx != 0
639
+ for p in font_types:
640
+ found[p] = 0
641
+ try:
642
+ for root, dirs, files in os.walk(fontpath):
643
+ for filename in files:
644
+ if not filename:
645
+ continue
646
+ short = basename(filename)
647
+ if not short:
648
+ continue
649
+ full_name = join(root, filename)
650
+ test = filename.lower()
651
+ for p in font_types:
652
+ if test.endswith(p):
653
+ if filename not in filelist:
654
+ font_family = ""
655
+ font_subfamily = ""
656
+ face_name = short
657
+ info = self.get_font_information(full_name)
658
+ if info:
659
+ # Tuple with font_family, font_subfamily, face_name
660
+ font_family, font_subfamily, face_name = info
661
+ else:
662
+ entry = font_types[p]
663
+ font_family = entry[0]
664
+ if face_name is None:
665
+ face_name = short
666
+ self._available_fonts.append(
667
+ (
668
+ str(full_name),
669
+ str(face_name),
670
+ font_family,
671
+ font_subfamily,
672
+ systemfont,
673
+ )
674
+ )
675
+ # print (face_name, font_family, font_subfamily, full_name)
676
+ filelist.append(filename)
677
+ found[p] += 1
678
+ break
679
+ except (OSError, FileNotFoundError, PermissionError):
680
+ continue
681
+ # for key, value in found.items():
682
+ # print(f"{key}: {value} - {fontpath}")
683
+ self._available_fonts.sort(key=lambda e: e[1])
684
+ try:
685
+ with open(cache, "w", encoding="utf-8") as f:
686
+ for p in self._available_fonts:
687
+ f.write(f"{p[0]}|{p[1]}|{p[2]}|{p[3]}|{p[4]}\n")
688
+ except (OSError, FileNotFoundError, PermissionError):
689
+ pass
690
+
691
+ busy.end()
692
+ # t1 = perf_counter()
693
+ # print (f"Ready, took {t1 - t0:.2f}sec")
694
+ return self._available_fonts
695
+
696
+
697
+ def plugin(kernel, lifecycle):
698
+ if lifecycle == "register":
699
+ _ = kernel.translation
700
+ context = kernel.root
701
+ # Generate setting for system-directories
702
+ directories = []
703
+ systype = platform.system()
704
+ if systype == "Windows":
705
+ if "WINDIR" in os.environ:
706
+ windir = os.environ["WINDIR"]
707
+ else:
708
+ windir = "c:\\windows"
709
+ directories.append(join(windir, "Fonts"))
710
+ if "LOCALAPPDATA" in os.environ:
711
+ appdir = os.environ["LOCALAPPDATA"]
712
+ directories.append(join(appdir, "Microsoft\\Windows\\Fonts"))
713
+ elif systype == "Linux":
714
+ directories.append("/usr/share/fonts")
715
+ directories.append("/usr/local/share/fonts")
716
+ directories.append("~/.local/share/fonts")
717
+ elif systype == "Darwin":
718
+ directories.append("/System/Library/Fonts")
719
+ directories.append("/Library/Fonts")
720
+ directories.append("~/Library/Fonts")
721
+ choices = [
722
+ {
723
+ "attr": "system_font_directories",
724
+ "object": context,
725
+ "page": "_95_Fonts",
726
+ "section": "_95_System font locations",
727
+ "default": directories,
728
+ "type": list,
729
+ "columns": [
730
+ {
731
+ "attr": "directory",
732
+ "type": str,
733
+ "label": _("Directory"),
734
+ "width": -1,
735
+ "editable": True,
736
+ },
737
+ ],
738
+ "label": "_00_",
739
+ "style": "chart",
740
+ "primary": "directory",
741
+ "allow_deletion": True,
742
+ "allow_duplication": True,
743
+ "tip": _("Places where MeerK40t will look for fonts."),
744
+ },
745
+ ]
746
+ kernel.register_choices("preferences", choices)
747
+ context.fonts = Meerk40tFonts(context=context)
748
+
749
+ # Register update routine for linetext
750
+ kernel.register("path_updater/linetext", context.fonts.update)
751
+ for idx, attrib in enumerate(
752
+ ("mkfontsize", "mkfontweld", "mkfontspacing", "mklinegap", "mkalign")
753
+ ):
754
+ kernel.register(f"registered_mk_svg_parameters/font{idx}", attrib)
755
+
756
+
757
+ @context.console_argument("x", type=Length, help=_("X-Coordinate"))
758
+ @context.console_argument("y", type=Length, help=_("Y-Coordinate"))
759
+ @context.console_argument("text", type=str, help=_("Text to render"))
760
+ @context.console_option("font", "f", type=str, help=_("Font file."))
761
+ @context.console_option(
762
+ "font_size", "s", type=Length, default="20px", help=_("Font size")
763
+ )
764
+ @context.console_option(
765
+ "font_spacing",
766
+ "g",
767
+ type=float,
768
+ default=1,
769
+ help=_("Character spacing factor"),
770
+ )
771
+ @context.console_command(
772
+ "linetext", help=_("linetext <font> <font_size> <text>")
773
+ )
774
+ def linetext(
775
+ command,
776
+ channel,
777
+ _,
778
+ x=None,
779
+ y=None,
780
+ text=None,
781
+ font=None,
782
+ font_size=None,
783
+ font_spacing=None,
784
+ remainder=None,
785
+ **kwargs,
786
+ ):
787
+ if x is None or y is None or text is None:
788
+ channel(_("linetext <x> <y> <text> - please provide all required arguments"))
789
+ registered_fonts = context.fonts.available_fonts()
790
+ for item in registered_fonts:
791
+ channel(f"{item[1]} ({item[0]})")
792
+ return
793
+ try:
794
+ x = float(Length(x))
795
+ y = float(Length(y))
796
+ except ValueError:
797
+ channel(_("Invalid coordinates"))
798
+ return
799
+ if text is None or text == "":
800
+ channel(_("No text given."))
801
+ return
802
+
803
+
804
+ if font_spacing is None:
805
+ font_spacing = 1
806
+
807
+ context.setting(str, "last_font", None)
808
+ if font is None:
809
+ font = context.last_font
810
+
811
+ font_name, font_path = context.fonts.retrieve_font(font)
812
+ if font_name is None:
813
+ channel(f"Could not find a valid font file for '{font}'")
814
+ registered_fonts = context.fonts.available_fonts()
815
+ for item in registered_fonts:
816
+ channel(f"{item[1]} ({item[0]})")
817
+ return
818
+
819
+ channel(f"Will use font '{font_name}' ({font_path})")
820
+
821
+ path_node = context.fonts.create_linetext_node(
822
+ x, y, text, font_path, font_size, font_spacing
823
+ )
824
+ # path_node = PathNode(
825
+ # path=path.path,
826
+ # matrix=Matrix.translate(0, float(font_size)),
827
+ # stroke=Color("black"),
828
+ # )
829
+ context.elements.elem_branch.add_node(path_node)
830
+ if context.elements.classify_new:
831
+ context.elements.classify([path_node])
832
+ context.elements.set_emphasis([path_node])
833
+
834
+ context.signal("element_added", path_node)
835
+ context.signal("refresh_scene", "Scene")