Source code for tecplot.layout.frame

import contextlib
import ctypes
import logging
import numbers

from ..tecutil import _tecutil, sv
from ..constant import *
from ..exception import *
from .. import annotation, constant, plot, session, tecutil


log = logging.getLogger(__name__)


[docs] class Frame(session.Style): """`Frame <tecplot.layout.Frame>` object within a `Page <tecplot.layout.Page>`, holding onto a `Dataset <data.Dataset>` and a `Plot`. Parameters: uid (`int`, optional): This must be a *valid* unique ID number pointing internally to a Frame object or `None`. A new `Frame <tecplot.layout.Frame>` is created if set to `None`. (default: `None`) page (`Page <tecplot.layout.Page>`, optional): The destination `Page <tecplot.layout.Page>` of this newly created `Frame <tecplot.layout.Frame>`. If `None`, the currently active `Page <tecplot.layout.Page>` is used. (default: `None`) Warning: Though it is possible to create a `Frame <tecplot.layout.Frame>` object using the constructor, it is usually sufficient to obtain a frame through `tecplot.active_frame()` or `Page.frame()`. One can also create a `Frame <tecplot.layout.Frame>` using a `Page <tecplot.layout.Page>` handle with `Page.add_frame()`. The concept of the `Frame <tecplot.layout.Frame>` is central to understanding the |Tecplot Engine|. The `Frame <tecplot.layout.Frame>` is what connects a `Dataset <data.Dataset>` to a `Plot` handle from which one manipulates the desired image as well as accessing the attached data: .. code-block:: python import tecplot frame = tecplot.active_frame() # will print: 'Frame "Frame 001"' print(frame) """ page = None """The `Page <tecplot.layout.Page>` containing this Frame. This provides access to the parent `Page <tecplot.layout.Page>`: .. code-block:: python import tecplot frame = tecplot.active_frame() page = frame.page # Will print: "Page 001" print(page.name) """ def __init__(self, uid, page): self.page = page self.uid = uid """The internal unique ID number of this Frame.""" super().__init__(sv.FRAMELAYOUT, uniqueid=self.uid) def __str__(self): """Brief string representation. Returns: `str`: Brief representation of this `Frame <tecplot.layout.Frame>`. Example: .. code-block:: python import tecplot frame = tecplot.active_frame() # will print: 'Frame: "Frame 001"' print(frame) """ return 'Frame: "{name}"'.format(name=self.name) def __repr__(self): """Executable string representation. Returns: `str`: Internal representation of this `Frame <tecplot.layout.Frame>`. The string returned can be executed to generate an identical copy of this `Frame <tecplot.layout.Frame>` object: .. code-block:: python import tecplot from tecplot.layout import Frame, Page frame = tecplot.active_frame() ''' The "repr" string of the Frame is executable code. The following will print: "Frame(uid=11, page=Page(uid=1))" ''' print(repr(frame)) frame2 = None exec('frame2 = '+repr(frame)) ''' At this point, frame2 is just another handle to the exact same frame object in the Tecplot Engine ''' assert frame2 == frame """ return 'Frame(uid={uid}, page={page})'.format(uid=self.uid, page=repr(self.page)) def __eq__(self, other): """Checks for `Frame <tecplot.layout.Frame>` equality in the |Tecplot Engine|. Returns: `bool`: `True` if the unique ID numbers are the same for both `Frames <tecplot.layout.Frame>`. Example: .. code-block:: python import tecplot page = tecplot.active_page() frame1 = page.active_frame() frame2 = page.add_frame() assert not (frame1 == frame2) assert page.active_frame() == frame2 """ return self.uid == other.uid @property def position(self): """`tuple`: ``(x,y)`` position of the `Frame <tecplot.layout.Frame>` in inches. The `Frame <tecplot.layout.Frame>` x position is relative to the left side of the paper. The `Frame <tecplot.layout.Frame>` y position is relative to the top of the paper. If x is `None`, the `Frame <tecplot.layout.Frame>` x position is not changed. If y is `None`, the `Frame <tecplot.layout.Frame>` y position is not changed. Set `Frame <tecplot.layout.Frame>` position 1 inch from the left side of the paper and two inches from the top of the paper:: >>> tp.active_frame().position=(1.0, 2.0) Move the active `Frame <tecplot.layout.Frame>` one inch to the right:: >>> tp.active_frame().position=(tp.active_frame().position.x+1, None) """ return session.XY(self, sv.XYPOS) @position.setter def position(self, pos): session.XY(self, sv.XYPOS)[:] = pos @property def aux_data(self): """Auxiliary data for this frame. Returns: `AuxData` This is the auxiliary data attached to the frame. Such data is written to the layout file by default and can be retrieved later. Example usage:: >>> aux = tp.active_frame().aux_data >>> aux['Result'] = '3.14159' >>> print(aux['Result']) 3.14159 """ return session.AuxData(self, AuxDataObjectType.Frame)
[docs] def texts(self): """Get an iterator for all `Text` objects in the frame. This example shows how to obtain a list of all red `Text` objects:: >>> from tecplot.constant import Color >>> all_red_text_objects = [T for T in tp.active_frame().texts() ... if T.color == Color.Red] """ return annotation.TextIterator(self)
[docs] def geometries(self): """Get an iterator for all geometry objects in the frame. This method provides access to all geometric shape instances attached to this `Frame <tecplot.layout.Frame>`. This includes `Circle`, `Polyline2D`, `Polyline3D` and all similar objects:: >>> for geom in frame.geometries(): ... print(type(geom)) <class 'tecplot.annotation.geometry.Circle'> <class 'tecplot.annotation.polyline.Polyline2D'> """ shapefilter = annotation.GeometryIterator.ShapeFilter return annotation.GeometryIterator(shapefilter, self)
[docs] def images(self): """Get an iterator for all image objects in the frame. This method provides access to all `Image` and `GeoreferencedImage` instances attached to this `Frame <tecplot.layout.Frame>`:: >>> for image in frame.images(): ... print(image.filename) image1.png image2.png """ imagefilter = annotation.GeometryIterator.ImageFilter return annotation.GeometryIterator(imagefilter, self)
[docs] @contextlib.contextmanager def activated(self): """Context for temporarily activating this `Frame <tecplot.layout.Frame>`. Example: .. code-block:: python import tecplot page = tecplot.active_page() frame1 = page.active_frame() frame2 = page.add_frame() assert frame2.active with frame1.activated(): # frame1 is active only during this context assert frame1.active # there is only one frame active at a time assert not frame2.active assert frame2.active """ frame_uid = _tecutil.FrameGetActiveID() if self.uid == frame_uid: yield else: page_uid = _tecutil.PageGetUniqueID() self.activate() try: yield finally: with tecutil.lock(): if _tecutil.PageGetUniqueID() != page_uid: _tecutil.PageSetCurrentByUniqueID(page_uid) _tecutil.FrameActivateByUniqueID(frame_uid)
[docs] @tecutil.lock() def load_stylesheet(self, filename, plot_style=True, text=True, geom=True, streams=True, contours=True, frame_geom=False, merge=False): """Apply a stylesheet settings file to this frame. Parameters: filename (`pathlib.Path` or `str`): The path to a stylesheet file. (See note below conerning absolute and relative paths.) plot_style (`bool`, optional): Apply the stylesheet's plot style. (default: `True`) text (`bool`, optional): Include the stylesheet's text objects. (default: `True`) geom (`bool`, optional): Include the stylesheet's geometry objects. (default: `True`) streams (`bool`, optional): Include the stylesheet's stream traces. (default: `True`) contours (`bool`, optional): Include the stylesheet's contour levels. (default: `True`) frame_geom (`bool`, optional): Apply the stylesheet's frame position and size. (default: `False`) merge (`bool`, optional): Merge with the frame's current style. (default: `False`) .. note:: **Absolute and relative paths with PyTecplot** Relative paths, when used within the PyTecplot API are always from Python's current working directory which can be obtained by calling :func:`os.getcwd()`. This is true for batch and `connected <tecplot.session.connect()>` modes. One exception to this is paths within a macro command or file which will be relative to the |Tecplot Engine|'s home directory, which is typically the |Tecplot 360| installation directory. Finally, when connected to a remote (non-local) instance of Tecplot 360, only absolute paths are allowed. Note that backslashes must be escaped which is especially important for windows paths such as ``"C:\\\\Users"`` or ``"\\\\\\\\server\\\\path"`` which will resolve to ``"C:\\Users"`` and ``"\\\\server\\path"`` respectively. Alternatively, one may use Python's raw strings: ``r"C:\\Users"`` and ``r"\\\\server\\path"`` Example usage:: >>> frame = tecplot.active_frame() >>> frame.load_stylesheet('my_style.sty') """ filepath = tecutil.normalize_path(filename) with self.activated(): log.debug(f'Applying style from {filepath} to frame') if not _tecutil.ReadStylesheet(str(filepath), plot_style, text, geom, streams, contours, merge, frame_geom): raise TecplotSystemError()
[docs] @tecutil.lock() def save_stylesheet(self, filename, plot_style=True, aux_data=True, text=True, geom=True, streams=True, contours=True, defaults=False, relative_paths=True, compress=False): """Save the frame's current style to a file. Parameters: filename (`pathlib.Path` or `str`): The path to a stylesheet file. (See note below conerning absolute and relative paths.) plot_style (`bool`, optional): Include the frame's plot style. (default: `True`) aux_data (`bool`, optional): Include auxiliary data. (default: `True`) text (`bool`, optional): Include text objects. (default: `True`) geom (`bool`, optional): Include geometry objects. (default: `True`) streams (`bool`, optional): Include stream traces. (default: `True`) contours (`bool`, optional): Include contour levels. (default: `True`) defaults (`bool`, optional): Include all factory defaults used by the current style. (default: `False`) relative_paths (`bool`, optional): Use relative paths. (default: `True`) compress (`bool`, optional): Compress the output of the style. (default: `False`) .. note:: **Absolute and relative paths with PyTecplot** Relative paths, when used within the PyTecplot API are always from Python's current working directory which can be obtained by calling :func:`os.getcwd()`. This is true for batch and `connected <tecplot.session.connect()>` modes. One exception to this is paths within a macro command or file which will be relative to the |Tecplot Engine|'s home directory, which is typically the |Tecplot 360| installation directory. Finally, when connected to a remote (non-local) instance of Tecplot 360, only absolute paths are allowed. Note that backslashes must be escaped which is especially important for windows paths such as ``"C:\\\\Users"`` or ``"\\\\\\\\server\\\\path"`` which will resolve to ``"C:\\Users"`` and ``"\\\\server\\path"`` respectively. Alternatively, one may use Python's raw strings: ``r"C:\\Users"`` and ``r"\\\\server\\path"`` Example usage:: >>> frame = tecplot.active_frame() >>> frame.save_stylesheet('my_style.sty') """ filepath = tecutil.normalize_path(filename) with self.activated(): with tecutil.ArgList() as arglist: arglist.update(( (sv.FNAME, str(filepath)), (sv.INCLUDEPLOTSTYLE, bool(plot_style)), (sv.INCLUDEAUXDATA, bool(aux_data)), (sv.INCLUDETEXT, bool(text)), (sv.INCLUDEGEOM, bool(geom)), (sv.INCLUDESTREAMPOSITIONS, bool(streams)), (sv.INCLUDECONTOURLEVELS, bool(contours)), (sv.INCLUDEFACTORYDEFAULTS, bool(defaults)), (sv.USERELATIVEPATHS, bool(relative_paths)), (sv.COMPRESS, bool(compress)))) if not _tecutil.WriteStylesheetX(arglist): raise TecplotSystemError()
@property def name(self): """`str`: Returns or sets the name. This is the name used when searching for `Frame <tecplot.layout.Frame>` objects in `Page.frames` and `Page.frame`. It does not have to be unique, even for multiple frames in a single `Page <tecplot.layout.Page>`: .. code-block:: python import tecplot frame = tecplot.active_frame() frame.name = '3D Data View' # will print: "this frame: 3D Data View" print('this frame:', frame.name) """ with self.activated(): return _tecutil.FrameGetName()[1] @name.setter @tecutil.lock() def name(self, name): with self.activated(): _tecutil.FrameSetName(name) @property def active(self): """Checks if this `Frame <tecplot.layout.Frame>` is active. Returns: `bool`: `True` if this `Frame <tecplot.layout.Frame>` is the active `Frame <tecplot.layout.Frame>`. """ return self.uid == _tecutil.FrameGetActiveID() @property def current(self): return self.uid == _tecutil.FrameGetUniqueID()
[docs] @tecutil.lock() def activate(self): """Causes this `Frame <tecplot.layout.Frame>` to become active. The parent `Page <tecplot.layout.Page>` is implicitly "activated" as a side-effect of this operation: .. code-block:: python import tecplot page1 = tecplot.active_page() frame1 = page1.active_frame() page2 = tecplot.add_page() frame2 = page2.active_frame() assert not (frame1.active and page1.active) assert frame2.active and page2.active frame1.activate() assert not (frame2.active or page2.active) assert frame1.active and page1.active """ if not self.active: if self.page is not None: self.page.activate() if not _tecutil.FrameActivateByUniqueID(self.uid): err = 'could not activate frame with uid {0}'.format(self.uid) raise TecplotSystemError(err)
@property def plot_type(self): """`constant.PlotType`: Returns or sets the current plot type. A `Frame <tecplot.layout.Frame>` can have only one active plot type at any given time. The types are enumerated by `constant.PlotType`: .. code-block:: python import os import tecplot from tecplot.constant import PlotType frame = tecplot.active_frame() assert frame.plot_type is PlotType.Sketch install_dir = tecplot.session.tecplot_install_directory() infile = os.path.join(install_dir, 'examples', 'SimpleData', 'SpaceShip.lpk') tecplot.load_layout(infile) frame = tecplot.active_frame() assert frame.plot_type is PlotType.Cartesian3D frame.plot_type = PlotType.Cartesian2D assert frame.plot_type is PlotType.Cartesian2D .. note:: Plot type cannot be set to `constant.PlotType.Automatic`. """ return _tecutil.FrameGetPlotTypeForFrame(self.uid) @plot_type.setter @tecutil.lock() def plot_type(self, plot_type): with self.activated(): try: res = _tecutil.FrameSetPlotType(plot_type.value) except TecplotSystemError as err: if not self.has_dataset: res = SetValueReturnCode.ContextError1 else: raise err if res not in [SetValueReturnCode.Ok, SetValueReturnCode.DuplicateValue]: if res is SetValueReturnCode.ContextError1: raise TecplotSystemError('no dataset attached to frame') raise TecplotSystemError(res)
[docs] def plot(self, plot_type=PlotType.Active): """The primary `Plot` style-control object. Returns: `Plot`: One of the possible `Plot` subclasses, depending on the ``plot_type`` specified. By default, the active plot type, obtained from `Frame.plot_type`, is used. The `Plot` object is the handle through which one can manipulate the style and visual representation of the `Dataset <data.Dataset>`. Possible return types are: `SketchPlot`, `Cartesian2DFieldPlot`, `Cartesian3DFieldPlot`, `PolarLinePlot` and `XYLinePlot`. Each of these have their own specific set of attributes and methods: .. code-block:: python import os import tecplot from tecplot.constant import PlotType install_dir = tecplot.session.tecplot_install_directory() infile = os.path.join(install_dir, 'examples', 'SimpleData', 'SpaceShip.lpk') tecplot.load_layout(infile) frame = tecplot.active_frame() assert frame.plot_type is PlotType.Cartesian3D plot3d = frame.plot() plot3d.show_contour = True """ if plot_type is PlotType.Active: plot_type = None _dispatch = { PlotType.Cartesian2D: plot.Cartesian2DFieldPlot, PlotType.Cartesian3D: plot.Cartesian3DFieldPlot, PlotType.XYLine: plot.XYLinePlot, PlotType.PolarLine: plot.PolarLinePlot, PlotType.Sketch: plot.SketchPlot} return _dispatch[plot_type or self.plot_type](self)
[docs] @tecutil.lock() def move_to_bottom(self): """Moves `Frame <tecplot.layout.Frame>` behind all others in `Page <tecplot.layout.Page>`. """ _tecutil.FrameMoveToBottomByUniqueID(self.uid)
[docs] @tecutil.lock() def move_to_top(self): """Moves `Frame <tecplot.layout.Frame>` in front of all others in `Page <tecplot.layout.Page>`. """ _tecutil.FrameMoveToTopByUniqueID(self.uid)
[docs] @tecutil.lock() def active_zones(self, *zones): """Returns or sets the active `Zones <data_access>`. Parameters: zones (`Zones <data_access>`, optional): The `Zone <data_access>` objects, which must be in the `Dataset <data.Dataset>` attached to this `Frame <tecplot.layout.Frame>`, that will be activated. All other `Zones <data_access>` will be deactivated. Returns: `Zones <data_access>`: This will return a generator of active `Zones <data_access>` in this `Frame <tecplot.layout.Frame>`. This should only be used on frames with an active plot type that contains a dataset with at least one zone. """ if __debug__: if self.plot_type is PlotType.Sketch: err = 'Active plot type is Sketch which has no active zones.' raise TecplotLogicError(err) if not self.has_dataset: raise TecplotLogicError('Frame has no dataset.') if self.dataset.num_zones == 0: raise TecplotLogicError('Dataset has no zones.') with self.activated(): if zones: with tecutil.IndexSet(*zones) as zoneset: _tecutil.ZoneSetActive(zoneset, AssignOp.Equals.value) else: zoneset = _tecutil.ZoneGetActiveForFrame(self.uid) zones = [self.dataset.zone(i) for i in zoneset] zoneset.dealloc() return zones
[docs] @tecutil.lock() def delete_geometry(self, geom): """Delete a geometry or image annotation object from the `Frame <tecplot.layout.Frame>`. Parameters: geom (`Circle`, `Image` or similar instance): The annotation instance to be removed from the `Frame <tecplot.layout.Frame>`. .. warning:: After the annotation is deleted, all handles to it will no longer be valid and all accessing any properties of the object will result in a `TecplotLogicError` being raised. Example usage: .. code-block:: python :emphasize-lines: 6 import tecplot from tecplot.constant import CoordSys frame = tecplot.active_frame() rectangle = frame.add_rectangle((0.5, 0.5), (0.1, 0.2), CoordSys.Frame) frame.delete_geometry(rectangle) """ with self.activated(): _tecutil.GeomDelete(geom.uid)
[docs] @tecutil.lock() def delete_image(self, img): """Delete an image annotation object from the `Frame <tecplot.layout.Frame>`. Parameters: img (`Image` or `GeoreferencedImage`): The annotation instance to be removed from the `Frame <tecplot.layout.Frame>`. .. warning:: After the annotation is deleted, all handles to it will no longer be valid and all accessing any properties of the object will result in a `TecplotLogicError` being raised. .. seealso:: `delete_geometry()` """ return self.delete_geometry(img)
[docs] @tecutil.lock() def delete_text(self, text): """Delete a `text <annotation.Text>` object from a frame. When deleted, the text object is no longer displayed in the frame and is permanently invalid. To display the text in the frame again, a new text object must be created by calling `add_text`. .. warning:: Use this method with care. After a text object has been deleted by calling this method, it is no longer valid, and all properties of the deleted text object will throw `TecplotLogicError` when accessed. Example usage: .. code-block:: python import tecplot as tp text = tp.active_frame().add_text("abc") tp.active_frame().delete_text(text) # The text object is no longer valid. # Any property access will throw TecplotLogicError try: print(text.text_string) except tp.exception.TecplotLogicError as e: print(e) .. seealso:: `add_text` """ with self.activated(): _tecutil.TextDelete(text.uid)
[docs] @tecutil.lock() def add_image(self, filename, position, height): """Add an image to the `Frame <tecplot.layout.Frame>`. Parameters: filename (`pathlib.Path` or `str`): The image source file. The format of this file must be Microsoft Windows Bitmap (*.bmp), JPEG (*.jpg or *.jpeg) or Portable Network Graphics (*.png). position (`tuple` of `floats <float>`): Position :math:`(x, y)` of the image in percentage frame coordinates. height (`float`): The initial size, or height, of the image in percentage frame units. The initial width in frame units is set automatically based on the width to height aspect ratio of the image. Returns: `annotation.Image` This example adds an image to the upper left quadrant of the frame:: >>> frame = tp.active_frame() >>> img = frame.add_image('myimage.png', (0, 50), 50) """ x, y = (float(p) for p in position) gid = _tecutil.GeomImageCreate(str(filename), x, y, height) if not gid: raise TecplotSystemError() return annotation.Image(gid, self)
[docs] @tecutil.lock() def add_georeferenced_image(self, image_filename, world_filename): """Add a geographic reference image and world file to the `Frame <tecplot.layout.Frame>`. Parameters: image_filename (`pathlib.Path` or `str`): The image source file. world_filename (`pathlib.Path` or `str`): The world file associated with the image. Returns: `annotation.GeoreferencedImage` Example usage:: >>> frame = tp.active_frame() >>> imgfile = 'region.png' >>> worldfile = 'region.pgw' >>> geoimg = frame.add_georeferenced_image(imgfile, worldfile) """ gid = _tecutil.GeoRefImageCreate(str(image_filename), str(world_filename)) if not gid: raise TecplotSystemError() return annotation.GeoreferencedImage(gid, self)
[docs] @tecutil.lock() def add_circle(self, center, radius, coord_sys): """Place a circle annotation on the `Frame <tecplot.layout.Frame>`. Parameters: center (`tuple` of `floats <float>`): Position :math:`(x, y)` of the center of the circle in the coordinates specified by **coord_sys**. radius (`float`): The size of the radius in the coordinates specified by **coord_sys**. coord_sys (`CoordSys`): The coordinate system to use for position and size of this annotation. Returns: `annotation.Circle` Example usage: .. code-block:: python import tecplot from tecplot.constant import CoordSys frame = tecplot.active_frame() circle = frame.add_circle((0.2, 0.2), 0.1, CoordSys.Frame) .. seealso:: `add_ellipse`, `add_rectangle`, `add_square`, `add_polyline` """ x, y = (float(i) for i in center) coord_sys = constant.CoordSys(coord_sys) gid = _tecutil.GeomCircleCreate(coord_sys.value, x, y, float(radius)) if not gid: raise TecplotSystemError() return annotation.Circle(gid, self)
[docs] @tecutil.lock() def add_ellipse(self, center, size, coord_sys): """Place an ellipse annotation on the `Frame <tecplot.layout.Frame>`. Parameters: center (`tuple` of `floats <float>`): Position :math:`(x, y)` of the ellipse in the coordinates specified by **coord_sys**. size (`tuple` of `floats <float>`): Lengths :math:`(h_{axis}, v_{axis})` of the horizontal and vertical axes in the coordinates specified by **coord_sys**. Both lengths must be non-zero. coord_sys (`CoordSys`): The coordinate system to use for position and size of this annotation. Returns: `annotation.Ellipse` Example usage: .. code-block:: python import tecplot from tecplot.constant import CoordSys frame = tecplot.active_frame() ellipse = frame.add_ellipse((0.5, 0.5), (0.1, 0.2), CoordSys.Frame) .. seealso:: `add_circle`, `add_rectangle`, `add_square`, `add_polyline` """ x, y = (float(i) for i in center) haxis, vaxis = (float(i) for i in size) coord_sys = constant.CoordSys(coord_sys) gid = _tecutil.GeomEllipseCreate(coord_sys.value, x, y, haxis, vaxis) if not gid: raise TecplotSystemError() return annotation.Ellipse(gid, self)
[docs] @tecutil.lock() def add_rectangle(self, corner, size, coord_sys): """Place a rectangle annotation on the `Frame <tecplot.layout.Frame>`. Parameters: center (`tuple` of `floats <float>`): Position :math:`(x, y)` of the rectangle in the coordinates specified by **coord_sys**. size (`tuple` of `floats <float>`): Size :math:`(width, height)` of the rectangle in the coordinates specified by **coord_sys**. coord_sys (`CoordSys`): The coordinate system to use for position and size of this annotation. Returns: `annotation.Rectangle` Example usage: .. code-block:: python import tecplot from tecplot.constant import CoordSys frame = tecplot.active_frame() rectangle = frame.add_rectangle((0.5, 0.5), (0.1, 0.2), CoordSys.Frame) .. seealso:: `add_circle`, `add_ellipse`, `add_square`, `add_polyline` """ x, y = (float(i) for i in corner) w, h = (float(i) for i in size) coord_sys = constant.CoordSys(coord_sys) gid = _tecutil.GeomRectangleCreate(coord_sys.value, x, y, w, h) if not gid: raise TecplotSystemError() return annotation.Rectangle(gid, self)
[docs] @tecutil.lock() def add_square(self, corner, size, coord_sys): """Place a square annotation on the `Frame <tecplot.layout.Frame>`. Parameters: corner (`tuple` of `floats <float>`): Position :math:`(x, y)` of the lower-left corner of the square in the coordinates specified by **coord_sys**. size (`float`): Side-length of the square in the coordinates specified by **coord_sys**. coord_sys (`CoordSys`): The coordinate system to use for position and size of this annotation. Returns: `annotation.Square` Example usage: .. code-block:: python import tecplot from tecplot.constant import CoordSys frame = tecplot.active_frame() square = frame.add_square((0.2, 0.2), 0.1, CoordSys.Frame) .. seealso:: `add_circle`, `add_ellipse`, `add_rectangle`, `add_polyline` """ x, y = (float(i) for i in corner) coord_sys = constant.CoordSys(coord_sys) gid = _tecutil.GeomSquareCreate(coord_sys.value, x, y, float(size)) if not gid: raise TecplotSystemError() return annotation.Square(gid, self)
[docs] @tecutil.lock() def add_polyline(self, *points, **kwargs): """Create a polyline annotation on this `Frame <tecplot.layout.Frame>`. Parameters: *points (`lists <list>` of points): Arrays of :math:`(x, y)` or :math:`(x, y, z)` positions of the points along this polyline in the coordinate system specified by **coord_sys**. If multiple lists are provided, they must be of the same dimension (2D or 3D), though they may be of different lengths. Keyword Parameters: coord_sys (`CoordSys`, optional): The coordinate system to use for the positions of this annotation. Only 2D polylines may use a coordinate system other than the data-coordinates ("grid"). (default: `CoordSys.Grid`) Returns: One of `annotation.Polyline2D`, `annotation.Polyline3D`, `annotation.MultiPolyline2D` or `annotation.MultiPolyline3D`. Example usage: .. code-block:: python import tecplot points = [ [1, 2, 3], [2, 4, 9], [3, 8, 27], ] frame = tecplot.active_frame() polyline = frame.add_polyline(points) .. seealso:: `add_circle`, `add_ellipse`, `add_rectangle`, `add_square` """ coord_sys = kwargs.get('coord_sys', constant.CoordSys.Grid) coord_sys = constant.CoordSys(coord_sys) num_lines = len(points) dims = [len(l[0]) for l in points] if any(d != dims[0] for d in dims): msg = 'dimensions of all polylines must be the same' raise TecplotLogicError(msg) lengths = (ctypes.c_int64 * num_lines)(*[len(l) for l in points]) if dims[0] == 3: if __debug__: if coord_sys not in [constant.CoordSys.Grid, constant.CoordSys.Grid3D]: msg = '3D polylines must be in "Grid" coordinates' raise TecplotLogicError(msg) gid = _tecutil.Geom3DMPolyCreate(num_lines, lengths) if not gid: raise TecplotSystemError() polyline = annotation.MultiPolyline3D(gid, self) else: gid = _tecutil.Geom2DMPolyCreate(coord_sys.value, num_lines, lengths) if not gid: raise TecplotSystemError() polyline = annotation.MultiPolyline2D(gid, self) for l, p in zip(polyline, points): l[:] = p if num_lines == 1: if dims[0] == 3: polyline = annotation.Polyline3D(0, polyline) else: polyline = annotation.Polyline2D(0, polyline) return polyline
[docs] @tecutil.lock() def add_text(self, text, position=None, coord_sys=None, text_type=None, typeface=None, bold=None, italic=None, size_units=None, size=None, color=None, angle=None, line_spacing=None, anchor=None, box_type=None, line_thickness=None, box_color=None, fill_color=None, margin=None, zone=None): """Adds a `text <annotation.Text>` to a `Frame <tecplot.layout.Frame>`. Parameters: text (`str`): The text to add to the `Frame <tecplot.layout.Frame>`. The text string must have a non-zero length. position (`tuple` of `floats <float>` (x,y), optional): The position of the anchor as a percentage of the specified coordinates. (default: (0,0)) coord_sys (`CoordSys`, optional): Coordinate system used to position the anchor of the text object. The possible values are: `CoordSys.Grid` or `CoordSys.Frame`. (default: `CoordSys.Frame`) text_type (`TextType`, optional): Type of text object to create. Options are `TextType.Regular` (default) and `TextType.LaTeX`. If set to `TextType.LaTeX`, most style-related options will be saved but ignored when the text is rendered. These options will only be used if the type of the text object is later changed to `TextType.Regular`. typeface (`str`, optional): The typeface name. For consistency across various platforms, Tecplot guarantees that the following standard typeface names are available: "Helvetica", "Times", "Courier", "Greek", "Math", and "User Defined". Other typefaces may or may not be available depending on the TrueType fonts available. If the typeface name or style is not available, a suitable replacement will be selected. (default: "Helvetica") bold (`bool`, optional): Use the bold variation of the specified typeface. (default: `True`) italic (`bool`, optional): Use the italic variation of the specified typeface. (default: `False`) size_units (`Units`, optional): Text sizing units. Possible values are: `Units.Grid`, `Units.Frame` or `Units.Point`. (default: `Units.Point`) size (`float`, optional): Text height in the specified units. (default: 14) color (`Color`, optional): Color of the text (default: `Color.Black`) angle (`float`, optional): Angle of the text baseline in degrees from -360 to 360. (default: 0) line_spacing (`float`, optional): Line spacing in units of line size. Can take values from 0 to 50. (default: 1) anchor (`TextAnchor`, optional): Anchor position with respect to the text box. Possible values are: `TextAnchor.Left`, `TextAnchor.Center`, `TextAnchor.Right`, `TextAnchor.MidLeft`, `TextAnchor.MidCenter`, `TextAnchor.MidRight`, `TextAnchor.HeadLeft`, `TextAnchor.HeadCenter`, `TextAnchor.HeadRight`, `TextAnchor.OnSide` (default: `TextAnchor.Left`) box_type (`constant.TextBox`, optional): Type of text box can be one of: `constant.TextBox.None_`, `constant.TextBox.Filled` or `constant.TextBox.Hollow`. (default: `constant.TextBox.None_`) line_thickness (`float`, optional): Text box boarder line thickness may be a value in the range from 0.0001 to 100. (default: 0.1) box_color (`Color`, optional): Text box border line color. See `Color` for possible values. (default: `Color.Black`) fill_color (`Color`, optional): Text box fill color. See `Color` for possible values. (default: `White`) margin (`float`, optional): Margin between the text and text box. May be in the range from 0 to 2000. (default: 20) zone (`Zone <data_access>`, optional): `Zone <data_access>` or `XYLinemap` to which the text will be attached. (default: None) Returns: `annotation.Text`: The resulting `text box <annotation.Text>` object. Example: .. code-block:: python :emphasize-lines: 5-6 import tecplot from tecplot.constant import Color frame = tecplot.active_frame() frame.add_text('Hello, World!', position=(35, 50), bold=True, italic=False, color=Color.Blue) .. seealso:: `add_latex`, `delete_text` """ if __debug__: tecutil.check_arglist_argtypes( 'frame.add_text', ([tuple], [position], ['position']), ([CoordSys], [coord_sys], ['coord_sys']), ([str], [typeface, text], ['typeface', 'text']), ([bool], [bold, italic], ['bold', 'italic']), ([Units], [size_units], ['size_units']), ([numbers.Number], [size, angle, line_thickness, margin, line_spacing], ['size', 'angle', 'line_thickness', 'margin', 'line_spacing']), ([Color], [color, box_color, fill_color], ['color', 'text_color', 'fill_color']), ([TextAnchor], [anchor], ['anchor']), ([TextBox], [box_type], ['box_type']), ) with tecutil.ArgList() as arglist: if zone is not None: arglist[sv.ATTACHTOZONE] = True arglist[sv.ZONE] = zone.index + 1 def optional(type_, value): return type_(value) if value is not None else None # Note that TecUtil calls SV_TEXTCOLOR the color of the text, # and SV_COLOR as the text *box* color. These names correspond # to the 'color' and 'box_color' parameters. arglist.update(( (sv.TEXT, text), (sv.POSITIONCOORDSYS, coord_sys), (sv.ISBOLD, bold), (sv.ISITALIC, italic), (sv.SIZEUNITS, size_units), (sv.ANCHOR, anchor), (sv.COLOR, box_color), (sv.TEXTCOLOR, color), (sv.FILLCOLOR, fill_color), (sv.BOXTYPE, box_type))) arglist[sv.TEXTTYPE] = optional(TextType, text_type) if position is not None: arglist[sv.XPOS] = optional(float, position[0]) arglist[sv.YPOS] = optional(float, position[1]) arglist[sv.HEIGHT] = optional(float, size) arglist[sv.ANGLE] = optional(float, angle) arglist[sv.LINETHICKNESS] = optional(float, line_thickness) arglist[sv.MARGIN] = optional(float, margin) arglist[sv.LINESPACING] = optional(float, line_spacing) tid = _tecutil.TextCreateX(arglist) if tid == 0: raise TecplotSystemError() return annotation.Text(tid, self)
[docs] def add_latex(self, text, position=None, coord_sys=None, size_units=None, size=None, anchor=None, zone=None): r"""Adds a `LaTeX text <annotation.Text>` annotation to the Frame. LaTeX is a computer language designed for typesetting. The most popular use of LaTeX is math and Greek fonts for technical purposes. See the User Manual for instruction on how to setup LaTeX for use with Tecplot 360. Once LaTeX is configured for the GUI version of Tecplot 360 no additional changes are needed for PyTecplot. Parameters: text (`str`): The text to add to the `Frame <tecplot.layout.Frame>`. The text string must have a non-zero length. position (`tuple` of `floats <float>` (x,y), optional): The position of the anchor as a percentage of the specified coordinates. (default: (0,0)) coord_sys (`CoordSys`, optional): Coordinate system used to position the anchor of the text object. The possible values are: `CoordSys.Grid` or `CoordSys.Frame`. (default: `CoordSys.Frame`) size_units (`Units`, optional): Text sizing units. Possible values are: `Units.Grid`, `Units.Frame` or `Units.Point`. (default: `Units.Point`) size (`float`, optional): Text height in the specified units. (default: 14) anchor (`TextAnchor`, optional): Anchor position with respect to the text box. Possible values are: `TextAnchor.Left`, `TextAnchor.Center`, `TextAnchor.Right`, `TextAnchor.MidLeft`, `TextAnchor.MidCenter`, `TextAnchor.MidRight`, `TextAnchor.HeadLeft`, `TextAnchor.HeadCenter`, `TextAnchor.HeadRight`, `TextAnchor.OnSide` (default: `TextAnchor.Left`) zone (`Zone <data_access>`, optional): `Zone <data_access>` or `XYLinemap` to which the text will be attached. (default: None) Returns: `annotation.Text`: The resulting `text box <annotation.Text>` object. Example: .. code-block:: python :emphasize-lines: 5-6 import tecplot as tp from tecplot.constant import TextAnchor frame = tp.active_frame() frame.add_latex(r'$$\zeta(s) = \sum_{n=1}^\infty\frac{1}{n^s}$$', (50,50), size=64, anchor=TextAnchor.Center) .. figure:: /_static/images/frame_add_latex.png :width: 300px :figwidth: 300px .. seealso:: `add_text`, `delete_text` """ return self.add_text(text, position=position, coord_sys=coord_sys, size_units=size_units, size=size, anchor=anchor, zone=zone, text_type=TextType.LaTeX)
[docs] @tecutil.lock() def create_dataset(self, name, var_names=None, reset_style=True): """Create an empty `Dataset <data.Dataset>`. This will create a new `Dataset <data.Dataset>` and replace the existing one, destroying all data associated with it. Parameters: name (`str`): Title of the new `Dataset <data.Dataset>`. This does not have to be unique. var_names (`list` of `strings <str>`, optional): `Variable` names. This only sets the names and not the data type or location. See `add_variable`. (default: `None`) reset_style (`bool`): Reset style of the active `Frame <tecplot.layout.Frame>` before loading the `Dataset <data.Dataset>`. (default: `True`) Returns: `Dataset <data.Dataset>`: The newly created `Dataset <data.Dataset>`. .. note:: Relationships between `Frame <tecplot.layout.Frame>` and `Dataset <data.Dataset>` A `Frame <tecplot.layout.Frame>` may only hold a single `Dataset <data.Dataset>`, though this `Dataset <data.Dataset>` may be shared between several `Frames <tecplot.layout.Frame>`. Therefore, this method will only *replace* the current dataset when **reset_style** is set to `True` and will fail otherwise. If this `Frame <tecplot.layout.Frame>` already has a `Dataset <data.Dataset>`, it may be more efficient to do one of the following instead: * Add a new (blank) frame with `Page.add_frame()` and create a `Dataset <data.Dataset>` for this. * Add new zones to the existing `Dataset <data.Dataset>` with `Dataset.add_ordered_zone()`, `Dataset.add_fe_zone()` or `Dataset.add_poly_zone()`. * Change the existing data `in-place <Array>`. .. note:: **Performance considerations for data manipulations.** When performing many data-manipulation operations including adding zones, adding variables, modifying field data or connectivity, and especially in connected mode, it is recommended to do this all with the `tecplot.session.suspend()`. This will prevent the Tecplot engine from trying to "keep up" with the changes. Tecplot will be notified of all changes made upon exit of this context. This may result in significant performance gains for long operations. """ with self.activated(): if var_names is not None: var_names = tecutil.StringList(*var_names) try: if not _tecutil.DataSetCreate(name, var_names, reset_style): raise TecplotSystemError() finally: if var_names is not None: var_names.dealloc() return self.dataset
@property def background_color(self): """`Color`: Color of the background.""" return self._get_style(constant.Color, sv.BACKGROUNDCOLOR) @background_color.setter def background_color(self, value): self._set_style(constant.Color(value), sv.BACKGROUNDCOLOR) @property def border_thickness(self): """`float`: The border thickness in units of `Frame.size_pos_units`.""" return self._get_style(float, sv.BORDERTHICKNESS) @border_thickness.setter def border_thickness(self, value): self._set_style(float(value), sv.BORDERTHICKNESS) @property def height(self): """`float`: The height in units of `Frame.size_pos_units`.""" return self._get_style(float, sv.HEIGHT) @height.setter def height(self, value): self._set_style(float(value), sv.HEIGHT) @property def show_border(self): """`bool`: Show or hide the `Frame <tecplot.layout.Frame>`'s border.""" return self._get_style(bool, sv.SHOWBORDER) @show_border.setter def show_border(self, value): self._set_style(bool(value), sv.SHOWBORDER) @property def show_header(self): """`bool`: Show or hide the `Frame <tecplot.layout.Frame>`'s header in the border.""" return self._get_style(bool, sv.SHOWHEADER) @show_header.setter def show_header(self, value): self._set_style(bool(value), sv.SHOWHEADER) @property def header_background_color(self): """`Color`: The header's background color.""" return self._get_style(constant.Color, sv.HEADERCOLOR) @header_background_color.setter def header_background_color(self, value): self._set_style(constant.Color(value), sv.HEADERCOLOR) @property def size_pos_units(self): """`FrameSizePosUnits`: The units used for size properties. Possible values: `Paper`, `Workspace <FrameSizePosUnits.Workspace>`. """ return self._get_style(constant.FrameSizePosUnits, sv.FRAMESIZEPOSUNITS) @size_pos_units.setter def size_pos_units(self, value): self._set_style(constant.FrameSizePosUnits(value), sv.FRAMESIZEPOSUNITS) @property def transparent(self): """`bool`: Use transparency within this `Frame <tecplot.layout.Frame>`.""" return self._get_style(bool, sv.ISTRANSPARENT) @transparent.setter def transparent(self, value): self._set_style(bool(value), sv.ISTRANSPARENT) @property def width(self): """`float`: The width in units of `Frame.size_pos_units`.""" return self._get_style(float, sv.WIDTH) @width.setter def width(self, value): self._set_style(float(value), sv.WIDTH)