FlatCAMDraw.py 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341
  1. from PyQt4 import QtGui, QtCore, Qt
  2. import FlatCAMApp
  3. from camlib import *
  4. from FlatCAMTool import FlatCAMTool
  5. from ObjectUI import LengthEntry
  6. from shapely.geometry import Polygon, LineString, Point, LinearRing
  7. from shapely.geometry import MultiPoint, MultiPolygon
  8. from shapely.geometry import box as shply_box
  9. from shapely.ops import cascaded_union, unary_union
  10. import shapely.affinity as affinity
  11. from shapely.wkt import loads as sloads
  12. from shapely.wkt import dumps as sdumps
  13. from shapely.geometry.base import BaseGeometry
  14. from numpy import arctan2, Inf, array, sqrt, pi, ceil, sin, cos, sign, dot
  15. from numpy.linalg import solve
  16. #from mpl_toolkits.axes_grid.anchored_artists import AnchoredDrawingArea
  17. from rtree import index as rtindex
  18. class BufferSelectionTool(FlatCAMTool):
  19. """
  20. Simple input for buffer distance.
  21. """
  22. toolName = "Buffer Selection"
  23. def __init__(self, app, fcdraw):
  24. FlatCAMTool.__init__(self, app)
  25. self.fcdraw = fcdraw
  26. ## Title
  27. title_label = QtGui.QLabel("<font size=4><b>%s</b></font>" % self.toolName)
  28. self.layout.addWidget(title_label)
  29. ## Form Layout
  30. form_layout = QtGui.QFormLayout()
  31. self.layout.addLayout(form_layout)
  32. ## Buffer distance
  33. self.buffer_distance_entry = LengthEntry()
  34. form_layout.addRow("Buffer distance:", self.buffer_distance_entry)
  35. ## Buttons
  36. hlay = QtGui.QHBoxLayout()
  37. self.layout.addLayout(hlay)
  38. hlay.addStretch()
  39. self.buffer_button = QtGui.QPushButton("Buffer")
  40. hlay.addWidget(self.buffer_button)
  41. self.layout.addStretch()
  42. ## Signals
  43. self.buffer_button.clicked.connect(self.on_buffer)
  44. def on_buffer(self):
  45. buffer_distance = self.buffer_distance_entry.get_value()
  46. self.fcdraw.buffer(buffer_distance)
  47. class DrawToolShape(object):
  48. """
  49. Encapsulates "shapes" under a common class.
  50. """
  51. @staticmethod
  52. def get_pts(o):
  53. """
  54. Returns a list of all points in the object, where
  55. the object can be a Polygon, Not a polygon, or a list
  56. of such. Search is done recursively.
  57. :param: geometric object
  58. :return: List of points
  59. :rtype: list
  60. """
  61. pts = []
  62. ## Iterable: descend into each item.
  63. try:
  64. for subo in o:
  65. pts += DrawToolShape.get_pts(subo)
  66. ## Non-iterable
  67. except TypeError:
  68. ## DrawToolShape: descend into .geo.
  69. if isinstance(o, DrawToolShape):
  70. pts += DrawToolShape.get_pts(o.geo)
  71. ## Descend into .exerior and .interiors
  72. elif type(o) == Polygon:
  73. pts += DrawToolShape.get_pts(o.exterior)
  74. for i in o.interiors:
  75. pts += DrawToolShape.get_pts(i)
  76. ## Has .coords: list them.
  77. else:
  78. pts += list(o.coords)
  79. return pts
  80. def __init__(self, geo=[]):
  81. # Shapely type or list of such
  82. self.geo = geo
  83. self.utility = False
  84. def get_all_points(self):
  85. return DrawToolShape.get_pts(self)
  86. class DrawToolUtilityShape(DrawToolShape):
  87. """
  88. Utility shapes are temporary geometry in the editor
  89. to assist in the creation of shapes. For example it
  90. will show the outline of a rectangle from the first
  91. point to the current mouse pointer before the second
  92. point is clicked and the final geometry is created.
  93. """
  94. def __init__(self, geo=[]):
  95. super(DrawToolUtilityShape, self).__init__(geo=geo)
  96. self.utility = True
  97. class DrawTool(object):
  98. """
  99. Abstract Class representing a tool in the drawing
  100. program. Can generate geometry, including temporary
  101. utility geometry that is updated on user clicks
  102. and mouse motion.
  103. """
  104. def __init__(self, draw_app):
  105. self.draw_app = draw_app
  106. self.complete = False
  107. self.start_msg = "Click on 1st point..."
  108. self.points = []
  109. self.geometry = None # DrawToolShape or None
  110. def click(self, point):
  111. """
  112. :param point: [x, y] Coordinate pair.
  113. """
  114. return ""
  115. def on_key(self, key):
  116. return None
  117. def utility_geometry(self, data=None):
  118. return None
  119. class FCShapeTool(DrawTool):
  120. """
  121. Abstarct class for tools that create a shape.
  122. """
  123. def __init__(self, draw_app):
  124. DrawTool.__init__(self, draw_app)
  125. def make(self):
  126. pass
  127. class FCCircle(FCShapeTool):
  128. """
  129. Resulting type: Polygon
  130. """
  131. def __init__(self, draw_app):
  132. DrawTool.__init__(self, draw_app)
  133. self.start_msg = "Click on CENTER ..."
  134. def click(self, point):
  135. self.points.append(point)
  136. if len(self.points) == 1:
  137. return "Click on perimeter to complete ..."
  138. if len(self.points) == 2:
  139. self.make()
  140. return "Done."
  141. return ""
  142. def utility_geometry(self, data=None):
  143. if len(self.points) == 1:
  144. p1 = self.points[0]
  145. p2 = data
  146. radius = sqrt((p1[0] - p2[0]) ** 2 + (p1[1] - p2[1]) ** 2)
  147. return DrawToolUtilityShape(Point(p1).buffer(radius))
  148. return None
  149. def make(self):
  150. p1 = self.points[0]
  151. p2 = self.points[1]
  152. radius = distance(p1, p2)
  153. self.geometry = DrawToolShape(Point(p1).buffer(radius))
  154. self.complete = True
  155. class FCArc(FCShapeTool):
  156. def __init__(self, draw_app):
  157. DrawTool.__init__(self, draw_app)
  158. self.start_msg = "Click on CENTER ..."
  159. # Direction of rotation between point 1 and 2.
  160. # 'cw' or 'ccw'. Switch direction by hitting the
  161. # 'o' key.
  162. self.direction = "cw"
  163. # Mode
  164. # C12 = Center, p1, p2
  165. # 12C = p1, p2, Center
  166. # 132 = p1, p3, p2
  167. self.mode = "c12" # Center, p1, p2
  168. self.steps_per_circ = 55
  169. def click(self, point):
  170. self.points.append(point)
  171. if len(self.points) == 1:
  172. return "Click on 1st point ..."
  173. if len(self.points) == 2:
  174. return "Click on 2nd point to complete ..."
  175. if len(self.points) == 3:
  176. self.make()
  177. return "Done."
  178. return ""
  179. def on_key(self, key):
  180. if key == 'o':
  181. self.direction = 'cw' if self.direction == 'ccw' else 'ccw'
  182. return 'Direction: ' + self.direction.upper()
  183. if key == 'p':
  184. if self.mode == 'c12':
  185. self.mode = '12c'
  186. elif self.mode == '12c':
  187. self.mode = '132'
  188. else:
  189. self.mode = 'c12'
  190. return 'Mode: ' + self.mode
  191. def utility_geometry(self, data=None):
  192. if len(self.points) == 1: # Show the radius
  193. center = self.points[0]
  194. p1 = data
  195. return DrawToolUtilityShape(LineString([center, p1]))
  196. if len(self.points) == 2: # Show the arc
  197. if self.mode == 'c12':
  198. center = self.points[0]
  199. p1 = self.points[1]
  200. p2 = data
  201. radius = sqrt((center[0] - p1[0]) ** 2 + (center[1] - p1[1]) ** 2)
  202. startangle = arctan2(p1[1] - center[1], p1[0] - center[0])
  203. stopangle = arctan2(p2[1] - center[1], p2[0] - center[0])
  204. return DrawToolUtilityShape([LineString(arc(center, radius, startangle, stopangle,
  205. self.direction, self.steps_per_circ)),
  206. Point(center)])
  207. elif self.mode == '132':
  208. p1 = array(self.points[0])
  209. p3 = array(self.points[1])
  210. p2 = array(data)
  211. center, radius, t = three_point_circle(p1, p2, p3)
  212. direction = 'cw' if sign(t) > 0 else 'ccw'
  213. startangle = arctan2(p1[1] - center[1], p1[0] - center[0])
  214. stopangle = arctan2(p3[1] - center[1], p3[0] - center[0])
  215. return DrawToolUtilityShape([LineString(arc(center, radius, startangle, stopangle,
  216. direction, self.steps_per_circ)),
  217. Point(center), Point(p1), Point(p3)])
  218. else: # '12c'
  219. p1 = array(self.points[0])
  220. p2 = array(self.points[1])
  221. # Midpoint
  222. a = (p1 + p2) / 2.0
  223. # Parallel vector
  224. c = p2 - p1
  225. # Perpendicular vector
  226. b = dot(c, array([[0, -1], [1, 0]], dtype=float32))
  227. b /= norm(b)
  228. # Distance
  229. t = distance(data, a)
  230. # Which side? Cross product with c.
  231. # cross(M-A, B-A), where line is AB and M is test point.
  232. side = (data[0] - p1[0]) * c[1] - (data[1] - p1[1]) * c[0]
  233. t *= sign(side)
  234. # Center = a + bt
  235. center = a + b * t
  236. radius = norm(center - p1)
  237. startangle = arctan2(p1[1] - center[1], p1[0] - center[0])
  238. stopangle = arctan2(p2[1] - center[1], p2[0] - center[0])
  239. return DrawToolUtilityShape([LineString(arc(center, radius, startangle, stopangle,
  240. self.direction, self.steps_per_circ)),
  241. Point(center)])
  242. return None
  243. def make(self):
  244. if self.mode == 'c12':
  245. center = self.points[0]
  246. p1 = self.points[1]
  247. p2 = self.points[2]
  248. radius = distance(center, p1)
  249. startangle = arctan2(p1[1] - center[1], p1[0] - center[0])
  250. stopangle = arctan2(p2[1] - center[1], p2[0] - center[0])
  251. self.geometry = DrawToolShape(LineString(arc(center, radius, startangle, stopangle,
  252. self.direction, self.steps_per_circ)))
  253. elif self.mode == '132':
  254. p1 = array(self.points[0])
  255. p3 = array(self.points[1])
  256. p2 = array(self.points[2])
  257. center, radius, t = three_point_circle(p1, p2, p3)
  258. direction = 'cw' if sign(t) > 0 else 'ccw'
  259. startangle = arctan2(p1[1] - center[1], p1[0] - center[0])
  260. stopangle = arctan2(p3[1] - center[1], p3[0] - center[0])
  261. self.geometry = DrawToolShape(LineString(arc(center, radius, startangle, stopangle,
  262. direction, self.steps_per_circ)))
  263. else: # self.mode == '12c'
  264. p1 = array(self.points[0])
  265. p2 = array(self.points[1])
  266. pc = array(self.points[2])
  267. # Midpoint
  268. a = (p1 + p2) / 2.0
  269. # Parallel vector
  270. c = p2 - p1
  271. # Perpendicular vector
  272. b = dot(c, array([[0, -1], [1, 0]], dtype=float32))
  273. b /= norm(b)
  274. # Distance
  275. t = distance(pc, a)
  276. # Which side? Cross product with c.
  277. # cross(M-A, B-A), where line is AB and M is test point.
  278. side = (pc[0] - p1[0]) * c[1] - (pc[1] - p1[1]) * c[0]
  279. t *= sign(side)
  280. # Center = a + bt
  281. center = a + b * t
  282. radius = norm(center - p1)
  283. startangle = arctan2(p1[1] - center[1], p1[0] - center[0])
  284. stopangle = arctan2(p2[1] - center[1], p2[0] - center[0])
  285. self.geometry = DrawToolShape(LineString(arc(center, radius, startangle, stopangle,
  286. self.direction, self.steps_per_circ)))
  287. self.complete = True
  288. class FCRectangle(FCShapeTool):
  289. """
  290. Resulting type: Polygon
  291. """
  292. def __init__(self, draw_app):
  293. DrawTool.__init__(self, draw_app)
  294. self.start_msg = "Click on 1st corner ..."
  295. def click(self, point):
  296. self.points.append(point)
  297. if len(self.points) == 1:
  298. return "Click on opposite corner to complete ..."
  299. if len(self.points) == 2:
  300. self.make()
  301. return "Done."
  302. return ""
  303. def utility_geometry(self, data=None):
  304. if len(self.points) == 1:
  305. p1 = self.points[0]
  306. p2 = data
  307. return DrawToolUtilityShape(LinearRing([p1, (p2[0], p1[1]), p2, (p1[0], p2[1])]))
  308. return None
  309. def make(self):
  310. p1 = self.points[0]
  311. p2 = self.points[1]
  312. #self.geometry = LinearRing([p1, (p2[0], p1[1]), p2, (p1[0], p2[1])])
  313. self.geometry = DrawToolShape(Polygon([p1, (p2[0], p1[1]), p2, (p1[0], p2[1])]))
  314. self.complete = True
  315. class FCPolygon(FCShapeTool):
  316. """
  317. Resulting type: Polygon
  318. """
  319. def __init__(self, draw_app):
  320. DrawTool.__init__(self, draw_app)
  321. self.start_msg = "Click on 1st point ..."
  322. def click(self, point):
  323. self.points.append(point)
  324. if len(self.points) > 0:
  325. return "Click on next point or hit SPACE to complete ..."
  326. return ""
  327. def utility_geometry(self, data=None):
  328. if len(self.points) == 1:
  329. temp_points = [x for x in self.points]
  330. temp_points.append(data)
  331. return DrawToolUtilityShape(LineString(temp_points))
  332. if len(self.points) > 1:
  333. temp_points = [x for x in self.points]
  334. temp_points.append(data)
  335. return DrawToolUtilityShape(LinearRing(temp_points))
  336. return None
  337. def make(self):
  338. # self.geometry = LinearRing(self.points)
  339. self.geometry = DrawToolShape(Polygon(self.points))
  340. self.complete = True
  341. class FCPath(FCPolygon):
  342. """
  343. Resulting type: LineString
  344. """
  345. def make(self):
  346. self.geometry = DrawToolShape(LineString(self.points))
  347. self.complete = True
  348. def utility_geometry(self, data=None):
  349. if len(self.points) > 1:
  350. temp_points = [x for x in self.points]
  351. temp_points.append(data)
  352. return DrawToolUtilityShape(LineString(temp_points))
  353. return None
  354. class FCSelect(DrawTool):
  355. def __init__(self, draw_app):
  356. DrawTool.__init__(self, draw_app)
  357. self.storage = self.draw_app.storage
  358. #self.shape_buffer = self.draw_app.shape_buffer
  359. self.selected = self.draw_app.selected
  360. self.start_msg = "Click on geometry to select"
  361. def click(self, point):
  362. try:
  363. _, closest_shape = self.storage.nearest(point)
  364. except StopIteration:
  365. return ""
  366. if self.draw_app.key != 'control':
  367. self.draw_app.selected = []
  368. self.draw_app.set_selected(closest_shape)
  369. self.draw_app.app.log.debug("Selected shape containing: " + str(closest_shape.geo))
  370. return ""
  371. class FCMove(FCShapeTool):
  372. def __init__(self, draw_app):
  373. FCShapeTool.__init__(self, draw_app)
  374. #self.shape_buffer = self.draw_app.shape_buffer
  375. self.origin = None
  376. self.destination = None
  377. self.start_msg = "Click on reference point."
  378. def set_origin(self, origin):
  379. self.origin = origin
  380. def click(self, point):
  381. if len(self.draw_app.get_selected()) == 0:
  382. return "Nothing to move."
  383. if self.origin is None:
  384. self.set_origin(point)
  385. return "Click on final location."
  386. else:
  387. self.destination = point
  388. self.make()
  389. return "Done."
  390. def make(self):
  391. # Create new geometry
  392. dx = self.destination[0] - self.origin[0]
  393. dy = self.destination[1] - self.origin[1]
  394. self.geometry = [DrawToolShape(affinity.translate(geom.geo, xoff=dx, yoff=dy))
  395. for geom in self.draw_app.get_selected()]
  396. # Delete old
  397. self.draw_app.delete_selected()
  398. # # Select the new
  399. # for g in self.geometry:
  400. # # Note that g is not in the app's buffer yet!
  401. # self.draw_app.set_selected(g)
  402. self.complete = True
  403. def utility_geometry(self, data=None):
  404. """
  405. Temporary geometry on screen while using this tool.
  406. :param data:
  407. :return:
  408. """
  409. if self.origin is None:
  410. return None
  411. if len(self.draw_app.get_selected()) == 0:
  412. return None
  413. dx = data[0] - self.origin[0]
  414. dy = data[1] - self.origin[1]
  415. return DrawToolUtilityShape([affinity.translate(geom.geo, xoff=dx, yoff=dy)
  416. for geom in self.draw_app.get_selected()])
  417. class FCCopy(FCMove):
  418. def make(self):
  419. # Create new geometry
  420. dx = self.destination[0] - self.origin[0]
  421. dy = self.destination[1] - self.origin[1]
  422. self.geometry = [DrawToolShape(affinity.translate(geom.geo, xoff=dx, yoff=dy))
  423. for geom in self.draw_app.get_selected()]
  424. self.complete = True
  425. ########################
  426. ### Main Application ###
  427. ########################
  428. class FlatCAMDraw(QtCore.QObject):
  429. def __init__(self, app, disabled=False):
  430. assert isinstance(app, FlatCAMApp.App)
  431. super(FlatCAMDraw, self).__init__()
  432. self.app = app
  433. self.canvas = app.plotcanvas
  434. self.axes = self.canvas.new_axes("draw")
  435. ### Drawing Toolbar ###
  436. self.drawing_toolbar = QtGui.QToolBar()
  437. self.drawing_toolbar.setDisabled(disabled)
  438. self.app.ui.addToolBar(self.drawing_toolbar)
  439. self.select_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/pointer32.png'), "Select 'Esc'")
  440. self.add_circle_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/circle32.png'), 'Add Circle')
  441. self.add_arc_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/arc32.png'), 'Add Arc')
  442. self.add_rectangle_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/rectangle32.png'), 'Add Rectangle')
  443. self.add_polygon_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/polygon32.png'), 'Add Polygon')
  444. self.add_path_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/path32.png'), 'Add Path')
  445. self.union_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/union32.png'), 'Polygon Union')
  446. self.intersection_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/intersection32.png'), 'Polygon Intersection')
  447. self.subtract_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/subtract32.png'), 'Polygon Subtraction')
  448. self.cutpath_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/cutpath32.png'), 'Cut Path')
  449. self.move_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/move32.png'), "Move Objects 'm'")
  450. self.copy_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/copy32.png'), "Copy Objects 'c'")
  451. self.delete_btn = self.drawing_toolbar.addAction(QtGui.QIcon('share/deleteshape32.png'), "Delete Shape '-'")
  452. ### Snap Toolbar ###
  453. self.snap_toolbar = QtGui.QToolBar()
  454. self.grid_snap_btn = self.snap_toolbar.addAction(QtGui.QIcon('share/grid32.png'), 'Snap to grid')
  455. self.grid_gap_x_entry = QtGui.QLineEdit()
  456. self.grid_gap_x_entry.setMaximumWidth(70)
  457. self.grid_gap_x_entry.setToolTip("Grid X distance")
  458. self.snap_toolbar.addWidget(self.grid_gap_x_entry)
  459. self.grid_gap_y_entry = QtGui.QLineEdit()
  460. self.grid_gap_y_entry.setMaximumWidth(70)
  461. self.grid_gap_y_entry.setToolTip("Grid Y distante")
  462. self.snap_toolbar.addWidget(self.grid_gap_y_entry)
  463. self.corner_snap_btn = self.snap_toolbar.addAction(QtGui.QIcon('share/corner32.png'), 'Snap to corner')
  464. self.snap_max_dist_entry = QtGui.QLineEdit()
  465. self.snap_max_dist_entry.setMaximumWidth(70)
  466. self.snap_max_dist_entry.setToolTip("Max. magnet distance")
  467. self.snap_toolbar.addWidget(self.snap_max_dist_entry)
  468. self.snap_toolbar.setDisabled(disabled)
  469. self.app.ui.addToolBar(self.snap_toolbar)
  470. ### Application menu ###
  471. self.menu = QtGui.QMenu("Drawing")
  472. self.app.ui.menu.insertMenu(self.app.ui.menutoolaction, self.menu)
  473. # self.select_menuitem = self.menu.addAction(QtGui.QIcon('share/pointer16.png'), "Select 'Esc'")
  474. # self.add_circle_menuitem = self.menu.addAction(QtGui.QIcon('share/circle16.png'), 'Add Circle')
  475. # self.add_arc_menuitem = self.menu.addAction(QtGui.QIcon('share/arc16.png'), 'Add Arc')
  476. # self.add_rectangle_menuitem = self.menu.addAction(QtGui.QIcon('share/rectangle16.png'), 'Add Rectangle')
  477. # self.add_polygon_menuitem = self.menu.addAction(QtGui.QIcon('share/polygon16.png'), 'Add Polygon')
  478. # self.add_path_menuitem = self.menu.addAction(QtGui.QIcon('share/path16.png'), 'Add Path')
  479. self.union_menuitem = self.menu.addAction(QtGui.QIcon('share/union16.png'), 'Polygon Union')
  480. self.intersection_menuitem = self.menu.addAction(QtGui.QIcon('share/intersection16.png'), 'Polygon Intersection')
  481. # self.subtract_menuitem = self.menu.addAction(QtGui.QIcon('share/subtract16.png'), 'Polygon Subtraction')
  482. self.cutpath_menuitem = self.menu.addAction(QtGui.QIcon('share/cutpath16.png'), 'Cut Path')
  483. # self.move_menuitem = self.menu.addAction(QtGui.QIcon('share/move16.png'), "Move Objects 'm'")
  484. # self.copy_menuitem = self.menu.addAction(QtGui.QIcon('share/copy16.png'), "Copy Objects 'c'")
  485. self.delete_menuitem = self.menu.addAction(QtGui.QIcon('share/deleteshape16.png'), "Delete Shape '-'")
  486. self.buffer_menuitem = self.menu.addAction(QtGui.QIcon('share/buffer16.png'), "Buffer selection 'b'")
  487. self.menu.addSeparator()
  488. self.buffer_menuitem.triggered.connect(self.on_buffer_tool)
  489. self.delete_menuitem.triggered.connect(self.on_delete_btn)
  490. self.union_menuitem.triggered.connect(self.union)
  491. self.intersection_menuitem.triggered.connect(self.intersection)
  492. self.cutpath_menuitem.triggered.connect(self.cutpath)
  493. ### Event handlers ###
  494. # Connection ids for Matplotlib
  495. self.cid_canvas_click = None
  496. self.cid_canvas_move = None
  497. self.cid_canvas_key = None
  498. self.cid_canvas_key_release = None
  499. # Connect the canvas
  500. #self.connect_canvas_event_handlers()
  501. self.union_btn.triggered.connect(self.union)
  502. self.intersection_btn.triggered.connect(self.intersection)
  503. self.subtract_btn.triggered.connect(self.subtract)
  504. self.cutpath_btn.triggered.connect(self.cutpath)
  505. self.delete_btn.triggered.connect(self.on_delete_btn)
  506. ## Toolbar events and properties
  507. self.tools = {
  508. "select": {"button": self.select_btn,
  509. "constructor": FCSelect},
  510. "circle": {"button": self.add_circle_btn,
  511. "constructor": FCCircle},
  512. "arc": {"button": self.add_arc_btn,
  513. "constructor": FCArc},
  514. "rectangle": {"button": self.add_rectangle_btn,
  515. "constructor": FCRectangle},
  516. "polygon": {"button": self.add_polygon_btn,
  517. "constructor": FCPolygon},
  518. "path": {"button": self.add_path_btn,
  519. "constructor": FCPath},
  520. "move": {"button": self.move_btn,
  521. "constructor": FCMove},
  522. "copy": {"button": self.copy_btn,
  523. "constructor": FCCopy}
  524. }
  525. ### Data
  526. self.active_tool = None
  527. self.storage = FlatCAMDraw.make_storage()
  528. self.utility = []
  529. ## List of selected shapes.
  530. self.selected = []
  531. self.move_timer = QtCore.QTimer()
  532. self.move_timer.setSingleShot(True)
  533. self.key = None # Currently pressed key
  534. def make_callback(thetool):
  535. def f():
  536. self.on_tool_select(thetool)
  537. return f
  538. for tool in self.tools:
  539. self.tools[tool]["button"].triggered.connect(make_callback(tool)) # Events
  540. self.tools[tool]["button"].setCheckable(True) # Checkable
  541. # for snap_tool in [self.grid_snap_btn, self.corner_snap_btn]:
  542. # snap_tool.triggered.connect(lambda: self.toolbar_tool_toggle("grid_snap"))
  543. # snap_tool.setCheckable(True)
  544. self.grid_snap_btn.setCheckable(True)
  545. self.grid_snap_btn.triggered.connect(lambda: self.toolbar_tool_toggle("grid_snap"))
  546. self.corner_snap_btn.setCheckable(True)
  547. self.corner_snap_btn.triggered.connect(lambda: self.toolbar_tool_toggle("corner_snap"))
  548. self.options = {
  549. "snap-x": 0.1,
  550. "snap-y": 0.1,
  551. "snap_max": 0.05,
  552. "grid_snap": False,
  553. "corner_snap": False,
  554. }
  555. self.grid_gap_x_entry.setText(str(self.options["snap-x"]))
  556. self.grid_gap_y_entry.setText(str(self.options["snap-y"]))
  557. self.snap_max_dist_entry.setText(str(self.options["snap_max"]))
  558. self.rtree_index = rtindex.Index()
  559. def entry2option(option, entry):
  560. self.options[option] = float(entry.text())
  561. self.grid_gap_x_entry.setValidator(QtGui.QDoubleValidator())
  562. self.grid_gap_x_entry.editingFinished.connect(lambda: entry2option("snap-x", self.grid_gap_x_entry))
  563. self.grid_gap_y_entry.setValidator(QtGui.QDoubleValidator())
  564. self.grid_gap_y_entry.editingFinished.connect(lambda: entry2option("snap-y", self.grid_gap_y_entry))
  565. self.snap_max_dist_entry.setValidator(QtGui.QDoubleValidator())
  566. self.snap_max_dist_entry.editingFinished.connect(lambda: entry2option("snap_max", self.snap_max_dist_entry))
  567. def activate(self):
  568. pass
  569. def connect_canvas_event_handlers(self):
  570. ## Canvas events
  571. self.cid_canvas_click = self.canvas.mpl_connect('button_press_event', self.on_canvas_click)
  572. self.cid_canvas_move = self.canvas.mpl_connect('motion_notify_event', self.on_canvas_move)
  573. self.cid_canvas_key = self.canvas.mpl_connect('key_press_event', self.on_canvas_key)
  574. self.cid_canvas_key_release = self.canvas.mpl_connect('key_release_event', self.on_canvas_key_release)
  575. def disconnect_canvas_event_handlers(self):
  576. self.canvas.mpl_disconnect(self.cid_canvas_click)
  577. self.canvas.mpl_disconnect(self.cid_canvas_move)
  578. self.canvas.mpl_disconnect(self.cid_canvas_key)
  579. self.canvas.mpl_disconnect(self.cid_canvas_key_release)
  580. def add_shape(self, shape):
  581. """
  582. Adds a shape to the shape storage.
  583. :param shape: Shape to be added.
  584. :type shape: DrawToolShape
  585. :return: None
  586. """
  587. # List of DrawToolShape?
  588. if isinstance(shape, list):
  589. for subshape in shape:
  590. self.add_shape(subshape)
  591. return
  592. assert isinstance(shape, DrawToolShape)
  593. assert shape.geo is not None
  594. assert (isinstance(shape.geo, list) and len(shape.geo) > 0) or not isinstance(shape.geo, list)
  595. if isinstance(shape, DrawToolUtilityShape):
  596. self.utility.append(shape)
  597. else:
  598. self.storage.insert(shape)
  599. def deactivate(self):
  600. self.disconnect_canvas_event_handlers()
  601. self.clear()
  602. self.drawing_toolbar.setDisabled(True)
  603. self.snap_toolbar.setDisabled(True) # TODO: Combine and move into tool
  604. def delete_utility_geometry(self):
  605. #for_deletion = [shape for shape in self.shape_buffer if shape.utility]
  606. #for_deletion = [shape for shape in self.storage.get_objects() if shape.utility]
  607. for_deletion = [shape for shape in self.utility]
  608. for shape in for_deletion:
  609. self.delete_shape(shape)
  610. def cutpath(self):
  611. selected = self.get_selected()
  612. tools = selected[1:]
  613. toolgeo = cascaded_union([shp.geo for shp in tools])
  614. target = selected[0]
  615. if type(target.geo) == Polygon:
  616. for ring in poly2rings(target.geo):
  617. self.add_shape(DrawToolShape(ring.difference(toolgeo)))
  618. self.delete_shape(target)
  619. elif type(target.geo) == LineString or type(target.geo) == LinearRing:
  620. self.add_shape(DrawToolShape(target.geo.difference(toolgeo)))
  621. self.delete_shape(target)
  622. else:
  623. self.app.log.warning("Not implemented.")
  624. self.replot()
  625. def toolbar_tool_toggle(self, key):
  626. self.options[key] = self.sender().isChecked()
  627. def clear(self):
  628. self.active_tool = None
  629. #self.shape_buffer = []
  630. self.selected = []
  631. self.storage = FlatCAMDraw.make_storage()
  632. self.replot()
  633. def edit_fcgeometry(self, fcgeometry):
  634. """
  635. Imports the geometry from the given FlatCAM Geometry object
  636. into the editor.
  637. :param fcgeometry: FlatCAMGeometry
  638. :return: None
  639. """
  640. assert isinstance(fcgeometry, Geometry)
  641. self.clear()
  642. self.connect_canvas_event_handlers()
  643. self.select_tool("select")
  644. # Link shapes into editor.
  645. for shape in fcgeometry.flatten():
  646. if shape is not None: # TODO: Make flatten never create a None
  647. self.add_shape(DrawToolShape(shape))
  648. self.replot()
  649. self.drawing_toolbar.setDisabled(False)
  650. self.snap_toolbar.setDisabled(False)
  651. def on_buffer_tool(self):
  652. buff_tool = BufferSelectionTool(self.app, self)
  653. buff_tool.run()
  654. def on_tool_select(self, tool):
  655. """
  656. Behavior of the toolbar. Tool initialization.
  657. :rtype : None
  658. """
  659. self.app.log.debug("on_tool_select('%s')" % tool)
  660. # This is to make the group behave as radio group
  661. if tool in self.tools:
  662. if self.tools[tool]["button"].isChecked():
  663. self.app.log.debug("%s is checked." % tool)
  664. for t in self.tools:
  665. if t != tool:
  666. self.tools[t]["button"].setChecked(False)
  667. self.active_tool = self.tools[tool]["constructor"](self)
  668. self.app.info(self.active_tool.start_msg)
  669. else:
  670. self.app.log.debug("%s is NOT checked." % tool)
  671. for t in self.tools:
  672. self.tools[t]["button"].setChecked(False)
  673. self.active_tool = None
  674. def on_canvas_click(self, event):
  675. """
  676. event.x and .y have canvas coordinates
  677. event.xdaya and .ydata have plot coordinates
  678. :param event: Event object dispatched by Matplotlib
  679. :return: None
  680. """
  681. if self.active_tool is not None:
  682. # Dispatch event to active_tool
  683. msg = self.active_tool.click(self.snap(event.xdata, event.ydata))
  684. self.app.info(msg)
  685. # If it is a shape generating tool
  686. if isinstance(self.active_tool, FCShapeTool) and self.active_tool.complete:
  687. self.on_shape_complete()
  688. return
  689. if isinstance(self.active_tool, FCSelect):
  690. self.app.log.debug("Replotting after click.")
  691. self.replot()
  692. else:
  693. self.app.log.debug("No active tool to respond to click!")
  694. def on_canvas_move(self, event):
  695. """
  696. event.x and .y have canvas coordinates
  697. event.xdaya and .ydata have plot coordinates
  698. :param event: Event object dispatched by Matplotlib
  699. :return:
  700. """
  701. self.on_canvas_move_effective(event)
  702. return None
  703. # self.move_timer.stop()
  704. #
  705. # if self.active_tool is None:
  706. # return
  707. #
  708. # # Make a function to avoid late evaluation
  709. # def make_callback():
  710. # def f():
  711. # self.on_canvas_move_effective(event)
  712. # return f
  713. # callback = make_callback()
  714. #
  715. # self.move_timer.timeout.connect(callback)
  716. # self.move_timer.start(500) # Stops if aready running
  717. def on_canvas_move_effective(self, event):
  718. """
  719. Is called after timeout on timer set in on_canvas_move.
  720. For details on animating on MPL see:
  721. http://wiki.scipy.org/Cookbook/Matplotlib/Animations
  722. event.x and .y have canvas coordinates
  723. event.xdaya and .ydata have plot coordinates
  724. :param event: Event object dispatched by Matplotlib
  725. :return: None
  726. """
  727. try:
  728. x = float(event.xdata)
  729. y = float(event.ydata)
  730. except TypeError:
  731. return
  732. if self.active_tool is None:
  733. return
  734. ### Snap coordinates
  735. x, y = self.snap(x, y)
  736. ### Utility geometry (animated)
  737. self.canvas.canvas.restore_region(self.canvas.background)
  738. geo = self.active_tool.utility_geometry(data=(x, y))
  739. if isinstance(geo, DrawToolShape) and geo.geo is not None:
  740. # Remove any previous utility shape
  741. self.delete_utility_geometry()
  742. # Add the new utility shape
  743. self.add_shape(geo)
  744. # Efficient plotting for fast animation
  745. #self.canvas.canvas.restore_region(self.canvas.background)
  746. elements = self.plot_shape(geometry=geo.geo,
  747. linespec="b--",
  748. linewidth=1,
  749. animated=True)
  750. for el in elements:
  751. self.axes.draw_artist(el)
  752. #self.canvas.canvas.blit(self.axes.bbox)
  753. # Pointer (snapped)
  754. elements = self.axes.plot(x, y, 'bo', animated=True)
  755. for el in elements:
  756. self.axes.draw_artist(el)
  757. self.canvas.canvas.blit(self.axes.bbox)
  758. def on_canvas_key(self, event):
  759. """
  760. event.key has the key.
  761. :param event:
  762. :return:
  763. """
  764. self.key = event.key
  765. ### Finish the current action. Use with tools that do not
  766. ### complete automatically, like a polygon or path.
  767. if event.key == ' ':
  768. if isinstance(self.active_tool, FCShapeTool):
  769. self.active_tool.click(self.snap(event.xdata, event.ydata))
  770. self.active_tool.make()
  771. if self.active_tool.complete:
  772. self.on_shape_complete()
  773. return
  774. ### Abort the current action
  775. if event.key == 'escape':
  776. # TODO: ...?
  777. #self.on_tool_select("select")
  778. self.app.info("Cancelled.")
  779. self.delete_utility_geometry()
  780. self.replot()
  781. # self.select_btn.setChecked(True)
  782. # self.on_tool_select('select')
  783. self.select_tool('select')
  784. return
  785. ### Delete selected object
  786. if event.key == '-':
  787. self.delete_selected()
  788. self.replot()
  789. ### Move
  790. if event.key == 'm':
  791. self.move_btn.setChecked(True)
  792. self.on_tool_select('move')
  793. self.active_tool.set_origin(self.snap(event.xdata, event.ydata))
  794. ### Copy
  795. if event.key == 'c':
  796. self.copy_btn.setChecked(True)
  797. self.on_tool_select('copy')
  798. self.active_tool.set_origin(self.snap(event.xdata, event.ydata))
  799. ### Snap
  800. if event.key == 'g':
  801. self.grid_snap_btn.trigger()
  802. if event.key == 'k':
  803. self.corner_snap_btn.trigger()
  804. ### Buffer
  805. if event.key == 'b':
  806. self.on_buffer_tool()
  807. ### Propagate to tool
  808. response = None
  809. if self.active_tool is not None:
  810. response = self.active_tool.on_key(event.key)
  811. if response is not None:
  812. self.app.info(response)
  813. def on_canvas_key_release(self, event):
  814. self.key = None
  815. def on_delete_btn(self):
  816. self.delete_selected()
  817. self.replot()
  818. def get_selected(self):
  819. """
  820. Returns list of shapes that are selected in the editor.
  821. :return: List of shapes.
  822. """
  823. #return [shape for shape in self.shape_buffer if shape["selected"]]
  824. return self.selected
  825. def delete_selected(self):
  826. tempref = [s for s in self.selected]
  827. for shape in tempref:
  828. self.delete_shape(shape)
  829. self.selected = []
  830. def plot_shape(self, geometry=None, linespec='b-', linewidth=1, animated=False):
  831. """
  832. Plots a geometric object or list of objects without rendering. Plotted objects
  833. are returned as a list. This allows for efficient/animated rendering.
  834. :param geometry: Geometry to be plotted (Any Shapely.geom kind or list of such)
  835. :param linespec: Matplotlib linespec string.
  836. :param linewidth: Width of lines in # of pixels.
  837. :param animated: If geometry is to be animated. (See MPL plot())
  838. :return: List of plotted elements.
  839. """
  840. plot_elements = []
  841. if geometry is None:
  842. geometry = self.active_tool.geometry
  843. try:
  844. for geo in geometry:
  845. plot_elements += self.plot_shape(geometry=geo,
  846. linespec=linespec,
  847. linewidth=linewidth,
  848. animated=animated)
  849. ## Non-iterable
  850. except TypeError:
  851. ## DrawToolShape
  852. if isinstance(geometry, DrawToolShape):
  853. plot_elements += self.plot_shape(geometry=geometry.geo,
  854. linespec=linespec,
  855. linewidth=linewidth,
  856. animated=animated)
  857. ## Polygon: Dscend into exterior and each interior.
  858. if type(geometry) == Polygon:
  859. plot_elements += self.plot_shape(geometry=geometry.exterior,
  860. linespec=linespec,
  861. linewidth=linewidth,
  862. animated=animated)
  863. plot_elements += self.plot_shape(geometry=geometry.interiors,
  864. linespec=linespec,
  865. linewidth=linewidth,
  866. animated=animated)
  867. if type(geometry) == LineString or type(geometry) == LinearRing:
  868. x, y = geometry.coords.xy
  869. element, = self.axes.plot(x, y, linespec, linewidth=linewidth, animated=animated)
  870. plot_elements.append(element)
  871. if type(geometry) == Point:
  872. x, y = geometry.coords.xy
  873. element, = self.axes.plot(x, y, 'bo', linewidth=linewidth, animated=animated)
  874. plot_elements.append(element)
  875. return plot_elements
  876. def plot_all(self):
  877. """
  878. Plots all shapes in the editor.
  879. Clears the axes, plots, and call self.canvas.auto_adjust_axes.
  880. :return: None
  881. :rtype: None
  882. """
  883. self.app.log.debug("plot_all()")
  884. self.axes.cla()
  885. for shape in self.storage.get_objects():
  886. if shape.geo is None: # TODO: This shouldn't have happened
  887. continue
  888. if shape in self.selected:
  889. self.plot_shape(geometry=shape.geo, linespec='k-', linewidth=2)
  890. continue
  891. self.plot_shape(geometry=shape.geo)
  892. for shape in self.utility:
  893. self.plot_shape(geometry=shape.geo, linespec='k--', linewidth=1)
  894. continue
  895. self.canvas.auto_adjust_axes()
  896. def on_shape_complete(self):
  897. self.app.log.debug("on_shape_complete()")
  898. # Add shape
  899. self.add_shape(self.active_tool.geometry)
  900. # Remove any utility shapes
  901. self.delete_utility_geometry()
  902. # Replot and reset tool.
  903. self.replot()
  904. self.active_tool = type(self.active_tool)(self)
  905. def delete_shape(self, shape):
  906. if shape in self.utility:
  907. self.utility.remove(shape)
  908. return
  909. self.storage.remove(shape)
  910. if shape in self.selected:
  911. self.selected.remove(shape)
  912. def replot(self):
  913. self.axes = self.canvas.new_axes("draw")
  914. self.plot_all()
  915. @staticmethod
  916. def make_storage():
  917. ## Shape storage.
  918. storage = FlatCAMRTreeStorage()
  919. storage.get_points = DrawToolShape.get_pts
  920. return storage
  921. def select_tool(self, toolname):
  922. """
  923. Selects a drawing tool. Impacts the object and GUI.
  924. :param toolname: Name of the tool.
  925. :return: None
  926. """
  927. self.tools[toolname]["button"].setChecked(True)
  928. self.on_tool_select(toolname)
  929. def set_selected(self, shape):
  930. # Remove and add to the end.
  931. if shape in self.selected:
  932. self.selected.remove(shape)
  933. self.selected.append(shape)
  934. def set_unselected(self, shape):
  935. if shape in self.selected:
  936. self.selected.remove(shape)
  937. def snap(self, x, y):
  938. """
  939. Adjusts coordinates to snap settings.
  940. :param x: Input coordinate X
  941. :param y: Input coordinate Y
  942. :return: Snapped (x, y)
  943. """
  944. snap_x, snap_y = (x, y)
  945. snap_distance = Inf
  946. ### Object (corner?) snap
  947. ### No need for the objects, just the coordinates
  948. ### in the index.
  949. if self.options["corner_snap"]:
  950. try:
  951. nearest_pt, shape = self.storage.nearest((x, y))
  952. nearest_pt_distance = distance((x, y), nearest_pt)
  953. if nearest_pt_distance <= self.options["snap_max"]:
  954. snap_distance = nearest_pt_distance
  955. snap_x, snap_y = nearest_pt
  956. except (StopIteration, AssertionError):
  957. pass
  958. ### Grid snap
  959. if self.options["grid_snap"]:
  960. if self.options["snap-x"] != 0:
  961. snap_x_ = round(x / self.options["snap-x"]) * self.options['snap-x']
  962. else:
  963. snap_x_ = x
  964. if self.options["snap-y"] != 0:
  965. snap_y_ = round(y / self.options["snap-y"]) * self.options['snap-y']
  966. else:
  967. snap_y_ = y
  968. nearest_grid_distance = distance((x, y), (snap_x_, snap_y_))
  969. if nearest_grid_distance < snap_distance:
  970. snap_x, snap_y = (snap_x_, snap_y_)
  971. return snap_x, snap_y
  972. def update_fcgeometry(self, fcgeometry):
  973. """
  974. Transfers the drawing tool shape buffer to the selected geometry
  975. object. The geometry already in the object are removed.
  976. :param fcgeometry: FlatCAMGeometry
  977. :return: None
  978. """
  979. fcgeometry.solid_geometry = []
  980. #for shape in self.shape_buffer:
  981. for shape in self.storage.get_objects():
  982. fcgeometry.solid_geometry.append(shape.geo)
  983. def union(self):
  984. """
  985. Makes union of selected polygons. Original polygons
  986. are deleted.
  987. :return: None.
  988. """
  989. results = cascaded_union([t.geo for t in self.get_selected()])
  990. # Delete originals.
  991. for_deletion = [s for s in self.get_selected()]
  992. for shape in for_deletion:
  993. self.delete_shape(shape)
  994. # Selected geometry is now gone!
  995. self.selected = []
  996. self.add_shape(DrawToolShape(results))
  997. self.replot()
  998. def intersection(self):
  999. """
  1000. Makes intersectino of selected polygons. Original polygons are deleted.
  1001. :return: None
  1002. """
  1003. shapes = self.get_selected()
  1004. results = shapes[0].geo
  1005. for shape in shapes[1:]:
  1006. results = results.intersection(shape.geo)
  1007. # Delete originals.
  1008. for_deletion = [s for s in self.get_selected()]
  1009. for shape in for_deletion:
  1010. self.delete_shape(shape)
  1011. # Selected geometry is now gone!
  1012. self.selected = []
  1013. self.add_shape(DrawToolShape(results))
  1014. self.replot()
  1015. def subtract(self):
  1016. selected = self.get_selected()
  1017. tools = selected[1:]
  1018. toolgeo = cascaded_union([shp.geo for shp in tools])
  1019. result = selected[0].geo.difference(toolgeo)
  1020. self.delete_shape(selected[0])
  1021. self.add_shape(DrawToolShape(result))
  1022. self.replot()
  1023. def buffer(self, buf_distance):
  1024. pre_buffer = cascaded_union([t.geo for t in self.get_selected()])
  1025. results = pre_buffer.buffer(buf_distance)
  1026. self.add_shape(DrawToolShape(results))
  1027. self.replot()
  1028. def distance(pt1, pt2):
  1029. return sqrt((pt1[0] - pt2[0]) ** 2 + (pt1[1] - pt2[1]) ** 2)
  1030. def mag(vec):
  1031. return sqrt(vec[0] ** 2 + vec[1] ** 2)
  1032. def poly2rings(poly):
  1033. return [poly.exterior] + [interior for interior in poly.interiors]