def createPipedViewerWindow(self, viewertype, title, visible, noalpha): ''' Creates a PipedViewer of viewertype as the window of this instance of the bindings. Arguments: viewertype: type of PipedViewer to use title: display title for the Window visible: display Window on start-up? noalpha: do not use the alpha channel in colors? Raises a RuntimeError if an active window is already associated with these bindings, or if there were problems with creating the window. Returns True. ''' if self.__window != None: raise RuntimeError("createWindow called from bindings " \ "with an active window") self.__window = PipedViewer(viewertype) self.__window.submitCommand({ "action": "setTitle", "title": str(title) }) if visible: self.__window.submitCommand({"action": "show"}) if noalpha: self.__window.submitCommand({"action": "noalpha"}) self.checkForErrorResponse() return True
class PyFerretBindings(AbstractPyFerretBindings): ''' Common methods in PipedViewer bindings for PyFerret graphical functions. The createWindow method should be defined in a subclass in order to create a valid bindings class for PyFerret. ''' def __init__(self): ''' Create an instance of the the PipedViewer bindings for PyFerret graphical functions. The createWindow method should be called to associate a new PipedViewer with these bindings. ''' super(PyFerretBindings, self).__init__() self.__window = None # wait time in seconds for error responses for "high-level" # (not drawing) commands that might have an error self.__errwait = 0.01 def createPipedViewerWindow(self, viewertype, title, visible, noalpha): ''' Creates a PipedViewer of viewertype as the window of this instance of the bindings. Arguments: viewertype: type of PipedViewer to use title: display title for the Window visible: display Window on start-up? noalpha: do not use the alpha channel in colors? Raises a RuntimeError if an active window is already associated with these bindings, or if there were problems with creating the window. Returns True. ''' if self.__window != None: raise RuntimeError("createWindow called from bindings " \ "with an active window") self.__window = PipedViewer(viewertype) self.__window.submitCommand({ "action": "setTitle", "title": str(title) }) if visible: self.__window.submitCommand({"action": "show"}) if noalpha: self.__window.submitCommand({"action": "noalpha"}) self.checkForErrorResponse() return True def submitCommand(self, cmnd): ''' Submits the given command to PipedViewer. Provided for use by functions defined in subclasses. ''' self.__window.submitCommand(cmnd) def checkForResponse(self, timeout=0.0): ''' Checks the response pipe for an object within the given timeout number of in seconds. If timeout is None, this method will wait indefinitely for something to be given on the responds pipe. If timeout is zero (the default) the method does not wait if there is no response waiting. If nothing is obtained within the given timeout, None is returned. ''' result = self.__window.checkForResponse(timeout) return result def checkForErrorResponse(self, timeout=0.0): ''' Checks the response pipe for a message. If anything is found, a RuntimeError is raised with the string of the full response. If timeout is zero (the default) the method does not wait if there is no response waiting; otherwise the method waits the given number of seconds for a reponse to arrive. An IllegalArgumentException if raised if timeout is None, as error responses should not be an expected result (thus an indefinite wait makes no sense). ''' if timeout is None: raise IllegalArgumentException( "timeout to checkForErrorResponse is None") fullresponse = None # Only wait (if timeout not zero) on the first check for a response response = self.__window.checkForResponse(timeout) while response: if fullresponse: fullresponse += '\n' fullresponse += str(response) else: fullresponse = str(response) # Any further messages associated with this error should be immediate response = self.__window.checkForResponse() if fullresponse: raise RuntimeError(fullresponse) # The remaining methods are common implementations of the required binding methods def deleteWindow(self): ''' Shuts down the PipedViewer. Returns True. ''' try: self.__window.submitCommand({"action": "exit"}) self.checkForErrorResponse(self.__errwait) self.__window.waitForViewerExit() finally: self.__window = None return True def setImageName(self, imagename, formatname): ''' Assigns the name and format of the image file to be created. Arguments: imagename - name for the image file (can be NULL) imgnamelen - actual length of imagename (zero if NULL) formatname - name of the image format (case insensitive, can be NULL) fmtnamelen - actual length of formatname (zero if NULL) If formatname is empty or NULL, the filename extension of imagename, if it exists and is recognized, will determine the format. This method only gives the default name of the image file to be created by the saveWindow method. The saveWindow method must be called to save the image. ''' cmnd = {"action": "imgname"} if imagename: cmnd["name"] = imagename if formatname: cmnd["format"] = formatname self.__window.submitCommand(cmnd) self.checkForErrorResponse(self.__errwait) def setAntialias(self, antialias): ''' Turns on (antilaias True) or off (antialias False) anti-aliasing in future drawing commands. ''' cmnd = {"action": "antialias", "antialias": bool(antialias)} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def beginView(self, leftfrac, bottomfrac, rightfrac, topfrac, clipit): ''' Start a view in the PipedViewer Window. The view fractions start at (0.0, 0.0) in the left top corner and increase to (1.0, 1.0) in the right bottom corner; thus leftfrac must be less than rightfrac and topfrac must be less than bottomfrac. Arguments: leftfrac: [0,1] fraction of the Window width for the left side of the View bottomfrac: [0,1] fraction of the Window height for the bottom side of the View rightfrac: [0,1] fraction of the Window width for the right side of the View topfrac: [0,1] fraction of the Window height for the top side of the View clipit: clip drawings to this View? ''' leftfracflt = float(leftfrac) bottomfracflt = float(bottomfrac) rightfracflt = float(rightfrac) topfracflt = float(topfrac) if (0.0 > leftfracflt) or (leftfracflt >= rightfracflt) or (rightfracflt > 1.0): raise ValueError("leftfrac (%f) and rightfrac (%f) must be in [0.0, 1.0] " \ "with leftfrac < rightfrac" % (leftfracflt, rightfracflt)) if (0.0 > topfracflt) or (topfracflt >= bottomfracflt) or (bottomfracflt > 1.0): raise ValueError("topfrac (%f) and bottomfrac (%f) must be in [0.0, 1.0] " \ "with topfrac < bottomfrac" % (topfracflt, bottomfracflt)) cmnd = { "action": "beginView", "viewfracs": { "left": leftfracflt, "right": rightfracflt, "top": topfracflt, "bottom": bottomfracflt }, "clip": bool(clipit) } self.__window.submitCommand(cmnd) self.checkForErrorResponse() def clipView(self, clipit): ''' Enable or disable clipping to the current View. Arguments: clipit: clip drawings to the current View? ''' cmnd = {"action": "clipView", "clip": bool(clipit)} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def endView(self): ''' Close a View in the PipedViewer Window ''' self.__window.submitCommand({"action": "endView"}) self.checkForErrorResponse() def beginSegment(self, segid): ''' Creates a "Segment object" for the given Window. A Segment is just a group of drawing commands. Arguments: segid: ID for the Segment ''' cmnd = {"action": "beginSegment", "segid": segid} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def endSegment(self): ''' End the current "Segment" for the Window. ''' cmnd = {"action": "endSegment"} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def deleteSegment(self, segid): ''' Deletes the drawing commands in the indicated Segment. Arguments: segid: ID for the Segment to be deleted ''' cmnd = {"action": "deleteSegment", "segid": segid} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def updateWindow(self): ''' Indicates the viewer should update the graphics displayed. ''' cmnd = {"action": "update"} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def clearWindow(self, bkgcolor): ''' Clears the Window of all drawings. The window is initialized to all bkgcolor (the background color). Arguments: bkgcolor: initialize (fill) the Window with this Color ''' if bkgcolor: # Make a copy of the bkgcolor dictionary cmnd = dict(bkgcolor) else: cmnd = {} cmnd["action"] = "clear" self.__window.submitCommand(cmnd) self.checkForErrorResponse() def redrawWindow(self, bkgcolor): ''' Redraw the current drawing except using bkgcolor as the background color (the initialization color for the Window). Arguments: bkgcolor: initialize (fill) the Window with this Color before redrawing the current drawing. ''' if bkgcolor: # Make a copy of the bkgcolor dictionary cmnd = dict(bkgcolor) else: cmnd = {} cmnd["action"] = "redraw" self.__window.submitCommand(cmnd) self.checkForErrorResponse() def resizeWindow(self, width, height): ''' Sets the current size of the Window. Arguments: width: width of the Window, in "device units" height: height of the window in "device units" "device units" is pixels at the current window DPI ''' cmnd = {"action": "resize", "width": width, "height": height} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def scaleWindow(self, scale): ''' Sets the current scaling factor for the Window. Arguments: scale: scaling factor to use ''' cmnd = {"action": "rescale", "factor": scale} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def windowScreenInfo(self): ''' Returns the four-tuple (dpix, dpiy, screenwidth, screenheight) for the default screen (display) of this Window dpix: dots (pixels) per inch, in the horizontal (X) direction dpiy: dots (pixels) per inch, in the vertical (Y) direction screenwidth: width of the screen (display) in pixels (dots) screenheight: height of the screen (display) in pixels (dots) ''' cmnd = {"action": "screenInfo"} self.__window.submitCommand(cmnd) response = None try: # Wait indefinitely for a response # Make sure it is a valid response response = self.__window.checkForResponse(None) if (type(response) != tuple) or (len(response) != 4): raise ValueError dpix = float(response[0]) dpiy = float(response[1]) screenwidth = int(response[2]) screenheight = int(response[3]) if (dpix <= 0.0) or (dpiy <= 0.0) or \ (screenwidth <= 0) or (screenheight <= 0): raise ValueError except Exception: if not response: # error raised before a response obtained raise fullresponse = str(response) response = self.__window.checkForResponse() while response: fullresponse += '\n' fullresponse += response response = self.__window.checkForResponse() raise RuntimeError(fullresponse) return (dpix, dpiy, screenwidth, screenheight) def showWindow(self, visible): ''' Display or hide a Window. Arguments: visible: display (if True) or hide (if False) the Window ''' if visible: cmnd = {"action": "show"} else: cmnd = {"action": "hide"} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def saveWindow(self, filename, fileformat, transparent, xinches, yinches, xpixels, ypixels, annotations): ''' Save the contents of the window to a file. This might be called when there is no image to save; in this case the call should be ignored. Arguments: filename: name of the file to create fileformat: name of the format to use transparent: use a transparent background? xinches: horizontal size of vector image in inches yinches: vertical size of vector image in inches xpixels: horizontal size of raster image in pixels ypixels: vertical size of raster image in pixels annotations: tuple of annotation strings If fileformat is None or empty, the fileformat is guessed from the filename extension. If transparent is False, the entire scene is initialized to the last clearing color. If transparent is True, the entire scene is initialized as transparent. If annotations is not None, the strings given in the tuple are to be displayed above the image. These annotations add height, as needed, to the saved image (i.e., yinches or ypixels is the height of the image below these annotations). ''' cmnd = {} cmnd["action"] = "save" cmnd["filename"] = filename if fileformat: cmnd["fileformat"] = fileformat cmnd["transparent"] = transparent cmnd["vectsize"] = {"width": xinches, "height": yinches} cmnd["rastsize"] = {"width": xpixels, "height": ypixels} cmnd["annotations"] = annotations self.__window.submitCommand(cmnd) self.checkForErrorResponse(self.__errwait) def createColor(self, redfrac, greenfrac, bluefrac, opaquefrac): ''' Returns a Color object from fractional [0.0, 1.0] intensities of the red, green, and blue channels. The opaquefrac is used to set the alpha channel. Arguments: redfrac: fractional [0.0, 1.0] red intensity greenfrac: fractional [0.0, 1.0] green intensity bluefrac: fractional [0.0, 1.0] blue intensity opaquefrac: fractional [0.0, 1.0] opaqueness (0.0 is transparent; 1.0 is opaque) of the color. For output that does not support an alpha channel, this will be silently ignored and the color will be completely opaque. Raises an error if unable to create the Color object. ''' if (redfrac < 0.0) or (redfrac > 1.0): raise ValueError("redfrac must be a value in [0.0, 1.0]") if (greenfrac < 0.0) or (greenfrac > 1.0): raise ValueError("greenfrac must be a value in [0.0, 1.0]") if (bluefrac < 0.0) or (bluefrac > 1.0): raise ValueError("bluefrac must be a value in [0.0, 1.0]") if (opaquefrac < 0.0) or (opaquefrac > 1.0): raise ValueError("opaquefrac must be a value in [0.0, 1.0]") redint = int(256.0 * redfrac) if redint == 256: redint = 255 greenint = int(256.0 * greenfrac) if greenint == 256: greenint = 255 blueint = int(256.0 * bluefrac) if blueint == 256: blueint = 255 colorint = (redint * 256 + greenint) * 256 + blueint opaqueint = int(256.0 * opaquefrac) if opaqueint == 256: opaqueint = 255 return {"color": colorint, "alpha": opaqueint} def deleteColor(self, color): ''' Delete a Color object created by createColor Arguments: color: Color to be deleted ''' del color def createFont(self, familyname, fontsize, italic, bold, underlined): ''' Returns a Font object. Arguments: familyname: name of the font family (e.g., "Helvetica", "Times"); None or an empty string uses the default font fontsize: desired size of the font (scales with view size) italic: use the italic version of the font? bold: use the bold version of the font? underlined: use the underlined version of the font? Raises an error if unable to create the Font object. ''' fontdict = { "size": fontsize, "italic": italic, "bold": bold, "underlined": underlined } if familyname: fontdict["family"] = familyname return fontdict def deleteFont(self, font): ''' Delete a Font object created by createFont Arguments: font: Font to be deleted ''' del font def createPen(self, color, width, style, capstyle, joinstyle): ''' Returns a Pen object. Arguments: color: Color to use width: line width (scales with view size) style: line style name (e.g., "solid", "dash") capstyle: end-cap style name (e.g., "square") joinstyle: join style name (e.g., "bevel") Raises an error if unable to create the Pen object. ''' if color: pen = dict(color) else: pen = {} if width: pen["width"] = width if style: pen["style"] = style if capstyle: pen["capstyle"] = capstyle if joinstyle: pen["joinstyle"] = joinstyle return pen def replacePenColor(self, pen, newcolor): ''' Replaces the color in pen with newcolor. Arguments: pen: Pen object to modify newcolor: Color to use Raises an error if unable to replace the Color in the Pen. ''' pen.update(newcolor) def deletePen(self, pen): ''' Delete a Pen object created by createPen Arguments: pen: Pen to be deleted ''' del pen def createBrush(self, color, style): ''' Returns a Brush object. Arguments: color: Color to use style: fill style name (e.g., "solid", "cross") Raises an error if unable to create the Brush object. ''' if color: brush = dict(color) else: brush = {} if style: brush["style"] = style return brush def replaceBrushColor(self, brush, newcolor): ''' Replaces the color in brush with newcolor. Arguments: brush: Brush object to modify newcolor: Color to use Raises an error if unable to replace the Color in the Brush. ''' brush.update(newcolor) def deleteBrush(self, brush): ''' Delete a Brush object created by createBrush Arguments: brush: Brush to be deleted ''' del brush def createSymbol(self, symbolname): ''' Returns a Symbol object. Arguments: symbolname: name of the symbol. Currently supported values are: '.' (period): filled circle 'o' (lowercase oh): unfilled circle '+': plus mark 'x' (lowercase ex): x mark '*': asterisk '^': triangle "#": square Raises an error if unable to create the Symbol object. ''' return symbolname def deleteSymbol(self, symbol): ''' Delete a Symbol object created by createSymbol Arguments: symbol: Symbol to be deleted ''' del symbol def setWidthFactor(self, widthfactor): ''' Assigns the scaling factor to be used for pen widths, symbols sizes, and font sizes Arguments: widthfactor: positive float giving the new scaling factor to use ''' newfactor = float(widthfactor) if newfactor <= 0.0: raise ValueError("Width scaling factor must be positive") cmnd = {"action": "setWidthFactor", "factor": widthfactor} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def drawMultiline(self, ptsx, ptsy, pen): ''' Draws connected line segments. Arguments: ptsx: X-coordinates of the endpoints ptsy: Y-coordinates of the endpoints pen: the Pen to use to draw the line segments Coordinates are measured from the upper left corner in "device units" (pixels at the current window DPI). ''' if len(ptsx) != len(ptsy): raise ValueError("the lengths of ptsx and ptsy are not the same") points = list(zip(ptsx, ptsy)) cmnd = {"action": "drawMultiline", "points": points, "pen": pen} self.__window.submitCommand(cmnd) self.checkForErrorResponse() def drawPoints(self, ptsx, ptsy, symbol, color, ptsize): ''' Draws discrete points. Arguments: ptsx: X-coordinates of the points ptsy: Y-coordinates of the points symbol: the Symbol to use to draw a point color: color of the Symbol (default color if None or empty) ptsize: size of the symbol (scales with view size) Coordinates are measured from the upper left corner in "device units" (pixels at the current window DPI). ''' if len(ptsx) != len(ptsy): raise ValueError("the lengths of ptsx and ptsy are not the same") points = list(zip(ptsx, ptsy)) if color: # make a copy of the color dictionary cmnd = dict(color) else: cmnd = {} cmnd["action"] = "drawPoints" cmnd["points"] = points cmnd["symbol"] = symbol cmnd["size"] = ptsize self.__window.submitCommand(cmnd) self.checkForErrorResponse() def drawPolygon(self, ptsx, ptsy, brush, pen): ''' Draws a polygon. Arguments: ptsx: X-coordinates of the vertices ptsy: Y-coordinates of the vertices brush: the Brush to use to fill the polygon; if None the polygon will not be filled pen: the Pen to use to outline the polygon; if None the polygon will not be outlined Coordinates are measured from the upper left corner in "device units" (pixels at the current window DPI). ''' if len(ptsx) != len(ptsy): raise ValueError("the lengths of ptsx and ptsy are not the same") points = list(zip(ptsx, ptsy)) cmnd = {"action": "drawPolygon", "points": points} if brush: cmnd["fill"] = brush if pen: cmnd["outline"] = pen self.__window.submitCommand(cmnd) self.checkForErrorResponse() def drawRectangle(self, left, bottom, right, top, brush, pen): ''' Draws a rectangle. Arguments: left: X-coordinate of the left edge bottom: Y-coordinate of the bottom edge right: X-coordinate of the right edge top: Y-coordinate of the top edge brush: the Brush to use to fill the polygon; if None the polygon will not be filled pen: the Pen to use to outline the polygon; if None the polygon will not be outlined Coordinates are measured from the upper left corner in "device units" (pixels at the current window DPI). ''' cmnd = { "action": "drawRectangle", "left": left, "bottom": bottom, "right": right, "top": top } if brush: cmnd["fill"] = brush if pen: cmnd["outline"] = pen self.__window.submitCommand(cmnd) self.checkForErrorResponse() def textSize(self, text, font): ''' Returns the width and height of the text if drawn in the given font. The width is such that continuing text should be positioned at the start of this text plus this width. The height will always be the ascent plus descent for the font and is independent of the text. Arguments: text: the text string to draw font: the font to use Returns: (width, height) of the text in "device units" (pixels at the current window DPI) ''' cmnd = {"action": "textSize", "text": text} if font: cmnd["font"] = font self.__window.submitCommand(cmnd) response = None try: # Wait indefinitely for a response # Make sure it is a valid response response = self.__window.checkForResponse(None) if (type(response) != tuple) or (len(response) != 2): raise ValueError width = float(response[0]) height = float(response[1]) if (width <= 0.0) or (height <= 0.0): raise ValueError except Exception: if not response: # error raised before a response obtained raise fullresponse = str(response) response = self.__window.checkForResponse() while response: fullresponse += '\n' fullresponse += response response = self.__window.checkForResponse() raise RuntimeError(fullresponse) return (width, height) def drawText(self, text, startx, starty, font, color, rotate): ''' Draws text. Arguments: text: the text string to draw startx: the X-coordinate of the beginning baseline starty: the Y-coordinate of the beginning baseline font: the font to use color: the color to use as a solid brush or pen rotate: the angle of the text baseline in degrees clockwise from horizontal Coordinates are measured from the upper left corner in "device units" (pixels at the current window DPI). ''' cmnd = { "action": "drawText", "text": text, "location": (startx, starty) } if font: cmnd["font"] = font if color: pen = dict(color) pen["style"] = "solid" cmnd["fill"] = pen if rotate != 0.0: cmnd["rotate"] = rotate self.__window.submitCommand(cmnd) self.checkForErrorResponse()