#!/usr/bin/env python
# -*- coding: utf-8 -*-
#
# layer.py
#
# Copyright 2016 notna <notna@apparat.org>
#
# 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 2 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, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
# MA 02110-1301, USA.
#
#
__all__ = ["Layer", "Layer2D", "Layer3D", "LayerGroup", "LayerWorld"]
import warnings
try:
import pyglet
except ImportError:
pass
from typing import TYPE_CHECKING, Any
if TYPE_CHECKING:
import peng3d
import peng3d.menu, peng3d.window
from . import camera
from .util.types import *
[docs]class Layer(object):
"""
Base layer class.
A Layer can be used to separate background from foreground or the 3d world from a 2d HUD.
This class by itself does nothing, you will need to subclass it and override the :py:meth:`draw()` method.
"""
def __init__(
self,
menu: "peng3d.menu.BasicMenu",
window: Any = None,
peng: Any = None,
):
if window is not None:
warnings.warn(
"Passing window to a layer is no longer necessary; the window parameter will be removed in peng3d 2.0",
DeprecationWarning,
3,
)
if peng is not None:
warnings.warn(
"Passing peng to a layer is no longer necessary; the peng parameter will be removed in peng3d 2.0",
DeprecationWarning,
3,
)
self.window: "peng3d.window.PengWindow" = menu.window
self.menu: "peng3d.menu.BasicMenu" = menu
self.peng: "peng3d.peng.Peng" = menu.peng
self.enabled: bool = True
self.should_redraw: bool = True
# Subclass overrides
[docs] def draw(self) -> None:
"""
Called when this layer needs to be drawn.
Override this method in subclasses to redefine behavior.
"""
pass
[docs] def predraw(self) -> None:
"""
Called before the :py:meth:`draw()` method is called.
This method is used in the :py:class:`Layer2D()` and :py:class:`Layer3D()` subclasses for setting OpenGL state.
Override this method in subclasses to redefine behavior.
"""
pass
[docs] def postdraw(self) -> None:
"""
Called after the :py:meth:`draw()` method is called.
This method can be used to reset OpenGL state to avoid conflicts with other code.
Override this method in subclasses to redefine behavior.
"""
pass
[docs] def on_redraw(self) -> None:
"""
Called whenever the Layer should redraw itself.
Note that this method should not be called manually, instead call :py:meth:`redraw()`\\ .
:return: None
"""
pass
# End subclass overrides
[docs] def redraw(self) -> None:
"""
Call this to redraw the layer.
Note that the redraw will not happen immediately, rather on the next frame that this
layer is rendered. This massively improves performance.
:return:
"""
self.should_redraw = True
# Event handlers
def _draw(self):
if not self.enabled:
return
if self.should_redraw:
self.on_redraw()
self.should_redraw = False
self.predraw()
try:
self.draw()
except Exception:
raise
finally:
self.postdraw()
[docs]class Layer2D(Layer):
"""
2D Variant of :py:class:`Layer()` and a subclass of the former.
This class makes use of the :py:meth:`predraw()` method to configure OpenGL to draw 2-Dimensionally.
This class uses :py:meth:`PengWindow.set2d()` to set the 2D mode.
When overriding the :py:meth:`predraw()` method, make sure to call the superclass.
"""
[docs] def predraw(self):
"""
Uses :py:meth:`PengWindow.set2d()` to enable a 2D OpenGL state.
"""
self.window.set2d()
[docs]class Layer3D(Layer):
"""
3D Variant of :py:class:`Layer()` and a subclass of the former.
This class works the same as :py:class:`Layer2D()`\\ , only for 3D drawing instead.
This class uses :py:meth:`PengWindow.set3d()` to set the 3D mode.
Also, the correct :py:func:`glTranslatef()` and :py:func:`glRotatef()` are applied to simplify drawing objects.
When writing the :py:meth:`draw()` method of this class, you will only need to use world coordinates, not camera coordinates.
This allows for easy building of Games using First-Person-Perspectives.
"""
def __init__(
self,
menu: "peng3d.menu.BasicMenu",
window: Any = None,
peng: Any = None,
cam: camera.Camera = None,
):
super(Layer3D, self).__init__(menu, window, peng)
if not isinstance(cam, camera.Camera):
raise TypeError("cam must be of type Camera!")
self.cam = cam
[docs] def predraw(self):
"""
Uses :py:meth:`PengWindow.set3d()` to enable a 3D OpenGL state.
"""
self.window.set3d(self.cam)
[docs]class LayerGroup(Layer):
"""
Layer variant wrapping the supplied pyglet group.
``group`` may only be an instance of :py:class:`pyglet.graphics.Group`\\ , else a :py:exc:`TypeError` will be raised.
Also note that both the :py:meth:`predraw() <Layer.predraw()>` and :py:meth:`postdraw() <Layer.postdraw()>` methods are overwritten by this class.
.. seealso::
For more information about pyglet groups, see `the pyglet docs <http://pyglet.readthedocs.io/en/latest/programming_guide/graphics.html#setting-the-opengl-state>`_\\ .
"""
def __init__(
self,
menu: "peng3d.menu.BasicMenu",
window: Any = None,
peng: Any = None,
group: pyglet.graphics.Group = None,
):
super(LayerGroup, self).__init__(menu, window, peng)
if not isinstance(group, pyglet.graphics.Group):
raise TypeError("group must be an instance of pyglet.graphics.Group")
self.group = group
[docs] def predraw(self):
"""
Sets the group state.
"""
self.group.set_state()
[docs] def postdraw(self):
"""
Re-sets the previous state.
"""
self.group.unset_state()
[docs]class LayerWorld(Layer3D):
"""
Subclass of :py:class:`Layer3D()` implementing a 3D Layer showing a specific :py:class:`WorldView`\\ .
All arguments passed to the constructor should be self-explanatory.
Note that you may not set any of the attributes directly, or crashes and bugs may appear indirectly within a certain during future re-drawing of the screen.
"""
def __init__(
self,
menu: "peng3d.menu.BasicMenu",
window: Any = None,
peng: Any = None,
world=None,
viewname: str = None,
):
super(LayerWorld, self).__init__(
menu, window, peng, world.getView(viewname).cam
)
self.world = world
self.viewname = viewname
self.view = self.world.getView(self.viewname)
[docs] def setView(self, name: str) -> None:
"""
Sets the view used to the specified ``name``\\ .
The name must be known to the world or else a :py:exc:`ValueError` is raised.
"""
if name not in self.world.views:
raise ValueError("Invalid viewname for world!")
self.viewname = name
self.view = self.world.getView(self.viewname)
[docs] def predraw(self):
"""
Sets up the attributes used by :py:class:`Layer3D()` and calls :py:meth:`Layer3D.predraw()`\\ .
"""
self.cam = self.view.cam
super(LayerWorld, self).predraw()
[docs] def draw(self):
"""
Draws the view using the :py:meth:`World.render3d()` method.
"""
self.world.render3d(self.view)