| 1 | using System; | 
 
 
 
 
 | 2 | using System.Collections.Generic; | 
 
 
 
 
 | 3 | using System.IO; | 
 
 
 
 
 | 4 | using System.Globalization; | 
 
 
 
 
 | 5 | using System.Text; | 
 
 
 
 
 | 6 | using System.Xml; | 
 
 
 
 
 | 7 |  | 
 
 
 
 
 | 8 | namespace Oni.Dae.IO | 
 
 
 
 
 | 9 | { | 
 
 
 
 
 | 10 | internal class DaeWriter | 
 
 
 
 
 | 11 | { | 
 
 
 
 
 | 12 | #region Private data | 
 
 
 
 
 | 13 | private XmlWriter xml; | 
 
 
 
 
 | 14 | private Scene mainScene; | 
 
 
 
 
 | 15 | private WriteVisitor visitor; | 
 
 
 
 
 | 16 | private Dictionary<Source, string> writtenSources = new Dictionary<Source, string>(); | 
 
 
 
 
 | 17 | #endregion | 
 
 
 
 
 | 18 |  | 
 
 
 
 
 | 19 | #region private class Animation | 
 
 
 
 
 | 20 |  | 
 
 
 
 
 | 21 | private class Animation : Entity | 
 
 
 
 
 | 22 | { | 
 
 
 
 
 | 23 | public readonly List<Source> Sources = new List<Source>(); | 
 
 
 
 
 | 24 | public readonly List<Sampler> Samplers = new List<Sampler>(); | 
 
 
 
 
 | 25 | public readonly List<AnimationChannel> Channels = new List<AnimationChannel>(); | 
 
 
 
 
 | 26 | } | 
 
 
 
 
 | 27 |  | 
 
 
 
 
 | 28 | #endregion | 
 
 
 
 
 | 29 | #region private class AnimationChannel | 
 
 
 
 
 | 30 |  | 
 
 
 
 
 | 31 | private class AnimationChannel | 
 
 
 
 
 | 32 | { | 
 
 
 
 
 | 33 | public readonly Sampler Sampler; | 
 
 
 
 
 | 34 | public readonly string TargetPath; | 
 
 
 
 
 | 35 |  | 
 
 
 
 
 | 36 | public AnimationChannel(Sampler sampler, string targetPath) | 
 
 
 
 
 | 37 | { | 
 
 
 
 
 | 38 | this.Sampler = sampler; | 
 
 
 
 
 | 39 | this.TargetPath = targetPath; | 
 
 
 
 
 | 40 | } | 
 
 
 
 
 | 41 | } | 
 
 
 
 
 | 42 |  | 
 
 
 
 
 | 43 | #endregion | 
 
 
 
 
 | 44 | #region private class AnimationSource | 
 
 
 
 
 | 45 |  | 
 
 
 
 
 | 46 | private class AnimationSource : Source | 
 
 
 
 
 | 47 | { | 
 
 
 
 
 | 48 | public readonly string[] Parameters; | 
 
 
 
 
 | 49 |  | 
 
 
 
 
 | 50 | public AnimationSource(float[] data, string[] parameters) | 
 
 
 
 
 | 51 | : base(data, parameters.Length) | 
 
 
 
 
 | 52 | { | 
 
 
 
 
 | 53 | this.Parameters = parameters; | 
 
 
 
 
 | 54 | } | 
 
 
 
 
 | 55 |  | 
 
 
 
 
 | 56 | public AnimationSource(string[] data, string[] parameters) | 
 
 
 
 
 | 57 | : base(data, parameters.Length) | 
 
 
 
 
 | 58 | { | 
 
 
 
 
 | 59 | this.Parameters = parameters; | 
 
 
 
 
 | 60 | } | 
 
 
 
 
 | 61 | } | 
 
 
 
 
 | 62 |  | 
 
 
 
 
 | 63 | #endregion | 
 
 
 
 
 | 64 |  | 
 
 
 
 
 | 65 | #region private class WriteVisitor | 
 
 
 
 
 | 66 |  | 
 
 
 
 
 | 67 | private class WriteVisitor : Visitor | 
 
 
 
 
 | 68 | { | 
 
 
 
 
 | 69 | private readonly Dictionary<Entity, string> entities = new Dictionary<Entity, string>(); | 
 
 
 
 
 | 70 | private readonly Dictionary<string, Entity> ids = new Dictionary<string, Entity>(StringComparer.Ordinal); | 
 
 
 
 
 | 71 | private readonly Dictionary<string, Sampler> samplers = new Dictionary<string, Sampler>(StringComparer.Ordinal); | 
 
 
 
 
 | 72 | private readonly Dictionary<string, Source> sources = new Dictionary<string, Source>(StringComparer.Ordinal); | 
 
 
 
 
 | 73 | private int uniqueEntityId = 1; | 
 
 
 
 
 | 74 |  | 
 
 
 
 
 | 75 | public readonly List<Image> Images = new List<Image>(); | 
 
 
 
 
 | 76 | public readonly List<Effect> Effects = new List<Effect>(); | 
 
 
 
 
 | 77 | public readonly List<Material> Materials = new List<Material>(); | 
 
 
 
 
 | 78 | public readonly List<Geometry> Geometries = new List<Geometry>(); | 
 
 
 
 
 | 79 | public readonly List<Scene> Scenes = new List<Scene>(); | 
 
 
 
 
 | 80 | public readonly List<Animation> Animations = new List<Animation>(); | 
 
 
 
 
 | 81 | public readonly List<Camera> Cameras = new List<Camera>(); | 
 
 
 
 
 | 82 |  | 
 
 
 
 
 | 83 | public override void VisitScene(Scene scene) | 
 
 
 
 
 | 84 | { | 
 
 
 
 
 | 85 | AddEntity(scene); | 
 
 
 
 
 | 86 |  | 
 
 
 
 
 | 87 | base.VisitScene(scene); | 
 
 
 
 
 | 88 | } | 
 
 
 
 
 | 89 |  | 
 
 
 
 
 | 90 | public override void VisitNode(Node node) | 
 
 
 
 
 | 91 | { | 
 
 
 
 
 | 92 | EnsureId(node); | 
 
 
 
 
 | 93 |  | 
 
 
 
 
 | 94 | foreach (var transform in node.Transforms.Where(t => t.HasAnimations)) | 
 
 
 
 
 | 95 | { | 
 
 
 
 
 | 96 | for (int i = 0; i < transform.Animations.Length; i++) | 
 
 
 
 
 | 97 | { | 
 
 
 
 
 | 98 | var sampler = transform.Animations[i]; | 
 
 
 
 
 | 99 |  | 
 
 
 
 
 | 100 | if (sampler != null) | 
 
 
 
 
 | 101 | AddAnimationChannel(sampler, node, transform, i); | 
 
 
 
 
 | 102 | } | 
 
 
 
 
 | 103 | } | 
 
 
 
 
 | 104 |  | 
 
 
 
 
 | 105 | base.VisitNode(node); | 
 
 
 
 
 | 106 | } | 
 
 
 
 
 | 107 |  | 
 
 
 
 
 | 108 | public override void VisitGeometry(Geometry geometry) | 
 
 
 
 
 | 109 | { | 
 
 
 
 
 | 110 | AddEntity(geometry); | 
 
 
 
 
 | 111 |  | 
 
 
 
 
 | 112 | string baseId = IdOf(geometry); | 
 
 
 
 
 | 113 |  | 
 
 
 
 
 | 114 | if (baseId.EndsWith("_geometry", StringComparison.Ordinal)) | 
 
 
 
 
 | 115 | baseId = baseId.Substring(0, baseId.Length - "_geometry".Length); | 
 
 
 
 
 | 116 |  | 
 
 
 
 
 | 117 | foreach (var input in geometry.Vertices) | 
 
 
 
 
 | 118 | EnsureId(input.Source, string.Format(CultureInfo.InvariantCulture, "{0}_{1}", baseId, input.Semantic.ToString().ToLowerInvariant())); | 
 
 
 
 
 | 119 |  | 
 
 
 
 
 | 120 | foreach (var input in geometry.Primitives.SelectMany(p => p.Inputs)) | 
 
 
 
 
 | 121 | EnsureId(input.Source, string.Format(CultureInfo.InvariantCulture, "{0}_{1}", baseId, input.Semantic.ToString().ToLowerInvariant())); | 
 
 
 
 
 | 122 |  | 
 
 
 
 
 | 123 | base.VisitGeometry(geometry); | 
 
 
 
 
 | 124 | } | 
 
 
 
 
 | 125 |  | 
 
 
 
 
 | 126 | public override void VisitMaterial(Material material) | 
 
 
 
 
 | 127 | { | 
 
 
 
 
 | 128 | AddEntity(material); | 
 
 
 
 
 | 129 |  | 
 
 
 
 
 | 130 | base.VisitMaterial(material); | 
 
 
 
 
 | 131 | } | 
 
 
 
 
 | 132 |  | 
 
 
 
 
 | 133 | public override void VisitEffect(Effect effect) | 
 
 
 
 
 | 134 | { | 
 
 
 
 
 | 135 | AddEntity(effect); | 
 
 
 
 
 | 136 |  | 
 
 
 
 
 | 137 | base.VisitEffect(effect); | 
 
 
 
 
 | 138 | } | 
 
 
 
 
 | 139 |  | 
 
 
 
 
 | 140 | public override void VisitImage(Image image) | 
 
 
 
 
 | 141 | { | 
 
 
 
 
 | 142 | AddEntity(image); | 
 
 
 
 
 | 143 |  | 
 
 
 
 
 | 144 | base.VisitImage(image); | 
 
 
 
 
 | 145 | } | 
 
 
 
 
 | 146 |  | 
 
 
 
 
 | 147 | public override void VisitCamera(Camera camera) | 
 
 
 
 
 | 148 | { | 
 
 
 
 
 | 149 | AddEntity(camera); | 
 
 
 
 
 | 150 |  | 
 
 
 
 
 | 151 | base.VisitCamera(camera); | 
 
 
 
 
 | 152 | } | 
 
 
 
 
 | 153 |  | 
 
 
 
 
 | 154 | private void AddEntity(Scene scene) | 
 
 
 
 
 | 155 | { | 
 
 
 
 
 | 156 | AddEntity(scene, Scenes); | 
 
 
 
 
 | 157 | } | 
 
 
 
 
 | 158 |  | 
 
 
 
 
 | 159 | private void AddEntity(Image image) | 
 
 
 
 
 | 160 | { | 
 
 
 
 
 | 161 | AddEntity(image, Images); | 
 
 
 
 
 | 162 | } | 
 
 
 
 
 | 163 |  | 
 
 
 
 
 | 164 | private void AddEntity(Effect effect) | 
 
 
 
 
 | 165 | { | 
 
 
 
 
 | 166 | AddEntity(effect, Effects); | 
 
 
 
 
 | 167 | } | 
 
 
 
 
 | 168 |  | 
 
 
 
 
 | 169 | private void AddEntity(Material material) | 
 
 
 
 
 | 170 | { | 
 
 
 
 
 | 171 | AddEntity(material, Materials); | 
 
 
 
 
 | 172 | } | 
 
 
 
 
 | 173 |  | 
 
 
 
 
 | 174 | private void AddEntity(Geometry geometry) | 
 
 
 
 
 | 175 | { | 
 
 
 
 
 | 176 | AddEntity(geometry, Geometries); | 
 
 
 
 
 | 177 | } | 
 
 
 
 
 | 178 |  | 
 
 
 
 
 | 179 | private void AddEntity(Animation animation) | 
 
 
 
 
 | 180 | { | 
 
 
 
 
 | 181 | AddEntity(animation, Animations); | 
 
 
 
 
 | 182 | } | 
 
 
 
 
 | 183 |  | 
 
 
 
 
 | 184 | private void AddEntity(Camera camera) | 
 
 
 
 
 | 185 | { | 
 
 
 
 
 | 186 | AddEntity(camera, Cameras); | 
 
 
 
 
 | 187 | } | 
 
 
 
 
 | 188 |  | 
 
 
 
 
 | 189 | private void AddEntity<T>(T entity, ICollection<T> entityCollection) | 
 
 
 
 
 | 190 | where T : Entity | 
 
 
 
 
 | 191 | { | 
 
 
 
 
 | 192 | if (EnsureId(entity)) | 
 
 
 
 
 | 193 | entityCollection.Add(entity); | 
 
 
 
 
 | 194 | } | 
 
 
 
 
 | 195 |  | 
 
 
 
 
 | 196 | private bool EnsureId(Entity entity) | 
 
 
 
 
 | 197 | { | 
 
 
 
 
 | 198 | if (entities.ContainsKey(entity)) | 
 
 
 
 
 | 199 | return false; | 
 
 
 
 
 | 200 |  | 
 
 
 
 
 | 201 | string name = entity.Name; | 
 
 
 
 
 | 202 | string id; | 
 
 
 
 
 | 203 |  | 
 
 
 
 
 | 204 | if (string.IsNullOrEmpty(name)) | 
 
 
 
 
 | 205 | { | 
 
 
 
 
 | 206 | do | 
 
 
 
 
 | 207 | { | 
 
 
 
 
 | 208 | id = string.Format(CultureInfo.InvariantCulture, | 
 
 
 
 
 | 209 | "unique_{0}", uniqueEntityId++, entity.GetType().Name.ToLowerInvariant()); | 
 
 
 
 
 | 210 | } | 
 
 
 
 
 | 211 | while (ids.ContainsKey(id)); | 
 
 
 
 
 | 212 | } | 
 
 
 
 
 | 213 | else | 
 
 
 
 
 | 214 | { | 
 
 
 
 
 | 215 | if (!ids.ContainsKey(name)) | 
 
 
 
 
 | 216 | { | 
 
 
 
 
 | 217 | id = name; | 
 
 
 
 
 | 218 | } | 
 
 
 
 
 | 219 | else | 
 
 
 
 
 | 220 | { | 
 
 
 
 
 | 221 | id = string.Format(CultureInfo.InvariantCulture, | 
 
 
 
 
 | 222 | "{0}_{1}", name, entity.GetType().Name.ToLowerInvariant()); | 
 
 
 
 
 | 223 |  | 
 
 
 
 
 | 224 | while (ids.ContainsKey(id)) | 
 
 
 
 
 | 225 | { | 
 
 
 
 
 | 226 | id = string.Format(CultureInfo.InvariantCulture, | 
 
 
 
 
 | 227 | "{0}_{1}_{2}", name, uniqueEntityId++, entity.GetType().Name.ToLowerInvariant()); | 
 
 
 
 
 | 228 | } | 
 
 
 
 
 | 229 | } | 
 
 
 
 
 | 230 | } | 
 
 
 
 
 | 231 |  | 
 
 
 
 
 | 232 | entities.Add(entity, id); | 
 
 
 
 
 | 233 | ids.Add(id, entity); | 
 
 
 
 
 | 234 |  | 
 
 
 
 
 | 235 | return true; | 
 
 
 
 
 | 236 | } | 
 
 
 
 
 | 237 |  | 
 
 
 
 
 | 238 | private bool EnsureId(Entity entity, string id) | 
 
 
 
 
 | 239 | { | 
 
 
 
 
 | 240 | if (entities.ContainsKey(entity)) | 
 
 
 
 
 | 241 | return false; | 
 
 
 
 
 | 242 |  | 
 
 
 
 
 | 243 | entities.Add(entity, id); | 
 
 
 
 
 | 244 | ids.Add(id, entity); | 
 
 
 
 
 | 245 |  | 
 
 
 
 
 | 246 | return true; | 
 
 
 
 
 | 247 | } | 
 
 
 
 
 | 248 |  | 
 
 
 
 
 | 249 | public string IdOf(Entity entity) | 
 
 
 
 
 | 250 | { | 
 
 
 
 
 | 251 | string id; | 
 
 
 
 
 | 252 | entities.TryGetValue(entity, out id); | 
 
 
 
 
 | 253 | return id; | 
 
 
 
 
 | 254 | } | 
 
 
 
 
 | 255 |  | 
 
 
 
 
 | 256 | public string UrlOf(Entity entity) | 
 
 
 
 
 | 257 | { | 
 
 
 
 
 | 258 | return string.Format("#{0}", IdOf(entity)); | 
 
 
 
 
 | 259 | } | 
 
 
 
 
 | 260 |  | 
 
 
 
 
 | 261 | private void AddAnimationChannel(Sampler sampler, Node node, Transform transform, int valueIndex) | 
 
 
 
 
 | 262 | { | 
 
 
 
 
 | 263 | Animation animation; | 
 
 
 
 
 | 264 |  | 
 
 
 
 
 | 265 | if (Animations.Count == 0) | 
 
 
 
 
 | 266 | { | 
 
 
 
 
 | 267 | animation = new Animation(); | 
 
 
 
 
 | 268 | Animations.Add(animation); | 
 
 
 
 
 | 269 | } | 
 
 
 
 
 | 270 | else | 
 
 
 
 
 | 271 | { | 
 
 
 
 
 | 272 | animation = Animations[0]; | 
 
 
 
 
 | 273 | } | 
 
 
 
 
 | 274 |  | 
 
 
 
 
 | 275 | EnsureId(sampler); | 
 
 
 
 
 | 276 |  | 
 
 
 
 
 | 277 | string nodeId = IdOf(node); | 
 
 
 
 
 | 278 | string samplerId = IdOf(sampler); | 
 
 
 
 
 | 279 | string valueName = transform.ValueIndexToValueName(valueIndex); | 
 
 
 
 
 | 280 | Sampler valueSampler; | 
 
 
 
 
 | 281 |  | 
 
 
 
 
 | 282 | if (!samplers.TryGetValue(samplerId + valueName, out valueSampler)) | 
 
 
 
 
 | 283 | { | 
 
 
 
 
 | 284 | valueSampler = new Sampler(); | 
 
 
 
 
 | 285 | EnsureId(valueSampler, string.Format("{0}_{1}_{2}", IdOf(node), transform.Sid, valueName)); | 
 
 
 
 
 | 286 | animation.Samplers.Add(valueSampler); | 
 
 
 
 
 | 287 |  | 
 
 
 
 
 | 288 | foreach (var input in sampler.Inputs) | 
 
 
 
 
 | 289 | { | 
 
 
 
 
 | 290 | var source = input.Source; | 
 
 
 
 
 | 291 | EnsureId(source); | 
 
 
 
 
 | 292 |  | 
 
 
 
 
 | 293 | string sourceId = IdOf(source) + (input.Semantic == Semantic.Output ? valueName : ""); | 
 
 
 
 
 | 294 |  | 
 
 
 
 
 | 295 | if (!sources.TryGetValue(sourceId, out source)) | 
 
 
 
 
 | 296 | { | 
 
 
 
 
 | 297 | source = input.Source; | 
 
 
 
 
 | 298 |  | 
 
 
 
 
 | 299 | switch (input.Semantic) | 
 
 
 
 
 | 300 | { | 
 
 
 
 
 | 301 | case Semantic.Input: | 
 
 
 
 
 | 302 | source = new AnimationSource(source.FloatData, new[] { "TIME" }); | 
 
 
 
 
 | 303 | break; | 
 
 
 
 
 | 304 | case Semantic.Output: | 
 
 
 
 
 | 305 | source = new AnimationSource(source.FloatData, new[] { valueName }); | 
 
 
 
 
 | 306 | break; | 
 
 
 
 
 | 307 | case Semantic.Interpolation: | 
 
 
 
 
 | 308 | source = new AnimationSource(source.NameData, new[] { "INTERPOLATION" }); | 
 
 
 
 
 | 309 | break; | 
 
 
 
 
 | 310 | case Semantic.OutTangent: | 
 
 
 
 
 | 311 | case Semantic.InTangent: | 
 
 
 
 
 | 312 | source = new AnimationSource(source.FloatData, new[] { "X", "Y" }); | 
 
 
 
 
 | 313 | break; | 
 
 
 
 
 | 314 | default: | 
 
 
 
 
 | 315 | throw new NotSupportedException(string.Format("Invalid semantic {0} for animation input", input.Semantic)); | 
 
 
 
 
 | 316 | } | 
 
 
 
 
 | 317 |  | 
 
 
 
 
 | 318 | sources.Add(sourceId, source); | 
 
 
 
 
 | 319 |  | 
 
 
 
 
 | 320 | EnsureId(source, string.Format(CultureInfo.InvariantCulture, "{0}_{1}", IdOf(valueSampler), input.Semantic.ToString().ToLowerInvariant())); | 
 
 
 
 
 | 321 | animation.Sources.Add(source); | 
 
 
 
 
 | 322 | } | 
 
 
 
 
 | 323 |  | 
 
 
 
 
 | 324 | valueSampler.Inputs.Add(new Input(input.Semantic, source)); | 
 
 
 
 
 | 325 | } | 
 
 
 
 
 | 326 | } | 
 
 
 
 
 | 327 |  | 
 
 
 
 
 | 328 | animation.Channels.Add(new AnimationChannel( | 
 
 
 
 
 | 329 | valueSampler, | 
 
 
 
 
 | 330 | string.Format(CultureInfo.InvariantCulture, "{0}/{1}.{2}", IdOf(node), transform.Sid, valueName))); | 
 
 
 
 
 | 331 | } | 
 
 
 
 
 | 332 | } | 
 
 
 
 
 | 333 |  | 
 
 
 
 
 | 334 | #endregion | 
 
 
 
 
 | 335 |  | 
 
 
 
 
 | 336 | public static void WriteFile(string filePath, Scene scene) | 
 
 
 
 
 | 337 | { | 
 
 
 
 
 | 338 | var writer = new DaeWriter(); | 
 
 
 
 
 | 339 | writer.visitor = new WriteVisitor(); | 
 
 
 
 
 | 340 | writer.visitor.VisitScene(scene); | 
 
 
 
 
 | 341 | writer.mainScene = scene; | 
 
 
 
 
 | 342 |  | 
 
 
 
 
 | 343 | var settings = new XmlWriterSettings { | 
 
 
 
 
 | 344 | CloseOutput = true, | 
 
 
 
 
 | 345 | ConformanceLevel = ConformanceLevel.Document, | 
 
 
 
 
 | 346 | Encoding = Encoding.UTF8, | 
 
 
 
 
 | 347 | Indent = true, | 
 
 
 
 
 | 348 | IndentChars = "\t" | 
 
 
 
 
 | 349 | }; | 
 
 
 
 
 | 350 |  | 
 
 
 
 
 | 351 | //AxisConverter.Convert(scene, Axis.Y, Axis.Z); | 
 
 
 
 
 | 352 |  | 
 
 
 
 
 | 353 | using (var stream = File.Create(filePath)) | 
 
 
 
 
 | 354 | using (writer.xml = XmlWriter.Create(stream, settings)) | 
 
 
 
 
 | 355 | writer.WriteRoot(); | 
 
 
 
 
 | 356 | } | 
 
 
 
 
 | 357 |  | 
 
 
 
 
 | 358 | private void WriteRoot() | 
 
 
 
 
 | 359 | { | 
 
 
 
 
 | 360 | WriteCollada(); | 
 
 
 
 
 | 361 |  | 
 
 
 
 
 | 362 | WriteLibrary("library_cameras", visitor.Cameras, WriteCamera); | 
 
 
 
 
 | 363 | WriteLibrary("library_images", visitor.Images, WriteImage); | 
 
 
 
 
 | 364 | WriteLibrary("library_effects", visitor.Effects, WriteEffect); | 
 
 
 
 
 | 365 | WriteLibrary("library_materials", visitor.Materials, WriteMaterial); | 
 
 
 
 
 | 366 | WriteLibrary("library_geometries", visitor.Geometries, WriteGeometry); | 
 
 
 
 
 | 367 | WriteLibrary("library_visual_scenes", visitor.Scenes, WriteScene); | 
 
 
 
 
 | 368 | WriteLibrary("library_animations", visitor.Animations, WriteAnimation); | 
 
 
 
 
 | 369 |  | 
 
 
 
 
 | 370 | WriteScene(); | 
 
 
 
 
 | 371 | } | 
 
 
 
 
 | 372 |  | 
 
 
 
 
 | 373 | private void WriteCollada() | 
 
 
 
 
 | 374 | { | 
 
 
 
 
 | 375 | xml.WriteStartDocument(); | 
 
 
 
 
 | 376 | xml.WriteStartElement("COLLADA", "http://www.collada.org/2005/11/COLLADASchema"); | 
 
 
 
 
 | 377 | xml.WriteAttributeString("version", "1.4.0"); | 
 
 
 
 
 | 378 |  | 
 
 
 
 
 | 379 | xml.WriteStartElement("asset"); | 
 
 
 
 
 | 380 | xml.WriteStartElement("contributor"); | 
 
 
 
 
 | 381 | //xml.WriteElementString("author", "OniSplit"); | 
 
 
 
 
 | 382 | xml.WriteElementString("authoring_tool", string.Format(CultureInfo.InvariantCulture, "OniSplit v{0}", Utils.Version)); | 
 
 
 
 
 | 383 | xml.WriteEndElement(); | 
 
 
 
 
 | 384 |  | 
 
 
 
 
 | 385 | xml.WriteStartElement("unit"); | 
 
 
 
 
 | 386 | xml.WriteAttributeString("meter", "0.1"); | 
 
 
 
 
 | 387 | xml.WriteAttributeString("name", "decimeter"); | 
 
 
 
 
 | 388 | xml.WriteEndElement(); | 
 
 
 
 
 | 389 | xml.WriteElementString("up_axis", "Y_UP"); | 
 
 
 
 
 | 390 | xml.WriteEndElement(); | 
 
 
 
 
 | 391 | } | 
 
 
 
 
 | 392 |  | 
 
 
 
 
 | 393 | private void WriteLibrary<T>(string name, ICollection<T> library, Action<T> entityWriter) | 
 
 
 
 
 | 394 | { | 
 
 
 
 
 | 395 | if (library.Count == 0) | 
 
 
 
 
 | 396 | return; | 
 
 
 
 
 | 397 |  | 
 
 
 
 
 | 398 | xml.WriteStartElement(name); | 
 
 
 
 
 | 399 |  | 
 
 
 
 
 | 400 | foreach (T entity in library) | 
 
 
 
 
 | 401 | entityWriter(entity); | 
 
 
 
 
 | 402 |  | 
 
 
 
 
 | 403 | xml.WriteEndElement(); | 
 
 
 
 
 | 404 | } | 
 
 
 
 
 | 405 |  | 
 
 
 
 
 | 406 | private void WriteScene() | 
 
 
 
 
 | 407 | { | 
 
 
 
 
 | 408 | xml.WriteStartElement("scene"); | 
 
 
 
 
 | 409 | xml.WriteStartElement("instance_visual_scene"); | 
 
 
 
 
 | 410 | xml.WriteAttributeString("url", visitor.UrlOf(mainScene)); | 
 
 
 
 
 | 411 | xml.WriteEndElement(); | 
 
 
 
 
 | 412 | xml.WriteEndElement(); | 
 
 
 
 
 | 413 | } | 
 
 
 
 
 | 414 |  | 
 
 
 
 
 | 415 | private void WriteImage(Image image) | 
 
 
 
 
 | 416 | { | 
 
 
 
 
 | 417 | BeginEntity("image", image); | 
 
 
 
 
 | 418 |  | 
 
 
 
 
 | 419 | string imageUrl; | 
 
 
 
 
 | 420 |  | 
 
 
 
 
 | 421 | if (Path.IsPathRooted(image.FilePath)) | 
 
 
 
 
 | 422 | imageUrl = "file:///" + image.FilePath.Replace('\\', '/'); | 
 
 
 
 
 | 423 | else | 
 
 
 
 
 | 424 | imageUrl = image.FilePath.Replace('\\', '/'); | 
 
 
 
 
 | 425 |  | 
 
 
 
 
 | 426 | xml.WriteElementString("init_from", imageUrl); | 
 
 
 
 
 | 427 |  | 
 
 
 
 
 | 428 | EndEntity(); | 
 
 
 
 
 | 429 | } | 
 
 
 
 
 | 430 |  | 
 
 
 
 
 | 431 | private void WriteEffect(Effect effect) | 
 
 
 
 
 | 432 | { | 
 
 
 
 
 | 433 | BeginEntity("effect", effect); | 
 
 
 
 
 | 434 | WriteEffectCommonProfile(effect); | 
 
 
 
 
 | 435 | EndEntity(); | 
 
 
 
 
 | 436 | } | 
 
 
 
 
 | 437 |  | 
 
 
 
 
 | 438 | private void WriteEffectCommonProfile(Effect effect) | 
 
 
 
 
 | 439 | { | 
 
 
 
 
 | 440 | xml.WriteStartElement("profile_COMMON"); | 
 
 
 
 
 | 441 |  | 
 
 
 
 
 | 442 | foreach (var parameter in effect.Parameters) | 
 
 
 
 
 | 443 | WriteEffectParameter(parameter); | 
 
 
 
 
 | 444 |  | 
 
 
 
 
 | 445 | WriteEffectTechnique(effect); | 
 
 
 
 
 | 446 |  | 
 
 
 
 
 | 447 | xml.WriteEndElement(); | 
 
 
 
 
 | 448 | } | 
 
 
 
 
 | 449 |  | 
 
 
 
 
 | 450 | private void WriteEffectParameter(EffectParameter parameter) | 
 
 
 
 
 | 451 | { | 
 
 
 
 
 | 452 | xml.WriteStartElement("newparam"); | 
 
 
 
 
 | 453 | xml.WriteAttributeString("sid", parameter.Sid); | 
 
 
 
 
 | 454 |  | 
 
 
 
 
 | 455 | if (!string.IsNullOrEmpty(parameter.Semantic)) | 
 
 
 
 
 | 456 | { | 
 
 
 
 
 | 457 | xml.WriteStartElement("semantic"); | 
 
 
 
 
 | 458 | xml.WriteString(parameter.Semantic); | 
 
 
 
 
 | 459 | xml.WriteEndElement(); | 
 
 
 
 
 | 460 | } | 
 
 
 
 
 | 461 |  | 
 
 
 
 
 | 462 | if (parameter.Value is float) | 
 
 
 
 
 | 463 | { | 
 
 
 
 
 | 464 | float value = (float)parameter.Value; | 
 
 
 
 
 | 465 | xml.WriteElementString("float", | 
 
 
 
 
 | 466 | XmlConvert.ToString(value)); | 
 
 
 
 
 | 467 | } | 
 
 
 
 
 | 468 | else if (parameter.Value is Vector2) | 
 
 
 
 
 | 469 | { | 
 
 
 
 
 | 470 | var value = (Vector2)parameter.Value; | 
 
 
 
 
 | 471 | xml.WriteElementString("float2", string.Format("{0} {1}", | 
 
 
 
 
 | 472 | XmlConvert.ToString(value.X), XmlConvert.ToString(value.Y))); | 
 
 
 
 
 | 473 | } | 
 
 
 
 
 | 474 | else if (parameter.Value is Vector3) | 
 
 
 
 
 | 475 | { | 
 
 
 
 
 | 476 | var value = (Vector3)parameter.Value; | 
 
 
 
 
 | 477 | xml.WriteElementString("float3", string.Format("{0} {1} {3}", | 
 
 
 
 
 | 478 | XmlConvert.ToString(value.X), XmlConvert.ToString(value.Y), XmlConvert.ToString(value.Z))); | 
 
 
 
 
 | 479 | } | 
 
 
 
 
 | 480 | else if (parameter.Value is EffectSurface) | 
 
 
 
 
 | 481 | { | 
 
 
 
 
 | 482 | var surface = (EffectSurface)parameter.Value; | 
 
 
 
 
 | 483 | xml.WriteStartElement("surface"); | 
 
 
 
 
 | 484 | xml.WriteAttributeString("type", "2D"); | 
 
 
 
 
 | 485 | xml.WriteElementString("init_from", visitor.IdOf(surface.InitFrom)); | 
 
 
 
 
 | 486 | xml.WriteEndElement(); | 
 
 
 
 
 | 487 | } | 
 
 
 
 
 | 488 | else if (parameter.Value is EffectSampler) | 
 
 
 
 
 | 489 | { | 
 
 
 
 
 | 490 | var sampler = (EffectSampler)parameter.Value; | 
 
 
 
 
 | 491 | xml.WriteStartElement("sampler2D"); | 
 
 
 
 
 | 492 | xml.WriteStartElement("source"); | 
 
 
 
 
 | 493 | xml.WriteString(sampler.Surface.DeclaringParameter.Sid); | 
 
 
 
 
 | 494 | xml.WriteEndElement(); | 
 
 
 
 
 | 495 |  | 
 
 
 
 
 | 496 | if (sampler.MinFilter != EffectSamplerFilter.None) | 
 
 
 
 
 | 497 | xml.WriteElementString("minfilter", sampler.MinFilter.ToString().ToUpperInvariant()); | 
 
 
 
 
 | 498 |  | 
 
 
 
 
 | 499 | if (sampler.MagFilter != EffectSamplerFilter.None) | 
 
 
 
 
 | 500 | xml.WriteElementString("magfilter", sampler.MagFilter.ToString().ToUpperInvariant()); | 
 
 
 
 
 | 501 |  | 
 
 
 
 
 | 502 | if (sampler.MipFilter != EffectSamplerFilter.None) | 
 
 
 
 
 | 503 | xml.WriteElementString("mipfilter", sampler.MipFilter.ToString().ToUpperInvariant()); | 
 
 
 
 
 | 504 |  | 
 
 
 
 
 | 505 | xml.WriteEndElement(); | 
 
 
 
 
 | 506 | } | 
 
 
 
 
 | 507 |  | 
 
 
 
 
 | 508 | xml.WriteEndElement(); | 
 
 
 
 
 | 509 | } | 
 
 
 
 
 | 510 |  | 
 
 
 
 
 | 511 | private void WriteEffectTechnique(Effect effect) | 
 
 
 
 
 | 512 | { | 
 
 
 
 
 | 513 | xml.WriteStartElement("technique"); | 
 
 
 
 
 | 514 | xml.WriteStartElement("phong"); | 
 
 
 
 
 | 515 |  | 
 
 
 
 
 | 516 | //WriteEffectTechniqueProperty("emission", effect.Emission); | 
 
 
 
 
 | 517 |  | 
 
 
 
 
 | 518 | WriteEffectTechniqueProperty("ambient", effect.Ambient); | 
 
 
 
 
 | 519 | WriteEffectTechniqueProperty("diffuse", effect.Diffuse); | 
 
 
 
 
 | 520 | WriteEffectTechniqueProperty("specular", effect.Specular); | 
 
 
 
 
 | 521 |  | 
 
 
 
 
 | 522 | //WriteEffectTechniqueProperty("shininess", effect.Shininess); | 
 
 
 
 
 | 523 | //WriteEffectTechniqueProperty("reflective", effect.Reflective); | 
 
 
 
 
 | 524 | //WriteEffectTechniqueProperty("reflectivity", effect.Reflectivity); | 
 
 
 
 
 | 525 |  | 
 
 
 
 
 | 526 | WriteEffectTechniqueProperty("transparent", effect.Transparent); | 
 
 
 
 
 | 527 |  | 
 
 
 
 
 | 528 | //WriteEffectTechniqueProperty("transparency", effect.Transparency); | 
 
 
 
 
 | 529 | //WriteEffectTechniqueProperty("index_of_refraction", effect.IndexOfRefraction); | 
 
 
 
 
 | 530 |  | 
 
 
 
 
 | 531 | xml.WriteEndElement(); | 
 
 
 
 
 | 532 | xml.WriteEndElement(); | 
 
 
 
 
 | 533 | } | 
 
 
 
 
 | 534 |  | 
 
 
 
 
 | 535 | private void WriteEffectTechniqueProperty(string name, EffectParameter value) | 
 
 
 
 
 | 536 | { | 
 
 
 
 
 | 537 | bool isTransparent = name == "transparent"; | 
 
 
 
 
 | 538 |  | 
 
 
 
 
 | 539 | if (isTransparent && value.Value == null) | 
 
 
 
 
 | 540 | return; | 
 
 
 
 
 | 541 |  | 
 
 
 
 
 | 542 | xml.WriteStartElement(name); | 
 
 
 
 
 | 543 |  | 
 
 
 
 
 | 544 | if (isTransparent) | 
 
 
 
 
 | 545 | xml.WriteAttributeString("opaque", "A_ONE"); | 
 
 
 
 
 | 546 |  | 
 
 
 
 
 | 547 | if (value.Reference != null) | 
 
 
 
 
 | 548 | { | 
 
 
 
 
 | 549 | xml.WriteStartElement("param"); | 
 
 
 
 
 | 550 | xml.WriteString(value.Reference); | 
 
 
 
 
 | 551 | xml.WriteEndElement(); | 
 
 
 
 
 | 552 | } | 
 
 
 
 
 | 553 | else if (value.Value is float) | 
 
 
 
 
 | 554 | { | 
 
 
 
 
 | 555 | float flt = (float)value.Value; | 
 
 
 
 
 | 556 | xml.WriteStartElement("float"); | 
 
 
 
 
 | 557 | xml.WriteAttributeString("sid", value.Sid); | 
 
 
 
 
 | 558 | xml.WriteString(XmlConvert.ToString(flt)); | 
 
 
 
 
 | 559 | xml.WriteEndElement(); | 
 
 
 
 
 | 560 | } | 
 
 
 
 
 | 561 | else if (value.Value is Vector4) | 
 
 
 
 
 | 562 | { | 
 
 
 
 
 | 563 | var color = (Vector4)value.Value; | 
 
 
 
 
 | 564 | xml.WriteStartElement("color"); | 
 
 
 
 
 | 565 | xml.WriteAttributeString("sid", value.Sid); | 
 
 
 
 
 | 566 | xml.WriteString(string.Format(CultureInfo.InvariantCulture, "{0} {1} {2} {3}", | 
 
 
 
 
 | 567 | XmlConvert.ToString(color.X), XmlConvert.ToString(color.Y), XmlConvert.ToString(color.Z), XmlConvert.ToString(color.W))); | 
 
 
 
 
 | 568 | xml.WriteEndElement(); | 
 
 
 
 
 | 569 | } | 
 
 
 
 
 | 570 | else if (value.Value is EffectTexture) | 
 
 
 
 
 | 571 | { | 
 
 
 
 
 | 572 | var texture = (EffectTexture)value.Value; | 
 
 
 
 
 | 573 | xml.WriteStartElement("texture"); | 
 
 
 
 
 | 574 | xml.WriteAttributeString("texture", texture.Sampler.Owner.Sid); | 
 
 
 
 
 | 575 | xml.WriteAttributeString("texcoord", texture.TexCoordSemantic); | 
 
 
 
 
 | 576 | xml.WriteEndElement(); | 
 
 
 
 
 | 577 | } | 
 
 
 
 
 | 578 |  | 
 
 
 
 
 | 579 | xml.WriteEndElement(); | 
 
 
 
 
 | 580 | } | 
 
 
 
 
 | 581 |  | 
 
 
 
 
 | 582 | private void WriteMaterial(Material matrial) | 
 
 
 
 
 | 583 | { | 
 
 
 
 
 | 584 | BeginEntity("material", matrial); | 
 
 
 
 
 | 585 |  | 
 
 
 
 
 | 586 | xml.WriteStartElement("instance_effect"); | 
 
 
 
 
 | 587 | xml.WriteAttributeString("url", visitor.UrlOf(matrial.Effect)); | 
 
 
 
 
 | 588 | xml.WriteEndElement(); | 
 
 
 
 
 | 589 |  | 
 
 
 
 
 | 590 | EndEntity(); | 
 
 
 
 
 | 591 | } | 
 
 
 
 
 | 592 |  | 
 
 
 
 
 | 593 | private void WriteGeometry(Geometry geometry) | 
 
 
 
 
 | 594 | { | 
 
 
 
 
 | 595 | BeginEntity("geometry", geometry); | 
 
 
 
 
 | 596 |  | 
 
 
 
 
 | 597 | xml.WriteStartElement("mesh"); | 
 
 
 
 
 | 598 |  | 
 
 
 
 
 | 599 | WriteGeometrySources(geometry); | 
 
 
 
 
 | 600 | WriteGeometryVertices(geometry); | 
 
 
 
 
 | 601 |  | 
 
 
 
 
 | 602 | foreach (var primitives in geometry.Primitives) | 
 
 
 
 
 | 603 | WriteGeometryPrimitives(geometry, primitives); | 
 
 
 
 
 | 604 |  | 
 
 
 
 
 | 605 | xml.WriteEndElement(); | 
 
 
 
 
 | 606 |  | 
 
 
 
 
 | 607 | EndEntity(); | 
 
 
 
 
 | 608 | } | 
 
 
 
 
 | 609 |  | 
 
 
 
 
 | 610 | private void WriteGeometrySources(Geometry geometry) | 
 
 
 
 
 | 611 | { | 
 
 
 
 
 | 612 | var sources = new Dictionary<Source, List<Semantic>>(); | 
 
 
 
 
 | 613 |  | 
 
 
 
 
 | 614 | foreach (var primitives in geometry.Primitives) | 
 
 
 
 
 | 615 | { | 
 
 
 
 
 | 616 | foreach (var input in primitives.Inputs) | 
 
 
 
 
 | 617 | { | 
 
 
 
 
 | 618 | List<Semantic> uses; | 
 
 
 
 
 | 619 |  | 
 
 
 
 
 | 620 | if (!sources.TryGetValue(input.Source, out uses)) | 
 
 
 
 
 | 621 | { | 
 
 
 
 
 | 622 | uses = new List<Semantic>(); | 
 
 
 
 
 | 623 | sources.Add(input.Source, uses); | 
 
 
 
 
 | 624 | } | 
 
 
 
 
 | 625 |  | 
 
 
 
 
 | 626 | if (!uses.Contains(input.Semantic)) | 
 
 
 
 
 | 627 | uses.Add(input.Semantic); | 
 
 
 
 
 | 628 | } | 
 
 
 
 
 | 629 | } | 
 
 
 
 
 | 630 |  | 
 
 
 
 
 | 631 | foreach (var pair in sources) | 
 
 
 
 
 | 632 | { | 
 
 
 
 
 | 633 | foreach (var semantic in pair.Value) | 
 
 
 
 
 | 634 | WriteSource(pair.Key, semantic); | 
 
 
 
 
 | 635 | } | 
 
 
 
 
 | 636 | } | 
 
 
 
 
 | 637 |  | 
 
 
 
 
 | 638 | private void WriteGeometryVertices(Geometry geometry) | 
 
 
 
 
 | 639 | { | 
 
 
 
 
 | 640 | string baseId = visitor.IdOf(geometry); | 
 
 
 
 
 | 641 |  | 
 
 
 
 
 | 642 | if (baseId.EndsWith("_geometry", StringComparison.Ordinal)) | 
 
 
 
 
 | 643 | baseId = baseId.Substring(0, baseId.Length - "_geometry".Length); | 
 
 
 
 
 | 644 |  | 
 
 
 
 
 | 645 | xml.WriteStartElement("vertices"); | 
 
 
 
 
 | 646 | xml.WriteAttributeString("id", baseId + "_vertices"); | 
 
 
 
 
 | 647 |  | 
 
 
 
 
 | 648 | foreach (var input in geometry.Vertices) | 
 
 
 
 
 | 649 | { | 
 
 
 
 
 | 650 | xml.WriteStartElement("input"); | 
 
 
 
 
 | 651 | WriteSemanticAttribute("semantic", input.Semantic); | 
 
 
 
 
 | 652 | xml.WriteAttributeString("source", visitor.UrlOf(input.Source)); | 
 
 
 
 
 | 653 | xml.WriteEndElement(); | 
 
 
 
 
 | 654 | } | 
 
 
 
 
 | 655 |  | 
 
 
 
 
 | 656 | xml.WriteEndElement(); | 
 
 
 
 
 | 657 | } | 
 
 
 
 
 | 658 |  | 
 
 
 
 
 | 659 | private void WriteGeometryPrimitives(Geometry geometry, MeshPrimitives primitives) | 
 
 
 
 
 | 660 | { | 
 
 
 
 
 | 661 | switch (primitives.PrimitiveType) | 
 
 
 
 
 | 662 | { | 
 
 
 
 
 | 663 | case MeshPrimitiveType.Lines: | 
 
 
 
 
 | 664 | case MeshPrimitiveType.LineStrips: | 
 
 
 
 
 | 665 | case MeshPrimitiveType.TriangleFans: | 
 
 
 
 
 | 666 | case MeshPrimitiveType.TriangleStrips: | 
 
 
 
 
 | 667 | throw new NotSupportedException(string.Format("Writing {0} is not supported", primitives.PrimitiveType)); | 
 
 
 
 
 | 668 | } | 
 
 
 
 
 | 669 |  | 
 
 
 
 
 | 670 | bool trianglesOnly = !primitives.VertexCounts.Exists(x => x != 3); | 
 
 
 
 
 | 671 |  | 
 
 
 
 
 | 672 | if (!trianglesOnly) | 
 
 
 
 
 | 673 | xml.WriteStartElement("polylist"); | 
 
 
 
 
 | 674 | else | 
 
 
 
 
 | 675 | xml.WriteStartElement("triangles"); | 
 
 
 
 
 | 676 |  | 
 
 
 
 
 | 677 | xml.WriteAttributeString("count", XmlConvert.ToString(primitives.VertexCounts.Count)); | 
 
 
 
 
 | 678 |  | 
 
 
 
 
 | 679 | if (!string.IsNullOrEmpty(primitives.MaterialSymbol)) | 
 
 
 
 
 | 680 | xml.WriteAttributeString("material", primitives.MaterialSymbol); | 
 
 
 
 
 | 681 |  | 
 
 
 
 
 | 682 | int offset = 0; | 
 
 
 
 
 | 683 | bool vertexInputWritten = false; | 
 
 
 
 
 | 684 |  | 
 
 
 
 
 | 685 | var inputs = new List<IndexedInput>(); | 
 
 
 
 
 | 686 |  | 
 
 
 
 
 | 687 | string baseUrl = visitor.UrlOf(geometry); | 
 
 
 
 
 | 688 |  | 
 
 
 
 
 | 689 | if (baseUrl.EndsWith("_geometry", StringComparison.Ordinal)) | 
 
 
 
 
 | 690 | baseUrl = baseUrl.Substring(0, baseUrl.Length - "_geometry".Length); | 
 
 
 
 
 | 691 |  | 
 
 
 
 
 | 692 | foreach (var input in primitives.Inputs) | 
 
 
 
 
 | 693 | { | 
 
 
 
 
 | 694 | if (geometry.Vertices.Any(x => x.Source == input.Source)) | 
 
 
 
 
 | 695 | { | 
 
 
 
 
 | 696 | if (!vertexInputWritten) | 
 
 
 
 
 | 697 | { | 
 
 
 
 
 | 698 | inputs.Add(input); | 
 
 
 
 
 | 699 |  | 
 
 
 
 
 | 700 | xml.WriteStartElement("input"); | 
 
 
 
 
 | 701 | xml.WriteAttributeString("semantic", "VERTEX"); | 
 
 
 
 
 | 702 | xml.WriteAttributeString("source", baseUrl + "_vertices"); | 
 
 
 
 
 | 703 | xml.WriteAttributeString("offset", XmlConvert.ToString(offset++)); | 
 
 
 
 
 | 704 | xml.WriteEndElement(); | 
 
 
 
 
 | 705 | } | 
 
 
 
 
 | 706 |  | 
 
 
 
 
 | 707 | vertexInputWritten = true; | 
 
 
 
 
 | 708 | } | 
 
 
 
 
 | 709 | else | 
 
 
 
 
 | 710 | { | 
 
 
 
 
 | 711 | inputs.Add(input); | 
 
 
 
 
 | 712 |  | 
 
 
 
 
 | 713 | xml.WriteStartElement("input"); | 
 
 
 
 
 | 714 | WriteSemanticAttribute("semantic", input.Semantic); | 
 
 
 
 
 | 715 | xml.WriteAttributeString("source", visitor.UrlOf(input.Source)); | 
 
 
 
 
 | 716 | xml.WriteAttributeString("offset", XmlConvert.ToString(offset++)); | 
 
 
 
 
 | 717 |  | 
 
 
 
 
 | 718 | if (input.Set != 0) | 
 
 
 
 
 | 719 | xml.WriteAttributeString("set", XmlConvert.ToString(input.Set)); | 
 
 
 
 
 | 720 |  | 
 
 
 
 
 | 721 | xml.WriteEndElement(); | 
 
 
 
 
 | 722 | } | 
 
 
 
 
 | 723 | } | 
 
 
 
 
 | 724 |  | 
 
 
 
 
 | 725 | if (!trianglesOnly) | 
 
 
 
 
 | 726 | { | 
 
 
 
 
 | 727 | xml.WriteStartElement("vcount"); | 
 
 
 
 
 | 728 | xml.WriteWhitespace("\n"); | 
 
 
 
 
 | 729 |  | 
 
 
 
 
 | 730 | int vertexCount = 0; | 
 
 
 
 
 | 731 | int c = 0; | 
 
 
 
 
 | 732 |  | 
 
 
 
 
 | 733 | foreach (int i in primitives.VertexCounts) | 
 
 
 
 
 | 734 | { | 
 
 
 
 
 | 735 | xml.WriteString(XmlConvert.ToString(i) + " "); | 
 
 
 
 
 | 736 | vertexCount += i; | 
 
 
 
 
 | 737 |  | 
 
 
 
 
 | 738 | c++; | 
 
 
 
 
 | 739 |  | 
 
 
 
 
 | 740 | if (c == 32) | 
 
 
 
 
 | 741 | { | 
 
 
 
 
 | 742 | xml.WriteWhitespace("\n"); | 
 
 
 
 
 | 743 | c = 0; | 
 
 
 
 
 | 744 | } | 
 
 
 
 
 | 745 | } | 
 
 
 
 
 | 746 |  | 
 
 
 
 
 | 747 | xml.WriteEndElement(); | 
 
 
 
 
 | 748 | } | 
 
 
 
 
 | 749 |  | 
 
 
 
 
 | 750 | xml.WriteStartElement("p"); | 
 
 
 
 
 | 751 | xml.WriteWhitespace("\n"); | 
 
 
 
 
 | 752 |  | 
 
 
 
 
 | 753 | int polygonStartIndex = 0; | 
 
 
 
 
 | 754 |  | 
 
 
 
 
 | 755 | foreach (int vertexCount in primitives.VertexCounts) | 
 
 
 
 
 | 756 | { | 
 
 
 
 
 | 757 | for (int index = 0; index < vertexCount; index++) | 
 
 
 
 
 | 758 | { | 
 
 
 
 
 | 759 | foreach (var input in inputs) | 
 
 
 
 
 | 760 | { | 
 
 
 
 
 | 761 | xml.WriteString(XmlConvert.ToString(input.Indices[polygonStartIndex + index])); | 
 
 
 
 
 | 762 |  | 
 
 
 
 
 | 763 | if (input != inputs.Last() || index != vertexCount - 1) | 
 
 
 
 
 | 764 | xml.WriteWhitespace(" "); | 
 
 
 
 
 | 765 | } | 
 
 
 
 
 | 766 | } | 
 
 
 
 
 | 767 |  | 
 
 
 
 
 | 768 | xml.WriteWhitespace("\n"); | 
 
 
 
 
 | 769 | polygonStartIndex += vertexCount; | 
 
 
 
 
 | 770 | } | 
 
 
 
 
 | 771 |  | 
 
 
 
 
 | 772 | xml.WriteEndElement(); | 
 
 
 
 
 | 773 |  | 
 
 
 
 
 | 774 | xml.WriteEndElement(); | 
 
 
 
 
 | 775 | } | 
 
 
 
 
 | 776 |  | 
 
 
 
 
 | 777 | private void WriteScene(Scene scene) | 
 
 
 
 
 | 778 | { | 
 
 
 
 
 | 779 | BeginEntity("visual_scene", scene); | 
 
 
 
 
 | 780 |  | 
 
 
 
 
 | 781 | foreach (var node in scene.Nodes) | 
 
 
 
 
 | 782 | WriteSceneNode(node); | 
 
 
 
 
 | 783 |  | 
 
 
 
 
 | 784 | EndEntity(); | 
 
 
 
 
 | 785 | } | 
 
 
 
 
 | 786 |  | 
 
 
 
 
 | 787 | private void WriteSceneNode(Node node) | 
 
 
 
 
 | 788 | { | 
 
 
 
 
 | 789 | BeginEntity("node", node); | 
 
 
 
 
 | 790 |  | 
 
 
 
 
 | 791 | foreach (var transform in node.Transforms) | 
 
 
 
 
 | 792 | WriteNodeTransform(transform); | 
 
 
 
 
 | 793 |  | 
 
 
 
 
 | 794 | foreach (var instance in node.Instances) | 
 
 
 
 
 | 795 | { | 
 
 
 
 
 | 796 | if (instance is GeometryInstance) | 
 
 
 
 
 | 797 | WriteGeometryInstance((GeometryInstance)instance); | 
 
 
 
 
 | 798 | else if (instance is CameraInstance) | 
 
 
 
 
 | 799 | WriteCameraInstance((CameraInstance)instance); | 
 
 
 
 
 | 800 | } | 
 
 
 
 
 | 801 |  | 
 
 
 
 
 | 802 | foreach (var child in node.Nodes) | 
 
 
 
 
 | 803 | WriteSceneNode(child); | 
 
 
 
 
 | 804 |  | 
 
 
 
 
 | 805 | EndEntity(); | 
 
 
 
 
 | 806 | } | 
 
 
 
 
 | 807 |  | 
 
 
 
 
 | 808 | private void WriteNodeTransform(Transform transform) | 
 
 
 
 
 | 809 | { | 
 
 
 
 
 | 810 | string type; | 
 
 
 
 
 | 811 |  | 
 
 
 
 
 | 812 | if (transform is TransformTranslate) | 
 
 
 
 
 | 813 | type = "translate"; | 
 
 
 
 
 | 814 | else if (transform is TransformRotate) | 
 
 
 
 
 | 815 | type = "rotate"; | 
 
 
 
 
 | 816 | else if (transform is TransformScale) | 
 
 
 
 
 | 817 | type = "scale"; | 
 
 
 
 
 | 818 | else | 
 
 
 
 
 | 819 | type = "matrix"; | 
 
 
 
 
 | 820 |  | 
 
 
 
 
 | 821 | xml.WriteStartElement(type); | 
 
 
 
 
 | 822 |  | 
 
 
 
 
 | 823 | if (!string.IsNullOrEmpty(transform.Sid)) | 
 
 
 
 
 | 824 | xml.WriteAttributeString("sid", transform.Sid); | 
 
 
 
 
 | 825 |  | 
 
 
 
 
 | 826 | var values = new StringBuilder(transform.Values.Length * 16); | 
 
 
 
 
 | 827 |  | 
 
 
 
 
 | 828 | foreach (float value in transform.Values) | 
 
 
 
 
 | 829 | values.AppendFormat(CultureInfo.InvariantCulture, "{0:f6} ", value); | 
 
 
 
 
 | 830 |  | 
 
 
 
 
 | 831 | if (values.Length > 0) | 
 
 
 
 
 | 832 | values.Length--; | 
 
 
 
 
 | 833 |  | 
 
 
 
 
 | 834 | xml.WriteValue(values.ToString()); | 
 
 
 
 
 | 835 |  | 
 
 
 
 
 | 836 | xml.WriteEndElement(); | 
 
 
 
 
 | 837 | } | 
 
 
 
 
 | 838 |  | 
 
 
 
 
 | 839 | private void WriteCameraInstance(CameraInstance instance) | 
 
 
 
 
 | 840 | { | 
 
 
 
 
 | 841 | xml.WriteStartElement("instance_camera"); | 
 
 
 
 
 | 842 | xml.WriteAttributeString("url", visitor.UrlOf(instance.Target)); | 
 
 
 
 
 | 843 | xml.WriteEndElement(); | 
 
 
 
 
 | 844 | } | 
 
 
 
 
 | 845 |  | 
 
 
 
 
 | 846 | private void WriteGeometryInstance(GeometryInstance instance) | 
 
 
 
 
 | 847 | { | 
 
 
 
 
 | 848 | xml.WriteStartElement("instance_geometry"); | 
 
 
 
 
 | 849 | xml.WriteAttributeString("url", visitor.UrlOf(instance.Target)); | 
 
 
 
 
 | 850 |  | 
 
 
 
 
 | 851 | if (instance.Materials.Count > 0) | 
 
 
 
 
 | 852 | { | 
 
 
 
 
 | 853 | xml.WriteStartElement("bind_material"); | 
 
 
 
 
 | 854 | xml.WriteStartElement("technique_common"); | 
 
 
 
 
 | 855 |  | 
 
 
 
 
 | 856 | foreach (var matInstance in instance.Materials) | 
 
 
 
 
 | 857 | WriteMaterialInstance(matInstance); | 
 
 
 
 
 | 858 |  | 
 
 
 
 
 | 859 | xml.WriteEndElement(); | 
 
 
 
 
 | 860 | xml.WriteEndElement(); | 
 
 
 
 
 | 861 | } | 
 
 
 
 
 | 862 |  | 
 
 
 
 
 | 863 | xml.WriteEndElement(); | 
 
 
 
 
 | 864 | } | 
 
 
 
 
 | 865 |  | 
 
 
 
 
 | 866 | private void WriteMaterialInstance(MaterialInstance matInstance) | 
 
 
 
 
 | 867 | { | 
 
 
 
 
 | 868 | xml.WriteStartElement("instance_material"); | 
 
 
 
 
 | 869 | xml.WriteAttributeString("symbol", matInstance.Symbol); | 
 
 
 
 
 | 870 | xml.WriteAttributeString("target", visitor.UrlOf(matInstance.Target)); | 
 
 
 
 
 | 871 |  | 
 
 
 
 
 | 872 | foreach (var binding in matInstance.Bindings) | 
 
 
 
 
 | 873 | { | 
 
 
 
 
 | 874 | xml.WriteStartElement("bind_vertex_input"); | 
 
 
 
 
 | 875 | xml.WriteAttributeString("semantic", binding.Semantic); | 
 
 
 
 
 | 876 | WriteSemanticAttribute("input_semantic", binding.VertexInput.Semantic); | 
 
 
 
 
 | 877 | xml.WriteAttributeString("input_set", XmlConvert.ToString(binding.VertexInput.Set)); | 
 
 
 
 
 | 878 | xml.WriteEndElement(); | 
 
 
 
 
 | 879 | } | 
 
 
 
 
 | 880 |  | 
 
 
 
 
 | 881 | xml.WriteEndElement(); | 
 
 
 
 
 | 882 | } | 
 
 
 
 
 | 883 |  | 
 
 
 
 
 | 884 | private void WriteAnimation(Animation animation) | 
 
 
 
 
 | 885 | { | 
 
 
 
 
 | 886 | BeginEntity("animation", animation); | 
 
 
 
 
 | 887 |  | 
 
 
 
 
 | 888 | foreach (var source in animation.Sources) | 
 
 
 
 
 | 889 | WriteSource(source, Semantic.None); | 
 
 
 
 
 | 890 |  | 
 
 
 
 
 | 891 | foreach (var sampler in animation.Samplers) | 
 
 
 
 
 | 892 | WriteAnimationSampler(animation, sampler); | 
 
 
 
 
 | 893 |  | 
 
 
 
 
 | 894 | foreach (var channel in animation.Channels) | 
 
 
 
 
 | 895 | WriteAnimationChannel(channel); | 
 
 
 
 
 | 896 |  | 
 
 
 
 
 | 897 | EndEntity(); | 
 
 
 
 
 | 898 | } | 
 
 
 
 
 | 899 |  | 
 
 
 
 
 | 900 | private void WriteAnimationSampler(Animation animation, Sampler sampler) | 
 
 
 
 
 | 901 | { | 
 
 
 
 
 | 902 | xml.WriteStartElement("sampler"); | 
 
 
 
 
 | 903 | xml.WriteAttributeString("id", visitor.IdOf(sampler)); | 
 
 
 
 
 | 904 |  | 
 
 
 
 
 | 905 | foreach (var input in sampler.Inputs) | 
 
 
 
 
 | 906 | { | 
 
 
 
 
 | 907 | xml.WriteStartElement("input"); | 
 
 
 
 
 | 908 | WriteSemanticAttribute("semantic", input.Semantic); | 
 
 
 
 
 | 909 | xml.WriteAttributeString("source", visitor.UrlOf(input.Source)); | 
 
 
 
 
 | 910 | xml.WriteEndElement(); | 
 
 
 
 
 | 911 | } | 
 
 
 
 
 | 912 |  | 
 
 
 
 
 | 913 | xml.WriteEndElement(); | 
 
 
 
 
 | 914 | } | 
 
 
 
 
 | 915 |  | 
 
 
 
 
 | 916 | private void WriteAnimationChannel(AnimationChannel channel) | 
 
 
 
 
 | 917 | { | 
 
 
 
 
 | 918 | xml.WriteStartElement("channel"); | 
 
 
 
 
 | 919 | xml.WriteAttributeString("source", visitor.UrlOf(channel.Sampler)); | 
 
 
 
 
 | 920 | xml.WriteAttributeString("target", channel.TargetPath); | 
 
 
 
 
 | 921 | xml.WriteEndElement(); | 
 
 
 
 
 | 922 | } | 
 
 
 
 
 | 923 |  | 
 
 
 
 
 | 924 | private void BeginEntity(string name, Entity entity) | 
 
 
 
 
 | 925 | { | 
 
 
 
 
 | 926 | xml.WriteStartElement(name); | 
 
 
 
 
 | 927 |  | 
 
 
 
 
 | 928 | string id = visitor.IdOf(entity); | 
 
 
 
 
 | 929 |  | 
 
 
 
 
 | 930 | if (!string.IsNullOrEmpty(id)) | 
 
 
 
 
 | 931 | xml.WriteAttributeString("id", id); | 
 
 
 
 
 | 932 |  | 
 
 
 
 
 | 933 | //if (!String.IsNullOrEmpty(entity.Name)) | 
 
 
 
 
 | 934 | //    xml.WriteAttributeString("name", entity.Name); | 
 
 
 
 
 | 935 | } | 
 
 
 
 
 | 936 |  | 
 
 
 
 
 | 937 | private void EndEntity() | 
 
 
 
 
 | 938 | { | 
 
 
 
 
 | 939 | xml.WriteEndElement(); | 
 
 
 
 
 | 940 | } | 
 
 
 
 
 | 941 |  | 
 
 
 
 
 | 942 | private void WriteSource(Source source, Semantic semantic) | 
 
 
 
 
 | 943 | { | 
 
 
 
 
 | 944 | if (writtenSources.ContainsKey(source)) | 
 
 
 
 
 | 945 | return; | 
 
 
 
 
 | 946 |  | 
 
 
 
 
 | 947 | string sourceId = visitor.IdOf(source); | 
 
 
 
 
 | 948 |  | 
 
 
 
 
 | 949 | writtenSources.Add(source, sourceId); | 
 
 
 
 
 | 950 |  | 
 
 
 
 
 | 951 | var animationSource = source as AnimationSource; | 
 
 
 
 
 | 952 |  | 
 
 
 
 
 | 953 | if (animationSource != null) | 
 
 
 
 
 | 954 | { | 
 
 
 
 
 | 955 | if (source.FloatData != null) | 
 
 
 
 
 | 956 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, animationSource.Parameters); | 
 
 
 
 
 | 957 | else | 
 
 
 
 
 | 958 | WriteSource(sourceId, source.NameData, x => x, source.Stride, animationSource.Parameters); | 
 
 
 
 
 | 959 |  | 
 
 
 
 
 | 960 | return; | 
 
 
 
 
 | 961 | } | 
 
 
 
 
 | 962 |  | 
 
 
 
 
 | 963 | switch (semantic) | 
 
 
 
 
 | 964 | { | 
 
 
 
 
 | 965 | case Semantic.Position: | 
 
 
 
 
 | 966 | case Semantic.Normal: | 
 
 
 
 
 | 967 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "X", "Y", "Z" }); | 
 
 
 
 
 | 968 | break; | 
 
 
 
 
 | 969 |  | 
 
 
 
 
 | 970 | case Semantic.TexCoord: | 
 
 
 
 
 | 971 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "S", "T" }); | 
 
 
 
 
 | 972 | break; | 
 
 
 
 
 | 973 |  | 
 
 
 
 
 | 974 | case Semantic.Color: | 
 
 
 
 
 | 975 | if (source.Stride == 4) | 
 
 
 
 
 | 976 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "R", "G", "B", "A" }); | 
 
 
 
 
 | 977 | else | 
 
 
 
 
 | 978 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "R", "G", "B" }); | 
 
 
 
 
 | 979 | break; | 
 
 
 
 
 | 980 |  | 
 
 
 
 
 | 981 | case Semantic.Input: | 
 
 
 
 
 | 982 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "TIME" }); | 
 
 
 
 
 | 983 | return; | 
 
 
 
 
 | 984 |  | 
 
 
 
 
 | 985 | case Semantic.Output: | 
 
 
 
 
 | 986 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "VALUE" }); | 
 
 
 
 
 | 987 | return; | 
 
 
 
 
 | 988 |  | 
 
 
 
 
 | 989 | case Semantic.Interpolation: | 
 
 
 
 
 | 990 | WriteSource(sourceId, source.NameData, x => x, source.Stride, new[] { "INTERPOLATION" }); | 
 
 
 
 
 | 991 | return; | 
 
 
 
 
 | 992 |  | 
 
 
 
 
 | 993 | case Semantic.InTangent: | 
 
 
 
 
 | 994 | case Semantic.OutTangent: | 
 
 
 
 
 | 995 | WriteSource(sourceId, source.FloatData, XmlConvert.ToString, source.Stride, new[] { "X", "Y" }); | 
 
 
 
 
 | 996 | break; | 
 
 
 
 
 | 997 |  | 
 
 
 
 
 | 998 | default: | 
 
 
 
 
 | 999 | throw new NotSupportedException(string.Format("Sources with semantic {0} are not supported", semantic)); | 
 
 
 
 
 | 1000 | } | 
 
 
 
 
 | 1001 | } | 
 
 
 
 
 | 1002 |  | 
 
 
 
 
 | 1003 | private void WriteSource<T>(string sourceId, T[] data, Func<T, string> toString, int stride, string[] paramNames) | 
 
 
 
 
 | 1004 | { | 
 
 
 
 
 | 1005 | string arrayId = sourceId + "_array"; | 
 
 
 
 
 | 1006 | string type = null; | 
 
 
 
 
 | 1007 |  | 
 
 
 
 
 | 1008 | if (typeof(T) == typeof(float)) | 
 
 
 
 
 | 1009 | type = "float"; | 
 
 
 
 
 | 1010 | else if (typeof(T) == typeof(string)) | 
 
 
 
 
 | 1011 | type = "Name"; | 
 
 
 
 
 | 1012 |  | 
 
 
 
 
 | 1013 | xml.WriteStartElement("source"); | 
 
 
 
 
 | 1014 | xml.WriteAttributeString("id", sourceId); | 
 
 
 
 
 | 1015 |  | 
 
 
 
 
 | 1016 | xml.WriteStartElement(type + "_array"); | 
 
 
 
 
 | 1017 | xml.WriteAttributeString("id", arrayId); | 
 
 
 
 
 | 1018 | xml.WriteAttributeString("count", XmlConvert.ToString(data.Length)); | 
 
 
 
 
 | 1019 | xml.WriteWhitespace("\n"); | 
 
 
 
 
 | 1020 |  | 
 
 
 
 
 | 1021 | int valuesPerLine = (stride == 1) ? 10 : stride; | 
 
 
 
 
 | 1022 |  | 
 
 
 
 
 | 1023 | for (int i = 0; i < data.Length; i++) | 
 
 
 
 
 | 1024 | { | 
 
 
 
 
 | 1025 | xml.WriteString(toString(data[i])); | 
 
 
 
 
 | 1026 |  | 
 
 
 
 
 | 1027 | if (i != data.Length - 1) | 
 
 
 
 
 | 1028 | { | 
 
 
 
 
 | 1029 | if (i % valuesPerLine == valuesPerLine - 1) | 
 
 
 
 
 | 1030 | xml.WriteWhitespace("\n"); | 
 
 
 
 
 | 1031 | else | 
 
 
 
 
 | 1032 | xml.WriteWhitespace(" "); | 
 
 
 
 
 | 1033 | } | 
 
 
 
 
 | 1034 | } | 
 
 
 
 
 | 1035 |  | 
 
 
 
 
 | 1036 | xml.WriteEndElement(); | 
 
 
 
 
 | 1037 |  | 
 
 
 
 
 | 1038 | xml.WriteStartElement("technique_common"); | 
 
 
 
 
 | 1039 | WriteSourceAccessor<T>(arrayId, data.Length / stride, stride, type, paramNames); | 
 
 
 
 
 | 1040 | xml.WriteEndElement(); | 
 
 
 
 
 | 1041 |  | 
 
 
 
 
 | 1042 | xml.WriteEndElement(); | 
 
 
 
 
 | 1043 | } | 
 
 
 
 
 | 1044 |  | 
 
 
 
 
 | 1045 | private void WriteSourceAccessor<T>(string arrayId, int count, int stride, string type, string[] paramNames) | 
 
 
 
 
 | 1046 | { | 
 
 
 
 
 | 1047 | xml.WriteStartElement("accessor"); | 
 
 
 
 
 | 1048 | xml.WriteAttributeString("source", "#" + arrayId); | 
 
 
 
 
 | 1049 | xml.WriteAttributeString("count", XmlConvert.ToString(count)); | 
 
 
 
 
 | 1050 | xml.WriteAttributeString("stride", XmlConvert.ToString(stride)); | 
 
 
 
 
 | 1051 |  | 
 
 
 
 
 | 1052 | for (int i = 0; i < stride; i++) | 
 
 
 
 
 | 1053 | { | 
 
 
 
 
 | 1054 | xml.WriteStartElement("param"); | 
 
 
 
 
 | 1055 | xml.WriteAttributeString("type", type); | 
 
 
 
 
 | 1056 | xml.WriteAttributeString("name", paramNames[i]); | 
 
 
 
 
 | 1057 | xml.WriteEndElement(); | 
 
 
 
 
 | 1058 | } | 
 
 
 
 
 | 1059 |  | 
 
 
 
 
 | 1060 | xml.WriteEndElement(); | 
 
 
 
 
 | 1061 | } | 
 
 
 
 
 | 1062 |  | 
 
 
 
 
 | 1063 | private void WriteSemanticAttribute(string name, Semantic semantic) | 
 
 
 
 
 | 1064 | { | 
 
 
 
 
 | 1065 | xml.WriteAttributeString(name, semantic.ToString().ToUpperInvariant()); | 
 
 
 
 
 | 1066 | } | 
 
 
 
 
 | 1067 |  | 
 
 
 
 
 | 1068 | private void WriteCamera(Camera camera) | 
 
 
 
 
 | 1069 | { | 
 
 
 
 
 | 1070 | BeginEntity("camera", camera); | 
 
 
 
 
 | 1071 |  | 
 
 
 
 
 | 1072 | xml.WriteStartElement("optics"); | 
 
 
 
 
 | 1073 | xml.WriteStartElement("technique_common"); | 
 
 
 
 
 | 1074 | xml.WriteStartElement("perspective"); | 
 
 
 
 
 | 1075 | xml.WriteElementString("xfov", XmlConvert.ToString(camera.XFov)); | 
 
 
 
 
 | 1076 | xml.WriteElementString("aspect_ratio", XmlConvert.ToString(camera.AspectRatio)); | 
 
 
 
 
 | 1077 | xml.WriteElementString("znear", XmlConvert.ToString(camera.ZNear)); | 
 
 
 
 
 | 1078 | xml.WriteElementString("zfar", XmlConvert.ToString(camera.ZFar)); | 
 
 
 
 
 | 1079 | xml.WriteEndElement(); | 
 
 
 
 
 | 1080 | xml.WriteEndElement(); | 
 
 
 
 
 | 1081 | xml.WriteEndElement(); | 
 
 
 
 
 | 1082 |  | 
 
 
 
 
 | 1083 | EndEntity(); | 
 
 
 
 
 | 1084 | } | 
 
 
 
 
 | 1085 | } | 
 
 
 
 
 | 1086 | } |