| 1 |
using System.Collections.Generic; |
| 2 |
using System.Globalization; |
| 3 |
using System.IO; |
| 4 |
using System.Xml; |
| 5 |
|
| 6 |
namespace Oni.Level |
| 7 |
{ |
| 8 |
using Akira; |
| 9 |
using Metadata; |
| 10 |
using Motoko; |
| 11 |
using Physics; |
| 12 |
using Xml; |
| 13 |
|
| 14 |
partial class LevelImporter |
| 15 |
{ |
| 16 |
private void ReadPhysics(XmlReader xml, string basePath) |
| 17 |
{ |
| 18 |
if (xml.SkipEmpty()) |
| 19 |
return; |
| 20 |
|
| 21 |
xml.ReadStartElement("Physics"); |
| 22 |
|
| 23 |
while (xml.IsStartElement()) |
| 24 |
ReadObjectSetup(xml, basePath); |
| 25 |
|
| 26 |
xml.ReadEndElement(); |
| 27 |
} |
| 28 |
|
| 29 |
private void ReadObjectSetup(XmlReader xml, string basePath) |
| 30 |
{ |
| 31 |
var scriptId = -1; |
| 32 |
var name = xml.GetAttribute("Name"); |
| 33 |
var position = Vector3.Zero; |
| 34 |
var rotation = Quaternion.Identity; |
| 35 |
var scale = 1.0f; |
| 36 |
var flags = ObjectSetupFlags.None; |
| 37 |
var physicsType = ObjectPhysicsType.None; |
| 38 |
var particles = new List<ObjectParticle>(); |
| 39 |
var nodes = new List<ObjectNode>(); |
| 40 |
string geometryName = null; |
| 41 |
string animationName = null; |
| 42 |
|
| 43 |
xml.ReadStartElement("Object"); |
| 44 |
|
| 45 |
while (xml.IsStartElement()) |
| 46 |
{ |
| 47 |
switch (xml.LocalName) |
| 48 |
{ |
| 49 |
case "Name": |
| 50 |
name = xml.ReadElementContentAsString(); |
| 51 |
break; |
| 52 |
case "ScriptId": |
| 53 |
scriptId = xml.ReadElementContentAsInt(); |
| 54 |
break; |
| 55 |
case "Flags": |
| 56 |
flags = xml.ReadElementContentAsEnum<ObjectSetupFlags>() & ~ObjectSetupFlags.InUse; |
| 57 |
break; |
| 58 |
case "Position": |
| 59 |
position = xml.ReadElementContentAsVector3(); |
| 60 |
break; |
| 61 |
case "Rotation": |
| 62 |
rotation = xml.ReadElementContentAsEulerXYZ(); |
| 63 |
break; |
| 64 |
case "Scale": |
| 65 |
scale = xml.ReadElementContentAsFloat(); |
| 66 |
break; |
| 67 |
case "Physics": |
| 68 |
physicsType = xml.ReadElementContentAsEnum<ObjectPhysicsType>(); |
| 69 |
break; |
| 70 |
case "Particles": |
| 71 |
particles.AddRange(ReadParticles(xml, basePath)); |
| 72 |
break; |
| 73 |
|
| 74 |
case "Geometry": |
| 75 |
geometryName = xml.ReadElementContentAsString(); |
| 76 |
if (nodes.Count > 0) |
| 77 |
{ |
| 78 |
error.WriteLine("Geometry cannot be used together with Import, ignoring"); |
| 79 |
break; |
| 80 |
} |
| 81 |
break; |
| 82 |
|
| 83 |
case "Animation": |
| 84 |
animationName = xml.ReadElementContentAsString(); |
| 85 |
if (nodes.Count > 0) |
| 86 |
{ |
| 87 |
error.WriteLine("Animation cannot be used together with Import, ignoring"); |
| 88 |
break; |
| 89 |
} |
| 90 |
break; |
| 91 |
|
| 92 |
case "Import": |
| 93 |
if (geometryName != null || animationName != null) |
| 94 |
{ |
| 95 |
error.WriteLine("Import cannot be used together with Geometry and Animation, ignoring"); |
| 96 |
break; |
| 97 |
} |
| 98 |
nodes.AddRange(ImportObjectGeometry(xml, basePath)); |
| 99 |
break; |
| 100 |
|
| 101 |
default: |
| 102 |
error.WriteLine("Unknown physics object element {0}", xml.LocalName); |
| 103 |
xml.Skip(); |
| 104 |
break; |
| 105 |
} |
| 106 |
} |
| 107 |
|
| 108 |
xml.ReadEndElement(); |
| 109 |
|
| 110 |
if (geometryName != null) |
| 111 |
{ |
| 112 |
var m3gm = FindSharedInstance(TemplateTag.M3GM, geometryName, objectLoadContext); |
| 113 |
var geometry = GeometryDatReader.Read(m3gm); |
| 114 |
var animation = new ObjectAnimation[0]; |
| 115 |
|
| 116 |
if (animationName != null) |
| 117 |
{ |
| 118 |
var oban = FindSharedInstance(TemplateTag.OBAN, animationName, objectLoadContext); |
| 119 |
animation = new[] { ObjectDatReader.ReadAnimation(oban) }; |
| 120 |
} |
| 121 |
|
| 122 |
nodes.Add(new ObjectNode(new[] { new ObjectGeometry(geometry) }) { |
| 123 |
FileName = Path.GetFileName(geometryName), |
| 124 |
Name = m3gm.Name, |
| 125 |
ScriptId = scriptId, |
| 126 |
Flags = flags, |
| 127 |
Animations = animation |
| 128 |
}); |
| 129 |
} |
| 130 |
|
| 131 |
for (int i = 0; i < nodes.Count; i++) |
| 132 |
{ |
| 133 |
var node = nodes[i]; |
| 134 |
|
| 135 |
var setup = new ObjectSetup { |
| 136 |
Name = node.Name, |
| 137 |
FileName = node.FileName, |
| 138 |
ScriptId = scriptId++, |
| 139 |
Flags = flags, |
| 140 |
PhysicsType = physicsType, |
| 141 |
}; |
| 142 |
|
| 143 |
setup.Particles.AddRange(particles); |
| 144 |
|
| 145 |
setup.Geometries = node.Geometries |
| 146 |
.Where(n => (n.Flags & GunkFlags.Invisible) == 0) |
| 147 |
.Select(n => n.Geometry.Name).ToArray(); |
| 148 |
|
| 149 |
foreach (var nodeGeometry in node.Geometries.Where(g => (g.Flags & GunkFlags.Invisible) == 0)) |
| 150 |
{ |
| 151 |
var writer = new DatWriter(); |
| 152 |
GeometryDatWriter.Write(nodeGeometry.Geometry, writer.ImporterFile); |
| 153 |
writer.Write(outputDirPath); |
| 154 |
} |
| 155 |
|
| 156 |
setup.Position = position; |
| 157 |
setup.Orientation = rotation; |
| 158 |
setup.Scale = scale; |
| 159 |
setup.Origin = Matrix.CreateFromQuaternion(setup.Orientation) |
| 160 |
* Matrix.CreateScale(setup.Scale) |
| 161 |
* Matrix.CreateTranslation(setup.Position); |
| 162 |
|
| 163 |
foreach (var animation in node.Animations) |
| 164 |
{ |
| 165 |
if (nodes.Count > 1) |
| 166 |
animation.Name += i.ToString("d2", CultureInfo.InvariantCulture); |
| 167 |
|
| 168 |
if ((animation.Flags & ObjectAnimationFlags.Local) != 0) |
| 169 |
{ |
| 170 |
//animation.Scale = Matrix.CreateScale(setup.Scale); |
| 171 |
|
| 172 |
foreach (var key in animation.Keys) |
| 173 |
{ |
| 174 |
key.Rotation = setup.Orientation * key.Rotation; |
| 175 |
key.Translation += setup.Position; |
| 176 |
} |
| 177 |
} |
| 178 |
|
| 179 |
if ((animation.Flags & ObjectAnimationFlags.AutoStart) != 0) |
| 180 |
{ |
| 181 |
setup.Animation = animation; |
| 182 |
setup.PhysicsType = ObjectPhysicsType.Animated; |
| 183 |
} |
| 184 |
|
| 185 |
var writer = new DatWriter(); |
| 186 |
ObjectDatWriter.WriteAnimation(animation, writer); |
| 187 |
writer.Write(outputDirPath); |
| 188 |
} |
| 189 |
|
| 190 |
if (setup.Animation == null && node.Animations.Length > 0) |
| 191 |
{ |
| 192 |
setup.Animation = node.Animations[0]; |
| 193 |
} |
| 194 |
|
| 195 |
if (setup.Animation != null) |
| 196 |
{ |
| 197 |
var frame0 = setup.Animation.Keys[0]; |
| 198 |
|
| 199 |
setup.Scale = frame0.Scale.X; |
| 200 |
setup.Orientation = frame0.Rotation; |
| 201 |
setup.Position = frame0.Translation; |
| 202 |
} |
| 203 |
|
| 204 |
level.physics.Add(setup); |
| 205 |
} |
| 206 |
} |
| 207 |
|
| 208 |
private IEnumerable<ObjectNode> ImportObjectGeometry(XmlReader xml, string basePath) |
| 209 |
{ |
| 210 |
var filePath = xml.GetAttribute("Path"); |
| 211 |
|
| 212 |
if (filePath == null) |
| 213 |
filePath = xml.GetAttribute("Url"); |
| 214 |
|
| 215 |
var scene = LoadScene(Path.Combine(basePath, filePath)); |
| 216 |
var animClips = new List<ObjectAnimationClip>(); |
| 217 |
|
| 218 |
if (!xml.SkipEmpty()) |
| 219 |
{ |
| 220 |
xml.ReadStartElement(); |
| 221 |
|
| 222 |
while (xml.IsStartElement()) |
| 223 |
{ |
| 224 |
switch (xml.LocalName) |
| 225 |
{ |
| 226 |
case "Animation": |
| 227 |
animClips.Add(ReadAnimationClip(xml)); |
| 228 |
break; |
| 229 |
|
| 230 |
default: |
| 231 |
error.WriteLine("Unknown element {0}", xml.LocalName); |
| 232 |
xml.Skip(); |
| 233 |
break; |
| 234 |
} |
| 235 |
} |
| 236 |
|
| 237 |
xml.ReadEndElement(); |
| 238 |
} |
| 239 |
|
| 240 |
var importer = new ObjectDaeImporter(textureImporter, null); |
| 241 |
importer.Import(scene); |
| 242 |
return importer.Nodes; |
| 243 |
} |
| 244 |
} |
| 245 |
} |