# SPDX-FileCopyrightText: Copyright (c) Kitware Inc.
# SPDX-License-Identifier: BSD-3-Clause
r"""servermanager is a module for using paraview server manager in Python.
One can always use the server manager API directly. However, this module
provides an interface easier to use from Python by wrapping several VTK
classes around Python classes.
Note that, upon load, this module will create several sub-modules: sources,
filters and rendering. These modules can be used to instantiate specific
proxy types. For a list, try "dir(servermanager.sources)"
Usually users should use the paraview.simple module instead as it provide a
more user friendly API.
A simple example::
from paraview.servermanager import *
# Creates a new built-in session and makes it the active session.
Connect()
# Creates a new render view on the active session.
renModule = CreateRenderView()
# Create a new sphere proxy on the active session and register it
# in the sources group.
sphere = sources.SphereSource(registrationGroup="sources", ThetaResolution=16, PhiResolution=32)
# Create a representation for the sphere proxy and adds it to the render
# module.
display = CreateRepresentation(sphere, renModule)
renModule.StillRender()
"""
from __future__ import print_function
import paraview, re, os, os.path, types, sys
# prefer `vtk` from `paraview` since it doesn't import all
# vtk modules.
from paraview import vtk
from paraview import _backwardscompatibilityhelper as _bc
from paraview.modules.vtkPVVTKExtensionsCore import *
from paraview.modules.vtkRemotingCore import *
from paraview.modules.vtkRemotingServerManager import *
from paraview.modules.vtkRemotingSettings import *
from paraview.modules.vtkRemotingApplication import *
try:
from paraview.modules.vtkRemotingViews import *
except ImportError:
pass
try:
from paraview.modules.vtkRemotingAnimation import *
except ImportError:
pass
try:
from paraview.modules.vtkRemotingExport import *
except ImportError:
pass
try:
from paraview.modules.vtkRemotingImport import *
except ImportError:
pass
try:
from paraview.modules.vtkRemotingMisc import *
except ImportError:
pass
try:
from paraview.modules.vtkRemotingLive import *
except ImportError:
pass
def _wrap_property(proxy, smproperty):
""" Internal function.
Given a server manager property and its domains, returns the
appropriate python object.
"""
property = None
if paraview.compatibility.GetVersion() >= (3, 5) and \
smproperty.IsA("vtkSMStringVectorProperty"):
arraySelectionDomain = smproperty.FindDomain("vtkSMArraySelectionDomain")
chartSeriesSelectionDomain = smproperty.FindDomain("vtkSMChartSeriesSelectionDomain")
arrayListDomain = smproperty.FindDomain("vtkSMArrayListDomain")
fileListDomain = smproperty.FindDomain("vtkSMFileListDomain")
stringListDomain = smproperty.FindDomain("vtkSMStringListDomain")
if arraySelectionDomain and smproperty.GetRepeatable():
property = ArrayListProperty(proxy, smproperty)
elif chartSeriesSelectionDomain and smproperty.GetRepeatable() and \
chartSeriesSelectionDomain.GetDefaultMode() == 1:
property = ArrayListProperty(proxy, smproperty)
elif arrayListDomain and smproperty.GetRepeatable():
# if it is repeatable, then it is not a single array selection... and if it happens
# to have 5 elements in the repeatable proxy, avoid an exception by testing this case
# first.
property = VectorProperty(proxy, smproperty)
elif arrayListDomain and smproperty.GetNumberOfElements() == 5:
property = ArraySelectionProperty(proxy, smproperty)
elif fileListDomain and fileListDomain.GetIsOptional() == 0:
# Refer to BUG #9710 to see why optional domains need to be ignored.
property = FileNameProperty(proxy, smproperty)
elif stringListDomain:
property = StringListProperty(proxy, smproperty)
else:
property = VectorProperty(proxy, smproperty)
elif smproperty.IsA("vtkSMVectorProperty"):
if smproperty.IsA("vtkSMIntVectorProperty") and \
smproperty.FindDomain("vtkSMEnumerationDomain"):
property = EnumerationProperty(proxy, smproperty)
else:
property = VectorProperty(proxy, smproperty)
elif smproperty.IsA("vtkSMInputProperty"):
property = InputProperty(proxy, smproperty)
elif smproperty.IsA("vtkSMProxyProperty"):
property = ProxyProperty(proxy, smproperty)
elif smproperty.IsA("vtkSMDoubleMapProperty"):
property = DoubleMapProperty(proxy, smproperty)
else:
property = Property(proxy, smproperty)
return property
[docs]class ParaViewPipelineController(object):
"""ParaViewPipelineController wraps vtkSMParaViewPipelineController class
to manage conversion of arguments passed around from Python Proxy objects to
vtkSMProxy instances are vice-versa."""
def __init__(self):
"""Constructor. Creates a new instance of
vtkSMParaViewPipelineController."""
self.SMController = vtkSMParaViewPipelineController()
def __ConvertArgumentsAndCall(self, *args):
newArgs = []
for arg in args:
# convert Proxy and ProxyManager to vtkSMProxy and
# vtkSMSessionProxyManager instances.
# FIXME: should handle session as well?
if issubclass(type(arg), Proxy) or isinstance(arg, Proxy):
newArgs.append(arg.SMProxy)
elif issubclass(type(arg), ProxyManager) or isinstance(arg, ProxyManager):
newArgs.append(arg.SMProxyManager)
else:
newArgs.append(arg)
func = getattr(self.SMController, self.__LastAttrName)
retVal = func(*newArgs)
if isinstance(retVal, vtkSMProxy):
# if this is a vtkObject and is a "vtkSMProxy", return a Proxy().
return _getPyProxy(retVal)
else:
return retVal
def __getattr__(self, name):
"""Returns attribute from the ParaViewPipelineController."""
try:
pmAttr = getattr(self.SMController, name)
self.__LastAttrName = name
return self.__ConvertArgumentsAndCall
except:
pass
return getattr(self.SMController, name)
[docs]class Proxy(object):
"""Proxy for a server side object. A proxy manages the lifetime of
one or more server manager objects. It also provides an interface
to set and get the properties of the server side objects. These
properties are presented as Python properties. For example,
you can set a property Foo using the following::
proxy.Foo = (1,2)
or::
proxy.Foo.SetData((1,2))
or::
proxy.Foo[0:2] = (1,2)
For more information, see the documentation of the property which
you can obtain with::
help(proxy.Foo).
This class also provides an iterator which can be used to iterate
over all properties, e.g.::
proxy = Proxy(proxy=smproxy)
for property in proxy:
print (property)
For advanced users:
This is a python class that wraps a vtkSMProxy. Makes it easier to
set/get properties.
Instead of::
proxy.GetProperty("Foo").SetElement(0, 1)
proxy.GetProperty("Foo").SetElement(0, 2)
you can do::
proxy.Foo = (1,2)
or::
proxy.Foo.SetData((1,2))
or::
proxy.Foo[0:2] = (1,2)
Instead of::
proxy.GetProperty("Foo").GetElement(0)
you can do::
proxy.Foo.GetData()[0]
or::
proxy.Foo[0]
For proxy properties, you can use append::
proxy.GetProperty("Bar").AddProxy(foo)
you can do::
proxy.Bar.append(foo)
Properties support most of the list API. See ``VectorProperty`` and
``ProxyProperty`` documentation for details.
Please note that some of the methods accessible through the Proxy
class are not listed by ``help()`` because the ``Proxy`` objects forward
unresolved attributes to the underlying object. To get the full list,
see also ``dir(proxy.SMProxy)``. See also the doxygen based documentation
of the ``vtkSMProxy`` C++ class.
"""
def __init__(self, **args):
""" Default constructor. It can be used to initialize properties
by passing keyword arguments where the key is the name of the
property. In addition registrationGroup and registrationName (optional)
can be specified (as keyword arguments) to automatically register
the proxy with the proxy manager. """
self.add_attribute('Observed', None)
self.add_attribute('ObserverTag', -1)
self.add_attribute('_Proxy__Properties', {})
self.add_attribute('_Proxy__LastAttrName', None)
self.add_attribute('SMProxy', None)
self.add_attribute('IgnoreUnknownSetRequests', False)
if 'port' in args:
self.add_attribute('Port', args['port'])
del args['port']
else:
self.add_attribute('Port', 0)
update = True
if 'no_update' in args:
if args['no_update']:
update = False
del args['no_update']
if 'proxy' in args:
self.InitializeFromProxy(args['proxy'])
del args['proxy']
else:
self.Initialize(None, update)
if 'registrationGroup' in args:
registrationGroup = args['registrationGroup']
del args['registrationGroup']
registrationName = self.SMProxy.GetGlobalIDAsString()
if 'registrationName' in args:
registrationName = args['registrationName']
del args['registrationName']
pxm = ProxyManager()
pxm.RegisterProxy(registrationGroup, registrationName, self.SMProxy)
if update:
self.UpdateVTKObjects()
for key in args.keys():
setattr(self, key, args[key])
# Visit all properties so that they are created
for prop in self:
pass
[docs] def add_attribute(self, name, value):
self.__dict__[name] = value
def __del__(self):
"""Destructor. Cleans up all observers as well as remove
the proxy from the _pyproxies dictionary"""
# Make sure that we remove observers we added
if self.Observed:
observed = self.Observed
tag = self.ObserverTag
self.Observed = None
self.ObserverTag = -1
observed.RemoveObserver(tag)
if _pyproxies and self.SMProxy and (self.SMProxy, self.Port) in _pyproxies:
del _pyproxies[(self.SMProxy, self.Port)]
[docs] def InitializeFromProxy(self, aProxy, update=True):
"""Constructor. Assigns proxy to self.SMProxy, updates the server
object as well as register the proxy in _pyproxies dictionary."""
import weakref
self.SMProxy = aProxy
if update:
self.SMProxy.UpdateVTKObjects()
_pyproxies[(self.SMProxy, self.Port)] = weakref.ref(self)
[docs] def Initialize(self):
"Overridden by the subclass created automatically"
pass
def __eq__(self, other):
"Returns true if the underlying SMProxies are the same."
if isinstance(other, Proxy):
try:
if self.Port != other.Port:
return False
except:
pass
return self.SMProxy == other.SMProxy
return self.SMProxy == other
def __ne__(self, other):
"Returns false if the underlying SMProxies are the same."
return not self.__eq__(other)
def __hash__(self):
return hash(self.SMProxy)
def __iter__(self):
"Creates an iterator for the properties."
return PropertyIterator(self)
[docs] def SetPropertyWithName(self, pname, arg):
"""Generic method for setting the value of a property."""
prop = self.GetProperty(pname)
if prop is None:
raise RuntimeError("Property %s does not exist. Please check the property name for typos." % pname)
prop.SetData(arg)
[docs] def GetPropertyValue(self, name):
"""Returns a scalar for properties with 1 elements, the property
itself for vectors."""
p = self.GetProperty(name)
if isinstance(p, VectorProperty) and paraview.compatibility.GetVersion() <= (4, 1) and name == "ColorArrayName":
# Return ColorArrayName as just the array name for backwards compatibility.
return p[1]
if isinstance(p, EnumerationProperty) or \
isinstance(p, ArraySelectionProperty) or \
isinstance(p, StringListProperty) or \
isinstance(p, ArrayListProperty):
# with domain based property, return the property to provide access to Available method
return p
elif isinstance(p, VectorProperty):
if p.GetNumberOfElements() == 1 and not p.GetRepeatable():
if p.SMProperty.IsA("vtkSMStringVectorProperty") or not p.GetArgumentIsArray():
return p[0]
elif isinstance(p, InputProperty):
if not p.GetMultipleInput():
if len(p) > 0:
return p[0]
else:
return None
elif isinstance(p, ProxyProperty):
if not p.GetRepeatable():
if len(p) > 0:
return p[0]
else:
return None
return p
[docs] def GetProperty(self, name):
"""Given a property name, returns the property object."""
if name in self.__Properties and self.__Properties[name]():
return self.__Properties[name]()
smproperty = self.SMProxy.GetProperty(name)
# Maybe they are looking by the label. Try to match that.
if not smproperty:
iter = PropertyIterator(self)
for prop in iter:
if name == _make_name_valid(iter.PropertyLabel):
smproperty = prop.SMProperty
break
if smproperty:
property = _wrap_property(self, smproperty)
if property is not None:
import weakref
self.__Properties[name] = weakref.ref(property)
return property
return None
[docs] def ListProperties(self):
"""Returns a list of all property names on this proxy."""
property_list = []
iter = self.__iter__()
for property in iter:
name = _make_name_valid(iter.PropertyLabel)
if name:
property_list.append(name)
return property_list
def __ConvertArgumentsAndCall(self, *args):
""" Internal function.
Used to call a function on SMProxy. Converts input and
output values as appropriate.
"""
newArgs = []
for arg in args:
if issubclass(type(arg), Proxy) or isinstance(arg, Proxy):
newArgs.append(arg.SMProxy)
else:
newArgs.append(arg)
func = getattr(self.SMProxy, self.__LastAttrName)
retVal = func(*newArgs)
if isinstance(retVal, vtkSMProxy):
return _getPyProxy(retVal)
elif isinstance(retVal, vtkSMProperty):
return _wrap_property(self, retVal)
else:
return retVal
def __GetActiveCamera(self):
""" This method handles GetActiveCamera specially.
We return a decorated vtkCamera object so that whenever
the Camera is directly modified using Python API,
we ensure that the Camera properties on the corresponding
view proxy are synchronized with the underlying vtkCamera.
"""
import weakref
camera = self.SMProxy.GetActiveCamera()
proxy = weakref.ref(self)
from functools import wraps
def _camera_sync(method):
@wraps(method)
def newfunc(*args, **kwargs):
result = method(*args, **kwargs)
if proxy():
proxy().SynchronizeCameraProperties()
return result
return newfunc
# Camera eventually inherit from Object
class _camera_wrapper(object):
def __getattribute__(self, s):
return _camera_sync(camera.__getattribute__(s))
# Calls to __dir__ bypass the __getattribute__ function, so override it here
# to delegate it to the vtkCameara class
def __dir__(self):
try:
return camera.__dir__()
except:
return []
return _camera_wrapper()
def __setattr__(self, name, value):
try:
setter = getattr(self.__class__, name)
setter = setter.__set__
except AttributeError:
paraview.print_debug_info("No attribute %s" % name)
# Let the backwards compatibility helper try to handle this
try:
_bc.setattr(self, name, value)
except _bc.Continue:
pass
except AttributeError:
if self.IgnoreUnknownSetRequests:
pass
else:
raise AttributeError("Attribute %s does not exist. " % name +
" This class does not allow addition of new attributes to avoid " +
"mistakes due to typos. Use add_attribute() if you really want " +
"to add this attribute.")
else:
paraview.print_debug_info("Setting '%s' as '%s'", name, value)
try:
setter(self, value)
except ValueError:
# Try without spaces
try:
setter(self, _make_name_valid(value))
except ValueError:
# Let the backwards compatibility helper try to handle this
try:
_bc.setattr_fix_value(self, name, value, setter)
except _bc.Continue:
pass
except ValueError:
raise ValueError("%s is not a valid value for attribute %s." % (value, name))
def __getattr__(self, name):
"""With the exception of a few overloaded methods,
returns the SMProxy method"""
if not self.SMProxy:
raise AttributeError("class %s has no attribute %s" % ("None", name))
return None
# Handle GetActiveCamera specially.
if name == "GetActiveCamera" and \
hasattr(self.SMProxy, "GetActiveCamera"):
return self.__GetActiveCamera
if name == "SaveDefinition" and hasattr(self.SMProxy, "SaveDefinition"):
return self.__SaveDefinition
try:
return _bc.getattr(self, name)
except _bc.Continue:
pass
# If not a property, see if SMProxy has the method
try:
proxyAttr = getattr(self.SMProxy, name)
self.__LastAttrName = name
return self.__ConvertArgumentsAndCall
except:
pass
return getattr(self.SMProxy, name)
[docs]class SourceProxy(Proxy):
"""Proxy for a source object. This class adds a few methods to Proxy
that are specific to sources. It also provides access to the output
ports. Output ports can be accessed by name or index:
> op = source[0]
or
> op = source['some name'].
"""
[docs] def UpdatePipeline(self, time=None):
"""This method updates the server-side VTK pipeline and the associated
data information. Make sure to update a source to validate the output
meta-data."""
if time != None:
self.SMProxy.UpdatePipeline(time)
else:
self.SMProxy.UpdatePipeline()
# This is here to cause a receive
# on the client side so that progress works properly.
if ActiveConnection and ActiveConnection.IsRemote():
self.SMProxy.GetDataInformation()
[docs] def FileNameChanged(self):
"Called when the filename of a source proxy is changed."
self.UpdatePipelineInformation()
def __getitem__(self, idx):
"""Given a slice, int or string, returns the corresponding
output port"""
if isinstance(idx, slice):
indices = idx.indices(self.SMProxy.GetNumberOfOutputPorts())
retVal = []
for i in range(*indices):
retVal.append(OutputPort(self, i))
return retVal
elif isinstance(idx, int):
if idx >= self.SMProxy.GetNumberOfOutputPorts() or idx < 0:
raise IndexError
return OutputPort(self, idx)
else:
return OutputPort(self, self.SMProxy.GetOutputPortIndex(idx))
PointData = property(GetPointDataInformation, None, None, "Returns point data information")
CellData = property(GetCellDataInformation, None, None, "Returns cell data information")
FieldData = property(GetFieldDataInformation, None, None, "Returns field data information")
[docs]class ExodusIIReaderProxy(SourceProxy):
"""Special class to define convenience functions for array
selection."""
if paraview.compatibility.GetVersion() >= (3, 5):
[docs] def FileNameChanged(self):
"Called when the filename changes. Selects all variables."
SourceProxy.FileNameChanged(self)
self.SelectAllVariables()
[docs] def SelectAllVariables(self):
"Select all available variables for reading."
for prop in ('PointVariables', 'EdgeVariables', 'FaceVariables',
'ElementVariables', 'GlobalVariables'):
f = getattr(self, prop)
f.SelectAll()
[docs] def DeselectAllVariables(self):
"Deselects all variables."
for prop in ('PointVariables', 'EdgeVariables', 'FaceVariables',
'ElementVariables', 'GlobalVariables'):
f = getattr(self, prop)
f.DeselectAll()
[docs]class MultiplexerSourceProxy(SourceProxy):
[docs] def UpdateDynamicProperties(self):
"""Update the instance to add properties of newly exposed
SMProperties. The current limitation is that help() still
doesn't work as one would expect."""
exclude = frozenset([p for p in dir(self.__class__) if isinstance(getattr(self.__class__, p), property)])
cdict = _createClassProperties(self, exclude)
for key, val in cdict.items():
self.add_attribute(key, val)
[docs]class ViewLayoutProxy(Proxy):
"""Special class to define convenience methods for View Layout"""
[docs] def SplitViewHorizontal(self, view, fraction=0.5):
"""Split the cell containing the specified view horizontally.
If no fraction is specified, the frame is split into equal parts.
On success returns a positive number that identifying the new cell
location that can be used to assign view to, or split further.
Return -1 on failure."""
location = self.GetViewLocation(view)
if location == -1:
raise RuntimeError("View is not present in this layout.")
if fraction < 0.0 or fraction > 1.0:
raise RuntimeError("'fraction' must be in the range [0.0, 1.0]")
return self.SMProxy.SplitHorizontal(location, fraction)
[docs] def SplitViewVertical(self, view=None, fraction=0.5):
"""Split the cell containing the specified view horizontally.
If no view is specified, active view is used.
If no fraction is specified, the frame is split into equal parts.
On success returns a positive number that identifying the new cell
location that can be used to assign view to, or split further.
Return -1 on failure."""
location = self.GetViewLocation(view)
if location == -1:
raise RuntimeError("View is not present in this layout.")
if fraction < 0.0 or fraction > 1.0:
raise RuntimeError("'fraction' must be in the range [0.0, 1.0]")
return self.SMProxy.SplitVertical(location, fraction)
[docs] def AssignView(self, location, view):
"""Assign a view at a particular location. Note that the view's position may
be changed by subsequent Split() calls. Returns true on success."""
viewproxy = None
if isinstance(view, Proxy):
view = view.SMProxy
return self.SMProxy.AssignView(location, view)
[docs] def GetViewLocation(self, view):
if isinstance(view, Proxy):
view = view.SMProxy
return self.SMProxy.GetViewLocation(view)
[docs]class Property(object):
"""Generic property object that provides access to one of the properties of
a server object. This class does not allow setting/getting any values but
provides an interface to update a property using __call__. This can be used
for command properties that correspond to function calls without arguments.
For example,
> proxy.Foo()
would push a Foo property which may cause the proxy to call a Foo method
on the actual VTK object.
For advanced users:
Python wrapper around a vtkSMProperty with a simple interface.
In addition to all method provided by vtkSMProperty (obtained by
forwarding unknown attributes requests to the underlying SMProxy),
Property and sub-class provide a list API.
Please note that some of the methods accessible through the Property
class are not listed by help() because the Property objects forward
unresolved attributes to the underlying object. To get the full list,
see also dir(proxy.SMProperty). See also the doxygen based documentation
of the vtkSMProperty C++ class.
"""
def __init__(self, proxy, smproperty):
"""Default constructor. Stores a reference to the proxy."""
import weakref
self.SMProperty = smproperty
self.Proxy = proxy
def __eq__(self, other):
"Returns true if the properties or properties values are the same."
return ((self is None and other is None) or
(self is not None and other is not None and self.__repr__() == other.__repr__()))
if sys.version_info < (3,):
def __ne__(self, other):
"Returns true if the properties or properties values are not the same."
return not self.__eq__(other)
def __repr__(self):
"""Returns a string representation containing property name
and value"""
if not type(self) is Property:
if self.GetData() is not None:
repr = self.GetData().__repr__()
else:
repr = "None"
else:
# this happens for command properties i.e. ones that do not store
# any state. For those, we use this alternate representation.
repr = "Property name= "
name = self.Proxy.GetPropertyName(self.SMProperty)
if name:
repr += name
else:
repr += "Unknown"
return repr
def __call__(self):
"""Forces a property update using InvokeCommand."""
if type(self) is Property:
self.Proxy.SMProxy.InvokeCommand(self._FindPropertyName())
else:
raise RuntimeError("Cannot invoke this property")
def _FindPropertyName(self):
"Returns the name of this property."
return self.Proxy.GetPropertyName(self.SMProperty)
def _UpdateProperty(self):
"Pushes the value of this property to the server."
# For now, we are updating all properties. This is due to an
# issue with the representations. Their VTK objects are not
# created until Input is set therefore, updating a property
# has no effect. Updating all properties everytime one is
# updated has the effect of pushing values set before Input
# when Input is updated.
# self.Proxy.SMProxy.UpdateProperty(self._FindPropertyName())
self.Proxy.SMProxy.UpdateVTKObjects()
def __getattr__(self, name):
"Unknown attribute requests get forwarded to SMProperty."
return getattr(self.SMProperty, name)
Name = property(_FindPropertyName, None, None,
"Returns the name for the property")
[docs]class GenericIterator(object):
"""Iterator for container type objects"""
def __init__(self, obj):
self.Object = obj
self.index = 0
def __iter__(self):
return self
def __next__(self):
if self.index >= len(self.Object):
raise StopIteration
idx = self.index
self.index += 1
return self.Object[idx]
[docs]class VectorProperty(Property):
"""A VectorProperty provides access to one or more values. You can use
a slice to get one or more property values:
> val = property[2]
or
> vals = property[0:5:2]
You can use a slice to set one or more property values:
> property[2] = val
or
> property[1:3] = (1,2)
"""
[docs] def ConvertValue(self, value):
return value
def __len__(self):
"""Returns the number of elements."""
return self.SMProperty.GetNumberOfElements()
def __iter__(self):
"""Implementation of the sequence API"""
return GenericIterator(self)
def __setitem__(self, idx, value):
"""Given a list or tuple of values, sets a slice of values [min, max)"""
if isinstance(idx, slice):
indices = idx.indices(len(self))
for i, j in zip(range(*indices), value):
self.SMProperty.SetElement(i, self.ConvertValue(j))
self._UpdateProperty()
elif idx >= len(self) or idx < 0:
raise IndexError
else:
self.SMProperty.SetElement(idx, self.ConvertValue(value))
self._UpdateProperty()
[docs] def GetElement(self, index):
return self.SMProperty.GetElement(index)
def __getitem__(self, idx):
"""Returns the range [min, max) of elements. Raises an IndexError
exception if an argument is out of bounds."""
ls = len(self)
if isinstance(idx, slice):
indices = idx.indices(ls)
retVal = []
for i in range(*indices):
retVal.append(self.GetElement(i))
return retVal
elif idx >= ls:
raise IndexError
elif idx < 0:
idx = ls + idx
if idx < 0:
raise IndexError
return self.GetElement(idx)
[docs] def GetData(self):
"Returns all elements as either a list or a single value."
property = self.SMProperty
if property.GetRepeatable() or \
property.GetNumberOfElements() > 1:
return self[0:len(self)]
elif property.GetNumberOfElements() == 1:
return self.GetElement(0)
[docs] def SetData(self, values):
"""Allows setting of all values at once. Requires a single value or
a iterable object."""
if self.SMProperty.GetInformationOnly() != 0:
raise RuntimeError("Cannot set an InformationOnly property!")
# Python3: str now has attr "__iter__", test separately
if (not hasattr(values, "__iter__")) or (type(values) == type("")):
values = (values,)
if not self.GetRepeatable() and len(values) != self.GetNumberOfElements():
raise RuntimeError("This property requires %d values." % self.GetNumberOfElements())
# map returns an iterable, must be converted to list.
convertedValues = list(map(self.ConvertValue, values))
if self.GetRepeatable():
# Clean up first
self.SMProperty.SetNumberOfElements(len(convertedValues))
try:
# SetElements() isn't available for all VectorProperty values.
# Try to use it so that the proxies are updated only once.
# Otherwise, call SetElement() for each element.
self.SMProperty.SetElements(convertedValues)
except TypeError as e:
idx = 0
for val in convertedValues:
self.SMProperty.SetElement(idx, val)
idx += 1
self._UpdateProperty()
[docs] def Clear(self):
"Removes all elements."
self.SMProperty().SetNumberOfElements(0)
self._UpdateProperty()
[docs]class DoubleMapProperty(Property):
"""A DoubleMapProperty provides access to a map of double vector values."""
def __len__(self):
"""Returns the number of elements."""
return self.SMProperty.GetNumberOfElements()
def __getitem__(self, key):
"""Returns the values for key."""
return self.GetData()[key]
def __setitem__(self, key, values):
"""Sets the values for key."""
for i, value in enumerate(values):
self.SMProperty.SetElementComponent(key, i, value)
self._UpdateProperty()
def __contains__(self, key):
"""Returns True if the property contains key."""
return key in self.keys()
[docs] def keys(self):
"""Returns the keys."""
return self.GetData().keys()
[docs] def items(self):
"""Iterates over the (key, value) pairs."""
return self.GetData().items()
[docs] def values(self):
"""Returns the values"""
return self.GetData().values()
[docs] def get(self, key, default_value=None):
"""Returns value of the given key, or the default_value if the key is not found
in the map."""
return self.GetData().get(key, default_value)
[docs] def GetData(self):
"""Returns all the elements as a dictionary"""
data = {}
iter = self.SMProperty.NewIterator()
while not iter.IsAtEnd():
values = []
for i in range(self.SMProperty.GetNumberOfComponents()):
values.append(iter.GetElementComponent(i))
data[iter.GetKey()] = values
iter.Next()
iter.UnRegister(None)
return data
[docs] def SetData(self, elements):
"""Sets all the elements at once."""
if self.SMProperty.GetInformationOnly() != 0:
raise RuntimeError("Cannot set an InformationOnly property!")
# first clear existing data
self.Clear()
for key, values in elements.items():
for i, value in enumerate(values):
self.SMProperty.SetElementComponent(key, i, value)
self._UpdateProperty()
[docs] def Clear(self):
"""Removes all elements."""
self.SMProperty.ClearElements()
self._UpdateProperty()
[docs]class EnumerationProperty(VectorProperty):
"""Subclass of VectorProperty that is applicable for enumeration type
properties."""
[docs] def GetElement(self, index):
"""Returns the text for the given element if available. Returns
the numerical values otherwise."""
val = self.SMProperty.GetElement(index)
domain = self.SMProperty.FindDomain("vtkSMEnumerationDomain")
if domain:
for i in range(domain.GetNumberOfEntries()):
if domain.GetEntryValue(i) == val:
return domain.GetEntryText(i)
return val
[docs] def ConvertValue(self, value):
"""Converts value to type suitable for vtSMProperty::SetElement()"""
if type(value) == str:
domain = self.SMProperty.FindDomain("vtkSMEnumerationDomain")
if domain:
if domain.HasEntryText(value):
return domain.GetEntryValueForText(value)
else:
raise ValueError("%s is not a valid value." % value)
return VectorProperty.ConvertValue(self, value)
[docs] def GetAvailable(self):
"Returns the list of available values for the property, if an enumeration domain is available."
retVal = []
domain = self.SMProperty.FindDomain("vtkSMEnumerationDomain")
if domain:
for i in range(domain.GetNumberOfEntries()):
retVal.append(domain.GetEntryText(i))
return retVal
Available = property(GetAvailable, None, None, \
"This read-only property contains the list of values that can be applied to this property.")
[docs]class FileNameProperty(VectorProperty):
"""Property to set/get one or more file names.
This property updates the pipeline information everytime its value changes.
This is used to keep the array lists up to date."""
def _UpdateProperty(self):
"Pushes the value of this property to the server."
VectorProperty._UpdateProperty(self)
try:
self.Proxy.FileNameChanged()
except AttributeError:
pass
[docs]class ArraySelectionProperty(VectorProperty):
"Property to select an array to be processed by a filter."
[docs] def GetAssociation(self):
val = self.GetElement(3)
if val == "":
return None
return GetAssociationAsString(int(val))
[docs] def GetArrayName(self):
return self.GetElement(4)
def __len__(self):
"""Returns the number of elements."""
return 2
def __setitem__(self, idx, value):
raise RuntimeError("This property cannot be accessed using __setitem__")
def __getitem__(self, idx):
"""Returns attribute type for index 0, array name for index 1"""
if isinstance(idx, slice):
indices = idx.indices(len(self))
retVal = []
for i in range(*indices):
if i >= 2 or i < 0:
raise IndexError
if i == 0:
retVal.append(self.GetAssociation())
else:
retVal.append(self.GetArrayName())
return retVal
elif idx >= 2 or idx < 0:
raise IndexError
if idx == 0:
return self.GetAssociation()
else:
return self.GetArrayName()
[docs] def SetData(self, values):
"""Allows setting of all values at once. Requires a single value,
a tuple or list."""
if self.SMProperty.GetInformationOnly() != 0:
raise RuntimeError("Cannot set an InformationOnly property!")
if not values:
# if values is None or empty list, we are resetting the selection.
self.SMProperty.SetElement(4, "")
self._UpdateProperty()
return
if not isinstance(values, tuple) and \
not isinstance(values, list):
values = (values,)
if len(values) == 1:
self.SMProperty.SetElement(4, values[0])
elif len(values) == 2:
if isinstance(values[0], str):
val = GetAssociationFromString(values[0])
else:
# In case user didn't specify valid association,
# just pick POINTS.
val = GetAssociationFromString("POINTS")
self.SMProperty.SetElement(3, str(val))
self.SMProperty.SetElement(4, values[1])
else:
raise RuntimeError("Expected 1 or 2 values.")
self._UpdateProperty()
[docs] def UpdateDefault(self):
"Helper method to set default values."
if self.SMProperty.GetNumberOfElements() != 5:
return
if self.GetElement(4) != '' or \
self.GetElement(3) != '':
return
for i in range(0, 3):
if self.GetElement(i) == '':
self.SMProperty.SetElement(i, '0')
self.SMProperty.ResetToDomainDefaults(False)
[docs]class StringListProperty(VectorProperty):
"""Property to set/get the a string with a string list domain.
This property provides an interface to get available strings."""
[docs] def GetAvailable(self):
"Returns the list of string values available for the property."
retVal = []
domain = self.SMProperty.FindDomain("vtkSMStringListDomain")
if domain:
for i in range(domain.GetNumberOfStrings()):
retVal.append(domain.GetString(i))
return retVal
Available = property(GetAvailable, None, None, \
"This read-only property contains the list of values that can be applied to this property.")
[docs]class ArrayListProperty(VectorProperty):
"""This property provides a simpler interface for selecting arrays.
Simply assign a list of arrays that should be loaded by the reader.
Use the Available property to get a list of available arrays."""
def __init__(self, proxy, smproperty):
VectorProperty.__init__(self, proxy, smproperty)
self.__arrays = []
[docs] def GetAvailable(self):
"Returns the list of available arrays"
dm = self.SMProperty.FindDomain("vtkSMArraySelectionDomain")
if not dm:
dm = self.SMProperty.FindDomain("vtkSMChartSeriesSelectionDomain")
retVal = []
if dm:
for i in range(dm.GetNumberOfStrings()):
retVal.append(dm.GetString(i))
return retVal
Available = property(GetAvailable, None, None, \
"This read-only property contains the list of items that can be read by a reader.")
[docs] def SelectAll(self):
"Selects all arrays."
self.SetData(self.Available)
[docs] def DeselectAll(self):
"Deselects all arrays."
self.SetData([])
def __iter__(self):
"""Implementation of the sequence API"""
return GenericIterator(self)
def __len__(self):
"""Returns the number of elements."""
return len(self.GetData())
def __setitem__(self, idx, value):
"""Given a list or tuple of values, sets a slice of values [min, max)"""
self.GetData()
if isinstance(idx, slice):
indices = idx.indices(len(self))
for i, j in zip(range(*indices), value):
self.__arrays[i] = j
self.SetData(self.__arrays)
elif idx >= len(self) or idx < 0:
raise IndexError
else:
self.__arrays[idx] = self.ConvertValue(value)
self.SetData(self.__arrays)
def __getitem__(self, idx):
"""Returns the range [min, max) of elements. Raises an IndexError
exception if an argument is out of bounds."""
self.GetData()
if isinstance(idx, slice):
indices = idx.indices(len(self))
retVal = []
for i in range(*indices):
retVal.append(self.__arrays[i])
return retVal
elif idx >= len(self) or idx < 0:
raise IndexError
return self.__arrays[idx]
[docs] def SetData(self, values):
"""Allows setting of all values at once. Requires a single value,
a tuple or list."""
if self.SMProperty.GetInformationOnly() != 0:
raise RuntimeError("Cannot set an InformationOnly property!")
# Clean up first
iup = self.SMProperty.GetImmediateUpdate()
self.SMProperty.SetImmediateUpdate(False)
# Clean up first
self.SMProperty.SetNumberOfElements(0)
if not isinstance(values, tuple) and \
not isinstance(values, list):
values = (values,)
fullvalues = []
# WARNING:
# The order of the two loops below are delibrately set in this way
# so that values passed in will take precedence.
# This is needed for backward compatibility of the
# property ElementBlocks for vtkExodusIIReader.
# If you attempt to change this, please verify that
# python state files for opening old .ex2 file (<=3.14) still works.
for array in self.Available:
if not values.__contains__(array):
fullvalues.append(array)
fullvalues.append('0')
for i in range(len(values)):
val = self.ConvertValue(values[i])
fullvalues.append(val)
fullvalues.append('1')
i = 0
for value in fullvalues:
self.SMProperty.SetElement(i, value)
i += 1
self._UpdateProperty()
self.SMProperty.SetImmediateUpdate(iup)
[docs] def GetData(self):
"Returns all elements as a list."
property = self.SMProperty
nElems = property.GetNumberOfElements()
if nElems % 2 != 0:
raise ValueError(
"The SMProperty with XML label '%s' has a size that is not a multiple of 2." % property.GetXMLLabel())
self.__arrays = []
for i in range(0, nElems, 2):
if self.GetElement(i + 1) != '0':
self.__arrays.append(self.GetElement(i))
return list(self.__arrays)
[docs]class ProxyProperty(Property):
"""A ProxyProperty provides access to one or more proxies. You can use
a slice to get one or more property values:
> proxy = property[2]
or
> proxies = property[0:5:2]
You can use a slice to set one or more property values:
> property[2] = proxy
or
> property[1:3] = (proxy1, proxy2)
You can also append and delete:
> property.append(proxy)
and
> del property[1:2]
You can also remove all elements with Clear().
Note that some properties expect only 1 proxy and will complain if
you set the number of values to be something else.
"""
def __init__(self, proxy, smproperty):
"""Default constructor. Stores a reference to the proxy. Also looks
at domains to find valid values."""
Property.__init__(self, proxy, smproperty)
# Check to see if there is a proxy list domain and, if so,
# initialize ourself. (Should this go in ProxyProperty?)
listdomain = self.FindDomain("vtkSMProxyListDomain")
if listdomain:
pm = ProxyManager()
group = "pq_helper_proxies." + proxy.GetGlobalIDAsString()
if listdomain.GetNumberOfProxies() == 0:
for i in range(listdomain.GetNumberOfProxyTypes()):
igroup = listdomain.GetProxyGroup(i)
name = listdomain.GetProxyName(i)
iproxy = CreateProxy(igroup, name)
listdomain.AddProxy(iproxy)
smproperty.ResetToDomainDefaults(False)
[docs] def GetAvailable(self):
"""If this proxy has a list domain, then this function returns the
strings you can use to select from the domain. If there is no such
list domain, the returned list is empty."""
listdomain = self.SMProperty.FindDomain("vtkSMProxyListDomain")
retval = []
if listdomain:
for i in range(listdomain.GetNumberOfProxies()):
proxy = listdomain.GetProxy(i)
retval.append(_make_name_valid(proxy.GetXMLLabel()))
return retval
Available = property(GetAvailable, None, None,
"""This read only property is a list of strings you can
use to select from the list domain. If there is no
such list domain, the array is empty.""")
def __iter__(self):
"""Implementation of the sequence API"""
return GenericIterator(self)
def __len__(self):
"""Returns the number of elements."""
return self.SMProperty.GetNumberOfProxies()
[docs] def remove(self, proxy):
"""Removes the first occurrence of the proxy from the property."""
self.SMProperty.RemoveProxy(proxy.SMProxy)
self._UpdateProperty()
def __setitem__(self, idx, value):
"""Given a list or tuple of values, sets a slice of values [min, max)"""
if isinstance(idx, slice):
indices = idx.indices(len(self))
for i, j in zip(range(*indices), value):
self.SMProperty.SetProxy(i, j.SMProxy)
self._UpdateProperty()
elif idx >= len(self) or idx < 0:
raise IndexError
else:
self.SMProperty.SetProxy(idx, value.SMProxy)
self._UpdateProperty()
def __delitem__(self, idx):
"""Removes the element idx"""
if isinstance(idx, slice):
indices = idx.indices(len(self))
# Collect the elements to delete to a new list first.
# Otherwise indices are screwed up during the actual
# remove loop.
toremove = []
for i in range(*indices):
toremove.append(self[i])
for i in toremove:
self.SMProperty.RemoveProxy(i.SMProxy)
self._UpdateProperty()
elif idx >= len(self) or idx < 0:
raise IndexError
else:
self.SMProperty.RemoveProxy(self[idx].SMProxy)
self._UpdateProperty()
def __getitem__(self, idx):
"""Returns the range [min, max) of elements. Raises an IndexError
exception if an argument is out of bounds."""
if isinstance(idx, slice):
indices = idx.indices(len(self))
retVal = []
for i in range(*indices):
retVal.append(_getPyProxy(self.SMProperty.GetProxy(i)))
return retVal
elif idx >= len(self) or idx < 0:
raise IndexError
return _getPyProxy(self.SMProperty.GetProxy(idx))
def __getattr__(self, name):
"Unknown attribute requests get forwarded to SMProperty."
return getattr(self.SMProperty, name)
[docs] def index(self, proxy):
idx = 0
for px in self:
if proxy == px:
return idx
idx += 1
raise ValueError("proxy is not in the list.")
[docs] def append(self, proxy):
"Appends the given proxy to the property values."
self.SMProperty.AddProxy(proxy.SMProxy)
self._UpdateProperty()
[docs] def GetData(self):
"Returns all elements as either a list or a single value."
property = self.SMProperty
if property.GetRepeatable() or property.GetNumberOfProxies() > 1:
return self[0:len(self)]
else:
if property.GetNumberOfProxies() > 0:
return _getPyProxy(property.GetProxy(0))
return None
[docs] def SetData(self, values):
"""Allows setting of all values at once. Requires a single value,
a tuple or list."""
if self.SMProperty.GetInformationOnly() != 0:
raise RuntimeError("Cannot set an InformationOnly property!")
if isinstance(values, str):
position = -1
try:
position = self.Available.index(values)
except:
raise ValueError(values + " is not a valid object in the domain.")
listdomain = self.SMProperty.FindDomain("vtkSMProxyListDomain")
if listdomain:
values = listdomain.GetProxy(position)
if not isinstance(values, tuple) and \
not isinstance(values, list):
values = (values,)
self.SMProperty.RemoveAllProxies()
for value in values:
if isinstance(value, Proxy):
value_proxy = value.SMProxy
else:
value_proxy = value
self.SMProperty.AddProxy(value_proxy)
self._UpdateProperty()
[docs] def Clear(self):
"Removes all elements."
self.SMProperty.RemoveAllProxies()
self._UpdateProperty()
[docs]def OutputPort(proxy, outputPort=0):
if not Proxy:
return None
if isinstance(outputPort, str):
outputPort = proxy.GetOutputPortIndex(outputPort)
if outputPort >= proxy.GetNumberOfOutputPorts():
return None
if proxy.Port == outputPort:
return proxy
newinstance = _getPyProxy(proxy.SMProxy, outputPort)
newinstance.Port = outputPort
newinstance._Proxy__Properties = proxy._Proxy__Properties
return newinstance
[docs]class ProxyManager(object):
"""When running scripts from the python shell in the ParaView application,
registering proxies with the proxy manager is the only mechanism to
notify the graphical user interface (GUI) that a proxy
exists. Therefore, unless a proxy is registered, it will not show up in
the user interface. Also, the proxy manager is the only way to get
access to proxies created using the GUI. Proxies created using the GUI
are automatically registered under an appropriate group (sources,
filters, representations and views). To get access to these objects,
you can use proxyManager.GetProxy(group, name). The name is the same
as the name shown in the pipeline browser.
This class is a python wrapper for vtkSMProxyManager. Note that the
underlying vtkSMProxyManager is a singleton. All instances of this
class will refer to the same object. In addition to all methods provided by
vtkSMProxyManager (all unknown attribute requests are forwarded
to the vtkSMProxyManager), this class provides several convenience
methods.
Please note that some of the methods accessible through the ProxyManager
class are not listed by help() because the ProxyManager objects forwards
unresolved attributes to the underlying object. To get the full list,
see also dir(proxy.SMProxyManager). See also the doxygen based documentation
of the vtkSMProxyManager C++ class.
"""
def __init__(self, session=None):
"""Constructor. Assigned self.SMProxyManager to
vtkSMProxyManager.GetProxyManager()."""
global ActiveConnection
if not session:
session = ActiveConnection.Session
self.SMProxyManager = session.GetSessionProxyManager()
[docs] def RegisterProxy(self, group, name, aProxy):
"""Registers a proxy (either SMProxy or proxy) with the
server manager"""
if isinstance(aProxy, Proxy):
self.SMProxyManager.RegisterProxy(group, name, aProxy.SMProxy)
else:
self.SMProxyManager.RegisterProxy(group, name, aProxy)
[docs] def NewProxy(self, group, name):
"""Creates a new proxy of given group and name and returns an SMProxy.
Note that this is a server manager object. You should normally create
proxies using the class objects. For example:
obj = servermanager.sources.SphereSource()"""
if not self.SMProxyManager:
return None
aProxy = self.SMProxyManager.NewProxy(group, name)
if not aProxy:
return None
aProxy.UnRegister(None)
return aProxy
[docs] def GetProxy(self, group, name):
"""Returns a Proxy registered under a group and name"""
if not self.SMProxyManager:
return None
aProxy = self.SMProxyManager.GetProxy(group, name)
if not aProxy:
return None
return _getPyProxy(aProxy)
[docs] def GetPrototypeProxy(self, group, name):
"""Returns a prototype proxy given a group and name. This is an
SMProxy. This is a low-level method. You should not normally
have to call it."""
if not self.SMProxyManager:
return None
aProxy = self.SMProxyManager.GetPrototypeProxy(group, name)
if not aProxy:
return None
return aProxy
[docs] def GetProxiesInGroup(self, groupname):
"""Returns a map of proxies in a particular group."""
proxies = {}
iter = self.NewGroupIterator(groupname)
for aProxy in iter:
proxies[(iter.GetKey(), aProxy.GetGlobalIDAsString())] = aProxy
return proxies
[docs] def UnRegisterProxy(self, groupname, proxyname, aProxy):
"""Unregisters a proxy."""
if not self.SMProxyManager:
return
if aProxy != None and isinstance(aProxy, Proxy):
aProxy = aProxy.SMProxy
if aProxy:
self.SMProxyManager.UnRegisterProxy(groupname, proxyname, aProxy)
[docs] def GetProxies(self, groupname, proxyname):
"""Returns all proxies registered under the given group with the
given name. Note that it is possible to register more than one
proxy with the same name in the same group. Because the proxies
are different, there is no conflict. Use this method instead of
GetProxy() if you know that there are more than one proxy registered
with this name."""
if not self.SMProxyManager:
return []
collection = vtk.vtkCollection()
result = []
self.SMProxyManager.GetProxies(groupname, proxyname, collection)
for i in range(0, collection.GetNumberOfItems()):
aProxy = _getPyProxy(collection.GetItemAsObject(i))
if aProxy:
result.append(aProxy)
return result
def __iter__(self):
"""Returns a new ProxyIterator."""
iter = ProxyIterator()
iter.Begin()
return iter
[docs] def NewGroupIterator(self, group_name):
"""Returns a ProxyIterator for a group. The resulting object
can be used to traverse the proxies that are in the given
group."""
iter = self.__iter__()
iter.SetModeToOneGroup()
iter.Begin(group_name)
return iter
[docs] def NewDefinitionIterator(self, groupname=None):
"""Returns an iterator that can be used to iterate over
all groups and types of proxies that the proxy manager
can create."""
iter = None
if groupname != None:
iter = ProxyDefinitionIterator(self.GetProxyDefinitionManager().NewSingleGroupIterator(groupname, 0))
else:
iter = ProxyDefinitionIterator(self.GetProxyDefinitionManager().NewIterator(0))
return iter
def __ConvertArgumentsAndCall(self, *args):
newArgs = []
for arg in args:
if issubclass(type(arg), Proxy) or isinstance(arg, Proxy):
newArgs.append(arg.SMProxy)
else:
newArgs.append(arg)
func = getattr(self.SMProxyManager, self.__LastAttrName)
retVal = func(*newArgs)
if isinstance(retVal, vtkSMProxy):
return _getPyProxy(retVal)
else:
return retVal
def __getattr__(self, name):
"""Returns attribute from the ProxyManager"""
try:
pmAttr = getattr(self.SMProxyManager, name)
self.__LastAttrName = name
return self.__ConvertArgumentsAndCall
except:
pass
return getattr(self.SMProxyManager, name)
[docs] def LoadState(self, filename, loader=None, location=vtkPVSession.CLIENT):
self.SMProxyManager.LoadXMLState(filename, loader, location)
[docs] def SaveState(self, filename, location=vtkPVSession.CLIENT):
self.SMProxyManager.SaveXMLState(filename, location)
[docs]class PropertyIterator(object):
"""Wrapper for a vtkSMPropertyIterator class to satisfy
the python iterator protocol. Note that the list of
properties can also be obtained from the class object's
dictionary.
See the doxygen documentation for vtkSMPropertyIterator C++
class for details.
"""
def __init__(self, aProxy):
self.SMIterator = aProxy.NewPropertyIterator()
if self.SMIterator:
self.SMIterator.UnRegister(None)
self.SMIterator.Begin()
self.Key = None
self.PropertyLabel = None
self.Proxy = aProxy
def __iter__(self):
return self
def __next__(self):
if not self.SMIterator:
raise StopIteration
if self.SMIterator.IsAtEnd():
self.Key = None
raise StopIteration
self.Key = self.SMIterator.GetKey()
self.PropertyLabel = self.SMIterator.GetPropertyLabel()
self.SMIterator.Next()
return self.Proxy.GetProperty(self.Key)
[docs] def GetProxy(self):
"""Returns the proxy for the property last returned by the call to
'__next__()'"""
return self.Proxy
[docs] def GetKey(self):
"""Returns the key for the property last returned by the call to
'__next__()' """
return self.Key
[docs] def GetProperty(self):
"""Returns the property last returned by the call to '__next__()' """
return self.Proxy.GetProperty(self.Key)
def __getattr__(self, name):
"""returns attributes from the vtkSMPropertyIterator."""
return getattr(self.SMIterator, name)
[docs]class ProxyDefinitionIterator(object):
"""Wrapper for a vtkPVProxyDefinitionIterator class to satisfy
the python iterator protocol.
See the doxygen documentation of the vtkPVProxyDefinitionIterator
C++ class for more information."""
def __init__(self, iter):
self.SMIterator = iter
if self.SMIterator:
self.SMIterator.UnRegister(None)
self.SMIterator.InitTraversal()
self.Group = None
self.Key = None
self.XML = None
def __iter__(self):
return self
def __next__(self):
if self.SMIterator.IsDoneWithTraversal():
self.Group = None
self.Key = None
raise StopIteration
self.Group = self.SMIterator.GetGroupName()
self.Key = self.SMIterator.GetProxyName()
self.XML = self.SMIterator.GetProxyDefinition()
self.SMIterator.GoToNextItem()
return {"group": self.Group, "key": self.Key, "xml": self.XML}
[docs] def GetProxyName(self):
"""Returns the key for the proxy definition last returned by the call
to '__next__()' """
return self.Key
[docs] def GetGroup(self):
"""Returns the group for the proxy definition last returned by the
call to '__next__()' """
return self.Group
[docs] def GetXML(self):
"""Returns the proxy definition XML (as vtkPVXMLElement)"""
return self.XML
def __getattr__(self, name):
"""returns attributes from the vtkPVProxyDefinitionIterator."""
return getattr(self.SMIterator, name)
[docs]class ProxyIterator(object):
"""Wrapper for a vtkSMProxyIterator class to satisfy the
python iterator protocol.
See the doxygen documentation of vtkSMProxyIterator C++ class for
more information.
"""
def __init__(self):
self.SMIterator = vtkSMProxyIterator()
self.SMIterator.SetSession(ActiveConnection.Session)
self.SMIterator.Begin()
self.AProxy = None
self.Group = None
self.Key = None
def __iter__(self):
return self
def __next__(self):
if self.SMIterator.IsAtEnd():
self.AProxy = None
self.Group = None
self.Key = None
raise StopIteration
return None
self.AProxy = _getPyProxy(self.SMIterator.GetProxy())
self.Group = self.SMIterator.GetGroup()
self.Key = self.SMIterator.GetKey()
self.SMIterator.Next()
return self.AProxy
[docs] def GetProxy(self):
"""Returns the proxy last returned by the call to '__next__()'"""
return self.AProxy
[docs] def GetKey(self):
"""Returns the key for the proxy last returned by the call to
'__next__()' """
return self.Key
[docs] def GetGroup(self):
"""Returns the group for the proxy last returned by the call to
'__next__()' """
return self.Group
def __getattr__(self, name):
"""returns attributes from the vtkSMProxyIterator."""
return getattr(self.SMIterator, name)
# Caution: Observers must be global methods otherwise we run into memory
# leak when the interpreter get reset from the C++ layer.
[docs]class Connection(object):
"""
This is a python representation for a session/connection.
"""
def __init__(self, connectionId, session):
"""Default constructor. Creates a Connection with the given
ID, all other data members initialized to None."""
global ActiveConnection
self.ID = connectionId
self.Session = session
self.Alive = True
self.DefinitionObserverTag = 0
self.CustomDefinitionObserverTag = 0
# newly created connection always become the active connection.
ActiveConnection = self
# Build the list of available proxies for this connection.
self.ProxiesNS = ConnectionProxyNamespaces(self.Session)
ActiveConnection = None
# SetActiveConnection(self)
def __eq__(self, other):
"Returns true if the connection ids are the same."
# Python3 - this is now called when self is None
return ((self is None and other is None) or
((self is not None and other is not None) and (self.ID == other.ID)))
def __repr__(self):
"""User friendly string representation"""
return "Connection (%s) [%d]" % (self.Session.GetURI(), self.ID)
[docs] def GetURI(self):
"""Get URI of the connection"""
return self.Session.GetURI()
[docs] def IsRemote(self):
"""Returns True if the connection to a remote server, False if
it is local (built-in)"""
if self.Session.IsA("vtkSMSessionClient"):
return True
return False
[docs] def GetNumberOfDataPartitions(self):
"""Returns the number of partitions on the data server for this
connection"""
return self.Session.GetServerInformation().GetNumberOfProcesses()
[docs] def close(self):
if self.DefinitionObserverTag:
self.Session.GetProxyDefinitionManager().RemoveObserver(self.DefinitionObserverTag)
self.Session.GetProxyDefinitionManager().RemoveObserver(self.CustomDefinitionObserverTag)
self.Session = None
self.Modules = None
self.Alive = False
def __del__(self):
if self.Alive:
self.close()
[docs]def SaveState(filename, location=vtkPVSession.CLIENT):
"""Given a state filename, saves the state of objects registered
with the proxy manager."""
pm = ProxyManager()
pm.SaveState(filename, location)
[docs]def LoadState(filename, connection=None, location=vtkPVSession.CLIENT):
"""Given a state filename and an optional connection, loads the server
manager state."""
if not connection:
connection = ActiveConnection
if not connection:
raise RuntimeError("Cannot load state without a connection")
pm = ProxyManager()
pm.LoadState(filename, None, location)
views = GetRenderViews()
for view in views:
# Make sure that the client window size matches the
# ViewSize property. In paraview, the GUI takes care
# of this.
if view.GetClassName() == "vtkSMIceTDesktopRenderViewProxy":
view.GetRenderWindow().SetSize(view.ViewSize[0], \
view.ViewSize[1])
[docs]def Connect(ds_host=None, ds_port=11111, rs_host=None, rs_port=22221, timeout=60):
"""
Use this function call to create a new session. On success,
it returns a vtkSMSession object that abstracts the connection.
Otherwise, it returns None.
There are several ways in which this function can be called:
* When called with no hosts, it creates a new session
to the built-in server on the client itself.
* When called with ds_host and ds_port arguments, it
attempts to connect to a server(data and render server on the same server)
on the indicated host:port.
* When called with ds_host, ds_port, rs_host, rs_port, it
creates a new connection to the data server on ds_host:ds_port and to the
render server on rs_host: rs_port.
* All these connection types support a timeout argument in second.
Default is 60. 0 means no retry, -1 means infinite retries.
"""
ret = True;
if ds_host == None:
session = vtkSMSession()
elif rs_host == None:
session = vtkSMSessionClient()
ret = session.Connect("cs://%s:%d" % (ds_host, ds_port), timeout)
else:
session = vtkSMSessionClient()
ret = session.Connect("cdsrs://%s:%d/%s:%d" % (ds_host, ds_port, rs_host, rs_port), timeout)
if ret:
id = vtkProcessModule.GetProcessModule().RegisterSession(session)
connection = GetConnectionFromId(id)
# This shouldn't be needed. However, it's needed for old Python scripts that
# directly import servermanager.py without simple.py
SetActiveConnection(connection)
return connection
else:
return None
[docs]def ConnectToCatalyst(ds_host='localhost', ds_port=22222):
"""
Use this function to create a new catalyst session.
"""
# Create an InsituLink
insituLink = CreateProxy('coprocessing', 'LiveInsituLink')
insituLink.GetProperty('InsituPort').SetElement(0, ds_port)
insituLink.GetProperty('Hostname').SetElement(0, ds_host)
# set process type to Visualization
insituLink.GetProperty('ProcessType').SetElement(0, 0)
insituLink.UpdateVTKObjects()
# Create dummy session
id = vtkSMSession.ConnectToCatalyst()
connection = GetConnectionFromId(id)
insituLink.SetInsituProxyManager(connection.Session.GetSessionProxyManager())
return insituLink
[docs]def ReverseConnect(port=11111):
"""
Use this function call to create a new session. On success,
it returns a Session object that abstracts the connection.
Otherwise, it returns None.
In reverse connection mode, the client waits for a connection
from the server (client has to be started first). The server
then connects to the client (run pvserver with -rc and -ch
option).
The optional port specified the port to listen to.
"""
session = vtkSMSessionClient()
session.Connect("csrc://hostname:" + port)
id = vtkProcessModule.GetProcessModule().RegisterSession(session)
connection = GetConnectionFromId(id)
# This shouldn't be needed. However, it's needed for old Python scripts that
# directly import servermanager.py without simple.py
SetActiveConnection(connection)
return connection
[docs]def Disconnect(connection=None):
"""Disconnects the connection. Make sure to clear the proxy manager
first."""
global ActiveConnection
if not connection:
connection = ActiveConnection
if connection:
vtkSMSession.Disconnect(connection.ID)
[docs]def CreateProxy(xml_group, xml_name, session=None):
"""Creates a proxy. If session is set, the proxy's session is
set accordingly. If session is None, the current Session is used, if
present. You should not have to use method normally. Instantiate the
appropriate class from the appropriate module, for example:
sph = servermanager.sources.SphereSource()"""
global ActiveConnection
if not session:
session = ActiveConnection.Session
if not session:
raise RuntimeError("Cannot create objects without a session.")
pxm = ProxyManager(session)
return pxm.NewProxy(xml_group, xml_name)
[docs]def GetRenderView(connection=None):
"""Return the render view in use. If more than one render view is in
use, return the first one."""
render_module = None
for aProxy in ProxyManager():
if aProxy.IsA("vtkSMRenderViewProxy"):
render_module = aProxy
break
return render_module
[docs]def GetRenderViews(connection=None):
"""Returns the set of all render views."""
render_modules = []
for aProxy in ProxyManager().GetProxiesInGroup("views").values():
if aProxy.IsA("vtkSMRenderViewProxy"):
render_modules.append(aProxy)
return render_modules
[docs]def GetContextViews(connection=None):
"""Returns the set of all context views."""
context_modules = []
for aProxy in ProxyManager():
if aProxy.IsA("vtkSMContextViewProxy"):
context_modules.append(aProxy)
return context_modules
[docs]def CreateRenderView(session=None, **extraArgs):
"""Creates a render window on the particular session. If session
is not specified, then the active session is used, if available.
This method can also be used to initialize properties by passing
keyword arguments where the key is the name of the property. In addition
registrationGroup and registrationName (optional) can be specified (as
keyword arguments) to automatically register the proxy with the proxy
manager."""
return _create_view("RenderView", session, **extraArgs)
def _create_view(view_xml_name, session=None, **extraArgs):
"""Creates a view on the particular session. If session
is not specified, then the active session is used, if available.
This method can also be used to initialize properties by passing
keyword arguments where the key is the name of the property."""
if not session:
session = ActiveConnection.Session
if not session:
raise RuntimeError("Cannot create view without session.")
pxm = ProxyManager()
view_module = None
if view_xml_name:
view_module = CreateProxy("views", view_xml_name, session)
if not view_module:
return None
return _getPyProxy(view_module)
[docs]def GetRepresentation(aProxy, view):
if view:
return view.FindRepresentation(aProxy.SMProxy, aProxy.Port)
return None
[docs]def CreateRepresentation(aProxy, view, **extraArgs):
"""Creates a representation for the proxy and adds it to the render
module.
This method can also be used to initialize properties by passing
keyword arguments where the key is the name of the property.In addition
registrationGroup and registrationName (optional) can be specified (as
keyword arguments) to automatically register the proxy with the proxy
manager.
This method tries to create the best possible representation for the given
proxy in the given view. Additionally, the user can specify proxyName
(optional) to create a representation of a particular type."""
global rendering
if not aProxy:
raise RuntimeError("proxy argument cannot be None.")
if not view:
raise RuntimeError("view argument cannot be None.")
if "proxyName" in extraArgs:
display = CreateProxy("representations", extraArgs['proxyName'], None)
del extraArgs['proxyName']
else:
display = view.SMProxy.CreateDefaultRepresentation(aProxy.SMProxy, 0)
if display:
display.UnRegister(None)
if not display:
return None
proxy = _getPyProxy(display)
proxy.Input = aProxy
for param in extraArgs.items():
setattr(proxy, items[0], items[1])
proxy.UpdateVTKObjects()
view.Representations.append(proxy)
return proxy
import importlib, importlib.abc
[docs]class ParaViewLoader(importlib.abc.InspectLoader):
def _info(self, fullname):
return vtkPVPythonModule.GetModule(fullname)
[docs] def is_package(self, fullname):
return self._info(fullname).GetIsPackage()
[docs] def get_source(self, fullname):
return self._info(fullname).GetSource()
[docs]def LoadPlugin(filename, remote=True, connection=None):
""" Given a filename and a session (optional, otherwise uses
ActiveConnection), loads a plugin. It then updates the sources,
filters and rendering modules.
remote=True has no effect when the connection is not remote.
"""
if not connection:
connection = ActiveConnection
if not connection:
raise RuntimeError("Cannot load a plugin without a connection.")
plm = vtkSMProxyManager.GetProxyManager().GetPluginManager()
if remote and connection.IsRemote():
status = plm.LoadRemotePlugin(filename, connection.Session)
else:
status = plm.LoadLocalPlugin(filename)
# shouldn't the extension check happen before attempting to load the plugin?
if not status:
raise RuntimeError("Problem loading plugin %s" % (filename))
[docs]def Fetch(input, arg1=None, arg2=None, idx=0):
"""
A convenience method that moves data from the server to the client,
optionally performing some operation on the data as it moves.
The input argument is the name of the (proxy for a) source or filter
whose output is needed on the client.
You can use Fetch to do three things:
If arg1 is None (the default) then all of the data is brought to the client.
In parallel runs an appropriate append Filter merges the
data on each processor into one data object. The filter chosen will be
vtkAppendPolyData for vtkPolyData, vtkAppendRectilinearGrid for
vtkRectilinearGrid, vtkMultiBlockDataGroupFilter for vtkCompositeData,
and vtkAppendFilter for anything else.
If arg1 is an integer then one particular processor's output is brought to
the client. In serial runs the arg is ignored. If you have a filter that
computes results in parallel and brings them to the root node, then set
arg to be 0.
If arg1 and arg2 are a algorithms, for example vtkMinMax, the algorithm
will be applied to the data to obtain some result. Here arg1 will be
applied pre-gather and arg2 will be applied post-gather. In parallel
runs the algorithm will be run on each processor to make intermediate
results and then again on the root processor over all of the
intermediate results to create a global result.
Optional argument idx is used to specify the output port number to fetch the
data from. Default is port 0.
"""
import sys
if sys.version_info < (3,):
integer_types = (int, long,)
else:
integer_types = (int,)
reducer = filters.ReductionFilter(Input=OutputPort(input, idx))
# create the pipeline that reduces and transmits the data
if arg1 == None:
dinfo = input.GetDataInformation(idx)
if dinfo.IsCompositeDataSet():
paraview.print_debug_info("use composite data append")
reducer.PostGatherHelperName = "vtkMultiBlockDataGroupFilter"
elif dinfo.GetDataClassName() == "vtkPolyData":
paraview.print_debug_info("use append poly data filter")
reducer.PostGatherHelperName = "vtkAppendPolyData"
elif dinfo.GetDataClassName() == "vtkRectilinearGrid":
paraview.print_debug_info("use append rectilinear grid filter")
reducer.PostGatherHelperName = "vtkAppendRectilinearGrid"
elif dinfo.IsA("vtkDataSet"):
paraview.print_debug_info("use unstructured append filter")
reducer.PostGatherHelperName = "vtkAppendFilter"
elif type(arg1) in integer_types:
reducer.PassThrough = arg1
else:
reducer.PreGatherHelper = arg1
reducer.PostGatherHelper = arg2
# reduce
reducer.UpdatePipeline()
dataInfo = reducer.GetDataInformation(0)
dataType = dataInfo.GetDataSetType()
if dataInfo.GetCompositeDataSetType() > 0:
dataType = dataInfo.GetCompositeDataSetType()
fetcher = filters.ClientServerMoveData(Input=reducer)
fetcher.OutputDataType = dataType
fetcher.WholeExtent = dataInfo.GetExtent()[:]
# fetch
fetcher.UpdatePipeline()
op = fetcher.GetClientSideObject().GetOutputDataObject(0)
opc = op.NewInstance()
opc.ShallowCopy(op)
opc.UnRegister(None)
return opc
[docs]def AnimateReader(reader, view):
"""This is a utility function that, given a reader and a view
animates over all time steps of the reader.
It creates an AnimationScene and add a new TimeAnimationCue in it.
When running from the GUI and python shell, prefer using the
existing scene and cue to maintain time consistency in the
application."""
if not reader:
raise RuntimeError("No reader was specified, cannot animate.")
if not view:
raise RuntimeError("No view was specified, cannot animate.")
# Create an animation scene
# This is why it is not recommended to use this method from the GUI. See #18984
scene = animation.AnimationScene()
# We need to have the reader and the view registered with
# the time keeper. This is how the scene gets its time values.
try:
tk = next(iter(ProxyManager().GetProxiesInGroup("timekeeper").values()))
scene.TimeKeeper = tk
except IndexError:
tk = misc.TimeKeeper()
scene.TimeKeeper = tk
if not reader in tk.TimeSources:
tk.TimeSources.append(reader)
if not view in tk.Views:
tk.Views.append(view)
# with 1 view
scene.ViewModules = [view]
# Update the reader to get the time information
reader.UpdatePipelineInformation()
# Animate from 1st time step to last
scene.StartTime = reader.TimestepValues.GetData()[0]
scene.EndTime = reader.TimestepValues.GetData()[-1]
# Each frame will correspond to a time step
scene.PlayMode = 2 # Snap To Timesteps
# Create a special animation cue for time.
cue = animation.TimeAnimationCue()
cue.AnimatedProxy = view
cue.AnimatedPropertyName = "ViewTime"
scene.Cues = [cue]
scene.Play()
return scene
[docs]def GetProgressPrintingIsEnabled():
return progressObserverTag is not None
[docs]def SetProgressPrintingEnabled(value):
"""Turn on/off printing of progress (by default, it is on). You can
always turn progress off and add your own observer to the process
module to handle progress in a custom way. See _printProgress for
an example event observer."""
global progressObserverTag
# If value is true and progress printing is currently off...
if value and not GetProgressPrintingIsEnabled():
if paraview.fromGUI:
raise RuntimeError("Printing progress in the GUI is not supported.")
if ActiveConnection and ActiveConnection.Session:
progressObserverTag = ActiveConnection.Session.GetProgressHandler().AddObserver(
"ProgressEvent", _printProgress)
# If value is false and progress printing is currently on...
elif GetProgressPrintingIsEnabled():
if ActiveConnection and ActiveConnection.Session:
ActiveConnection.Session.GetProgressHandler().RemoveObserver(progressObserverTag)
progressObserverTag = None
[docs]def ToggleProgressPrinting():
"""Turn on/off printing of progress. See SetProgressPrintingEnabled."""
SetProgressPrintingEnabled(not GetProgressPrintingIsEnabled())
[docs]def Finalize():
"""Although not required, this can be called at exit to cleanup."""
global progressObserverTag
# Make sure to remove the observer
if progressObserverTag:
ToggleProgressPrinting()
vtkInitializationHelper.Finalize()
# Internal methods
def _getPyProxy(smproxy, outputPort=0):
"""Returns a python wrapper for a server manager proxy. This method
first checks if there is already such an object by looking in the
_pyproxies group and returns it if found. Otherwise, it creates a
new one. Proxies register themselves in _pyproxies upon creation."""
global ActiveConnection, _pyproxies
if isinstance(smproxy, Proxy):
# if already a pyproxy, do nothing.
return smproxy
if not smproxy:
return None
if smproxy.IsA("vtkSMOutputPort"):
return _getPyProxy(smproxy.GetSourceProxy(), smproxy.GetPortIndex())
if (smproxy, outputPort) in _pyproxies:
return _pyproxies[(smproxy, outputPort)]()
classForProxy = ActiveConnection.ProxiesNS.getClass(smproxy)
if not classForProxy:
raise RuntimeError("Failed to locate proxy class for proxy (%s, %s)" % \
(smproxy.GetXMLGroup(), smproxy.GetXMLName()))
return classForProxy(proxy=smproxy, port=outputPort)
def _createInitialize(group, name):
"""Internal method to create an Initialize() method for the sub-classes
of Proxy"""
pgroup = group
pname = name
def aInitialize(self, connection=None, update=True):
if not connection:
connection = ActiveConnection
if not connection:
raise RuntimeError('Cannot create a proxy without a session.')
if not connection.Session.GetProxyDefinitionManager().HasDefinition(pgroup, pname):
error_msg = "The connection does not provide any definition for %s." % pname
raise RuntimeError(error_msg)
self.InitializeFromProxy( \
CreateProxy(pgroup, pname, connection.Session), update)
return aInitialize
def _createGetProperty(pName):
"""Internal method to create a GetXXX() method where XXX == pName."""
propName = pName
def getProperty(self):
if paraview.compatibility.GetVersion() >= (3, 5):
return self.GetPropertyValue(propName)
else:
return self.GetProperty(propName)
return getProperty
def _createSetProperty(pName):
"""Internal method to create a SetXXX() method where XXX == pName."""
propName = pName
def setProperty(self, value):
return self.SetPropertyWithName(propName, value)
return setProperty
def _printProgress(caller, event):
"""The default event handler for progress. Prints algorithm
name and 1 '.' per 10% progress."""
global currentAlgorithm, currentProgress
pm = vtkProcessModule.GetProcessModule()
progress = caller.GetLastProgress()
alg = caller.GetLastProgressText()
if alg != currentAlgorithm and alg:
if currentAlgorithm:
while currentProgress <= 10:
import sys
sys.stdout.write(".")
currentProgress += 1
print("]")
currentProgress = 0
print(alg, ": [ ", end="")
currentAlgorithm = alg
while currentProgress <= progress:
import sys
sys.stdout.write(".")
# sys.stdout.write("%d " % pm.GetLastProgress())
currentProgress += 1
if progress == 10:
print("]")
currentAlgorithm = None
currentProgress = 0
[docs]class ProxyNamespace:
"""
This class is a container for class types representing known proxies types.
Conceptually, one can think of this a Python equivalent for a proxy group,
however, due to legacy reasons, it can represent proxies in more than one
proxy-group.
Class types are created on demand. All known proxies types can be determined
using `dir()` call on this instance. This avoids having to define classes
for all proxy types during initialization, thus speeding up initialization.
"""
def __init__(self, xmlGroups, session):
self.xmlGroups = set(xmlGroups) if xmlGroups else set()
self.session = session
@staticmethod
def _getPyName(xml=None, smproxy=None, proxyname=None):
"""In ProxyNamespace, the proxies are named not using their XML name as
provided in the definition, but using a sanitized version of their
label. This change happened in ParaView 3.5. This method returns the
name to use given a proxy definition XML or a proxy itself based on the
compatibility version set."""
assert xml or smproxy or proxyname
if proxyname:
name = proxyname
elif paraview.compatibility.GetVersion() >= (3, 5):
if xml:
name = xml.GetAttributeOrDefault("label", xml.GetAttribute("name"))
elif smproxy:
name = smproxy.GetXMLLabel()
else:
if xml:
name = xml.GetAttribute("name")
elif smproxy:
name = smproxy.GetXMLName()
assert name
return _make_name_valid(name)
def _findProxy(self, name=None, xmlname=None):
assert name or xmlname
pdm = self.session.GetProxyDefinitionManager()
for group in self.xmlGroups:
pditer = ProxyDefinitionIterator(pdm.NewSingleGroupIterator(group, pdm.ALL_DEFINITIONS))
for item in pditer:
if (item["xml"].GetAttribute("name")):
pname = self._getPyName(xml=item["xml"])
else:
pname = self._getPyName(proxyname=pditer.GetProxyName())
if name and pname == name:
return item
elif xmlname and xmlname == item["key"]:
return item
return None
def __dir__(self):
s = set()
pdm = self.session.GetProxyDefinitionManager()
for group in self.xmlGroups:
pditer = ProxyDefinitionIterator(pdm.NewSingleGroupIterator(group, pdm.ALL_DEFINITIONS))
for item in pditer:
if (item["xml"].GetAttribute("name")):
s.add(self._getPyName(xml=item["xml"]))
else:
s.add(self._getPyName(proxyname=pditer.GetProxyName()))
return s
def __getattr__(self, name):
"""called when default lookup has failed; this implies that a Proxy type
is requested which has not been created yet.
"""
ptype = self._findProxy(name=name)
if not ptype:
raise AttributeError("No Proxy type named '%s' found." % name)
cls = _createClass(ptype["group"], ptype["key"], apxm=self.session.GetSessionProxyManager())
# store class definition to avoid creating new type for each proxy.
setattr(self, name, cls)
return cls
[docs] def getDocumentation(self, name):
ptype = self._findProxy(name=name)
if not ptype:
raise RuntimeError("Invalid proxy '%s'" % name)
xml = ptype["xml"]
doc = xml.FindNestedElementByName("Documentation")
return doc.GetCharacterData() if doc and doc.GetCharacterData() else ""
[docs] def findClass(self, smproxy):
"""Given a vtkSMProxy, returns the class type to use for this proxy
under this namespace."""
# shortcut. if the proxy's XML group is not something we are responsible
# for, skip it.
if self.xmlGroups and smproxy.GetXMLGroup() not in self.xmlGroups:
return None
# determine the 'name' that will be used for the class type for this proxy.
pname = self._getPyName(smproxy=smproxy)
# now, attempt to access it.
return getattr(self, pname) if pname else None
[docs]class ConnectionProxyNamespaces(object):
"""
This class stores class types for all known proxies on a particular session.
The classes are maintained in separate namespaces accessible as read-only properties
on an instance of this class.
"""
def __init__(self, session):
self._session = session
self._namespaces = {}
def _get_proxy_types(self, name, xmlGroups=None):
if name not in self._namespaces:
groups = xmlGroups if xmlGroups else [name]
self._namespaces[name] = ProxyNamespace(groups, self._session)
return self._namespaces[name]
@classmethod
def _getNamespaceNames(cls):
return [item[0] for item in vars(cls).items() if type(item[1]) == property]
[docs] def getNamespaces(self):
retval = {}
for n in self._getNamespaceNames():
retval[n] = getattr(self, n)
return retval
[docs] def getClass(self, smproxy):
"""
Given a vtkSMProxy returns a `Proxy` class subclass type appropriate for
the proxy. This will atttempt to reuse the class type is already seen,
otherwise a new one will be created.
"""
for name in self._getNamespaceNames():
ns = getattr(self, name)
cls = ns.findClass(smproxy)
if cls:
return cls
nsname = smproxy.GetXMLGroup()
# if not found in prefined collections, create a new collection for the
# group. note, this modifies the class so all existing instances will
# inherit the new property being added, but that's perfectly acceptable.
cls = self.__class__
assert not hasattr(cls, nsname)
setattr(cls, nsname, property(lambda self: self._get_proxy_types(nsname),
doc="%s proxies" % nsname))
ns = getattr(self, nsname)
return ns.findClass(smproxy)
# Here, we have some predefined proxy groups for which we setup standard
# namespaces. Besides the groups which had defined that combine multiple xml
# groups e.g. rendering, animation, and the ones used to create functions
# e.g sources, filters, writers, I am not sure this necessary anymore.
sources = property(lambda self: self._get_proxy_types("sources"), doc="source proxies")
filters = property(lambda self: self._get_proxy_types("filters"), doc="filter proxies")
writers = property(lambda self: self._get_proxy_types("writers"), doc="writers proxies")
rendering = property(lambda self: self._get_proxy_types("rendering",
["annotations", "cameramanipulators", "lookup_tables",
"representations", "textures", "views"]),
doc="rendering proxies")
animation = property(lambda self: self._get_proxy_types("animation",
["animation", "animation_keyframes"]),
doc="animation proxies")
implicit_functions = property(lambda self: self._get_proxy_types("implicit_functions"),
doc="implicit function proxies")
piecewise_functions = property(lambda self: self._get_proxy_types("piecewise_functions"),
doc="piece-wise functions")
extended_sources = property(lambda self: self._get_proxy_types("extended_sources"),
doc="extented sources")
misc = property(lambda self: self._get_proxy_types("misc",
["misc", "incremental_point_locators", "point_locators"]),
doc="miscellaneous proxies")
def _make_name_valid(name):
return paraview.make_name_valid(name)
def _createClassProperties(proto, excludeset=frozenset()):
"""Builds a dict of properties for all SMProperties on the `proto` proxy.
If excludeset is not empty, then it is expected to be names of properties
to exclude."""
cdict = {}
iter = PropertyIterator(proto)
# Add all properties as python properties.
for prop in iter:
propName = iter.GetKey()
if paraview.compatibility.GetVersion() >= (3, 5):
if (prop.GetInformationOnly() and propName != "TimestepValues" \
and prop.GetPanelVisibility() == "never") or prop.GetIsInternal():
continue
names = [propName]
if paraview.compatibility.GetVersion() >= (3, 5):
names = [iter.PropertyLabel]
propDoc = None
if prop.GetDocumentation():
propDoc = prop.GetDocumentation().GetDescription()
for name in names:
name = _make_name_valid(name)
if name and name not in excludeset:
cdict[name] = property(_createGetProperty(propName),
_createSetProperty(propName),
None,
propDoc)
return cdict
def _createClass(groupName, proxyName, apxm=None, prototype=None):
"""Defines a new class type for the proxy."""
if prototype is None:
pxm = ProxyManager() if not apxm else apxm
proto = pxm.GetPrototypeProxy(groupName, proxyName)
else:
proto = prototype
if not proto:
paraview.print_error("Error while loading %s %s" % (groupName, proxyName))
return None
pname = proxyName
if paraview.compatibility.GetVersion() >= (3, 5) and proto.GetXMLLabel():
pname = proto.GetXMLLabel()
pname = _make_name_valid(pname)
if not pname:
return None
cdict = {}
# Create an Initialize() method for this sub-class.
cdict['Initialize'] = _createInitialize(groupName, proxyName)
cdict.update(_createClassProperties(proto))
# Add the documentation as the class __doc__
if proto.GetDocumentation() and \
proto.GetDocumentation().GetDescription():
doc = proto.GetDocumentation().GetDescription()
else:
doc = Proxy.__doc__
cdict['__doc__'] = doc
# Create the new type
if proto.GetXMLName() == "ExodusIIReader":
superclasses = (ExodusIIReaderProxy,)
elif proto.IsA("vtkSMMultiplexerSourceProxy"):
superclasses = (MultiplexerSourceProxy,)
elif proto.IsA("vtkSMSourceProxy"):
superclasses = (SourceProxy,)
elif proto.IsA("vtkSMViewLayoutProxy"):
superclasses = (ViewLayoutProxy,)
else:
superclasses = (Proxy,)
cobj = type(pname, superclasses, cdict)
return cobj
def __determineGroup(proxy):
"""Internal method"""
if not proxy:
return None
xmlgroup = proxy.GetXMLGroup()
xmlname = proxy.GetXMLName()
if xmlgroup == "sources":
if xmlname in ["BlockSelectionSource",
"BlockSelectorsSelectionSource",
"SelectionQuerySource",
"FrustumSelectionSource",
"GlobalIDSelectionSource",
"PedigreeIDSelectionSource",
"IDSelectionSource",
"CompositeDataIDSelectionSource",
"HierarchicalDataIDSelectionSource",
"ThresholdSelectionSource",
"LocationSelectionSource"]:
return "selection_sources"
return "sources"
elif xmlgroup == "filters":
if xmlname == "AppendSelections":
return "selection_sources"
return "sources"
elif xmlgroup == "representations":
if xmlname == "ScalarBarWidgetRepresentation":
return "scalar_bars"
return "representations"
elif xmlgroup == "animation_keyframes":
return "animation"
return xmlgroup
[docs]def ResetSession():
"""Reset the session in the active connection to its initial state."""
global ActiveConnection
# Simulate disconnect
pxm = ProxyManager()
session = ActiveConnection.Session
pxm.UnRegisterProxies()
pm = vtkProcessModule.GetProcessModule()
pm.UnRegisterSession(session)
id = pm.RegisterSession(session)
connection = GetConnectionFromId(id)
return connection
[docs]def demo1():
"""This simple demonstration creates a sphere, renders it and delivers
it to the client using Fetch. It returns a tuple of (data, render
view)"""
if not ActiveConnection:
Connect()
if paraview.compatibility.GetVersion() <= (3, 4):
ss = sources.SphereSource(Radius=2, ThetaResolution=32)
shr = filters.ShrinkFilter(Input=OutputPort(ss, 0))
cs = sources.ConeSource()
app = filters.Append()
else:
ss = sources.Sphere(Radius=2, ThetaResolution=32)
shr = filters.Shrink(Input=OutputPort(ss, 0))
cs = sources.Cone()
app = filters.AppendDatasets()
app.Input = [shr, cs]
rv = CreateRenderView()
rep = CreateRepresentation(app, rv)
rv.ResetCamera()
rv.StillRender()
data = Fetch(ss)
return (data, rv)
[docs]def demo2(fname="/Users/berk/Work/ParaViewData/Data/disk_out_ref.ex2"):
"""This method demonstrates the user of a reader, representation and
view. It also demonstrates how meta-data can be obtained using proxies.
Make sure to pass the full path to an exodus file. Also note that certain
parameters are hard-coded for disk_out_ref.ex2 which can be found
in ParaViewData. This method returns the render view."""
if not ActiveConnection:
Connect()
# Create the exodus reader and specify a file name
reader = sources.ExodusIIReader(FileName=fname)
# Get the list of point arrays.
if paraview.compatibility.GetVersion() <= (3, 4):
arraySelection = reader.PointResultArrayStatus
else:
arraySelection = reader.PointVariables
print(arraySelection.Available)
# Select all arrays
arraySelection.SetData(arraySelection.Available)
# Next create a default render view appropriate for the session type.
rv = CreateRenderView()
# Create the matching representation
rep = CreateRepresentation(reader, rv)
rep.Representation = 1 # Wireframe
# Black background is not pretty
rv.Background = [0.4, 0.4, 0.6]
rv.StillRender()
# Reset the camera to include the whole thing
rv.ResetCamera()
rv.StillRender()
# Change the elevation of the camera. See VTK documentation of vtkCamera
# for camera parameters.
c = rv.GetActiveCamera()
c.Elevation(45)
rv.StillRender()
# Now that the reader execute, let's get some information about it's
# output.
pdi = reader[0].PointData
# This prints a list of all read point data arrays as well as their
# value ranges.
print('Number of point arrays:', len(pdi))
for i in range(len(pdi)):
ai = pdi[i]
print("----------------")
print("Array:", i, ai.Name, ":")
numComps = ai.GetNumberOfComponents()
print("Number of components:", numComps)
for j in range(numComps):
if paraview.compatibility.GetVersion() <= (3, 4):
print("Range:", ai.Range(j))
else:
print("Range:", ai.GetRange(j))
# White is boring. Let's color the geometry using a variable.
# First create a lookup table. This object controls how scalar
# values are mapped to colors. See VTK documentation for
# details.
lt = rendering.PVLookupTable()
# Assign it to the representation
rep.LookupTable = lt
# Color by point array called Pres
rep.ColorArrayName = ("POINTS", "Pres")
# Add to RGB points. These are tuples of 4 values. First one is
# the scalar values, the other 3 the RGB values. This list has
# 2 points: Pres: 0.00678, color: blue, Pres: 0.0288, color: red
lt.RGBPoints = [0.00678, 0, 0, 1, 0.0288, 1, 0, 0]
lt.ColorSpace = 1 # HSV
rv.StillRender()
return rv
[docs]def demo3():
"""This method demonstrates the use of servermanager with numpy as
well as pylab for plotting. It creates an artificial data sources,
probes it with a line, delivers the result to the client using Fetch
and plots it using pylab. This demo requires numpy and pylab installed.
It returns a tuple of (data, render view)."""
from vtkmodules.util import numpy_support
import pylab
if not ActiveConnection:
Connect()
# Create a synthetic data source
if paraview.compatibility.GetVersion() <= (3, 4):
source = sources.RTAnalyticSource()
else:
source = sources.Wavelet()
# Let's get some information about the data. First, for the
# source to execute
source.UpdatePipeline()
di = source.GetDataInformation()
print("Data type:", di.GetPrettyDataTypeString())
print("Extent:", di.GetExtent())
print("Array name:", \
source[0].PointData[0].Name)
rv = CreateRenderView()
rep1 = CreateRepresentation(source, rv)
rep1.Representation = 3 # outline
# Let's apply a contour filter
cf = filters.Contour(Input=source, ContourValues=[200])
# Select the array to contour by
# cf.SelectInputScalars = 'RTData'
rep2 = CreateRepresentation(cf, rv)
rv.Background = (0.4, 0.4, 0.6)
# Reset the camera to include the whole thing
rv.StillRender()
rv.ResetCamera()
rv.StillRender()
# Now, let's probe the data
if paraview.compatibility.GetVersion() <= (3, 4):
probe = filters.Probe(Input=source)
# with a line
line = sources.LineSource(Resolution=60)
else:
probe = filters.ResampleWithDataset(SourceDataArrays=source)
# with a line
line = sources.Line(Resolution=60)
# that spans the dataset
bounds = di.GetBounds()
print("Bounds: ", bounds)
line.Point1 = bounds[0:6:2]
line.Point2 = bounds[1:6:2]
probe.Source = line
# Render with the line
rep3 = CreateRepresentation(line, rv)
rv.StillRender()
# Now deliver it to the client. Remember, this is for small data.
data = Fetch(probe)
# Convert it to a numpy array
data = numpy_support.vtk_to_numpy(
data.GetPointData().GetArray("RTData"))
# Plot it using matplotlib
pylab.plot(data)
pylab.show()
return (data, rv, probe)
[docs]def demo4(fname="/Users/berk/Work/ParaViewData/Data/can.ex2"):
"""This method demonstrates the user of AnimateReader for
creating animations."""
if not ActiveConnection:
Connect()
reader = sources.ExodusIIReader(FileName=fname)
view = CreateRenderView()
repr = CreateRepresentation(reader, view)
view.StillRender()
view.ResetCamera()
view.StillRender()
c = view.GetActiveCamera()
c.Elevation(95)
return AnimateReader(reader, view)
[docs]def demo5():
""" Simple sphere animation"""
if not ActiveConnection:
Connect()
if paraview.compatibility.GetVersion() <= (3, 4):
sphere = sources.SphereSource()
else:
sphere = sources.Sphere()
view = CreateRenderView()
repr = CreateRepresentation(sphere, view)
view.StillRender()
view.ResetCamera()
view.StillRender()
# Create an animation scene
scene = animation.AnimationScene()
# Add 1 view
scene.ViewModules = [view]
# Create a cue to animate the StartTheta property
cue = animation.KeyFrameAnimationCue()
cue.AnimatedProxy = sphere
cue.AnimatedPropertyName = "StartTheta"
# Add it to the scene's cues
scene.Cues = [cue]
# Create 2 keyframes for the StartTheta track
keyf0 = animation.CompositeKeyFrame()
keyf0.Type = 2 # Set keyframe interpolation type to Ramp.
# At time = 0, value = 0
keyf0.KeyTime = 0
keyf0.KeyValues = [0]
keyf1 = animation.CompositeKeyFrame()
# At time = 1.0, value = 200
keyf1.KeyTime = 1.0
keyf1.KeyValues = [200]
# Add keyframes.
cue.KeyFrames = [keyf0, keyf1]
scene.Play()
return scene
ASSOCIATIONS = {'POINTS': 0, 'CELLS': 1, 'FIELD': 2, 'VERTICES': 4, 'EDGES': 5, 'ROWS': 6}
_LEGACY_ASSOCIATIONS = {'POINT_DATA': 0, 'CELL_DATA': 1}
[docs]def GetAssociationAsString(val):
"""Returns array association string from its integer value"""
global ASSOCIATIONS
if not type(val) == int:
raise ValueError("argument must be of type 'int'")
for k in ASSOCIATIONS:
if ASSOCIATIONS[k] == val:
return k
raise RuntimeError("invalid association type '%d'" % val)
[docs]def GetAssociationFromString(val):
"""Returns array association integer value from its string representation"""
global ASSOCIATIONS, _LEGACY_ASSOCIATIONS
val = str(val).upper()
try:
return ASSOCIATIONS[val]
except KeyError:
try:
return _LEGACY_ASSOCIATIONS[val]
except KeyError:
raise RuntimeError("invalid association string '%s'" % val)
# Users can set the active connection which will be used by API
# to create proxies etc when no connection argument is passed.
# Connect() automatically sets this if it is not already set.
ActiveConnection = None
"""Keeps track of all connection objects. Unless the process was run with
--multi-servers flag set to True, this will generally be just 1 item long at the
most."""
Connections = []
[docs]def SetActiveConnection(connection=None):
"""Set the active connection. If the process was run without multi-server
enabled and this method is called with a non-None argument while an
ActiveConnection is present, it will raise a RuntimeError."""
global ActiveConnection
# supports_simutaneous_connections =\
# vtkProcessModule.GetProcessModule().GetMultipleSessionsSupport()
# print ("updating active connection", connection)
ActiveConnection = connection
# This will ensure that servemanager.sources.* will point to the right
# constructors.
__exposeActiveModules__()
# If this method was initiated from Python, we need to pass that info to the
# ServerManager.
session = None
if ActiveConnection:
session = ActiveConnection.Session
# This is a no-op if the session is unchanged.
pxm = vtkSMProxyManager.GetProxyManager()
pxm.SetActiveSession(session)
return ActiveConnection
# Needs to be called when paraview module is loaded from python instead
# of pvpython, pvbatch or GUI. If running in parallel this will
# also set up the vtkMPIController automatically as well. Users
# should specify paraview.options.{batch,symmetric} to be true
# or false to set up ParaView properly. If MPI was initialized, calling
# servermanager.Finalize() may also be needed to exit properly without
# VTK_DEBUG_LEAKS reporting memory leaks.
if not vtkProcessModule.GetProcessModule():
slist = vtkStringList()
slist.AddString(sys.executable)
if paraview.options.batch:
ptype = vtkProcessModule.PROCESS_BATCH
if paraview.options.symmetric:
slist.AddString("--symmetric")
else:
ptype = vtkProcessModule.PROCESS_CLIENT
# ensure that we don't init MPI on the client process
slist.AddString("--no-mpi")
if not vtkInitializationHelper.Initialize(slist, ptype):
raise RuntimeError("Failed to initialize ParaView engine!")
# In case of non-symmetric mode, if we are a satellite,
# we should lock right away and wait for the requests
# from master
pm = vtkProcessModule.GetProcessModule()
if not pm.GetSymmetricMPIMode() and pm.GetPartitionId() > 0:
paraview.options.satelite = True
sid = vtkSMSession.ConnectToSelf()
pm.GetGlobalController().ProcessRMIs()
pm.UnRegisterSession(sid)
# since we initialized paraview, lets ensure that we finalize it too
import atexit
atexit.register(Finalize)
# Initialize progress printing. Can be turned off by calling
# ToggleProgressPrinting() again.
progressObserverTag = None
currentAlgorithm = False
currentProgress = 0
if not paraview.fromGUI:
ToggleProgressPrinting()
_pyproxies = {}
# Set up our custom importer (if possible)
finder = ParaViewMetaPathFinder()
sys.meta_path.append(finder)
def __exposeActiveModules__():
"""Update servermanager submodules to point to the current
ActiveConnection.Modules.*"""
global ActiveConnection
# Expose all active module to the current servermanager module
if ActiveConnection:
g = globals()
for item in ActiveConnection.ProxiesNS.getNamespaces().items():
g[item[0]] = item[1]
# need to delete obsolete groups? that will never happen since when a
# group is registered, it gets registered for all ConnectionProxyNamespaces
# instances.
[docs]def GetConnectionFromId(id):
"""Returns the Connection object corresponding a connection identified by
the id."""
global Connections
for connection in Connections:
if connection.ID == id:
return connection
return None
[docs]def GetConnectionFromSession(session):
"""Returns the Connection object corresponding to a vtkSMSession instance."""
global Connections
for connection in Connections:
if connection.Session == session:
return connection
pm = vtkProcessModule.GetProcessModule()
sid = pm.GetSessionID(session)
if session and sid:
# it implies that the we simply may not have received the
# ConnectionCreatedEvent event yet. Create a connection object.
c = Connection(sid, session)
Connections.append(c)
return c
return None
def __connectionCreatedCallback(obj, string):
"""Callback called when a new session is created."""
global Connections
pm = vtkProcessModule.GetProcessModule()
sid = pm.GetEventCallDataSessionId()
# this creates the Connection object if needed.
GetConnectionFromSession(pm.GetSession(sid))
def __connectionClosedCallback(obg, string):
"""Callback called when a new session is closed."""
global Connections, ActiveConnection
pm = vtkProcessModule.GetProcessModule()
sid = pm.GetEventCallDataSessionId()
if sid:
c = GetConnectionFromId(sid)
if c:
for cc in range(len(Connections)):
if Connections[cc] == c:
del Connections[cc]
break
c.close()
del c
import gc
gc.collect()
def __initialize():
"""Does initialization of the module, ensuring that the module's state
correctly reflects that of the ProcessModule/ServerManager."""
global ActiveConnection, Connections
# Monitor connection creations/deletions on the ProcessModule.
pm = vtkProcessModule.GetProcessModule()
pm.AddObserver("ConnectionCreatedEvent", __connectionCreatedCallback)
pm.AddObserver("ConnectionClosedEvent", __connectionClosedCallback)
# Iterate over existing connections, if any, and set the datastructures up.
iter = vtkProcessModule.GetProcessModule().NewSessionIterator();
iter.UnRegister(None)
iter.InitTraversal()
firstSession = None
while not iter.IsDoneWithTraversal():
c = Connection(iter.GetCurrentSessionId(), iter.GetCurrentSession())
Connections.append(c)
iter.GoToNextItem()
# Update active session.
activeConnection = GetConnectionFromSession( \
vtkSMProxyManager.GetProxyManager().GetActiveSession())
SetActiveConnection(activeConnection)
__initialize()
if hasattr(sys, "ps1"):
# session is interactive.
paraview.print_debug_info(vtkSMProxyManager.GetParaViewSourceVersion());