Skip to content
Snippets Groups Projects
export_x3d.py 69.4 KiB
Newer Older
  • Learn to ignore specific revisions
  • # ##### BEGIN GPL LICENSE BLOCK #####
    #
    #  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.
    #
    # ##### END GPL LICENSE BLOCK #####
    
    # <pep8 compliant>
    
    # Contributors: bart:neeneenee*de, http://www.neeneenee.de/vrml, Campbell Barton
    
    """
    This script exports to X3D format.
    
    Usage:
    Run this script from "File->Export" menu.  A pop-up will ask whether you
    want to export only selected or all relevant objects.
    
    Known issues:
        Doesn't handle multiple materials (don't use material indices);<br>
        Doesn't handle multiple UV textures on a single mesh (create a mesh for each texture);<br>
        Can't get the texture array associated with material * not the UV ones;
    """
    
    import math
    import os
    
    import bpy
    import mathutils
    
    
    from bpy_extras.io_utils import create_derived_objects, free_derived_objects
    
    x3d_names_reserved = {'Anchor', 'Appearance', 'Arc2D', 'ArcClose2D', 'AudioClip', 'Background', 'Billboard',
                          'BooleanFilter', 'BooleanSequencer', 'BooleanToggle', 'BooleanTrigger', 'Box', 'Circle2D',
                          'Collision', 'Color', 'ColorInterpolator', 'ColorRGBA', 'component', 'Cone', 'connect',
                          'Contour2D', 'ContourPolyline2D', 'Coordinate', 'CoordinateDouble', 'CoordinateInterpolator',
                          'CoordinateInterpolator2D', 'Cylinder', 'CylinderSensor', 'DirectionalLight', 'Disk2D',
                          'ElevationGrid', 'EspduTransform', 'EXPORT', 'ExternProtoDeclare', 'Extrusion', 'field',
                          'fieldValue', 'FillProperties', 'Fog', 'FontStyle', 'GeoCoordinate', 'GeoElevationGrid',
                          'GeoLocationLocation', 'GeoLOD', 'GeoMetadata', 'GeoOrigin', 'GeoPositionInterpolator',
                          'GeoTouchSensor', 'GeoViewpoint', 'Group', 'HAnimDisplacer', 'HAnimHumanoid', 'HAnimJoint',
                          'HAnimSegment', 'HAnimSite', 'head', 'ImageTexture', 'IMPORT', 'IndexedFaceSet',
                          'IndexedLineSet', 'IndexedTriangleFanSet', 'IndexedTriangleSet', 'IndexedTriangleStripSet',
                          'Inline', 'IntegerSequencer', 'IntegerTrigger', 'IS', 'KeySensor', 'LineProperties', 'LineSet',
                          'LoadSensor', 'LOD', 'Material', 'meta', 'MetadataDouble', 'MetadataFloat', 'MetadataInteger',
                          'MetadataSet', 'MetadataString', 'MovieTexture', 'MultiTexture', 'MultiTextureCoordinate',
                          'MultiTextureTransform', 'NavigationInfo', 'Normal', 'NormalInterpolator', 'NurbsCurve',
                          'NurbsCurve2D', 'NurbsOrientationInterpolator', 'NurbsPatchSurface',
                          'NurbsPositionInterpolator', 'NurbsSet', 'NurbsSurfaceInterpolator', 'NurbsSweptSurface',
                          'NurbsSwungSurface', 'NurbsTextureCoordinate', 'NurbsTrimmedSurface', 'OrientationInterpolator',
                          'PixelTexture', 'PlaneSensor', 'PointLight', 'PointSet', 'Polyline2D', 'Polypoint2D',
                          'PositionInterpolator', 'PositionInterpolator2D', 'ProtoBody', 'ProtoDeclare', 'ProtoInstance',
                          'ProtoInterface', 'ProximitySensor', 'ReceiverPdu', 'Rectangle2D', 'ROUTE', 'ScalarInterpolator',
                          'Scene', 'Script', 'Shape', 'SignalPdu', 'Sound', 'Sphere', 'SphereSensor', 'SpotLight', 'StaticGroup',
                          'StringSensor', 'Switch', 'Text', 'TextureBackground', 'TextureCoordinate', 'TextureCoordinateGenerator',
                          'TextureTransform', 'TimeSensor', 'TimeTrigger', 'TouchSensor', 'Transform', 'TransmitterPdu',
                          'TriangleFanSet', 'TriangleSet', 'TriangleSet2D', 'TriangleStripSet', 'Viewpoint', 'VisibilitySensor',
                          'WorldInfo', 'X3D', 'XvlShell', 'VertexShader', 'FragmentShader', 'MultiShaderAppearance', 'ShaderAppearance'}
    
    # h3d defines
    H3D_TOP_LEVEL = 'TOP_LEVEL_TI'
    
    H3D_CAMERA_FOLLOW = 'CAMERA_FOLLOW_TRANSFORM'
    
    H3D_VIEW_MATRIX = 'view_matrix'
    
    
    def clamp_color(col):
        return tuple([max(min(c, 1.0), 0.0) for c in col])
    
        return (matrix.to_3x3() * mathutils.Vector((0.0, 0.0, -1.0))).normalized()[:]
    
    def prefix_quoted_str(value, prefix):
        return value[0] + prefix + value[1:]
    
    def suffix_quoted_str(value, suffix):
        return value[:-1] + suffix + value[-1:]
    
    
    
        if txt[0] in "1234567890+-":
            txt = "_" + txt
        return txt.translate({  # control characters 0x0-0x1f
    
    Campbell Barton's avatar
    Campbell Barton committed
                                # 0x00: "_",
    
                              0x01: "_",
                              0x02: "_",
                              0x03: "_",
                              0x04: "_",
                              0x05: "_",
                              0x06: "_",
                              0x07: "_",
                              0x08: "_",
                              0x09: "_",
                              0x0a: "_",
                              0x0b: "_",
                              0x0c: "_",
                              0x0d: "_",
                              0x0e: "_",
                              0x0f: "_",
                              0x10: "_",
                              0x11: "_",
                              0x12: "_",
                              0x13: "_",
                              0x14: "_",
                              0x15: "_",
                              0x16: "_",
                              0x17: "_",
                              0x18: "_",
                              0x19: "_",
                              0x1a: "_",
                              0x1b: "_",
                              0x1c: "_",
                              0x1d: "_",
                              0x1e: "_",
                              0x1f: "_",
    
    
    Campbell Barton's avatar
    Campbell Barton committed
                              0x7f: "_",  # 127
    
                              0x20: "_",  # space
                              0x22: "_",  # "
                              0x27: "_",  # '
                              0x23: "_",  # #
                              0x2c: "_",  # ,
                              0x2e: "_",  # .
                              0x5b: "_",  # [
                              0x5d: "_",  # ]
                              0x5c: "_",  # \
                              0x7b: "_",  # {
                              0x7d: "_",  # }
    
    Campbell Barton's avatar
    Campbell Barton committed
        """ returns parent child relationships, skipping
    
        """
        objects_set = set(objects)
        par_lookup = {}
    
        def test_parent(parent):
            while (parent is not None) and (parent not in objects_set):
                parent = parent.parent
            return parent
    
    
        for obj in objects:
    
            par_lookup.setdefault(test_parent(obj.parent), []).append((obj, []))
    
        for parent, children in par_lookup.items():
            for obj, subchildren in children:
                subchildren[:] = par_lookup.get(obj, [])
    
    
        return par_lookup.get(None, [])
    
    # -----------------------------------------------------------------------------
    # H3D Functions
    # -----------------------------------------------------------------------------
    
    def h3d_shader_glsl_frag_patch(filepath, scene, global_vars, frag_uniform_var_map):
    
        h3d_file = open(filepath, 'r', encoding='utf-8')
    
        lines = []
    
        last_transform = None
    
        for l in h3d_file:
    
            if l.startswith("void main(void)"):
                lines.append("\n")
                lines.append("// h3d custom vars begin\n")
                for v in global_vars:
                    lines.append("%s\n" % v)
                lines.append("// h3d custom vars end\n")
                lines.append("\n")
            elif l.lstrip().startswith("lamp_visibility_other("):
                w = l.split(', ')
    
                last_transform = w[1] + "_transform"  # XXX - HACK!!!
    
                w[1] = '(view_matrix * %s_transform * vec4(%s.x, %s.y, %s.z, 1.0)).xyz' % (w[1], w[1], w[1], w[1])
                l = ", ".join(w)
    
            elif l.lstrip().startswith("lamp_visibility_sun_hemi("):
                w = l.split(', ')
                w[0] = w[0][len("lamp_visibility_sun_hemi(") + 1:]
    
                if not h3d_is_object_view(scene, frag_uniform_var_map[w[0]]):
                    w[0] = '(mat3(normalize(view_matrix[0].xyz), normalize(view_matrix[1].xyz), normalize(view_matrix[2].xyz)) * -%s)' % w[0]
                else:
                    w[0] = ('(mat3(normalize((view_matrix*%s)[0].xyz), normalize((view_matrix*%s)[1].xyz), normalize((view_matrix*%s)[2].xyz)) * -%s)' %
                            (last_transform, last_transform, last_transform, w[0]))
    
    
                l = "\tlamp_visibility_sun_hemi(" + ", ".join(w)
            elif l.lstrip().startswith("lamp_visibility_spot_circle("):
                w = l.split(', ')
                w[0] = w[0][len("lamp_visibility_spot_circle(") + 1:]
    
                if not h3d_is_object_view(scene, frag_uniform_var_map[w[0]]):
                    w[0] = '(mat3(normalize(view_matrix[0].xyz), normalize(view_matrix[1].xyz), normalize(view_matrix[2].xyz)) * -%s)' % w[0]
                else:
                    w[0] = ('(mat3(normalize((view_matrix*%s)[0].xyz), normalize((view_matrix*%s)[1].xyz), normalize((view_matrix*%s)[2].xyz)) * %s)' %
                        (last_transform, last_transform, last_transform, w[0]))
    
    
                l = "\tlamp_visibility_spot_circle(" + ", ".join(w)
    
            lines.append(l)
    
        h3d_file.close()
    
    
        h3d_file = open(filepath, 'w', encoding='utf-8')
    
        h3d_file.writelines(lines)
        h3d_file.close()
    
    
    
    def h3d_is_object_view(scene, obj):
        camera = scene.camera
        parent = obj.parent
        while parent:
            if parent == camera:
                return True
            parent = parent.parent
        return False
    
    
    
    # -----------------------------------------------------------------------------
    
    # Functions for writing output file
    
    # -----------------------------------------------------------------------------
    
    def export(file,
               global_matrix,
               scene,
               use_apply_modifiers=False,
               use_selection=True,
    
               use_triangulate=False,
               use_normals=False,
    
    Campbell Barton's avatar
    Campbell Barton committed
               use_h3d=False,
    
               path_mode='AUTO',
    
    Campbell Barton's avatar
    Campbell Barton committed
        # -------------------------------------------------------------------------
    
    Campbell Barton's avatar
    Campbell Barton committed
        # -------------------------------------------------------------------------
    
        import bpy_extras
    
        from bpy_extras.io_utils import unique_name
    
        from xml.sax.saxutils import quoteattr, escape
    
        if name_decorations:
            # If names are decorated, the uuid map can be split up
            # by type for efficiency of collision testing
            # since objects of different types will always have
            # different decorated names.
            uuid_cache_object = {}    # object
            uuid_cache_lamp = {}      # 'LA_' + object.name
            uuid_cache_view = {}      # object, different namespace
            uuid_cache_mesh = {}      # mesh
            uuid_cache_material = {}  # material
            uuid_cache_image = {}     # image
            uuid_cache_world = {}     # world
            CA_ = 'CA_'
            OB_ = 'OB_'
            ME_ = 'ME_'
            IM_ = 'IM_'
            WO_ = 'WO_'
            MA_ = 'MA_'
            LA_ = 'LA_'
            group_ = 'group_'
        else:
            # If names are not decorated, it may be possible for two objects to
    
    Campbell Barton's avatar
    Campbell Barton committed
            # have the same name, so there has to be a unified dictionary to
    
            # prevent uuid collisions.
    
            uuid_cache = {}
            uuid_cache_object = uuid_cache           # object
            uuid_cache_lamp = uuid_cache             # 'LA_' + object.name
            uuid_cache_view = uuid_cache             # object, different namespace
            uuid_cache_mesh = uuid_cache             # mesh
            uuid_cache_material = uuid_cache         # material
            uuid_cache_image = uuid_cache            # image
            uuid_cache_world = uuid_cache            # world
            del uuid_cache
    
            CA_ = ''
            OB_ = ''
            ME_ = ''
            IM_ = ''
            WO_ = ''
            MA_ = ''
            LA_ = ''
            group_ = ''
    
        _TRANSFORM = '_TRANSFORM'
    
        # store files to copy
        copy_set = set()
    
    
        # store names of newly cerated meshes, so we dont overlap
        mesh_name_set = set()
    
    
        base_src = os.path.dirname(bpy.data.filepath)
        base_dst = os.path.dirname(file.name)
        filename_strip = os.path.splitext(os.path.basename(file.name))[0]
    
    Campbell Barton's avatar
    Campbell Barton committed
        gpu_shader_cache = {}
    
        if use_h3d:
            import gpu
    
            gpu_shader_dummy_mat = bpy.data.materials.new('X3D_DYMMY_MAT')
    
    Campbell Barton's avatar
    Campbell Barton committed
            gpu_shader_cache[None] = gpu.export_shader(scene, gpu_shader_dummy_mat)
    
            h3d_material_route = []
    
    Campbell Barton's avatar
    Campbell Barton committed
    
    
        # -------------------------------------------------------------------------
        # File Writing Functions
        # -------------------------------------------------------------------------
    
        def writeHeader(ident):
    
            filepath_quoted = quoteattr(os.path.basename(file.name))
            blender_ver_quoted = quoteattr('Blender %s' % bpy.app.version_string)
    
    
            fw('%s<?xml version="1.0" encoding="UTF-8"?>\n' % ident)
    
    Campbell Barton's avatar
    Campbell Barton committed
            if use_h3d:
    
                fw('%s<X3D profile="H3DAPI" version="1.4">\n' % ident)
    
    Campbell Barton's avatar
    Campbell Barton committed
            else:
    
                fw('%s<!DOCTYPE X3D PUBLIC "ISO//Web3D//DTD X3D 3.0//EN" "http://www.web3d.org/specifications/x3d-3.0.dtd">\n' % ident)
                fw('%s<X3D version="3.0" profile="Immersive" xmlns:xsd="http://www.w3.org/2001/XMLSchema-instance" xsd:noNamespaceSchemaLocation="http://www.web3d.org/specifications/x3d-3.0.xsd">\n' % ident)
    
            ident += '\t'
            fw('%s<head>\n' % ident)
            ident += '\t'
    
            fw('%s<meta name="filename" content=%s />\n' % (ident, filepath_quoted))
            fw('%s<meta name="generator" content=%s />\n' % (ident, blender_ver_quoted))
    
    Campbell Barton's avatar
    Campbell Barton committed
            # this info was never updated, so blender version should be enough
            # fw('%s<meta name="translator" content="X3D exporter v1.55 (2006/01/17)" />\n' % ident)
    
            fw('%s</head>\n' % ident)
            fw('%s<Scene>\n' % ident)
            ident += '\t'
    
    
            if use_h3d:
                # outputs the view matrix in glModelViewMatrix field
                fw('%s<TransformInfo DEF="%s" outputGLMatrices="true" />\n' % (ident, H3D_TOP_LEVEL))
    
    
        def writeFooter(ident):
    
            if use_h3d:
                # global
                for route in h3d_material_route:
                    fw('%s%s\n' % (ident, route))
    
    
            fw('%s</Scene>\n' % ident)
    
            fw('%s</X3D>' % ident)
    
        def writeViewpoint(ident, obj, matrix, scene):
    
            view_id = quoteattr(unique_name(obj, CA_ + obj.name, uuid_cache_view, clean_func=clean_def, sep="_"))
    
            loc, rot, scale = matrix.decompose()
            rot = rot.to_axis_angle()
    
    Campbell Barton's avatar
    Campbell Barton committed
    
            ident_step = ident + (' ' * (-len(ident) + \
            fw('%s<Viewpoint ' % ident)))
    
    Campbell Barton's avatar
    Campbell Barton committed
            fw(ident_step + 'centerOfRotation="0 0 0"\n')
            fw(ident_step + 'position="%3.2f %3.2f %3.2f"\n' % loc[:])
    
            fw(ident_step + 'orientation="%3.2f %3.2f %3.2f %3.2f"\n' % rot)
            fw(ident_step + 'fieldOfView="%.3f"\n' % obj.data.angle)
    
    Campbell Barton's avatar
    Campbell Barton committed
            fw(ident_step + '/>\n')
    
        def writeFog(ident, world):
    
            if world:
                mtype = world.mist_settings.falloff
                mparam = world.mist_settings
            else:
                return
    
    Campbell Barton's avatar
    Campbell Barton committed
                ident_step = ident + (' ' * (-len(ident) + \
                fw('%s<Fog ' % ident)))
                fw('fogType="%s"\n' % ('LINEAR' if (mtype == 'LINEAR') else 'EXPONENTIAL'))
    
                fw(ident_step + 'color="%.3f %.3f %.3f"\n' % clamp_color(world.horizon_color))
                fw(ident_step + 'visibilityRange="%.3f"\n' % mparam.depth)
    
    Campbell Barton's avatar
    Campbell Barton committed
                fw(ident_step + '/>\n')
    
        def writeNavigationInfo(ident, scene):
    
    Campbell Barton's avatar
    Campbell Barton committed
            ident_step = ident + (' ' * (-len(ident) + \
            fw('%s<NavigationInfo ' % ident)))
            fw('headlight="false"\n')
            fw(ident_step + 'visibilityLimit="0.0"\n')
            fw(ident_step + 'type=\'"EXAMINE", "ANY"\'\n')
            fw(ident_step + 'avatarSize="0.25, 1.75, 0.75"\n')
            fw(ident_step + '/>\n')
    
        def writeTransform_begin(ident, matrix, def_id):
            ident_step = ident + (' ' * (-len(ident) + \
            fw('%s<Transform ' % ident)))
            if def_id is not None:
                fw('DEF=%s\n' % def_id)
            else:
                fw('\n')
    
    
            loc, rot, sca = matrix.decompose()
            rot = rot.to_axis_angle()
            rot = rot[0][:] + (rot[1], )
    
            fw(ident_step + 'translation="%.6f %.6f %.6f"\n' % loc[:])
            # fw(ident_step + 'center="%.6f %.6f %.6f"\n' % (0, 0, 0))
            fw(ident_step + 'scale="%.6f %.6f %.6f"\n' % sca[:])
            fw(ident_step + 'rotation="%.6f %.6f %.6f %.6f"\n' % rot)
    
            fw(ident_step + '>\n')
            ident += '\t'
            return ident
    
        def writeTransform_end(ident):
            ident = ident[:-1]
            fw('%s</Transform>\n' % ident)
            return ident
    
    
        def writeSpotLight(ident, obj, matrix, lamp, world):
            # note, lamp_id is not re-used
    
            lamp_id = quoteattr(unique_name(obj, LA_ + obj.name, uuid_cache_lamp, clean_func=clean_def, sep="_"))
    
            if world:
                ambi = world.ambient_color
    
                amb_intensity = ((ambi[0] + ambi[1] + ambi[2]) / 3.0) / 2.5
    
    
            # compute cutoff and beamwidth
            intensity = min(lamp.energy / 1.75, 1.0)
            beamWidth = lamp.spot_size * 0.37
            # beamWidth=((lamp.spotSize*math.pi)/180.0)*.37
            cutOffAngle = beamWidth * 1.3
    
    
            orientation = matrix_direction_neg_z(matrix)
    
    
            radius = lamp.distance * math.cos(beamWidth)
            # radius = lamp.dist*math.cos(beamWidth)
    
    Campbell Barton's avatar
    Campbell Barton committed
            ident_step = ident + (' ' * (-len(ident) + \
            fw('%s<SpotLight ' % ident)))
    
            fw(ident_step + 'radius="%.4f"\n' % radius)
            fw(ident_step + 'ambientIntensity="%.4f"\n' % amb_intensity)
            fw(ident_step + 'intensity="%.4f"\n' % intensity)
            fw(ident_step + 'color="%.4f %.4f %.4f"\n' % clamp_color(lamp.color))
            fw(ident_step + 'beamWidth="%.4f"\n' % beamWidth)
            fw(ident_step + 'cutOffAngle="%.4f"\n' % cutOffAngle)
            fw(ident_step + 'direction="%.4f %.4f %.4f"\n' % orientation)
            fw(ident_step + 'location="%.4f %.4f %.4f"\n' % location)
    
    Campbell Barton's avatar
    Campbell Barton committed
            fw(ident_step + '/>\n')
    
        def writeDirectionalLight(ident, obj, matrix, lamp, world):
            # note, lamp_id is not re-used
    
            lamp_id = quoteattr(unique_name(obj, LA_ + obj.name, uuid_cache_lamp, clean_func=clean_def, sep="_"))
    
            if world:
                ambi = world.ambient_color
                # ambi = world.amb
    
                amb_intensity = ((float(ambi[0] + ambi[1] + ambi[2])) / 3.0) / 2.5
    
    
            intensity = min(lamp.energy / 1.75, 1.0)
    
            orientation = matrix_direction_neg_z(matrix)
    
    Campbell Barton's avatar
    Campbell Barton committed
            ident_step = ident + (' ' * (-len(ident) + \
            fw('%s<DirectionalLight ' % ident)))
    
            fw(ident_step + 'ambientIntensity="%.4f"\n' % amb_intensity)
            fw(ident_step + 'color="%.4f %.4f %.4f"\n' % clamp_color(lamp.color))
            fw(ident_step + 'intensity="%.4f"\n' % intensity)
            fw(ident_step + 'direction="%.4f %.4f %.4f"\n' % orientation)
    
    Campbell Barton's avatar
    Campbell Barton committed
            fw(ident_step + '/>\n')
    
        def writePointLight(ident, obj, matrix, lamp, world):
            # note, lamp_id is not re-used
    
            lamp_id = quoteattr(unique_name(obj, LA_ + obj.name, uuid_cache_lamp, clean_func=clean_def, sep="_"))
    
            if world:
                ambi = world.ambient_color
                # ambi = world.amb
    
                amb_intensity = ((float(ambi[0] + ambi[1] + ambi[2])) / 3.0) / 2.5
    
                ambi = 0.0
                amb_intensity = 0.0
    
            intensity = min(lamp.energy / 1.75, 1.0)
    
    Campbell Barton's avatar
    Campbell Barton committed
            ident_step = ident + (' ' * (-len(ident) + \
            fw('%s<PointLight ' % ident)))
    
            fw(ident_step + 'ambientIntensity="%.4f"\n' % amb_intensity)
            fw(ident_step + 'color="%.4f %.4f %.4f"\n' % clamp_color(lamp.color))
    
            fw(ident_step + 'intensity="%.4f"\n' % intensity)
            fw(ident_step + 'radius="%.4f" \n' % lamp.distance)
            fw(ident_step + 'location="%.4f %.4f %.4f"\n' % location)
    
    Campbell Barton's avatar
    Campbell Barton committed
            fw(ident_step + '/>\n')
    
        def writeIndexedFaceSet(ident, obj, mesh, matrix, world):
    
            obj_id = quoteattr(unique_name(obj, OB_ + obj.name, uuid_cache_object, clean_func=clean_def, sep="_"))
            mesh_id = quoteattr(unique_name(mesh, ME_ + mesh.name, uuid_cache_mesh, clean_func=clean_def, sep="_"))
            mesh_id_group = prefix_quoted_str(mesh_id, group_)
    
            mesh_id_coords = prefix_quoted_str(mesh_id, 'coords_')
            mesh_id_normals = prefix_quoted_str(mesh_id, 'normals_')
    
            # tessellation faces may not exist
    
            if not mesh.tessfaces and mesh.polygons:
    
                mesh.update(calc_tessface=True)
    
    
            use_collnode = bool([mod for mod in obj.modifiers
                                 if mod.type == 'COLLISION'
                                 if mod.show_viewport])
    
            if use_collnode:
    
                fw('%s<Collision enabled="true">\n' % ident)
    
                ident += '\t'
    
            # use _ifs_TRANSFORM suffix so we dont collide with transform node when
            # hierarchys are used.
    
            ident = writeTransform_begin(ident, matrix, suffix_quoted_str(obj_id, "_ifs" + _TRANSFORM))
    
                fw('%s<Group USE=%s />\n' % (ident, mesh_id_group))
    
                fw('%s<Group DEF=%s>\n' % (ident, mesh_id_group))
    
                ident += '\t'
    
                is_uv = bool(mesh.tessface_uv_textures.active)
    
                # is_col, defined for each material
    
                is_coords_written = False
    
                mesh_materials = mesh.materials[:]
                if not mesh_materials:
                    mesh_materials = [None]
    
                mesh_material_tex = [None] * len(mesh_materials)
                mesh_material_mtex = [None] * len(mesh_materials)
                mesh_material_images = [None] * len(mesh_materials)
    
                for i, material in enumerate(mesh_materials):
                    if material:
                        for mtex in material.texture_slots:
                            if mtex:
                                tex = mtex.texture
                                if tex and tex.type == 'IMAGE':
                                    image = tex.image
                                    if image:
                                        mesh_material_tex[i] = tex
                                        mesh_material_mtex[i] = mtex
                                        mesh_material_images[i] = image
                                        break
    
    
                mesh_materials_use_face_texture = [getattr(material, 'use_face_texture', True) for material in mesh_materials]
    
                mesh_vertices = mesh.vertices[:]
    
                mesh_faces_materials = [f.material_index for f in mesh_faces]
    
                mesh_faces_vertices = [f.vertices[:] for f in mesh_faces]
    
                if is_uv and True in mesh_materials_use_face_texture:
    
                    mesh_faces_image = [(fuv.image
                                         if mesh_materials_use_face_texture[mesh_faces_materials[i]]
                                         else mesh_material_images[mesh_faces_materials[i]])
                                         for i, fuv in enumerate(mesh.tessface_uv_textures.active.data)]
    
    
                    mesh_faces_image_unique = set(mesh_faces_image)
                elif len(set(mesh_material_images) | {None}) > 1:  # make sure there is at least one image
                    mesh_faces_image = [mesh_material_images[material_index] for material_index in mesh_faces_materials]
                    mesh_faces_image_unique = set(mesh_faces_image)
                else:
                    mesh_faces_image = [None] * len(mesh_faces)
                    mesh_faces_image_unique = {None}
    
                # group faces
                face_groups = {}
                for material_index in range(len(mesh_materials)):
                    for image in mesh_faces_image_unique:
                        face_groups[material_index, image] = []
                del mesh_faces_image_unique
    
                for i, (material_index, image) in enumerate(zip(mesh_faces_materials, mesh_faces_image)):
                    face_groups[material_index, image].append(i)
    
    
                # same as face_groups.items() but sorted so we can get predictable output.
                face_groups_items = list(face_groups.items())
    
                face_groups_items.sort(key=lambda m: (m[0][0], getattr(m[0][1], 'name', '')))
    
    
                for (material_index, image), face_group in face_groups_items:  # face_groups.items()
    
                    if face_group:
                        material = mesh_materials[material_index]
    
    
                        fw('%s<Shape>\n' % ident)
                        ident += '\t'
    
                        is_col = (mesh.tessface_vertex_colors.active and (material is None or material.use_vertex_color_paint))
    
    
                        # kludge but as good as it gets!
                        for i in face_group:
                            if mesh_faces[i].use_smooth:
                                is_smooth = True
                                break
    
    
                        # UV's and VCols split verts off which effects smoothing
                        # force writing normals in this case.
    
                        # Also, creaseAngle is not supported for IndexedTriangleSet,
                        # so write normals when is_smooth (otherwise
                        # IndexedTriangleSet can have only all smooth/all flat shading).
                        is_force_normals = use_triangulate and (is_smooth or is_uv or is_col)
    
    Campbell Barton's avatar
    Campbell Barton committed
                        if use_h3d:
                            gpu_shader = gpu_shader_cache.get(material)  # material can be 'None', uses dummy cache
                            if gpu_shader is None:
                                gpu_shader = gpu_shader_cache[material] = gpu.export_shader(scene, material)
    
    Campbell Barton's avatar
    Campbell Barton committed
                                if 1:  # XXX DEBUG
                                    gpu_shader_tmp = gpu.export_shader(scene, material)
                                    import pprint
    
                                    print('\nWRITING MATERIAL:', material.name)
    
    Campbell Barton's avatar
    Campbell Barton committed
                                    del gpu_shader_tmp['fragment']
                                    del gpu_shader_tmp['vertex']
                                    pprint.pprint(gpu_shader_tmp, width=120)
                                    #pprint.pprint(val['vertex'])
                                    del gpu_shader_tmp
    
    
                        fw('%s<Appearance>\n' % ident)
                        ident += '\t'
    
    Campbell Barton's avatar
    Campbell Barton committed
                        if image and not use_h3d:
    
                            writeImageTexture(ident, image)
    
    
                            if mesh_materials_use_face_texture[material_index]:
                                if image.use_tiles:
    
                                    fw('%s<TextureTransform scale="%s %s" />\n' % (ident, image.tiles_x, image.tiles_y))
    
                            else:
                                # transform by mtex
                                loc = mesh_material_mtex[material_index].offset[:2]
    
                                # mtex_scale * tex_repeat
                                sca_x, sca_y = mesh_material_mtex[material_index].scale[:2]
    
                                sca_x *= mesh_material_tex[material_index].repeat_x
                                sca_y *= mesh_material_tex[material_index].repeat_y
    
                                # flip x/y is a sampling feature, convert to transform
                                if mesh_material_tex[material_index].use_flip_axis:
                                    rot = math.pi / -2.0
                                    sca_x, sca_y = sca_y, -sca_x
                                else:
                                    rot = 0.0
    
    
    Campbell Barton's avatar
    Campbell Barton committed
                                ident_step = ident + (' ' * (-len(ident) + \
                                fw('%s<TextureTransform ' % ident)))
    
                                # fw('center="%.6f %.6f" ' % (0.0, 0.0))
                                fw(ident_step + 'translation="%.6f %.6f"\n' % loc)
                                fw(ident_step + 'scale="%.6f %.6f"\n' % (sca_x, sca_y))
                                fw(ident_step + 'rotation="%.6f"\n' % rot)
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw(ident_step + '/>\n')
    
    Campbell Barton's avatar
    Campbell Barton committed
                        if use_h3d:
                            mat_tmp = material if material else gpu_shader_dummy_mat
    
                            writeMaterialH3D(ident, mat_tmp, world,
    
                                             obj, gpu_shader)
    
    Campbell Barton's avatar
    Campbell Barton committed
                            del mat_tmp
                        else:
    
                            if material:
                                writeMaterial(ident, material, world)
    
                        fw('%s</Appearance>\n' % ident)
    
                        mesh_faces_col = mesh.tessface_vertex_colors.active.data if is_col else None
                        mesh_faces_uv = mesh.tessface_uv_textures.active.data if is_uv else None
    
                        #-- IndexedFaceSet or IndexedLineSet
    
                        if use_triangulate:
    
    Campbell Barton's avatar
    Campbell Barton committed
                            ident_step = ident + (' ' * (-len(ident) + \
                            fw('%s<IndexedTriangleSet ' % ident)))
    
    
                            # --- Write IndexedTriangleSet Attributes (same as IndexedFaceSet)
    
                            fw('solid="%s"\n' % ('true' if material and material.game_settings.use_backface_culling else 'false'))
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw(ident_step + 'normalPerVertex="true"\n')
    
                            else:
                                # Tell X3D browser to generate flat (per-face) normals
                                fw(ident_step + 'normalPerVertex="false"\n')
    
                            slot_uv = None
                            slot_col = None
    
                            if is_uv and is_col:
                                slot_uv = 0
                                slot_col = 1
    
                                def vertex_key(fidx, f_cnr_idx):
                                    return (
                                        mesh_faces_uv[fidx].uv[f_cnr_idx][:],
                                        getattr(mesh_faces_col[fidx], "color%d" % (f_cnr_idx + 1))[:],
                                    )
                            elif is_uv:
                                slot_uv = 0
    
                                def vertex_key(fidx, f_cnr_idx):
                                    return (
    
                                        mesh_faces_uv[fidx].uv[f_cnr_idx][:],
    
                                    )
                            elif is_col:
                                slot_col = 0
    
                                def vertex_key(fidx, f_cnr_idx):
                                    return (
    
    Campbell Barton's avatar
    Campbell Barton committed
                                        getattr(mesh_faces_col[fidx], "color%d" % (f_cnr_idx + 1))[:],
    
                                    )
                            else:
                                # ack, not especially efficient in this case
                                def vertex_key(fidx, f_cnr_idx):
                                    return None
    
                            # build a mesh mapping dict
                            vertex_hash = [{} for i in range(len(mesh.vertices))]
                            # worst case every face is a quad
    
                            face_tri_list = [[None, None, None] for i in range(len(mesh.tessfaces) * 2)]
    
                            vert_tri_list = []
                            totvert = 0
                            totface = 0
                            temp_face = [None] * 4
                            for i in face_group:
                                fv = mesh_faces_vertices[i]
                                for j, v_idx in enumerate(fv):
                                    key = vertex_key(i, j)
                                    vh = vertex_hash[v_idx]
                                    x3d_v = vh.get(key)
                                    if x3d_v is None:
                                        x3d_v = key, v_idx, totvert
                                        vh[key] = x3d_v
                                        # key / original_vertex / new_vertex
                                        vert_tri_list.append(x3d_v)
                                        totvert += 1
                                    temp_face[j] = x3d_v
    
                                if len(fv) == 4:
                                    f_iter = ((0, 1, 2), (0, 2, 3))
                                else:
                                    f_iter = ((0, 1, 2), )
    
                                for f_it in f_iter:
                                    # loop over a quad as 2 tris
                                    f_tri = face_tri_list[totface]
                                    for ji, j in enumerate(f_it):
                                        f_tri[ji] = temp_face[j]
                                    # quads run this twice
                                    totface += 1
    
                            # clear unused faces
                            face_tri_list[totface:] = []
    
    
    Campbell Barton's avatar
    Campbell Barton committed
                            fw(ident_step + 'index="')
    
                                fw('%i %i %i ' % (x3d_f[0][2], x3d_f[1][2], x3d_f[2][2]))
    
    Campbell Barton's avatar
    Campbell Barton committed
                            fw('"\n')
    
    Campbell Barton's avatar
    Campbell Barton committed
                            fw(ident_step + '>\n')
                            ident += '\t'
    
                            fw('%s<Coordinate ' % ident)
                            fw('point="')
    
                                fw('%.6f %.6f %.6f ' % mesh_vertices[x3d_v[1]].co[:])
    
                                fw('%s<Normal ' % ident)
                                fw('vector="')
    
                                for x3d_v in vert_tri_list:
    
                                    fw('%.6f %.6f %.6f ' % mesh_vertices[x3d_v[1]].normal[:])
    
                                fw('%s<TextureCoordinate point="' % ident)
    
                                    fw('%.4f %.4f ' % x3d_v[0][slot_uv])
    
                                fw('%s<Color color="' % ident)
    
                                    fw('%.3f %.3f %.3f ' % x3d_v[0][slot_col])
    
    Campbell Barton's avatar
    Campbell Barton committed
                            if use_h3d:
                                # write attributes
                                for gpu_attr in gpu_shader['attributes']:
    
    Campbell Barton's avatar
    Campbell Barton committed
                                    # UVs
                                    if gpu_attr['type'] == gpu.CD_MTFACE:
                                        if gpu_attr['datatype'] == gpu.GPU_DATA_2F:
    
                                            fw('%s<FloatVertexAttribute ' % ident)
    
    Campbell Barton's avatar
    Campbell Barton committed
                                            fw('name="%s" ' % gpu_attr['varname'])
                                            fw('numComponents="2" ')
                                            fw('value="')
                                            for x3d_v in vert_tri_list:
    
                                                fw('%.4f %.4f ' % x3d_v[0][slot_uv])
    
    Campbell Barton's avatar
    Campbell Barton committed
                                            fw('" />\n')
                                        else:
                                            assert(0)
    
                                    elif gpu_attr['type'] == gpu.CD_MCOL:
                                        if gpu_attr['datatype'] == gpu.GPU_DATA_4UB:
                                            pass  # XXX, H3D can't do
                                        else:
                                            assert(0)
    
    
    Campbell Barton's avatar
    Campbell Barton committed
                            ident = ident[:-1]
    
    
                            fw('%s</IndexedTriangleSet>\n' % ident)
    
    Campbell Barton's avatar
    Campbell Barton committed
                            ident_step = ident + (' ' * (-len(ident) + \
                            fw('%s<IndexedFaceSet ' % ident)))
    
                            # --- Write IndexedFaceSet Attributes (same as IndexedTriangleSet)
    
                            fw('solid="%s"\n' % ('true' if material and material.game_settings.use_backface_culling else 'false'))
    
                            if is_smooth:
                                # use Auto-Smooth angle, if enabled. Otherwise make
                                # the mesh perfectly smooth by creaseAngle > pi.
                                fw(ident_step + 'creaseAngle="%.4f"\n' % (mesh.auto_smooth_angle if mesh.use_auto_smooth else 4.0))
    
                            if use_normals:
                                # currently not optional, could be made so:
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw(ident_step + 'normalPerVertex="true"\n')
    
                            # IndexedTriangleSet assumes true
                            if is_col:
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw(ident_step + 'colorPerVertex="false"\n')
    
                            # for IndexedTriangleSet we use a uv per vertex so this isnt needed.
                            if is_uv:
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw(ident_step + 'texCoordIndex="')
    
                                    if len(mesh_faces_vertices[i]) == 4:
    
                                        fw('%d %d %d %d -1 ' % (j, j + 1, j + 2, j + 3))
    
                                        fw('%d %d %d -1 ' % (j, j + 1, j + 2))
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw('"\n')
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw(ident_step + 'coordIndex="')
    
                                        fw('%i %i %i -1 ' % fv)
    
                                        fw('%i %i %i %i -1 ' % fv)
    
    Campbell Barton's avatar
    Campbell Barton committed
                                fw('"\n')
    
    Campbell Barton's avatar
    Campbell Barton committed
                            fw(ident_step + '>\n')
                            ident += '\t'
    
                            # --- Write IndexedFaceSet Elements
                            if True:
                                if is_coords_written:
    
                                    fw('%s<Coordinate USE=%s />\n' % (ident, mesh_id_coords))
    
                                    if use_normals:
    
                                        fw('%s<Normal USE=%s />\n' % (ident, mesh_id_normals))
    
    Campbell Barton's avatar
    Campbell Barton committed
                                    ident_step = ident + (' ' * (-len(ident) + \
                                    fw('%s<Coordinate ' % ident)))
    
    Campbell Barton's avatar
    Campbell Barton committed
                                    fw(ident_step + 'point="')
    
    Campbell Barton's avatar
    Campbell Barton committed
                                    fw('"\n')
                                    fw(ident_step + '/>\n')
    
                                    if use_normals:
    
    Campbell Barton's avatar
    Campbell Barton committed
                                        ident_step = ident + (' ' * (-len(ident) + \
                                        fw('%s<Normal ' % ident)))
    
    Campbell Barton's avatar
    Campbell Barton committed
                                        fw(ident_step + 'vector="')
    
                                        for v in mesh.vertices:
    
                                            fw('%.6f %.6f %.6f ' % v.normal[:])
    
    Campbell Barton's avatar
    Campbell Barton committed
                                        fw('"\n')
                                        fw(ident_step + '/>\n')
    
                                fw('%s<TextureCoordinate point="' % ident)
    
                                for i in face_group:
                                    for uv in mesh_faces_uv[i].uv:
    
                                fw('%s<Color color="' % ident)
    
                                    fw('%.3f %.3f %.3f ' % mesh_faces_col[i].color1[:])
    
                            #--- output closing braces
                            ident = ident[:-1]
    
                            fw('%s</IndexedFaceSet>\n' % ident)
    
                        fw('%s</Shape>\n' % ident)
    
                #fw('%s<PythonScript DEF="PS" url="object.py" >\n' % ident)
                #fw('%s    <ShaderProgram USE="MA_Material.005" containerField="references"/>\n' % ident)
                #fw('%s</PythonScript>\n' % ident)
    
                fw('%s</Group>\n' % ident)
    
                fw('%s</Collision>\n' % ident)
    
        def writeMaterial(ident, material, world):
    
            material_id = quoteattr(unique_name(material, MA_ + material.name, uuid_cache_material, clean_func=clean_def, sep="_"))
    
            # look up material name, use it if available
    
            if material.tag:
                fw('%s<Material USE=%s />\n' % (ident, material_id))
    
                emit = material.emit
                ambient = material.ambient / 3.0
                diffuseColor = material.diffuse_color[:]
    
                    ambiColor = ((material.ambient * 2.0) * world.ambient_color)[:]
    
                else:
                    ambiColor = 0.0, 0.0, 0.0
    
                emitColor = tuple(((c * emit) + ambiColor[i]) / 2.0 for i, c in enumerate(diffuseColor))
    
                shininess = material.specular_hardness / 512.0
                specColor = tuple((c + 0.001) / (1.25 / (material.specular_intensity + 0.001)) for c in material.specular_color)
                transp = 1.0 - material.alpha
    
                    ambient = 1.0
                    shininess = 0.0
                    specColor = emitColor = diffuseColor
    
    
    Campbell Barton's avatar
    Campbell Barton committed
                ident_step = ident + (' ' * (-len(ident) + \
                fw('%s<Material ' % ident)))
    
                fw(ident_step + 'diffuseColor="%.3f %.3f %.3f"\n' % clamp_color(diffuseColor))
                fw(ident_step + 'specularColor="%.3f %.3f %.3f"\n' % clamp_color(specColor))
                fw(ident_step + 'emissiveColor="%.3f %.3f %.3f"\n' % clamp_color(emitColor))
                fw(ident_step + 'ambientIntensity="%.3f"\n' % ambient)