Source code for taurus.qt.qtgui.graphic.jdraw.jdraw_view

#!/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 view widget for jdraw files"""
from __future__ import absolute_import

from builtins import str

import os
import traceback

from future.utils import string_types

import taurus
from taurus.external.qt import Qt, compat
from taurus.core.taurusbasetypes import TaurusElementType
from taurus.core import taurushelper
from taurus.qt.qtgui.graphic.taurusgraphic import parseTangoUri, TaurusGraphicsItem, SynopticSelectionStyle
from taurus.qt.qtcore.mimetypes import TAURUS_ATTR_MIME_TYPE, TAURUS_DEV_MIME_TYPE, TAURUS_MODEL_MIME_TYPE
from taurus.qt.qtgui.base import TaurusBaseWidget

from taurus.qt.qtgui.graphic.jdraw.jdraw_parser import parse

__all__ = ["TaurusJDrawSynopticsView"]

__docformat__ = 'restructuredtext'


[docs]class TaurusJDrawSynopticsView(Qt.QGraphicsView, TaurusBaseWidget): ''' Taurus Class that visualizes Synoptics drawn with the JDraw tool (by ESRF). It is equivalent to ATK Synoptic Player (Java). After initialization call setModel('/your/file.jdw') to parse the synoptic file and connect to controlled objects. Arguments to TaurusJDrawSynopticsView() creator are: - designMode; used by Qt Designer - updateMode; controls Qt Viewport refresh (disabled by default) - alias; a dictionary of name replacements to be applied on graphical objects - resizable: whether to allow resizing or not - panelClass: class object, class name or shell command to be shown when an object is clicked (None will show default panel, '' or 'noPanel' will disable it) TaurusJDrawSynopticsView and TaurusGraphicsScene signals/slots External events:: Slot selectGraphicItem(const QString &) displays a selection mark around the TaurusGraphicsItem that matches the argument passed. Mouse Left-button events:: Signal graphicItemSelected(QString) is triggered, passing the selected TaurusGraphicsItem.name() as argument. Mouse Right-button events:: TaurusGraphicsItem.setContextMenu([(ActionName,ActionMethod(device_name))] allows to configure custom context menus for graphic items using a list of tuples. Empty tuples will insert separators in the menu. ''' itemsChanged = Qt.pyqtSignal('QString', dict) modelsChanged = Qt.pyqtSignal(list) graphicItemSelected = Qt.pyqtSignal('QString') graphicSceneClicked = Qt.pyqtSignal('QPoint') def __init__(self, parent=None, designMode=False, updateMode=None, alias=None, resizable=True, panelClass=None): name = self.__class__.__name__ self.call__init__wo_kw(Qt.QGraphicsView, parent) self.call__init__(TaurusBaseWidget, name, designMode=designMode) self._currF = self.modelName self.path = '' self.w_scene = None self.h_scene = None self._fileName = "Root" self._mousePos = (0, 0) self._selectionStyle = SynopticSelectionStyle.OUTLINE self.setResizable(resizable) self.setInteractive(True) self.setAlias(alias) self.setDragEnabled(True) self.setPanelClass(panelClass) # By default the items will update the view when necessary. # This default value is much more efficient then the QQraphicsView default # value, so if you decide to change then expect a lot of processor to be # used by your application. if updateMode is None: self.setViewportUpdateMode(Qt.QGraphicsView.NoViewportUpdate) else: self.setViewportUpdateMode(updateMode)
[docs] def defineStyle(self): self.updateStyle()
#-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- # TaurusBaseWidget over writing #-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-
[docs] def isReadOnly(self): return True
[docs] def update(self): # self.emit_signal() self.emitColors()
[docs] def openJDraw(self): ifile, _ = compat.getOpenFileName( self, 'Load JDraw File', '', 'JDraw File (*.jdw)') if not ifile: return fileName = ifile.split("/") self._fileName = fileName[-1] self.setModel(ifile) return fileName[-1]
[docs] def setAlias(self, alias): """ Assigning a dictionary like {'Tag':'Value'} with tags to be replaced in object names while parsing. """ if (isinstance(alias, dict) or hasattr(alias, 'items')) and alias: self.alias = alias else: self.alias = None return
[docs] def get_item_list(self): return [item._name for item in self.scene().items() if hasattr(item, '_name') and item._name]
[docs] def get_device_list(self): items = [(item, parseTangoUri(item)) for item in self.get_item_list()] return list(set(v['_devslashname'] for k, v in items if v))
[docs] def get_item_colors(self, emit=False): item_colors = {} try: for item in self.scene().items(): if not getattr(item, '_name', '') or not getattr(item, '_currBgBrush', None): continue item_colors[item._name] = item._currBgBrush.color().name() if emit: self.itemsChanged.emit(self.modelName.split( '/')[-1].split('.')[0], item_colors) except: self.warning('Unable to emitColors: %s' % traceback.format_exc()) return item_colors
@Qt.pyqtSlot(object) @Qt.pyqtSlot('QString')
[docs] def selectGraphicItem(self, item_name): self.scene().selectGraphicItem(item_name) return False
def _graphicItemSelected(self, item_name): self.debug(' => graphicItemSelected(QString)(%s)' % item_name) self.graphicItemSelected.emit(item_name) def _graphicSceneClicked(self, point): self.debug('In TaurusJDrawSynopticsView.graphicSceneClicked(%s,%s)' % ( point.x(), point.y())) self.graphicSceneClicked.emit(point) def __modelsChanged(self): items = self.get_item_list() self.debug('modelsChanged(%s)' % len(items)) self.modelsChanged.emit(items)
[docs] def emitColors(self): '''emit signal which is used to refresh the tree and colors of icons depend of the current status in jdrawSynoptic''' self.get_item_colors(True)
[docs] def get_sizes(self): srect = self.scene().sceneRect() sizes = [x for s in (self.size(), self.sizeHint(), srect.size()) for x in (s.width(), s.height())] try: s = self.parent().size() sizes.extend([s.width(), s.height()]) except: sizes.extend([0, 0]) return tuple(sizes)
[docs] def fitting(self, ADJUST_FRAME=False): """ Parent size is the size of the bigger panel (desn't keep ratio) Rect size never changes (fixed by the graphics objects) Size and SizeHint move one around the other the method works well until an object is clicked, then the whole reference changes and doesn't work again. """ srect = self.scene().sceneRect() w, h = (srect.width(), srect.height()) offset = self.mapToGlobal(Qt.QPoint(srect.x(), srect.y())) x0, y0 = (offset.x(), offset.y()) self.debug('\n\nIn TauJDrawSynopticsView.fitting()') self.debug(self.get_sizes()) self.debug( '\tAdjusting SizeHint: size(%s,%s),hint(%s,%s),srect(%s,%s),parent(%s,%s)' % self.get_sizes()) self.debug('\toffset = %s,%s ; size = %s,%s' % (x0, y0, w, h)) self.fitInView(x0, y0, w, h, Qt.Qt.KeepAspectRatio) # ------------------------------------------------------------------- # This block seems to be a poorly executed workaround to some # issue, but it is itself causing bugs (see # https://github.com/taurus-org/taurus/issues/484 ). # We are not removing it yet to preserve bck-compat # TODO: Deprecate the ADJUST_FRAME kwarg? if ADJUST_FRAME: # This additional resizing adjust the "white" frame around the synoptic self.debug( '\tResizing: size(%s,%s),hint(%s,%s),srect(%s,%s),parent(%s,%s)' % self.get_sizes()) self.resize(self.sizeHint() + Qt.QSize(5, 5)) # ------------------------------------------------------------------- # THIS LINE MUST BE ALWAYS EXECUTED, It prevents the UP/DOWN resize BUG!!! # apparently Qt needs this 2 fitInView calls to be aware of it, maybe # first offset was not good self.debug( '\tFitting:: size(%s,%s),hint(%s,%s),srect(%s,%s),parent(%s,%s)' % self.get_sizes()) self.fitInView(x0, y0, w, h, Qt.Qt.KeepAspectRatio) self.debug( 'Done: size(%s,%s),hint(%s,%s),srect(%s,%s),parent(%s,%s)\n\n' % self.get_sizes())
[docs] def resizeEvent(self, event): """ It has been needed to reimplent size policies """ if not self.resizable() or not self.scene() or isinstance(self.parent(), Qt.QScrollArea) or not self.isVisible(): self.debug('In TaurusJDrawSynopticsView(' + self._fileName + ').resizeEvent(): Disabled') return try: self.debug('In TaurusJDrawSynopticsView(' + self._fileName + ').resizeEvent()') if self.size() == self.sizeHint(): self.debug('\tSize already fits: %s' % self.size()) return self.setVerticalScrollBarPolicy(Qt.Qt.ScrollBarAlwaysOff) self.setHorizontalScrollBarPolicy(Qt.Qt.ScrollBarAlwaysOff) self.fitting() self.emitColors() except Exception: self.warning('Exception in JDrawView(' + self._fileName + ').resizeEvent: %s' % traceback.format_exc()) pass
[docs] def refreshModel(self): self.setModel(self.getModelName())
[docs] def updateStyle(self): self.repaint()
[docs] def repaint(self): Qt.QGraphicsView.repaint(self)
# self.fitting()
[docs] def getGraphicsFactory(self, delayed=False): from . import jdraw # self.parent()) return jdraw.TaurusJDrawGraphicsFactory(self, alias=(self.alias or None), delayed=delayed)
###########################################################################
[docs] def getFramed(self): try: parent = self.parent() except: parent = None frame = Qt.QFrame(parent) self.setFrameStyle(self.StyledPanel | self.Raised) self.setLineWidth(2) self.setParent(frame) return frame
[docs] def setResizable(self, resizable): self._resizable = resizable
[docs] def resizable(self): return self._resizable
[docs] def mousePressEvent(self, event): """ Records last event position to use it for DragEvents """ try: self.mousePos = event.scenePos().x(), event.scenePos().y() except: self.mousePos = event.x(), event.y() self.debug('MouseEvent received is not a GraphicsScene event, using raw position %s' % str( self.mousePos)) TaurusBaseWidget.mousePressEvent(self, event)
[docs] def getModelMimeData(self): """ Used for drag events """ model, mimeData = '', None try: #model = getattr(self.scene().itemAt(*self.mousePos),'_name','') selected = self.scene()._selectedItems if not selected: self.debug( 'jdrawView.getModelMimeData(%s): nothing to drag' % model) return model = getattr(([s for s in selected if s.isUnderMouse() and getattr( s, '_name', '')] or [selected])[0], '_name', '') self.debug('getModelMimeData(%s)' % model) mimeData = Qt.QMimeData() if model: taurusType = taurushelper.getValidTypesForName(model, False) model = str(model).encode('utf8') if TaurusElementType.Device in taurusType: mimeData.setData(TAURUS_DEV_MIME_TYPE, model) if TaurusElementType.Attribute in taurusType: mimeData.setData(TAURUS_ATTR_MIME_TYPE, model) else: mimeData.setData(TAURUS_MODEL_MIME_TYPE, model) except: self.debug( 'jdrawView.getModelMimeData(%s): unable to get MimeData' % model) self.debug(traceback.format_exc()) return mimeData
#-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- # QT properties #-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~-~- @classmethod
[docs] def setDefaultPanelClass(klass, other): """ This method returns the Class used to open new object panels on double-click (TaurusDevicePanel by default) """ klass._defaultClass = other
@classmethod
[docs] def defaultPanelClass(klass): """ This method assigns the Class used to open new object panels on double-click (TaurusDevicePanel by default) If an string is used it can be either a Taurus class or an OS launcher """ if not hasattr(klass, '_defaultClass'): from taurus.qt.qtgui.panel import TaurusDevicePanel # 'taurusdevicepanel' #You can use an executable or a class klass._defaultClass = TaurusDevicePanel obj = klass._defaultClass return obj
[docs] def setPanelClass(self, widget): self._panelClass = widget
[docs] def panelClass(self): if self._panelClass is None: return self.defaultPanelClass() else: return self._panelClass
@Qt.pyqtSlot('QString')
[docs] def setModel(self, model, alias=None, delayed=False, trace=False): self.modelName = str(model) self._currF = str(model) if alias is not None: self.setAlias(alias) ll = taurus.getLogLevel() if trace: taurus.setLogLevel(taurus.Debug) # self.debug('setModel("%s")'%model) if self._currF: #filename = str(self._currFile.absoluteFilePath()) filename = self._currF filename = os.path.realpath(filename) if os.path.isfile(filename): self.debug("Starting to parse %s" % filename) self.path = os.path.dirname(filename) factory = self.getGraphicsFactory(delayed=delayed) scene = parse(filename, factory) scene.setSelectionStyle(self._selectionStyle) self.debug("Obtained %s(%s)", type(scene).__name__, filename) if not scene: self.warning("TaurusJDrawSynopticsView.setModel(%s): Unable to parse %s!!!" % ( model, filename)) elif self.w_scene is None and scene.sceneRect(): self.w_scene = scene.sceneRect().width() self.h_scene = scene.sceneRect().height() else: self.debug('JDrawView.sceneRect() is NONE!!!') self.setScene(scene) self.scene().graphicItemSelected.connect(self._graphicItemSelected) self.scene().graphicSceneClicked.connect(self._graphicSceneClicked) # Qt.QApplication.instance().lastWindowClosed.connect(self.close) #It caused a # segfault! self.__modelsChanged() self.setWindowTitle(self.modelName) # The emitted signal contains the filename and a dictionary # with the name of items and its color self.emitColors() # get_item_colors(emit=True) self.fitting() else: self.setScene(None) #self.debug('out of setModel()') taurus.setLogLevel(ll)
[docs] def closeEvent(self, event=None): if self.scene(): self.scene().closeAllPanels() Qt.QGraphicsView.closeEvent(self, event)
[docs] def setModels(self): """ This method triggers item.setModel(item._name) in all internal items. """ for item in self.scene().items(): if item._name and isinstance(item, TaurusGraphicsItem): self.debug( 'TaurusJDrawGraphicsFactory.setModels(): calling item.setModel(%s)' % (item._name)) item.setModel(item._name)
[docs] def getModel(self): return self._currF
@classmethod
[docs] def getQtDesignerPluginInfo(cls): ret = TaurusBaseWidget.getQtDesignerPluginInfo() ret['group'] = 'Taurus Display' ret['module'] = 'taurus.qt.qtgui.graphic' ret['icon'] = "designer:graphicsview.png" return ret
model = Qt.pyqtProperty("QString", getModel, setModel)
[docs] def setSelectionStyle(self, selectionStyle): if isinstance(selectionStyle, string_types): selectionStyle = str(selectionStyle).upper() try: selectionStyle = SynopticSelectionStyle[selectionStyle] except: self.debug('invalid selectionStyle "%s"', selectionStyle) return if self.scene() is not None: self.scene().setSelectionStyle(selectionStyle) self._selectionStyle = selectionStyle
[docs] def getSelectionStyle(self): return self._selectionStyle
[docs] def getSelectionStyleName(self): return SynopticSelectionStyle.whatis(self.getSelectionStyle())
[docs] def resetSelectionStyle(self): self.setSelectionStyle(SynopticSelectionStyle.OUTLINE)
selectionStyle = Qt.pyqtProperty("QString", getSelectionStyleName, setSelectionStyle, resetSelectionStyle)
#!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! def jdraw_view_main(): import sys import time import taurus.qt.qtgui.graphic taurus.setLogLevel(taurus.Info) from taurus.qt.qtgui.application import TaurusApplication app = TaurusApplication(sys.argv) #form = Qt.QDialog() # ly=Qt.QVBoxLayout(form) # container=Qt.QWidget() # ly.addWidget(container) # for m in sys.argv[1:]: #tv=TaurusJDrawSynopticsView(container, designMode=False) # tv.setModel(m) form = taurus.qt.qtgui.graphic.TaurusJDrawSynopticsView(designMode=False) form.show() # print '%s setModel(%s)'%(time.ctime(),sys.argv[1]) form.setModel(sys.argv[1]) form.setWindowTitle(sys.argv[1].rsplit('.', 1)[0]) sys.exit(app.exec_()) if __name__ == "__main__": jdraw_view_main()