Source code for opengl.decors

#
##
##  SPDX-FileCopyrightText: © 2007-2023 Benedict Verhegghe <bverheg@gmail.com>
##  SPDX-License-Identifier: GPL-3.0-or-later
##
##  This file is part of pyFormex 3.4  (Thu Nov 16 18:07:39 CET 2023)
##  pyFormex is a tool for generating, manipulating and transforming 3D
##  geometrical models by sequences of mathematical operations.
##  Home page: https://pyformex.org
##  Project page: https://savannah.nongnu.org/projects/pyformex/
##  Development: https://gitlab.com/bverheg/pyformex
##  Distributed under the GNU General Public License version 3 or later.
##
##  This program is free software: you can redistribute it and/or modify
##  it under the terms of the GNU General Public License as published by
##  the Free Software Foundation, either version 3 of the License, or
##  (at your option) any later version.
##
##  This program is distributed in the hope that it will be useful,
##  but WITHOUT ANY WARRANTY; without even the implied warranty of
##  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
##  GNU General Public License for more details.
##
##  You should have received a copy of the GNU General Public License
##  along with this program.  If not, see http://www.gnu.org/licenses/.
##
"""Decorations for the OpenGL canvas.

This module contains a collection of predefined decorations that
can be useful additions to a geometry scene rendering.
"""
import numpy as np

import pyformex as pf
from pyformex import simple
from pyformex.formex import Formex
from pyformex import colors
from .drawable import Actor
from .textext import FontTexture
from .sanitize import saneLineWidth


__all__ = ['BboxActor', 'Rectangle', 'Line', 'Lines', 'Grid2D', 'ColorLegend',
            'Grid']

### Decorations ###############################################


[docs]class BboxActor(Actor): """Create a bounding box actor. A bounding box is a hexaeder in global axes. The hexaeder is drawn in wireframe mode with default color black. Parameters ---------- bbox: :term:`coords_like` A Coords with two points: the minimal and maximal coordinates of the bounding box to be drawn. **kargs: keyword parameters Keyword parameters like in the :func:`draw` funnction. The `mode`, `lighting`, `opak` parameters are fixed and can not be used. Returns ------- :class:`Actor` An Actor representing a bounding box. """ def __init__(self, bbox, **kargs): F = simple.cuboid(*bbox) Actor.__init__(self, F, mode='wireframe', lighting=False, opak=True, **kargs)
[docs]class Rectangle(Actor): """A 2D-rectangle on the canvas.""" def __init__(self, x1, y1, x2, y2, **kargs): F = Formex([[[x1, y1], [x2, y1], [x2, y2], [x1, y2]]]) Actor.__init__(self, F, rendertype=2, **kargs)
[docs]class Line(Actor): """A 2D-line on the canvas. Parameters: - `x1, y1, x2, y2`: floats: the viewport coordinates of the endpoints of the line - `kargs`: keyword arguments to be passed to the :class:`Actor`. """ def __init__(self, x1, y1, x2, y2, **kargs): F = Formex([[[x1, y1], [x2, y2]]]) Actor.__init__(self, F, rendertype=2, **kargs)
[docs]class Lines(Actor): """A collection of straight lines on the canvas. Parameters: - `data`: data that can initialize a 2-plex Formex: the viewport coordinates of the 2 endpoints of the n lines. The third coordinate is ignored. - `kargs`: keyword arguments to be passed to the :class:`Actor`. """ def __init__(self, data, color=None, linewidth=None, **kargs): """Initialize a Lines.""" F = Formex(data) Actor.__init__(self, F, rendertype=2, **kargs)
[docs]class Grid2D(Actor): """A 2D-grid on the canvas.""" def __init__(self, x1, y1, x2, y2, nx=1, ny=1, lighting=False, rendertype=2, **kargs): F = Formex([[[x1, y1], [x1, y2]]]).replic(nx+1, step=float(x2-x1)/nx, dir=0) + \ Formex([[[x1, y1], [x2, y1]]]).replic(ny+1, step=float(y2-y1)/ny, dir=1) Actor.__init__(self, F, rendertype=rendertype, lighting=lighting, **kargs)
[docs]class ColorLegend(Actor): """A labeled colorscale legend. When showing the distribution of some variable over a domain by means of a color encoding, the viewer expects some labeled colorscale as a guide to decode the colors. The ColorLegend decoration provides such a color legend. This class only provides the visual details of the scale. The conversion of the numerical values to the matching colors is provided by the :class:`~gui.colorscale.ColorScale` class. Parameters: - `colorscale`: a :class:`~gui.colorscale.ColorScale` instance providing conversion between numerical values and colors - `ncolors`: int: the number of different colors to use. - `x,y,w,h`: four integers specifying the position and size of the color bar rectangle - `ngrid`: int: number of intervals for the grid lines to be shown. If > 0, grid lines are drawn around the color bar and between the ``ngrid`` intervals. If = 0, no grid lines are drawn. If < 0 (default), the value is set equal to the number of colors or to 0 if this number is higher than 50. - `linewidth`: float: width of the grid lines. If not specified, the current canvas line width is used. - `nlabel`: int: number of intervals for the labels to be shown. If > 0, labels will be displayed at `nlabel` interval borders, if possible. The number of labels displayed thus will be ``nlabel+1``, or less if the labels would otherwise be too close or overlapping. If 0, no labels are shown. If < 0 (default), a default number of labels is shown. - `size`: font size to be used for the labels - `font`: font to be used for the labels. It can be a :class:`~opengl.textext.FontTexture` or a string with the path to a monospace .ttf font. If unspecified, the default font is used. - `dec`: int: number of decimals to be used in the labels - `scale`: int: exponent of 10 for the scaling factor of the label values. The displayed values will be equal to the real values multiplied with ``10**scale``. - `lefttext`: bool: if True, the labels will be drawn to the left of the color bar. The default is to draw the labels at the right. Some practical guidelines: - Large numbers of colors result in a quasi continuous color scheme. - With a high number of colors, grid lines disturb the image, so either use ``ngrid=0`` or ``ngrid=`` to only draw a border around the colors. - With a small number of colors, set ``ngrid = len(colorlegend.colors)`` to add gridlines between each color. Without it, the individual colors in the color bar may seem to be not constant, due to an optical illusion. Adding the grid lines reduces this illusion. - When using both grid lines and labels, set both ``ngrid`` and ``nlabel`` to the same number or make one a multiple of the other. Not doing so may result in a very confusing picture. - The best practice is to either use a low number of colors (<=20) and the default ``ngrid`` and ``nlabel``, or to use a high number of colors (>=200) and the default values or a low value for ``nlabel``. The `ColorScale` example script provides opportunity to experiment with different settings. """ def __init__(self, colorscale, ncolors, x, y, w, h, ngrid=0, linewidth=None, nlabel=-1, size=18, font=None, textcolor=None, dec=2, scale=0, lefttext=False, **kargs): """Initialize the ColorLegend.""" from pyformex.gui import colorscale as cs self.cl = cs.ColorLegend(colorscale, ncolors) self.x = float(x) self.y = float(y) self.w = float(w) self.h = float(h) self.ngrid = int(ngrid) if self.ngrid < 0: self.ngrid = ncolors if self.ngrid > 50: self.ngrid = 0 self.linewidth = saneLineWidth(linewidth) self.nlabel = int(nlabel) if self.nlabel < 0: if self.ngrid > 0: self.nlabel = self.ngrid else: self.nlabel = ncolors self.dec = dec # number of decimals self.scale = 10 ** scale # scale all numbers with 10**scale self.lefttext = lefttext self.xgap = 4 # hor. gap between color bar and labels self.ygap = 4 # (min) vert. gap between labels F = simple.rectangle(1, ncolors, self.w, self.h).trl([self.x, self.y, 0.]) Actor.__init__(self, F, rendertype=2, lighting=False, color=self.cl.colors, **kargs) if self.ngrid > 0: F = simple.rectangle(1, self.ngrid, self.w, self.h).trl([self.x, self.y, 0.]) G = Actor(F, rendertype=2, lighting=False, color=colors.black, linewidth=self.linewidth, mode='wireframe') self.children.append(G) # labels # print("LABELS: %s" % self.nlabel) if self.nlabel > 0: from .import textext if font is None: font = FontTexture.default() elif isinstance(font, str): font = FontTexture(font, size) self.font = font fh = self.font.height pf.debug("FONT HEIGHT %s" % fh, pf.DEBUG.DRAW) if textcolor is None: textcolor = pf.canvas.settings['fgcolor'] if self.lefttext: x = F.coords[0, 0, 0] - self.xgap gravity = 'W' else: x = F.coords[0, 1, 0] + self.xgap gravity = 'E' # minimum label distance dh = fh + self.ygap da = self.h / self.nlabel # Check if labels will fit if da < dh and self.nlabel == ncolors: # reduce number of labels self.nlabel = int(self.h/dh) da = self.h / self.nlabel vals = cs.ColorLegend(colorscale, self.nlabel).limits #print("VALS",vals) ypos = self.y + da * np.arange(self.nlabel+1) yok = self.y - 0.01*dh for (y, v) in zip(ypos, vals): #print(y,v,yok) if y >= yok: t = textext.Text(("%%.%df" % self.dec) % (v*self.scale), (x, y), size=size, gravity=gravity, color=textcolor) self.children.append(t) yok = y + dh
[docs]def Grid(nx=(1, 1, 1), ox=(0.0, 0.0, 0.0), dx=(1.0, 1.0, 1.0), planes='b', lines='b', planecolor=colors.white, linecolor=colors.black, alpha=0.3, name='_grid_', **kargs): """Creates a (set of) grid(s) in (some of) the coordinate planes. Parameters ---------- nx: tuple of int A tuple of three ints, specifying the number of divisions of the grid in the three coordinate directions. A zero value may be specified to avoid the grid to extend in that direction. Thus, setting the last value to zero will result in a planar grid in the xy-plane. ox: tuple of float The coordinates of the origin of the grid. dx: tuple of float The step size in each coordinate direction. plane: str One of 'first', 'box', 'all', 'no'. The string can be shortened to the first character. Specifies how many planes to draw in each direction: 'first' only draws the first, 'box' draws the first and the last, 'all' draws all planes, 'no' draws no planes. lines: str One of 'first', 'box', 'all', 'no'. The string can be shortened to the first character. Specifies how many lines to draw in each direction: 'first' only draws the first, 'box' draws the first and the last, 'all' draws all lines, 'no' draws no lines. plane_color: :term:`color_like` The color to use for drawing the planes line_color: :term:`color_like` The color to use for drawing the lines alpha: float Alpha transparency value (0.0 <= alpha <= 1.0) name: str Base name for the attributes set on the items in the returned List. Returns ------- :class:`List` A List with up to two Meshes: the planes and the lines. The Meshes come with the following :class:`~attributes.Attributes` set: - name: the provided name with '_planes_' or '_lines_' appended. - color: the provided plane or line color - alpha: the provided alpha value for the planes, 0.6 for the lines. - mode: 'flat' - lighting: False The user can of course change these before drawing the List. """ from pyformex import simple from pyformex.olist import List from pyformex.mesh import Mesh G = List() planes = planes[:1].lower() P = [] L = [] for i in range(3): n0, n1, n2 = np.roll(nx, i) d0, d1, d2 = np.roll(dx, i) if n0*n1: if planes != 'n': M = simple.rectangle(b=n0*d0, h=n1*d1) if n2: if planes == 'b': M = M.replic(2, dir=2, step=n2*d2) elif planes == 'a': M = M.replic(n2+1, dir=2, step=d2) P.append(M.rollAxes(-i).toMesh()) if lines != 'n': M = Formex('l:1').scale(n0*d0).replic(n1+1, dir=1, step=d1) + \ Formex('l:2').scale(n1*d1).replic(n0+1, dir=0, step=d0) if n2: if lines == 'b': M = M.replic(2, dir=2, step=n2*d2) elif lines == 'a': M = M.replic(n2+1, dir=2, step=d2) L.append(M.rollAxes(-i).toMesh()) if P: M = Mesh.concatenate(P) M.attrib(name=f'{name}_planes_', mode='flat', lighting=False, color=planecolor, alpha=alpha, **kargs) G.append(M) if L: M = Mesh.concatenate(L) M.attrib(name=f'{name}_lines_', mode='flat', lighting=False, color=linecolor, alpha=0.6, **kargs) G.append(M) return G.trl(ox)
# End