pyffi.formats.egm

Last Built: Mar 06, 2020

Source code for pyffi.formats.egm

"""
:mod:`pyffi.formats.egm` --- EGM (.egm)
=======================================

An .egm file contains facial shape modifiers, that is, morphs that modify
static properties of the face, such as nose size, chin shape, and so on.

Implementation
--------------

.. autoclass:: EgmFormat
   :show-inheritance:
   :members:

Regression tests
----------------

Read a EGM file
^^^^^^^^^^^^^^^

>>> # check and read egm file
>>> from os.path import dirname
>>> dirpath = __file__
>>> for i in range(4): #recurse up to root repo dir
...     dirpath = dirname(dirpath)
>>> repo_root = dirpath
>>> format_root = os.path.join(repo_root, 'tests', 'formats', 'egm')
>>> file = os.path.join(format_root, 'mmouthxivilai.egm')
>>> stream = open(file, 'rb')
>>> data = EgmFormat.Data()
>>> data.inspect_quick(stream)
>>> data.version
2
>>> data.inspect(stream)
>>> data.header.num_vertices
89
>>> data.header.num_sym_morphs
50
>>> data.header.num_asym_morphs
30
>>> data.header.time_date_stamp
2001060901
>>> data.read(stream)
>>> data.sym_morphs[0].vertices[0].x
17249

Parse all EGM files in a directory tree
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

>>> for stream, data in EgmFormat.walkData(format_root):
...     try:
...         # the replace call makes the doctest also pass on windows
...         os_path = stream.name
...         split = (os_path.split(os.sep))[-4:]
...         rejoin = os.path.join(*split).replace(os.sep, "/")
...         print("reading %s" % rejoin)
...     except Exception:
...         print(
...             "Warning: read failed due corrupt file,"
...             " corrupt format description, or bug.") # doctest: +REPORT_NDIFF
reading tests/formats/egm/mmouthxivilai.egm

Create an EGM file from scratch and write to file
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

>>> data = EgmFormat.Data(num_vertices=10)
>>> data.header.num_vertices
10
>>> morph = data.add_sym_morph()
>>> len(morph.vertices)
10
>>> morph.scale = 0.4
>>> morph.vertices[0].z = 123
>>> morph.vertices[9].x = -30000
>>> morph = data.add_asym_morph()
>>> morph.scale = 2.3
>>> morph.vertices[3].z = -5
>>> morph.vertices[4].x = 99
>>> from tempfile import TemporaryFile
>>> stream = TemporaryFile()
>>> data.write(stream)
"""

# ***** BEGIN LICENSE BLOCK *****
#
# Copyright (c) 2007-2012, Python File Format Interface
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
#
#    * Redistributions of source code must retain the above copyright
#      notice, this list of conditions and the following disclaimer.
#
#    * Redistributions in binary form must reproduce the above
#      copyright notice, this list of conditions and the following
#      disclaimer in the documentation and/or other materials provided
#      with the distribution.
#
#    * Neither the name of the Python File Format Interface
#      project nor the names of its contributors may be used to endorse
#      or promote products derived from this software without specific
#      prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
#
# ***** END LICENSE BLOCK *****


import struct
import os
import re

import pyffi.object_models.xml
import pyffi.object_models.common
from pyffi.object_models.xml.basic import BasicBase
import pyffi.object_models
from pyffi.utils.graph import EdgeFilter

[docs]class EgmFormat(pyffi.object_models.xml.FileFormat): """This class implements the EGM format.""" xml_file_name = 'egm.xml' # where to look for egm.xml and in what order: # EGMXMLPATH env var, or EgmFormat module directory xml_file_path = [os.getenv('EGMXMLPATH'), os.path.dirname(__file__)] # file name regular expression match RE_FILENAME = re.compile(r'^.*\.egm$', re.IGNORECASE) # basic types int = pyffi.object_models.common.Int uint = pyffi.object_models.common.UInt byte = pyffi.object_models.common.Byte ubyte = pyffi.object_models.common.UByte char = pyffi.object_models.common.Char short = pyffi.object_models.common.Short ushort = pyffi.object_models.common.UShort float = pyffi.object_models.common.Float # implementation of egm-specific basic types
[docs] class FileSignature(BasicBase): """Basic type which implements the header of a EGM file.""" def __init__(self, **kwargs): BasicBase.__init__(self, **kwargs) def __str__(self): return 'FREGM'
[docs] def get_detail_display(self): return self.__str__()
[docs] def get_hash(self, data=None): """Return a hash value for this value. :return: An immutable object that can be used as a hash. """ return None
[docs] def read(self, stream, data): """Read header string from stream and check it. :param stream: The stream to read from. :type stream: file """ hdrstr = stream.read(5) # check if the string is correct if hdrstr != "FREGM".encode("ascii"): raise ValueError( "invalid EGM header: expected 'FREGM' but got '%s'" % hdrstr)
[docs] def write(self, stream, data): """Write the header string to stream. :param stream: The stream to write to. :type stream: file """ stream.write("FREGM".encode("ascii"))
[docs] def get_size(self, data=None): """Return number of bytes the header string occupies in a file. :return: Number of bytes. """ return 5
[docs] class FileVersion(BasicBase):
[docs] def get_value(self): raise NotImplementedError
[docs] def set_value(self, value): raise NotImplementedError
def __str__(self): return 'XXX'
[docs] def get_size(self, data=None): return 3
[docs] def get_hash(self, data=None): return None
[docs] def read(self, stream, data): ver = stream.read(3) if ver != ('%03i' % data.version).encode("ascii"): raise ValueError( "Invalid version number: expected b'%03i' but got %s." % (data.version, ver))
[docs] def write(self, stream, data): stream.write(('%03i' % data.version).encode("ascii"))
[docs] def get_detail_display(self): return 'XXX'
[docs] @staticmethod def version_number(version_str): """Converts version string into an integer. :param version_str: The version string. :type version_str: str :return: A version integer. >>> EgmFormat.version_number('002') 2 >>> EgmFormat.version_number('XXX') -1 """ try: # note: always '002' in all files seen so far return int(version_str) except ValueError: # not supported return -1
[docs] class Data(pyffi.object_models.FileFormat.Data): """A class to contain the actual egm data.""" def __init__(self, version=2, num_vertices=0): self.header = EgmFormat.Header() self.header.num_vertices = num_vertices self.sym_morphs = [] self.asym_morphs = [] self.version = version
[docs] def inspect_quick(self, stream): """Quickly checks if stream contains EGM data, and gets the version, by looking at the first 8 bytes. :param stream: The stream to inspect. :type stream: file """ pos = stream.tell() try: hdrstr = stream.read(5) if hdrstr != "FREGM".encode("ascii"): raise ValueError("Not an EGM file.") self.version = EgmFormat.version_number(stream.read(3)) finally: stream.seek(pos)
# overriding pyffi.object_models.FileFormat.Data methods
[docs] def inspect(self, stream): """Quickly checks if stream contains EGM data, and reads the header. :param stream: The stream to inspect. :type stream: file """ pos = stream.tell() try: self.inspect_quick(stream) self.header.read(stream, self) finally: stream.seek(pos)
[docs] def read(self, stream): """Read a egm file. :param stream: The stream from which to read. :type stream: ``file`` """ # read the file self.inspect_quick(stream) self.header.read(stream, self) self.sym_morphs = [ EgmFormat.MorphRecord(argument=self.header.num_vertices) for i in range(self.header.num_sym_morphs)] self.asym_morphs = [ EgmFormat.MorphRecord(argument=self.header.num_vertices) for i in range(self.header.num_asym_morphs)] for morph in self.sym_morphs + self.asym_morphs: morph.read(stream, self) # check if we are at the end of the file if stream.read(1): raise ValueError( 'end of file not reached: corrupt egm file?')
[docs] def write(self, stream): """Write a egm file. :param stream: The stream to which to write. :type stream: ``file`` """ # write the file self.header.num_sym_morphs = len(self.sym_morphs) self.header.num_asym_morphs = len(self.asym_morphs) self.header.write(stream, self) for morph in self.sym_morphs + self.asym_morphs: if morph.arg != self.header.num_vertices: raise ValueError("invalid morph length") morph.write(stream, self)
[docs] def add_sym_morph(self): """Add a symmetric morph, and return it.""" morph = EgmFormat.MorphRecord(argument=self.header.num_vertices) self.sym_morphs.append(morph) self.header.num_sym_morphs = len(self.sym_morphs) return morph
[docs] def add_asym_morph(self): """Add an asymmetric morph, and return it.""" morph = EgmFormat.MorphRecord(argument=self.header.num_vertices) self.asym_morphs.append(morph) self.header.num_asym_morphs = len(self.asym_morphs) return morph
[docs] def apply_scale(self, scale): """Apply scale factor to all morphs.""" for morph in self.sym_morphs + self.asym_morphs: morph.apply_scale(scale)
# DetailNode
[docs] def get_detail_child_nodes(self, edge_filter=EdgeFilter()): return self.header.get_detail_child_nodes(edge_filter=edge_filter)
[docs] def get_detail_child_names(self, edge_filter=EdgeFilter()): return self.header.get_detail_child_names(edge_filter=edge_filter)
# GlobalNode
[docs] def get_global_child_nodes(self, edge_filter=EdgeFilter()): for morph in self.sym_morphs: yield morph for morph in self.asym_morphs: yield morph
def get_global_child_names(self, edge_filter=EdgeFilter()): for morph in self.sym_morphs: yield "Sym Morph" for morph in self.asym_morphs: yield "Asym Morph"
[docs] class MorphRecord: """ >>> # create morph with 3 vertices. >>> morph = EgmFormat.MorphRecord(argument=3) >>> morph.set_relative_vertices( ... [(3, 5, 2), (1, 3, 2), (-9, 3, -1)]) >>> # scale should be 9/32768.0 = 0.0002746... >>> morph.scale # doctest: +ELLIPSIS 0.0002746... >>> for vert in morph.get_relative_vertices(): ... print([int(1000 * x + 0.5) for x in vert]) [3000, 5000, 2000] [1000, 3000, 2000] [-8999, 3000, -999] """ def get_relative_vertices(self): for vert in self.vertices: yield (vert.x * self.scale, vert.y * self.scale, vert.z * self.scale) def set_relative_vertices(self, vertices): # copy to list vertices = list(vertices) # check length if len(vertices) != self.arg: raise ValueError("expected %i vertices, but got %i" % (self.arg, len(vertices))) # get extreme values of morph max_value = max(max(abs(value) for value in vert) for vert in vertices) # calculate scale self.scale = max_value / 32767.0 inv_scale = 1 / self.scale # set vertices for vert, self_vert in zip(vertices, self.vertices): self_vert.x = int(vert[0] * inv_scale) self_vert.y = int(vert[1] * inv_scale) self_vert.z = int(vert[2] * inv_scale)
[docs] def apply_scale(self, scale): """Apply scale factor to data. >>> # create morph with 3 vertices. >>> morph = EgmFormat.MorphRecord(argument=3) >>> morph.set_relative_vertices( ... [(3, 5, 2), (1, 3, 2), (-9, 3, -1)]) >>> morph.apply_scale(2) >>> for vert in morph.get_relative_vertices(): ... print([int(1000 * x + 0.5) for x in vert]) [6000, 10000, 4000] [2000, 6000, 4000] [-17999, 6000, -1999] """ self.scale *= scale