#!/usr/bin/env python
# ###########################################################################
#
# This file is part of Taurus
#
# http://taurus-scada.org
#
# Copyright 2011 CELLS / ALBA Synchrotron, Bellaterra, Spain
#
# Taurus is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Taurus 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 Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with Taurus. If not, see <http://www.gnu.org/licenses/>.
#
# ###########################################################################
"""This module contains the graphics factory for the jdraw file format
"""
import os
import traceback
from taurus.external.qt import Qt
from taurus.core.util.log import Logger
from taurus.core.util.singleton import Singleton
from taurus.core.util.containers import CaselessDict
from taurus.qt.qtgui.graphic import (
TaurusBaseGraphicsFactory,
TaurusGraphicsScene,
TaurusGraphicsItem,
)
__docformat__ = "restructuredtext"
LINESTYLE_JDW2QT = {
0: Qt.Qt.SolidLine,
1: Qt.Qt.DotLine,
2: Qt.Qt.DashLine,
3: Qt.Qt.DashLine,
4: Qt.Qt.DashDotLine,
}
FILLSTYLE_JDW2QT = {
0: Qt.Qt.NoBrush,
1: Qt.Qt.SolidPattern,
2: Qt.Qt.FDiagPattern,
3: Qt.Qt.BDiagPattern,
4: Qt.Qt.DiagCrossPattern,
5: Qt.Qt.FDiagPattern,
6: Qt.Qt.BDiagPattern,
7: Qt.Qt.Dense5Pattern,
8: Qt.Qt.Dense7Pattern,
9: Qt.Qt.Dense6Pattern,
10: Qt.Qt.Dense4Pattern,
11: Qt.Qt.LinearGradientPattern,
}
TEXTHINT_JDW2QT = CaselessDict(
{
"helvetica": Qt.QFont.Helvetica,
"serif": Qt.QFont.Serif,
"sansserif": Qt.QFont.SansSerif,
"courier": Qt.QFont.Courier,
"Monospaced": Qt.QFont.Courier,
"times": Qt.QFont.Times,
"": Qt.QFont.AnyStyle,
}
)
ALIGNMENT = {
0: Qt.Qt.AlignHCenter,
1: Qt.Qt.AlignLeft,
2: Qt.Qt.AlignRight,
}
VALIGNMENT = {
0: Qt.Qt.AlignVCenter,
1: Qt.Qt.AlignTop,
2: Qt.Qt.AlignBottom,
}
[docs]class TaurusJDrawGraphicsFactory(Singleton, TaurusBaseGraphicsFactory, Logger):
def __init__(self, parent, alias=None, delayed=False):
"""Initialization. Nothing to be done here for now."""
self.myparent = parent
self.call__init__wo_kw(TaurusBaseGraphicsFactory, parent)
self._zBufferLevel = 0
self._delayed = delayed
self.alias = alias if alias is not None else {}
[docs] def init(self, *args, **kwargs):
"""Singleton instance initialization."""
self.call__init__(Logger, self.__class__.__name__)
self.call__init__(TaurusBaseGraphicsFactory)
[docs] def getZBufferLevel(self):
return self._zBufferLevel
[docs] def incZBufferLevel(self):
self._zBufferLevel += 1
return self._zBufferLevel
[docs] def setZBufferLevel(self, level):
self._zBufferLevel = level
[docs] def resetZBufferLevel(self):
self.setZBufferLevel(0)
[docs] def getSceneObj(self, items):
scene = TaurusGraphicsScene(self.myparent)
for item in items:
try:
if isinstance(item, Qt.QWidget):
scene.addWidget(item)
elif isinstance(item, Qt.QGraphicsItem):
scene.addItem(item)
except Exception:
self.warning("Unable to add item %s to scene" % str(item))
self.debug("Details:", exc_info=1)
return scene
[docs] def getObj(self, name, params):
method_name = "get" + name.lstrip("JD") + "Obj"
try:
method = getattr(self, method_name)
obj = method(params)
obj.setZValue(self.incZBufferLevel())
return obj
except Exception:
self.warning("Error fetching object")
self.info("Details:", exc_info=1)
pass
return None
[docs] def getRectangleObj(self, params):
item = self.getGraphicsItem("Rectangle", params)
x1, y1, x2, y2 = params.get("summit")
width = x2 - x1
height = y2 - y1
# item.setPos(x1,y1)
item.setRect(x1, y1, width, height)
return item
[docs] def getBarObj(self, params):
# TODO: properly implement JDBar support
# As a workaround, use a filled rectangle as a substitute of a JDBar
self.warning("JDBar not yet supported. Using a JDRectangle instead")
if "fillStyle" not in params:
params["fillStyle"] = 1
return self.getRectangleObj(params)
[docs] def getRoundRectangleObj(self, params):
item = self.getGraphicsItem("RoundRectangle", params)
x1, y1, x2, y2 = params.get("summit")
width = x2 - x1
height = y2 - y1
cornerWidth = params.get("cornerWidth", 24)
nbPoints = params.get("step", 6)
# item.setPos(x1,y1)
item.setRect(x1, y1, width, height)
item.setCornerWidth(cornerWidth, nbPoints)
return item
[docs] def getLineObj(self, params):
item = self.getGraphicsItem("Line", params)
x1, y1, x2, y2 = params.get("summit")
item.setLine(x1, y1, x2, y2)
return item
[docs] def getEllipseObj(self, params):
item = self.getGraphicsItem("Ellipse", params)
x1, y1, x2, y2 = params.get("summit")
width = x2 - x1
height = y2 - y1
item.setRect(x1, y1, width, height)
return item
[docs] def getPolylineObj(self, params):
item = self.getGraphicsItem("Polyline", params)
polygon = Qt.QPolygonF()
p = params.get("summit")
for i in range(0, len(p), 2):
polygon.append(Qt.QPointF(p[i], p[i + 1]))
item.setPolygon(polygon)
return item
[docs] def getSplineObj(self, params):
item = self.getGraphicsItem("Spline", params)
p = params.get("summit")
p = [Qt.QPointF(p[i], p[i + 1]) for i in range(0, len(p), 2)]
item.setControlPoints(p)
isClosed = params.get("isClosed", True)
item.setClose(isClosed)
return item
[docs] def getLabelObj(self, params):
item = self.getGraphicsItem("Label", params)
self.readLabelObj(item, params)
return item
[docs] def readLabelObj(self, item, params):
origin = params.get("origin")
item.setPos(origin[0], origin[1])
summit = params.get("summit")
x, y = summit[0] - origin[0], summit[1] - origin[1]
width, height = summit[2] - summit[0], summit[3] - summit[1]
item.setRect(x, y, width, height)
# it is parsed as a float
vAlignment = int(params.get("vAlignment", 0))
hAlignment = int(params.get("hAlignment", 0))
assert vAlignment in VALIGNMENT
assert hAlignment in ALIGNMENT
vAlignment = VALIGNMENT[vAlignment]
hAlignment = ALIGNMENT[hAlignment]
item.setAlignment(hAlignment | vAlignment)
fnt = params.get("font", None)
if fnt:
family, style, size = fnt
f = Qt.QFont(family, int(0.85 * size), Qt.QFont.Light, False)
f.setStyleHint(TEXTHINT_JDW2QT.get(family, Qt.QFont.AnyStyle))
f.setStyleStrategy(Qt.QFont.PreferMatch)
if style == 1:
f.setWeight(Qt.QFont.DemiBold)
elif style == 2:
f.setItalic(True)
elif style == 3:
f.setWeight(Qt.QFont.DemiBold)
f.setItalic(True)
# TODO: Improve code in order to be able to set a suitable font
item.setFont(f)
fg = params.get("foreground", (0, 0, 0))
color = Qt.QColor(fg[0], fg[1], fg[2])
item.setDefaultTextColor(color)
txt = params.get("text")
if txt:
if any(
isinstance(txt, t) for t in (list, tuple, set)
): # Parsing several lines of text
txt = "\n".join(txt)
item.setPlainText(str(txt))
item._currText = txt
[docs] def getGroupObj(self, params):
item = self.getGraphicsItem("Group", params)
s = params.get("summit")
x1, y1 = s[0], s[1]
item.setPos(x1, y1)
children = params.get("children")
if children:
for child in children:
if child:
item.addToGroup(child)
if item._fillStyle:
self.set_item_filling(item, expand=True)
return item
[docs] def getSwingObjectObj(self, params):
item = self.getGraphicsItem("SwingObject", params)
s = params.get("summit")
x1, y1 = s[0], s[1]
item.setPos(x1, y1)
className = params.get("className")
if className == "fr.esrf.tangoatk.widget.attribute.SimpleScalarViewer":
self.readSimpleScalarViewerObj(item, params)
return item
[docs] def readSimpleScalarViewerObj(self, item, params):
self.readLabelObj(item, params)
ext = params.get("extensions")
c = ext.get("validBackground")
c = [int(x) for x in c.split(",")]
if c:
validBackground = Qt.QColor(*c)
item.setValidBackground(validBackground)
invalidText = ext.get("invalidText", "-----")
item.setNoneValue(invalidText)
alarmEnabled = ext.get("alarmEnabled", True)
alarmEnabled = alarmEnabled.lower().strip() in ["yes", "true", "1"]
item.setShowQuality(alarmEnabled)
unitVisible = ext.get("unitVisible", True)
unitVisible = unitVisible.lower().strip() in ["yes", "true", "1"]
item.setUnitVisible(unitVisible)
userFormat = ext.get("userFormat", None)
item.setUserFormat(userFormat)
[docs] def getImageObj(self, params):
item = self.getGraphicsItem("Image", params)
s = params.get("summit")
x1, y1, x2, y2 = s
item.setPos(x1, y1)
fname = params.get("file_name")
if fname:
if os.path.isfile(fname):
fname = os.path.realpath(fname)
elif hasattr(self.myparent, "path"):
# self.info('using path param ...')
fname = self.myparent.path + os.path.sep + fname
self.debug(
"Opening JDImage(%s) = x,y,w,h=%f,%f,%f,%f"
% (fname, x1, y1, x2 - x1, y2 - y1)
)
pixmap = Qt.QPixmap(fname)
item.setPixmap(pixmap.scaled(int(x2 - x1), int(y2 - y1)))
# item.scale(float(w)/pixmap.width(), float(h)/pixmap.height())
else:
self.warning("No filename for image!?!")
return item
[docs] def set_common_params(self, item, params):
if not item:
return
item._params = params
name = params.get("name")
if name.lower() == "ignorerepaint":
name = ""
if "extensions" not in params:
params["extensions"] = {}
params.get("extensions")["ignoreRepaint"] = "true"
if self.alias:
for k, v in self.alias.items():
name = str(name).replace(k, v)
# Forcing not-Taurus items to have a name and be able to trigger events
setattr(item, "_name", name)
if name and not self._delayed:
if isinstance(item, TaurusGraphicsItem):
item.setModel(name)
else:
self.debug(
"TaurusJDrawGraphicsFactory.set_common_params(%s): "
+ "%s is not a TaurusGraphicsItem",
name,
type(item).__name__,
)
visibilitymapper = params.get("visibilitymapper")
if visibilitymapper is not None:
# mapping_type = visibilitymapper["mapping_type"]
# mode = visibilitymapper["mode"]
default = visibilitymapper["default"]
item._default = default
item._map = visibilitymapper["map"]
visible = params.get("visible")
if visible is not None:
item.setVisible(visible)
extensions = params.get("extensions")
if extensions:
item._extensions = extensions
try:
getattr(item, "setPen")
fg = params.get("foreground", (0, 0, 0))
lineWidth = params.get("lineWidth", 1)
if lineWidth == 0:
pen = Qt.QPen(Qt.Qt.NoPen)
else:
pen = Qt.QPen(Qt.QColor(int(fg[0]), int(fg[1]), int(fg[2])))
pen.setWidth(int(lineWidth))
pen.setStyle(LINESTYLE_JDW2QT[params.get("lineStyle", 0)])
item.setPen(pen)
except AttributeError:
pass
except Exception:
self.warning(
"jdraw.set_common_params(%s(%s)).(foreground,width,style) "
+ "failed!: \n\t%s",
type(item).__name__,
name,
traceback.format_exc(),
)
fillStyle = FILLSTYLE_JDW2QT[params.get("fillStyle", 0)]
item._fillStyle = fillStyle
if hasattr(item, "brush"):
brush = Qt.QBrush()
if fillStyle == Qt.Qt.LinearGradientPattern:
ox, oy = params.get("origin", (0, 0))
gradient = Qt.QLinearGradient(
ox + params.get("gradX1", 0),
oy + params.get("gradY1", 0),
ox + params.get("gradX2", 0),
oy + params.get("gradY2", 0),
)
c = params.get("gradC1", (0, 0, 0))
gradient.setColorAt(0, Qt.QColor(c[0], c[1], c[2]))
c = params.get("gradC2", (255, 255, 255))
gradient.setColorAt(1, Qt.QColor(c[0], c[1], c[2]))
gradCyclic = params.get("gradCyclic", False)
if gradCyclic:
gradient.setSpread(Qt.QGradient.ReflectSpread)
brush = Qt.QBrush(gradient)
else:
brush.setStyle(fillStyle)
bg = params.get("background", (255, 255, 255))
brush.setColor(Qt.QColor(bg[0], bg[1], bg[2]))
item.setBrush(brush)
[docs] def set_item_filling(
self, item, pattern=Qt.Qt.Dense4Pattern, expand=False
):
item._fillStyle = item._fillStyle or pattern
if hasattr(item, "brush"):
br = item.brush()
br.setStyle(item._fillStyle)
item.setBrush(br)
if expand:
for c in item.childItems():
if not c._fillStyle:
self.set_item_filling(c, pattern=pattern, expand=True)
return
if __name__ == "__main__":
from taurus.qt.qtgui.graphic.jdraw import jdraw_view
jdraw_view.jdraw_view_main()