Source code for sierra.plugins.engine.argos.variables.cameras

# Copyright 2020 John Harwell, All rights reserved.
#
#  SPDX-License-Identifier: MIT

"""Classes for specifying ARGoS cameras.

Positions, timeline, and interpolation, for manipulating the frame
capture/rendering perspective.

"""

# Core packages
import typing as tp
import math

# 3rd party packages

# Project packages
from sierra.core.variables.base_variable import IBaseVariable
from sierra.core.utils import ArenaExtent
from sierra.core.experiment import definition
from sierra.core import types
from sierra.core.vector import Vector3D
import sierra.plugins.engine.argos.variables.exp_setup as exp


[docs] class QTCameraTimeline(IBaseVariable): """Defines when/how to switch between camera perspectives within ARGoS. Attributes: interpolate: Should we interpolate between camera positions on our timeline ? setup: Simulation experiment definitions. extents: List of (X,Y,Zs) tuple of dimensions of arena areas to generate camera definitions for. """ # If this default changes in ARGoS, it will need to be updated here too. N_CAMERAS = 12 def __init__( self, setup: exp.ExpSetup, cmdline: str, extents: list[ArenaExtent] ) -> None: self.cmdline = cmdline self.extents = extents self.setup = setup self.element_adds = [] # type: tp.List[definition.ElementAddList]
[docs] def gen_attr_changelist(self) -> list[definition.AttrChangeSet]: """ No effect. All tags/attributes are either deleted or added. """ return []
[docs] def gen_tag_rmlist(self) -> list[definition.ElementRmList]: """Remove the ``<camera>`` tag if it exists. Obviously you *must* call this function BEFORE adding new definitions. """ return [ definition.ElementRmList( definition.ElementRm("./visualization/qt-opengl", "camera") ) ]
[docs] def gen_element_addlist(self) -> list[definition.ElementAddList]: if not self.element_adds: adds = definition.ElementAddList( definition.ElementAdd("./visualization/qt-opengl", "camera", {}, False), definition.ElementAdd( "./visualization/qt-opengl/camera", "placements", {}, False ), ) in_ticks = self.setup.n_secs_per_run * self.setup.n_ticks_per_sec adds.append( definition.ElementAdd( ".//qt-opengl/camera", "timeline", {"loop": str(in_ticks)}, False ) ) for ext in self.extents: # generate keyframes for switching between camera perspectives self._gen_keyframes(adds, self.N_CAMERAS, in_ticks) info = [ self._gen_camera_config(ext, c, self.N_CAMERAS) for c in range(0, self.N_CAMERAS) ] for index, up, look_at, pos in info: camera = definition.ElementAdd( ".//camera/placements", "placement", { "index": f"{index}", "up": f"{up.x},{up.y},{up.z}", "position": f"{pos.x},{pos.y},{pos.z}", "look_at": f"{look_at.x},{look_at.y},{look_at.z}", }, True, ) adds.append(camera) self.element_adds = [adds] return self.element_adds
[docs] def gen_files(self) -> None: pass
def _gen_keyframes( self, adds: definition.ElementAddList, n_cameras: int, cycle_length: int ) -> None: for c in range(0, n_cameras): index = c % n_cameras adds.append( definition.ElementAdd( ".//qt-opengl/camera/timeline", "keyframe", { "placement": str(index), "step": str(int(cycle_length / n_cameras * c)), }, True, ) ) if "interp" in self.cmdline and c < n_cameras: adds.append( definition.ElementAdd( ".//qt-opengl/camera/timeline", "interpolate", {}, True ) ) def _gen_camera_config(self, ext: ArenaExtent, index: int, n_cameras) -> tuple: angle = (index % n_cameras) * (2.0 * math.pi / n_cameras) look_at = Vector3D(ext.xsize() / 2.0, ext.ysize() / 2.0, 0.0) hyp = math.sqrt(2 * max(look_at.x, look_at.y) ** 2) pos_x = hyp * math.cos(angle) + look_at.x pos_y = hyp * math.sin(angle) + look_at.y pos_z = max(ext.xsize(), ext.ysize()) * 0.50 pos = Vector3D(pos_x, pos_y, pos_z) # This is what the ARGoS source does for the up vector for the default # camera configuration up = Vector3D(0, 0, 1) return index, up, look_at, pos
[docs] class QTCameraOverhead(IBaseVariable): """Defines a single overhead camera perspective within ARGoS. Attributes: extents: List of (X,Y,Z) tuple of dimensions of arena areas to generate camera definitions for. """ def __init__(self, extents: list[ArenaExtent]) -> None: self.extents = extents self.element_adds = [] # type: tp.List[definition.ElementAddList]
[docs] def gen_attr_changelist(self) -> list[definition.AttrChangeSet]: """No effect. All tags/attributes are either deleted or added. """ return []
[docs] def gen_tag_rmlist(self) -> list[definition.ElementRmList]: """Remove the ``<camera>`` tag if it exists. Obviously you *must* call this function BEFORE adding new definitions. """ return [ definition.ElementRmList( definition.ElementRm("./visualization/qt-opengl", "camera") ) ]
[docs] def gen_element_addlist(self) -> list[definition.ElementAddList]: if not self.element_adds: adds = definition.ElementAddList( definition.ElementAdd("./visualization/qt-opengl", "camera", {}, False), definition.ElementAdd( "./visualization/qt-opengl/camera", "placements", {}, False ), ) for ext in self.extents: height = max(ext.xsize(), ext.ysize()) * 0.75 camera = definition.ElementAdd( ".//camera/placements", "placement", { "index": "0", "position": "{}, {}, {}".format( ext.xsize() / 2.0, ext.ysize() / 2.0, height ), "look_at": "{}, {}, 0".format( ext.xsize() / 2.0, ext.ysize() / 2.0 ), }, True, ) adds.append(camera) self.element_adds = [adds] return self.element_adds
[docs] def gen_files(self) -> None: pass
def factory(cmdopts: types.Cmdopts, extents: list[ArenaExtent]): """Create cameras for a list of arena extents.""" if cmdopts["camera_config"] == "overhead": return QTCameraOverhead(extents) return QTCameraTimeline( exp.factory(cmdopts["exp_setup"]), cmdopts["camera_config"], extents ) __all__ = [ "QTCameraOverhead", "QTCameraTimeline", ]