# -*- coding: utf-8 -*- # # AWL simulator - FUP - Shift element classes # # Copyright 2017-2018 Michael Buesch # # This program 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 2 of the License, or # (at your option) any later version. # # This program 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 this program; if not, write to the Free Software Foundation, Inc., # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. # from __future__ import division, absolute_import, print_function, unicode_literals #from awlsim.common.cython_support cimport * #@cy from awlsim.common.compat import * from awlsim.common.xmlfactory import * from awlsim.gui.fup.fup_base import * from awlsim.gui.fup.fup_elem import * from awlsim.gui.fup.fup_elemoperand import * class FupElem_SHIFT_factory(FupElem_factory): def parser_open(self, tag): assert(tag) x = tag.getAttrInt("x") y = tag.getAttrInt("y") subType = tag.getAttr("subtype") uuid = tag.getAttr("uuid", None) enabled = tag.getAttrBool("enabled", True) elemClass = { FupElem_SSI.OP_SYM_NAME : FupElem_SSI, FupElem_SSD.OP_SYM_NAME : FupElem_SSD, FupElem_SLW.OP_SYM_NAME : FupElem_SLW, FupElem_SRW.OP_SYM_NAME : FupElem_SRW, FupElem_SLD.OP_SYM_NAME : FupElem_SLD, FupElem_SRD.OP_SYM_NAME : FupElem_SRD, FupElem_RLD.OP_SYM_NAME : FupElem_RLD, FupElem_RRD.OP_SYM_NAME : FupElem_RRD, }.get(subType) if not elemClass: raise self.Error("Shift subtype '%s' is not known " "to the element parser." % ( subType)) self.elem = elemClass(x=x, y=y, uuid=uuid, enabled=enabled) self.elem.grid = self.grid XmlFactory.parser_open(self, tag) def parser_beginTag(self, tag): if tag.name == "connections": self.parser_switchTo(FupConn.factory(elem=self.elem)) return XmlFactory.parser_beginTag(self, tag) def parser_endTag(self, tag): if tag.name == "element": # Insert the element into the grid. if not self.grid.placeElem(self.elem): raise self.Error(" caused " "a grid collision.") self.parser_finish() return XmlFactory.parser_endTag(self, tag) def composer_getTags(self): elem = self.elem connTags = [] for inp in elem.inputs: connTags.extend(inp.factory(conn=inp).composer_getTags()) for out in elem.outputs: connTags.extend(out.factory(conn=out).composer_getTags()) return [ self.Tag(name="element", attrs={ "type" : "shift", "subtype" : elem.OP_SYM_NAME, "x" : str(elem.x), "y" : str(elem.y), "uuid" : str(elem.uuid), "enabled" : "0" if not elem.enabled else "", }, tags=[ self.Tag(name="connections", tags=connTags), ]) ] class FupElem_SHIFT(FupElem): """Shift FUP/FBD element base class""" factory = FupElem_SHIFT_factory FIXED_INPUTS = [ "EN", "N", "IN", ] FIXED_OUTPUTS = [ "ENO", ] OPTIONAL_CONNS = { "EN", "LOB", "ENO", } BLANK_CONNS = { "OUT", } # Sequence of special connections. __CONN_OUT_SEQUENCE = ( "LOB", "ENO", ) def __init__(self, x, y, nrOutputs=1, **kwargs): FupElem.__init__(self, x, y, **kwargs) self.inputs = [ FupConnIn(self, text=text) for text in self.FIXED_INPUTS ] self.outputs = [ FupConnOut(self, text=("OUT%d" % i)) for i in range(nrOutputs) ] self.outputs.extend(FupConnOut(self, text=text) for text in self.FIXED_OUTPUTS ) self.__renumberConns() def __renumberConns(self): i = 0 for conn in self.outputs: if not conn.text or conn.text.startswith("OUT"): conn.text = "OUT%d" % i i += 1 def __inferOutInsertIndex(self, conn): connText = conn.text.upper() connOutSeq = self.__CONN_OUT_SEQUENCE if connText in connOutSeq: # This is one of the special outputs. # Add it to its fixed position (see connOutSeq). if connText in (c.text.upper() for c in self.outputs): return -1 # We already have this one. beforeIndex = len(self.outputs) for idx, c in enumerate(self.outputs): if c.text.upper() not in connOutSeq: continue if connOutSeq.index(connText) <\ connOutSeq.index(c.text.upper()): beforeIndex = idx break else: # This is a regular OUTx output. # Add it just before the special outputs. beforeIndex = len(self.outputs) for idx, c in enumerate(self.outputs): if c.text.upper() in connOutSeq: beforeIndex = idx break return beforeIndex # Overridden method. For documentation see base class. def insertConn(self, beforeIndex, conn): if not conn: return False if conn.IN: return False if conn.OUT: if beforeIndex is None: beforeIndex = self.__inferOutInsertIndex(conn) if beforeIndex < 0: return False ok = FupElem.insertConn(self, beforeIndex, conn) if ok: self.__renumberConns() return ok # Overridden method. For documentation see base class. def getAreaViaPixCoord(self, pixelX, pixelY): if self.grid: cellWidth = self.grid.cellPixWidth cellHeight = self.grid.cellPixHeight totalWidth = cellWidth totalHeight = cellHeight * self.height xpad, ypad = self._xpadding, self._ypadding if pixelY >= ypad and\ pixelY < totalHeight - ypad: if pixelX < xpad: # inputs idx = pixelY // cellHeight lastIdx = len(self.inputs) - 1 if idx <= lastIdx: return self.AREA_INPUT, idx elif pixelX >= totalWidth - xpad: # outputs idx = pixelY // cellHeight firstIdx = self.height - len(self.outputs) if idx >= firstIdx: idx -= firstIdx return self.AREA_OUTPUT, idx else: # body return self.AREA_BODY, 0 return self.AREA_NONE, 0 # Overridden method. For documentation see base class. def getConnRelCoords(self, conn): x, y = 0, -1 if conn.IN: y = self.inputs.index(conn) elif conn.OUT: y = self.outputs.index(conn) if y >= 0: y = self.height - len(self.outputs) + y if x >= 0 and y >= 0: return x, y return FupElem.getConnRelCoords(self, conn) # Overridden method. For documentation see base class. @property def height(self): return max(len(self.inputs), len(self.outputs)) # Overridden method. For documentation see base class. def draw(self, painter): grid = self.grid if not grid: return cellWidth = grid.cellPixWidth cellHeight = grid.cellPixHeight xpad, ypad = self._xpadding, self._ypadding elemHeight = cellHeight * self.height elemWidth = cellWidth selected = self.selected # Draw body painter.setPen(self._outlineSelPen if selected else self._outlinePen) painter.setBrush(self._bgSelBrush if selected else self._bgBrush) (tlX, tlY), (trX, trY), (blX, blY), (brX, brY) = self._calcBodyBox() painter.drawRoundedRect(tlX, tlY, trX - tlX, blY - tlY, self.BODY_CORNER_RADIUS, self.BODY_CORNER_RADIUS) # Draw inputs painter.setBrush(self._bgSelBrush if selected else self._bgBrush) painter.setFont(self.getFont(8)) for i, conn in enumerate(self.inputs): cellIdx = i connPen = self._connPen\ if (conn.isConnected or conn.text in self.OPTIONAL_CONNS)\ else self._connOpenPen x = conn.drawOffset y = (cellIdx * cellHeight) + (cellHeight // 2) painter.setPen(connPen) painter.drawLine(x, y, xpad, y) if conn.text and\ not any(conn.text.startswith(b) for b in self.BLANK_CONNS): painter.setPen(connPen) x = xpad + 2 y = (cellIdx * cellHeight) painter.drawText(x, y, elemWidth - xpad - 2, cellHeight, Qt.AlignLeft | Qt.AlignVCenter, conn.text) # Draw outputs painter.setFont(self.getFont(8)) for i, conn in enumerate(self.outputs): cellIdx = self.height - len(self.outputs) + i connPen = self._connPen\ if (conn.isConnected or conn.text in self.OPTIONAL_CONNS)\ else self._connOpenPen x = cellWidth - conn.drawOffset y = (cellIdx * cellHeight) + (cellHeight // 2) painter.setPen(connPen) painter.setBrush(self._bgSelBrush if selected else self._bgBrush) painter.drawLine(cellWidth - xpad, y, x, y) if conn.text and\ not any(conn.text.startswith(b) for b in self.BLANK_CONNS): x = 0 y = (cellIdx * cellHeight) painter.drawText(x, y, elemWidth - xpad - 2, cellHeight, Qt.AlignRight | Qt.AlignVCenter, conn.text) # Draw symbol text painter.setPen(self._outlineSelPen if selected else self._outlinePen) painter.setBrush(self._bgSelBrush if selected else self._bgBrush) painter.setFont(self.getFont(12, bold=True)) painter.drawText(0, 0, elemWidth, elemHeight, Qt.AlignVCenter | Qt.AlignHCenter, self.OP_SYM) # Draw disable-marker self._drawDisableMarker(painter) # Overridden method. For documentation see base class. def prepareContextMenu(self, menu, area=None, conn=None): menu.enableAddOutput(True) if conn: normalOutputs = [ c for c in self.outputs if c.text.upper() not in self.__CONN_OUT_SEQUENCE ] if conn.OUT: if conn.text.upper() != "ENO": if conn.text.upper() in self.__CONN_OUT_SEQUENCE: menu.enableRemoveConn(True) else: menu.enableRemoveConn(len(normalOutputs) > 1) menu.enableDisconnWire(conn.isConnected) if not conn or conn.OUT: existing = set(c.text.upper() for c in self.outputs) if "LOB" not in existing: menu.enableCustomAction(0, True, text="Add LOB (Last shifted Out Bit) output") def __addStateOutput(self, name): return self.addConn(FupConnOut(text=name)) def __handleAddLOB(self, index): return self.__addStateOutput("LOB") CUSTOM_ACTIONS = ( __handleAddLOB, # index 0 ) class FupElem_SSI(FupElem_SHIFT): """SSI FUP/FBD element""" OP_SYM = "I>>" OP_SYM_NAME = "ssi" # XML ABI name class FupElem_SSD(FupElem_SHIFT): """SSD FUP/FBD element""" OP_SYM = "D>>" OP_SYM_NAME = "ssd" # XML ABI name class FupElem_SLW(FupElem_SHIFT): """SLW FUP/FBD element""" OP_SYM = "W<<" OP_SYM_NAME = "slw" # XML ABI name class FupElem_SRW(FupElem_SHIFT): """SRW FUP/FBD element""" OP_SYM = "W>>" OP_SYM_NAME = "srw" # XML ABI name class FupElem_SLD(FupElem_SHIFT): """SLD FUP/FBD element""" OP_SYM = "DW<<" OP_SYM_NAME = "sld" # XML ABI name class FupElem_SRD(FupElem_SHIFT): """SRD FUP/FBD element""" OP_SYM = "DW>>" OP_SYM_NAME = "srd" # XML ABI name class FupElem_RLD(FupElem_SHIFT): """RLD FUP/FBD element""" OP_SYM = "RLD" OP_SYM_NAME = "rld" # XML ABI name class FupElem_RRD(FupElem_SHIFT): """RRD FUP/FBD element""" OP_SYM = "RRD" OP_SYM_NAME = "rrd" # XML ABI name