# This file is part of the Printrun suite. # # Printrun is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # Printrun is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Printrun. If not, see . import wx, time from printrun import gcoder from printrun_utils import imagefile ID_ABOUT = 101 ID_EXIT = 110 class window(wx.Frame): def __init__(self, f, size = (600, 600), build_dimensions = [200, 200, 100, 0, 0, 0], grid = (10, 50), extrusion_width = 0.5): wx.Frame.__init__(self, None, title = "Gcode view, shift to move view, mousewheel to set layer", size = size) self.CreateStatusBar(1); self.SetStatusText("Layer number and Z position show here when you scroll"); panel = wx.Panel(self, -1) self.p = gviz(panel, size = size, build_dimensions = build_dimensions, grid = grid, extrusion_width = extrusion_width, realparent = self) vbox = wx.BoxSizer(wx.VERTICAL) toolbar = wx.ToolBar(panel, -1, style = wx.TB_HORIZONTAL | wx.NO_BORDER) toolbar.AddSimpleTool(1, wx.Image(imagefile('zoom_in.png'), wx.BITMAP_TYPE_PNG).ConvertToBitmap(), 'Zoom In [+]', '') toolbar.AddSimpleTool(2, wx.Image(imagefile('zoom_out.png'), wx.BITMAP_TYPE_PNG).ConvertToBitmap(), 'Zoom Out [-]', '') toolbar.AddSeparator() toolbar.AddSimpleTool(3, wx.Image(imagefile('arrow_up.png'), wx.BITMAP_TYPE_PNG).ConvertToBitmap(), 'Move Up a Layer [U]', '') toolbar.AddSimpleTool(4, wx.Image(imagefile('arrow_down.png'), wx.BITMAP_TYPE_PNG).ConvertToBitmap(), 'Move Down a Layer [D]', '') toolbar.AddSimpleTool(5, wx.Image(imagefile('reset.png'), wx.BITMAP_TYPE_PNG).ConvertToBitmap(), 'Reset view', '') toolbar.AddSeparator() #toolbar.AddSimpleTool(6, wx.Image(imagefile('inject.png'), wx.BITMAP_TYPE_PNG).ConvertToBitmap(), 'Insert Code at start of this layer', '') toolbar.Realize() vbox.Add(toolbar, 0, border = 5) vbox.Add(self.p, 1, wx.EXPAND) panel.SetSizer(vbox) self.SetMinSize(self.ClientToWindowSize(vbox.GetMinSize())) self.Bind(wx.EVT_TOOL, lambda x:self.p.zoom(-1, -1, 1.2), id = 1) self.Bind(wx.EVT_TOOL, lambda x:self.p.zoom(-1, -1, 1/1.2), id = 2) self.Bind(wx.EVT_TOOL, lambda x:self.p.layerup(), id = 3) self.Bind(wx.EVT_TOOL, lambda x:self.p.layerdown(), id = 4) self.Bind(wx.EVT_TOOL, self.resetview, id = 5) #self.Bind(wx.EVT_TOOL, lambda x:self.p.inject(), id = 6) self.initpos = [0, 0] self.p.Bind(wx.EVT_KEY_DOWN, self.key) self.Bind(wx.EVT_KEY_DOWN, self.key) self.p.Bind(wx.EVT_MOUSEWHEEL, self.zoom) self.Bind(wx.EVT_MOUSEWHEEL, self.zoom) self.p.Bind(wx.EVT_MOUSE_EVENTS, self.mouse) self.Bind(wx.EVT_MOUSE_EVENTS, self.mouse) if f: gcode = gcoder.GCode(f) self.p.addfile(gcode) def resetview(self, event): self.p.translate = [0.0, 0.0] self.p.scale = self.p.basescale self.p.zoom(0, 0, 1.0) def mouse(self, event): if event.ButtonUp(wx.MOUSE_BTN_LEFT): if self.initpos is not None: self.initpos = None elif event.Dragging(): e = event.GetPositionTuple() if self.initpos is None or not hasattr(self, "basetrans"): self.initpos = e self.basetrans = self.p.translate self.p.translate = [self.basetrans[0] + (e[0] - self.initpos[0]), self.basetrans[1] + (e[1] - self.initpos[1])] self.p.dirty = 1 wx.CallAfter(self.p.Refresh) else: event.Skip() def key(self, event): # Keycode definitions kup = [85, 315] # Up keys kdo = [68, 317] # Down Keys kzi = [388, 316, 61] # Zoom In Keys kzo = [390, 314, 45] # Zoom Out Keys x = event.GetKeyCode() #print "Key event - "+str(x) #if event.ShiftDown(): cx, cy = self.p.translate # if x == wx.WXK_UP: # self.p.zoom(cx, cy, 1.2) # if x == wx.WXK_DOWN: # self.p.zoom(cx, cy, 1/1.2) #else: # if x == wx.WXK_UP: # self.p.layerup() # if x == wx.WXK_DOWN: # self.p.layerdown() if x in kup: self.p.layerup() if x in kdo: self.p.layerdown() if x in kzi: self.p.zoom(cx, cy, 1.2) if x in kzo: self.p.zoom(cx, cy, 1/1.2) #print p.lines.keys() def zoom(self, event): z = event.GetWheelRotation() if event.ShiftDown(): if z > 0: self.p.layerdown() elif z < 0: self.p.layerup() else: if z > 0: self.p.zoom(event.GetX(), event.GetY(), 1.2) elif z < 0: self.p.zoom(event.GetX(), event.GetY(), 1/1.2) class gviz(wx.Panel): # Mark canvas as dirty when setting showall _showall = 0 def _get_showall(self): return self._showall def _set_showall(self, showall): if showall != self._showall: self.dirty = 1 self._showall = showall showall = property(_get_showall, _set_showall) def __init__(self, parent, size = (200, 200), build_dimensions = [200, 200, 100, 0, 0, 0], grid = (10, 50), extrusion_width = 0.5, realparent = None): wx.Panel.__init__(self, parent, -1, size = size) self.SetMinSize((150, 150)) self.parent = realparent if realparent else parent self.size = size self.build_dimensions = build_dimensions self.grid = grid self.lastpos = [0, 0, 0, 0, 0, 0, 0] self.hilightpos = self.lastpos[:] self.Bind(wx.EVT_PAINT, self.paint) self.Bind(wx.EVT_SIZE, self.resize) self.lines = {} self.pens = {} self.arcs = {} self.arcpens = {} self.layers = [] self.layerindex = 0 self.filament_width = extrusion_width # set it to 0 to disable scaling lines with zoom self.basescale = [min(float(size[0])/build_dimensions[0], float(size[1])/build_dimensions[1])]*2 self.scale = self.basescale penwidth = max(1.0, self.filament_width*((self.scale[0]+self.scale[1])/2.0)) self.translate = [0.0, 0.0] self.mainpen = wx.Pen(wx.Colour(0, 0, 0), penwidth) self.arcpen = wx.Pen(wx.Colour(255, 0, 0), penwidth) self.travelpen = wx.Pen(wx.Colour(10, 80, 80), penwidth) self.hlpen = wx.Pen(wx.Colour(200, 50, 50), penwidth) self.fades = [wx.Pen(wx.Colour(250-0.6**i*100, 250-0.6**i*100, 200-0.4**i*50), penwidth) for i in xrange(6)] self.penslist = [self.mainpen, self.travelpen, self.hlpen]+self.fades self.showall = 0 self.hilight = [] self.hilightarcs = [] self.dirty = 1 self.blitmap = wx.EmptyBitmap(self.GetClientSize()[0], self.GetClientSize()[1],-1) def inject(self): #import pdb; pdb.set_trace() print"Inject code here..." print "Layer "+str(self.layerindex +1)+" - Z = "+str(self.layers[self.layerindex])+" mm" def clear(self): self.lastpos = [0, 0, 0, 0, 0, 0, 0] self.lines = {} self.pens = {} self.arcs = {} self.arcpens = {} self.layers = [] self.hilight = [] self.hilightarcs = [] self.layerindex = 0 self.showall = 0 self.dirty = 1 wx.CallAfter(self.Refresh) def layerup(self): if(self.layerindex+10): self.layerindex-=1 # Display layer info on statusbar (Jezmy) self.parent.SetStatusText("Layer "+str(self.layerindex + 1)+" - Going Down - Z = "+str(self.layers[self.layerindex])+ " mm", 0) self.dirty = 1 wx.CallAfter(self.Refresh) def setlayer(self, layer): try: self.layerindex = self.layers.index(layer) self.dirty = 1 wx.CallAfter(self.Refresh) self.showall = 0 except: pass def resize(self, event): oldside = max(1.0, min(self.size)) self.size = self.GetClientSizeTuple() newside = max(1.0, min(self.size)) self.basescale = 2*[min(float(self.size[0])/self.build_dimensions[0], float(self.size[1])/self.build_dimensions[1])] zoomratio = float(newside) / oldside wx.CallAfter(self.zoom, 0, 0, zoomratio) def zoom(self, x, y, factor): if x == -1 and y == -1: side = min(self.size) x = y = side / 2 self.scale = [s * factor for s in self.scale] self.translate = [ x - (x-self.translate[0]) * factor, y - (y-self.translate[1]) * factor] penwidth = max(1.0, self.filament_width*((self.scale[0]+self.scale[1])/2.0)) for pen in self.penslist: pen.SetWidth(penwidth) self.dirty = 1 wx.CallAfter(self.Refresh) def repaint(self): self.blitmap = wx.EmptyBitmap(self.GetClientSize()[0], self.GetClientSize()[1],-1) dc = wx.MemoryDC() dc.SelectObject(self.blitmap) dc.SetBackground(wx.Brush((250, 250, 200))) dc.Clear() dc.SetPen(wx.Pen(wx.Colour(180, 180, 150))) for grid_unit in self.grid: if grid_unit > 0: for x in xrange(int(self.build_dimensions[0]/grid_unit)+1): dc.DrawLine(self.translate[0]+x*self.scale[0]*grid_unit, self.translate[1], self.translate[0]+x*self.scale[0]*grid_unit, self.translate[1]+self.scale[1]*self.build_dimensions[1]) for y in xrange(int(self.build_dimensions[1]/grid_unit)+1): dc.DrawLine(self.translate[0], self.translate[1]+y*self.scale[1]*grid_unit, self.translate[0]+self.scale[0]*self.build_dimensions[0], self.translate[1]+y*self.scale[1]*grid_unit) dc.SetPen(wx.Pen(wx.Colour(0, 0, 0))) if not self.showall: dc.SetBrush(wx.Brush((43, 144, 255))) dc.DrawRectangle(self.size[0]-15, 0, 15, self.size[1]) dc.SetBrush(wx.Brush((0, 255, 0))) if len(self.layers): dc.DrawRectangle(self.size[0]-14, (1.0-(1.0*(self.layerindex+1))/len(self.layers))*self.size[1], 13, self.size[1]-1) def _drawlines(lines, pens): def _scaler(x): return (self.scale[0]*x[0]+self.translate[0], self.scale[1]*x[1]+self.translate[1], self.scale[0]*x[2]+self.translate[0], self.scale[1]*x[3]+self.translate[1],) scaled_lines = map(_scaler, lines) dc.DrawLineList(scaled_lines, pens) def _drawarcs(arcs, pens): def _scaler(x): return (self.scale[0]*x[0]+self.translate[0], self.scale[1]*x[1]+self.translate[1], self.scale[0]*x[2]+self.translate[0], self.scale[1]*x[3]+self.translate[1], self.scale[0]*x[4]+self.translate[0], self.scale[1]*x[5]+self.translate[1],) scaled_arcs = map(_scaler, arcs) for i in range(len(scaled_arcs)): dc.SetPen(pens[i] if type(pens).__name__ == 'list' else pens) dc.SetBrush(wx.TRANSPARENT_BRUSH) dc.DrawArc(*scaled_arcs[i]) if self.showall: l = [] for i in self.layers: dc.DrawLineList(l, self.fades[0]) _drawlines(self.lines[i], self.pens[i]) _drawarcs(self.arcs[i], self.arcpens[i]) return if self.layerindex