Serialization.cs 62 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Xml;
  5. using System.Xml.Serialization;
  6. using FarseerPhysics.Collision.Shapes;
  7. using FarseerPhysics.Dynamics;
  8. using FarseerPhysics.Dynamics.Joints;
  9. using Microsoft.Xna.Framework;
  10. namespace FarseerPhysics.Common
  11. {
  12. public static class WorldSerializer
  13. {
  14. public static void Serialize(World world, string filename)
  15. {
  16. using (FileStream fs = new FileStream(filename, FileMode.Create))
  17. {
  18. new WorldXmlSerializer().Serialize(world, fs);
  19. }
  20. }
  21. public static void Deserialize(World world, string filename)
  22. {
  23. using (FileStream fs = new FileStream(filename, FileMode.Open))
  24. {
  25. new WorldXmlDeserializer().Deserialize(world, fs);
  26. }
  27. }
  28. public static World Deserialize(string filename)
  29. {
  30. using (FileStream fs = new FileStream(filename, FileMode.Open))
  31. {
  32. return new WorldXmlDeserializer().Deserialize(fs);
  33. }
  34. }
  35. }
  36. ///<summary>
  37. ///</summary>
  38. public class WorldXmlSerializer
  39. {
  40. private List<Body> _bodies = new List<Body>();
  41. private List<Fixture> _serializedFixtures = new List<Fixture>();
  42. private List<Shape> _serializedShapes = new List<Shape>();
  43. private XmlWriter _writer;
  44. private void SerializeShape(Shape shape)
  45. {
  46. _writer.WriteStartElement("Shape");
  47. _writer.WriteAttributeString("Type", shape.ShapeType.ToString());
  48. switch (shape.ShapeType)
  49. {
  50. case ShapeType.Circle:
  51. {
  52. CircleShape circle = (CircleShape)shape;
  53. _writer.WriteElementString("Radius", circle.Radius.ToString());
  54. WriteElement("Position", circle.Position);
  55. }
  56. break;
  57. case ShapeType.Polygon:
  58. {
  59. PolygonShape poly = (PolygonShape)shape;
  60. _writer.WriteStartElement("Vertices");
  61. foreach (Vector2 v in poly.Vertices)
  62. WriteElement("Vertex", v);
  63. _writer.WriteEndElement();
  64. WriteElement("Centroid", poly.MassData.Centroid);
  65. }
  66. break;
  67. case ShapeType.Edge:
  68. {
  69. EdgeShape poly = (EdgeShape)shape;
  70. WriteElement("Vertex1", poly.Vertex1);
  71. WriteElement("Vertex2", poly.Vertex2);
  72. }
  73. break;
  74. default:
  75. throw new Exception();
  76. }
  77. _writer.WriteEndElement();
  78. }
  79. private void SerializeFixture(Fixture fixture)
  80. {
  81. _writer.WriteStartElement("Fixture");
  82. _writer.WriteElementString("Shape", FindShapeIndex(fixture.Shape).ToString());
  83. _writer.WriteElementString("Density", fixture.Shape.Density.ToString());
  84. _writer.WriteStartElement("FilterData");
  85. _writer.WriteElementString("CategoryBits", ((int)fixture.CollisionCategories).ToString());
  86. _writer.WriteElementString("MaskBits", ((int)fixture.CollidesWith).ToString());
  87. _writer.WriteElementString("GroupIndex", fixture.CollisionGroup.ToString());
  88. _writer.WriteEndElement();
  89. _writer.WriteElementString("Friction", fixture.Friction.ToString());
  90. _writer.WriteElementString("IsSensor", fixture.IsSensor.ToString());
  91. _writer.WriteElementString("Restitution", fixture.Restitution.ToString());
  92. if (fixture.UserData != null)
  93. {
  94. _writer.WriteStartElement("UserData");
  95. WriteDynamicType(fixture.UserData.GetType(), fixture.UserData);
  96. _writer.WriteEndElement();
  97. }
  98. _writer.WriteEndElement();
  99. }
  100. private void SerializeBody(Body body)
  101. {
  102. _writer.WriteStartElement("Body");
  103. _writer.WriteAttributeString("Type", body.BodyType.ToString());
  104. _writer.WriteElementString("Active", body.Enabled.ToString());
  105. _writer.WriteElementString("AllowSleep", body.SleepingAllowed.ToString());
  106. _writer.WriteElementString("Angle", body.Rotation.ToString());
  107. _writer.WriteElementString("AngularDamping", body.AngularDamping.ToString());
  108. _writer.WriteElementString("AngularVelocity", body.AngularVelocity.ToString());
  109. _writer.WriteElementString("Awake", body.Awake.ToString());
  110. _writer.WriteElementString("Bullet", body.IsBullet.ToString());
  111. _writer.WriteElementString("FixedRotation", body.FixedRotation.ToString());
  112. _writer.WriteElementString("LinearDamping", body.LinearDamping.ToString());
  113. WriteElement("LinearVelocity", body.LinearVelocity);
  114. WriteElement("Position", body.Position);
  115. if (body.UserData != null)
  116. {
  117. _writer.WriteStartElement("UserData");
  118. WriteDynamicType(body.UserData.GetType(), body.UserData);
  119. _writer.WriteEndElement();
  120. }
  121. _writer.WriteStartElement("Fixtures");
  122. for (int i = 0; i < body.FixtureList.Count; i++)
  123. {
  124. _writer.WriteElementString("ID", FindFixtureIndex(body.FixtureList[i]).ToString());
  125. }
  126. _writer.WriteEndElement();
  127. _writer.WriteEndElement();
  128. }
  129. private void SerializeJoint(Joint joint)
  130. {
  131. if (joint.IsFixedType())
  132. return;
  133. _writer.WriteStartElement("Joint");
  134. _writer.WriteAttributeString("Type", joint.JointType.ToString());
  135. WriteElement("BodyA", FindBodyIndex(joint.BodyA));
  136. WriteElement("BodyB", FindBodyIndex(joint.BodyB));
  137. WriteElement("CollideConnected", joint.CollideConnected);
  138. WriteElement("Breakpoint", joint.Breakpoint);
  139. if (joint.UserData != null)
  140. {
  141. _writer.WriteStartElement("UserData");
  142. WriteDynamicType(joint.UserData.GetType(), joint.UserData);
  143. _writer.WriteEndElement();
  144. }
  145. switch (joint.JointType)
  146. {
  147. case JointType.Distance:
  148. {
  149. DistanceJoint djd = (DistanceJoint)joint;
  150. WriteElement("DampingRatio", djd.DampingRatio);
  151. WriteElement("FrequencyHz", djd.Frequency);
  152. WriteElement("Length", djd.Length);
  153. WriteElement("LocalAnchorA", djd.LocalAnchorA);
  154. WriteElement("LocalAnchorB", djd.LocalAnchorB);
  155. }
  156. break;
  157. case JointType.Friction:
  158. {
  159. FrictionJoint fjd = (FrictionJoint)joint;
  160. WriteElement("LocalAnchorA", fjd.LocalAnchorA);
  161. WriteElement("LocalAnchorB", fjd.LocalAnchorB);
  162. WriteElement("MaxForce", fjd.MaxForce);
  163. WriteElement("MaxTorque", fjd.MaxTorque);
  164. }
  165. break;
  166. case JointType.Gear:
  167. throw new Exception("Gear joint not supported by serialization");
  168. case JointType.Line:
  169. {
  170. LineJoint ljd = (LineJoint)joint;
  171. WriteElement("EnableMotor", ljd.MotorEnabled);
  172. WriteElement("LocalAnchorA", ljd.LocalAnchorA);
  173. WriteElement("LocalAnchorB", ljd.LocalAnchorB);
  174. WriteElement("MotorSpeed", ljd.MotorSpeed);
  175. WriteElement("DampingRatio", ljd.DampingRatio);
  176. WriteElement("MaxMotorTorque", ljd.MaxMotorTorque);
  177. WriteElement("FrequencyHz", ljd.Frequency);
  178. WriteElement("LocalXAxis", ljd.LocalXAxis);
  179. }
  180. break;
  181. case JointType.Prismatic:
  182. {
  183. PrismaticJoint pjd = (PrismaticJoint)joint;
  184. //NOTE: Does not conform with Box2DScene
  185. WriteElement("EnableLimit", pjd.LimitEnabled);
  186. WriteElement("EnableMotor", pjd.MotorEnabled);
  187. WriteElement("LocalAnchorA", pjd.LocalAnchorA);
  188. WriteElement("LocalAnchorB", pjd.LocalAnchorB);
  189. WriteElement("LocalXAxis1", pjd.LocalXAxis1);
  190. WriteElement("LowerTranslation", pjd.LowerLimit);
  191. WriteElement("UpperTranslation", pjd.UpperLimit);
  192. WriteElement("MaxMotorForce", pjd.MaxMotorForce);
  193. WriteElement("MotorSpeed", pjd.MotorSpeed);
  194. }
  195. break;
  196. case JointType.Pulley:
  197. {
  198. PulleyJoint pjd = (PulleyJoint)joint;
  199. WriteElement("GroundAnchorA", pjd.GroundAnchorA);
  200. WriteElement("GroundAnchorB", pjd.GroundAnchorB);
  201. WriteElement("LengthA", pjd.LengthA);
  202. WriteElement("LengthB", pjd.LengthB);
  203. WriteElement("LocalAnchorA", pjd.LocalAnchorA);
  204. WriteElement("LocalAnchorB", pjd.LocalAnchorB);
  205. WriteElement("MaxLengthA", pjd.MaxLengthA);
  206. WriteElement("MaxLengthB", pjd.MaxLengthB);
  207. WriteElement("Ratio", pjd.Ratio);
  208. }
  209. break;
  210. case JointType.Revolute:
  211. {
  212. RevoluteJoint rjd = (RevoluteJoint)joint;
  213. WriteElement("EnableLimit", rjd.LimitEnabled);
  214. WriteElement("EnableMotor", rjd.MotorEnabled);
  215. WriteElement("LocalAnchorA", rjd.LocalAnchorA);
  216. WriteElement("LocalAnchorB", rjd.LocalAnchorB);
  217. WriteElement("LowerAngle", rjd.LowerLimit);
  218. WriteElement("MaxMotorTorque", rjd.MaxMotorTorque);
  219. WriteElement("MotorSpeed", rjd.MotorSpeed);
  220. WriteElement("ReferenceAngle", rjd.ReferenceAngle);
  221. WriteElement("UpperAngle", rjd.UpperLimit);
  222. }
  223. break;
  224. case JointType.Weld:
  225. {
  226. WeldJoint wjd = (WeldJoint)joint;
  227. WriteElement("LocalAnchorA", wjd.LocalAnchorA);
  228. WriteElement("LocalAnchorB", wjd.LocalAnchorB);
  229. }
  230. break;
  231. //
  232. // Not part of Box2DScene
  233. //
  234. case JointType.Rope:
  235. {
  236. RopeJoint rjd = (RopeJoint)joint;
  237. WriteElement("LocalAnchorA", rjd.LocalAnchorA);
  238. WriteElement("LocalAnchorB", rjd.LocalAnchorB);
  239. WriteElement("MaxLength", rjd.MaxLength);
  240. }
  241. break;
  242. case JointType.Angle:
  243. {
  244. AngleJoint aj = (AngleJoint)joint;
  245. WriteElement("BiasFactor", aj.BiasFactor);
  246. WriteElement("MaxImpulse", aj.MaxImpulse);
  247. WriteElement("Softness", aj.Softness);
  248. WriteElement("TargetAngle", aj.TargetAngle);
  249. }
  250. break;
  251. case JointType.Slider:
  252. {
  253. SliderJoint sliderJoint = (SliderJoint)joint;
  254. WriteElement("DampingRatio", sliderJoint.DampingRatio);
  255. WriteElement("FrequencyHz", sliderJoint.Frequency);
  256. WriteElement("MaxLength", sliderJoint.MaxLength);
  257. WriteElement("MinLength", sliderJoint.MinLength);
  258. WriteElement("LocalAnchorA", sliderJoint.LocalAnchorA);
  259. WriteElement("LocalAnchorB", sliderJoint.LocalAnchorB);
  260. }
  261. break;
  262. default:
  263. throw new Exception("Joint not supported");
  264. }
  265. _writer.WriteEndElement();
  266. }
  267. private void WriteDynamicType(Type type, object val)
  268. {
  269. _writer.WriteElementString("Type", type.FullName);
  270. _writer.WriteStartElement("Value");
  271. XmlSerializer serializer = new XmlSerializer(type);
  272. XmlSerializerNamespaces xmlnsEmpty = new XmlSerializerNamespaces();
  273. xmlnsEmpty.Add("", "");
  274. serializer.Serialize(_writer, val, xmlnsEmpty);
  275. _writer.WriteEndElement();
  276. }
  277. private void WriteElement(string name, Vector2 vec)
  278. {
  279. _writer.WriteElementString(name, vec.X + " " + vec.Y);
  280. }
  281. private void WriteElement(string name, int val)
  282. {
  283. _writer.WriteElementString(name, val.ToString());
  284. }
  285. private void WriteElement(string name, bool val)
  286. {
  287. _writer.WriteElementString(name, val.ToString());
  288. }
  289. private void WriteElement(string name, float val)
  290. {
  291. _writer.WriteElementString(name, val.ToString());
  292. }
  293. public void Serialize(World world, Stream stream)
  294. {
  295. XmlWriterSettings settings = new XmlWriterSettings();
  296. settings.Indent = true;
  297. settings.NewLineOnAttributes = false;
  298. settings.OmitXmlDeclaration = true;
  299. _writer = XmlWriter.Create(stream, settings);
  300. _writer.WriteStartElement("World");
  301. _writer.WriteAttributeString("Version", "2");
  302. WriteElement("Gravity", world.Gravity);
  303. _writer.WriteStartElement("Shapes");
  304. for (int i = 0; i < world.BodyList.Count; i++)
  305. {
  306. Body body = world.BodyList[i];
  307. for (int j = 0; j < body.FixtureList.Count; j++)
  308. {
  309. Fixture fixture = body.FixtureList[j];
  310. bool alreadyThere = false;
  311. for (int k = 0; k < _serializedShapes.Count; k++)
  312. {
  313. Shape s2 = _serializedShapes[k];
  314. if (fixture.Shape.CompareTo(s2))
  315. {
  316. alreadyThere = true;
  317. break;
  318. }
  319. }
  320. if (!alreadyThere)
  321. {
  322. SerializeShape(fixture.Shape);
  323. _serializedShapes.Add(fixture.Shape);
  324. }
  325. }
  326. }
  327. _writer.WriteEndElement();
  328. _writer.WriteStartElement("Fixtures");
  329. for (int i = 0; i < world.BodyList.Count; i++)
  330. {
  331. Body body = world.BodyList[i];
  332. for (int j = 0; j < body.FixtureList.Count; j++)
  333. {
  334. Fixture fixture = body.FixtureList[j];
  335. bool alreadyThere = false;
  336. for (int k = 0; k < _serializedFixtures.Count; k++)
  337. {
  338. Fixture f2 = _serializedFixtures[k];
  339. if (fixture.CompareTo(f2))
  340. {
  341. alreadyThere = true;
  342. break;
  343. }
  344. }
  345. if (!alreadyThere)
  346. {
  347. SerializeFixture(fixture);
  348. _serializedFixtures.Add(fixture);
  349. }
  350. }
  351. }
  352. _writer.WriteEndElement();
  353. _writer.WriteStartElement("Bodies");
  354. for (int i = 0; i < world.BodyList.Count; i++)
  355. {
  356. Body body = world.BodyList[i];
  357. _bodies.Add(body);
  358. SerializeBody(body);
  359. }
  360. _writer.WriteEndElement();
  361. _writer.WriteStartElement("Joints");
  362. for (int i = 0; i < world.JointList.Count; i++)
  363. {
  364. Joint joint = world.JointList[i];
  365. SerializeJoint(joint);
  366. }
  367. _writer.WriteEndElement();
  368. _writer.WriteEndElement();
  369. _writer.Flush();
  370. _writer.Close();
  371. }
  372. private int FindBodyIndex(Body body)
  373. {
  374. for (int i = 0; i < _bodies.Count; ++i)
  375. if (_bodies[i] == body)
  376. return i;
  377. return -1;
  378. }
  379. private int FindFixtureIndex(Fixture fixture)
  380. {
  381. for (int i = 0; i < _serializedFixtures.Count; ++i)
  382. {
  383. if (_serializedFixtures[i].CompareTo(fixture))
  384. return i;
  385. }
  386. return -1;
  387. }
  388. private int FindShapeIndex(Shape shape)
  389. {
  390. for (int i = 0; i < _serializedShapes.Count; ++i)
  391. {
  392. if (_serializedShapes[i].CompareTo(shape))
  393. return i;
  394. }
  395. return -1;
  396. }
  397. }
  398. public class WorldXmlDeserializer
  399. {
  400. private List<Body> _bodies = new List<Body>();
  401. private List<Fixture> _fixtures = new List<Fixture>();
  402. private List<Joint> _joints = new List<Joint>();
  403. private List<Shape> _shapes = new List<Shape>();
  404. public World Deserialize(Stream stream)
  405. {
  406. World world = new World(Vector2.Zero);
  407. Deserialize(world, stream);
  408. return world;
  409. }
  410. public void Deserialize(World world, Stream stream)
  411. {
  412. world.Clear();
  413. XMLFragmentElement root = XMLFragmentParser.LoadFromStream(stream);
  414. if (root.Name.ToLower() != "world")
  415. throw new Exception();
  416. foreach (XMLFragmentElement main in root.Elements)
  417. {
  418. if (main.Name.ToLower() == "gravity")
  419. {
  420. world.Gravity = ReadVector(main);
  421. }
  422. }
  423. foreach (XMLFragmentElement shapeElement in root.Elements)
  424. {
  425. if (shapeElement.Name.ToLower() == "shapes")
  426. {
  427. foreach (XMLFragmentElement n in shapeElement.Elements)
  428. {
  429. if (n.Name.ToLower() != "shape")
  430. throw new Exception();
  431. ShapeType type = (ShapeType)Enum.Parse(typeof(ShapeType), n.Attributes[0].Value, true);
  432. switch (type)
  433. {
  434. case ShapeType.Circle:
  435. {
  436. CircleShape shape = new CircleShape();
  437. foreach (XMLFragmentElement sn in n.Elements)
  438. {
  439. switch (sn.Name.ToLower())
  440. {
  441. case "radius":
  442. shape.Radius = float.Parse(sn.Value);
  443. break;
  444. case "position":
  445. shape.Position = ReadVector(sn);
  446. break;
  447. default:
  448. throw new Exception();
  449. }
  450. }
  451. _shapes.Add(shape);
  452. }
  453. break;
  454. case ShapeType.Polygon:
  455. {
  456. PolygonShape shape = new PolygonShape();
  457. foreach (XMLFragmentElement sn in n.Elements)
  458. {
  459. switch (sn.Name.ToLower())
  460. {
  461. case "vertices":
  462. {
  463. List<Vector2> verts = new List<Vector2>();
  464. foreach (XMLFragmentElement vert in sn.Elements)
  465. verts.Add(ReadVector(vert));
  466. shape.Set(new Vertices(verts.ToArray()));
  467. }
  468. break;
  469. case "centroid":
  470. shape.MassData.Centroid = ReadVector(sn);
  471. break;
  472. }
  473. }
  474. _shapes.Add(shape);
  475. }
  476. break;
  477. case ShapeType.Edge:
  478. {
  479. EdgeShape shape = new EdgeShape();
  480. foreach (XMLFragmentElement sn in n.Elements)
  481. {
  482. switch (sn.Name.ToLower())
  483. {
  484. case "hasvertex0":
  485. shape.HasVertex0 = bool.Parse(sn.Value);
  486. break;
  487. case "hasvertex3":
  488. shape.HasVertex0 = bool.Parse(sn.Value);
  489. break;
  490. case "vertex0":
  491. shape.Vertex0 = ReadVector(sn);
  492. break;
  493. case "vertex1":
  494. shape.Vertex1 = ReadVector(sn);
  495. break;
  496. case "vertex2":
  497. shape.Vertex2 = ReadVector(sn);
  498. break;
  499. case "vertex3":
  500. shape.Vertex3 = ReadVector(sn);
  501. break;
  502. default:
  503. throw new Exception();
  504. }
  505. }
  506. _shapes.Add(shape);
  507. }
  508. break;
  509. }
  510. }
  511. }
  512. }
  513. foreach (XMLFragmentElement fixtureElement in root.Elements)
  514. {
  515. if (fixtureElement.Name.ToLower() == "fixtures")
  516. {
  517. foreach (XMLFragmentElement n in fixtureElement.Elements)
  518. {
  519. Fixture fixture = new Fixture();
  520. if (n.Name.ToLower() != "fixture")
  521. throw new Exception();
  522. foreach (XMLFragmentElement sn in n.Elements)
  523. {
  524. switch (sn.Name.ToLower())
  525. {
  526. case "shape":
  527. fixture.Shape = _shapes[int.Parse(sn.Value)];
  528. break;
  529. case "density":
  530. fixture.Shape.Density = float.Parse(sn.Value);
  531. break;
  532. case "filterdata":
  533. foreach (XMLFragmentElement ssn in sn.Elements)
  534. {
  535. switch (ssn.Name.ToLower())
  536. {
  537. case "categorybits":
  538. fixture._collisionCategories = (Category)int.Parse(ssn.Value);
  539. break;
  540. case "maskbits":
  541. fixture._collidesWith = (Category)int.Parse(ssn.Value);
  542. break;
  543. case "groupindex":
  544. fixture._collisionGroup = short.Parse(ssn.Value);
  545. break;
  546. }
  547. }
  548. break;
  549. case "friction":
  550. fixture.Friction = float.Parse(sn.Value);
  551. break;
  552. case "issensor":
  553. fixture.IsSensor = bool.Parse(sn.Value);
  554. break;
  555. case "restitution":
  556. fixture.Restitution = float.Parse(sn.Value);
  557. break;
  558. case "userdata":
  559. fixture.UserData = ReadSimpleType(sn, null, false);
  560. break;
  561. }
  562. }
  563. _fixtures.Add(fixture);
  564. }
  565. }
  566. }
  567. foreach (XMLFragmentElement bodyElement in root.Elements)
  568. {
  569. if (bodyElement.Name.ToLower() == "bodies")
  570. {
  571. foreach (XMLFragmentElement n in bodyElement.Elements)
  572. {
  573. Body body = new Body(world);
  574. if (n.Name.ToLower() != "body")
  575. throw new Exception();
  576. body.BodyType = (BodyType)Enum.Parse(typeof(BodyType), n.Attributes[0].Value, true);
  577. foreach (XMLFragmentElement sn in n.Elements)
  578. {
  579. switch (sn.Name.ToLower())
  580. {
  581. case "active":
  582. if (bool.Parse(sn.Value))
  583. body.Flags |= BodyFlags.Enabled;
  584. else
  585. body.Flags &= ~BodyFlags.Enabled;
  586. break;
  587. case "allowsleep":
  588. body.SleepingAllowed = bool.Parse(sn.Value);
  589. break;
  590. case "angle":
  591. {
  592. Vector2 position = body.Position;
  593. body.SetTransformIgnoreContacts(ref position, float.Parse(sn.Value));
  594. }
  595. break;
  596. case "angulardamping":
  597. body.AngularDamping = float.Parse(sn.Value);
  598. break;
  599. case "angularvelocity":
  600. body.AngularVelocity = float.Parse(sn.Value);
  601. break;
  602. case "awake":
  603. body.Awake = bool.Parse(sn.Value);
  604. break;
  605. case "bullet":
  606. body.IsBullet = bool.Parse(sn.Value);
  607. break;
  608. case "fixedrotation":
  609. body.FixedRotation = bool.Parse(sn.Value);
  610. break;
  611. case "lineardamping":
  612. body.LinearDamping = float.Parse(sn.Value);
  613. break;
  614. case "linearvelocity":
  615. body.LinearVelocity = ReadVector(sn);
  616. break;
  617. case "position":
  618. {
  619. float rotation = body.Rotation;
  620. Vector2 position = ReadVector(sn);
  621. body.SetTransformIgnoreContacts(ref position, rotation);
  622. }
  623. break;
  624. case "userdata":
  625. body.UserData = ReadSimpleType(sn, null, false);
  626. break;
  627. case "fixtures":
  628. {
  629. foreach (XMLFragmentElement v in sn.Elements)
  630. {
  631. Fixture blueprint = _fixtures[int.Parse(v.Value)];
  632. Fixture f = new Fixture(body, blueprint.Shape);
  633. f.Restitution = blueprint.Restitution;
  634. f.UserData = blueprint.UserData;
  635. f.Friction = blueprint.Friction;
  636. f.CollidesWith = blueprint.CollidesWith;
  637. f.CollisionCategories = blueprint.CollisionCategories;
  638. f.CollisionGroup = blueprint.CollisionGroup;
  639. }
  640. break;
  641. }
  642. }
  643. }
  644. _bodies.Add(body);
  645. }
  646. }
  647. }
  648. foreach (XMLFragmentElement jointElement in root.Elements)
  649. {
  650. if (jointElement.Name.ToLower() == "joints")
  651. {
  652. foreach (XMLFragmentElement n in jointElement.Elements)
  653. {
  654. Joint joint;
  655. if (n.Name.ToLower() != "joint")
  656. throw new Exception();
  657. JointType type = (JointType)Enum.Parse(typeof(JointType), n.Attributes[0].Value, true);
  658. int bodyAIndex = -1, bodyBIndex = -1;
  659. bool collideConnected = false;
  660. object userData = null;
  661. foreach (XMLFragmentElement sn in n.Elements)
  662. {
  663. switch (sn.Name.ToLower())
  664. {
  665. case "bodya":
  666. bodyAIndex = int.Parse(sn.Value);
  667. break;
  668. case "bodyb":
  669. bodyBIndex = int.Parse(sn.Value);
  670. break;
  671. case "collideconnected":
  672. collideConnected = bool.Parse(sn.Value);
  673. break;
  674. case "userdata":
  675. userData = ReadSimpleType(sn, null, false);
  676. break;
  677. }
  678. }
  679. Body bodyA = _bodies[bodyAIndex];
  680. Body bodyB = _bodies[bodyBIndex];
  681. switch (type)
  682. {
  683. case JointType.Distance:
  684. joint = new DistanceJoint();
  685. break;
  686. case JointType.Friction:
  687. joint = new FrictionJoint();
  688. break;
  689. case JointType.Line:
  690. joint = new LineJoint();
  691. break;
  692. case JointType.Prismatic:
  693. joint = new PrismaticJoint();
  694. break;
  695. case JointType.Pulley:
  696. joint = new PulleyJoint();
  697. break;
  698. case JointType.Revolute:
  699. joint = new RevoluteJoint();
  700. break;
  701. case JointType.Weld:
  702. joint = new WeldJoint();
  703. break;
  704. case JointType.Rope:
  705. joint = new RopeJoint();
  706. break;
  707. case JointType.Angle:
  708. joint = new AngleJoint();
  709. break;
  710. case JointType.Slider:
  711. joint = new SliderJoint();
  712. break;
  713. case JointType.Gear:
  714. throw new Exception("GearJoint is not supported.");
  715. default:
  716. throw new Exception("Invalid or unsupported joint.");
  717. }
  718. joint.CollideConnected = collideConnected;
  719. joint.UserData = userData;
  720. joint.BodyA = bodyA;
  721. joint.BodyB = bodyB;
  722. _joints.Add(joint);
  723. world.AddJoint(joint);
  724. foreach (XMLFragmentElement sn in n.Elements)
  725. {
  726. // check for specific nodes
  727. switch (type)
  728. {
  729. case JointType.Distance:
  730. {
  731. switch (sn.Name.ToLower())
  732. {
  733. case "dampingratio":
  734. ((DistanceJoint)joint).DampingRatio = float.Parse(sn.Value);
  735. break;
  736. case "frequencyhz":
  737. ((DistanceJoint)joint).Frequency = float.Parse(sn.Value);
  738. break;
  739. case "length":
  740. ((DistanceJoint)joint).Length = float.Parse(sn.Value);
  741. break;
  742. case "localanchora":
  743. ((DistanceJoint)joint).LocalAnchorA = ReadVector(sn);
  744. break;
  745. case "localanchorb":
  746. ((DistanceJoint)joint).LocalAnchorB = ReadVector(sn);
  747. break;
  748. }
  749. }
  750. break;
  751. case JointType.Friction:
  752. {
  753. switch (sn.Name.ToLower())
  754. {
  755. case "localanchora":
  756. ((FrictionJoint)joint).LocalAnchorA = ReadVector(sn);
  757. break;
  758. case "localanchorb":
  759. ((FrictionJoint)joint).LocalAnchorB = ReadVector(sn);
  760. break;
  761. case "maxforce":
  762. ((FrictionJoint)joint).MaxForce = float.Parse(sn.Value);
  763. break;
  764. case "maxtorque":
  765. ((FrictionJoint)joint).MaxTorque = float.Parse(sn.Value);
  766. break;
  767. }
  768. }
  769. break;
  770. case JointType.Line:
  771. {
  772. switch (sn.Name.ToLower())
  773. {
  774. case "enablemotor":
  775. ((LineJoint)joint).MotorEnabled = bool.Parse(sn.Value);
  776. break;
  777. case "localanchora":
  778. ((LineJoint)joint).LocalAnchorA = ReadVector(sn);
  779. break;
  780. case "localanchorb":
  781. ((LineJoint)joint).LocalAnchorB = ReadVector(sn);
  782. break;
  783. case "motorspeed":
  784. ((LineJoint)joint).MotorSpeed = float.Parse(sn.Value);
  785. break;
  786. case "dampingratio":
  787. ((LineJoint)joint).DampingRatio = float.Parse(sn.Value);
  788. break;
  789. case "maxmotortorque":
  790. ((LineJoint)joint).MaxMotorTorque = float.Parse(sn.Value);
  791. break;
  792. case "frequencyhz":
  793. ((LineJoint)joint).Frequency = float.Parse(sn.Value);
  794. break;
  795. case "localxaxis":
  796. ((LineJoint)joint).LocalXAxis = ReadVector(sn);
  797. break;
  798. }
  799. }
  800. break;
  801. case JointType.Prismatic:
  802. {
  803. switch (sn.Name.ToLower())
  804. {
  805. case "enablelimit":
  806. ((PrismaticJoint)joint).LimitEnabled = bool.Parse(sn.Value);
  807. break;
  808. case "enablemotor":
  809. ((PrismaticJoint)joint).MotorEnabled = bool.Parse(sn.Value);
  810. break;
  811. case "localanchora":
  812. ((PrismaticJoint)joint).LocalAnchorA = ReadVector(sn);
  813. break;
  814. case "localanchorb":
  815. ((PrismaticJoint)joint).LocalAnchorB = ReadVector(sn);
  816. break;
  817. case "local1axis1":
  818. ((PrismaticJoint)joint).LocalXAxis1 = ReadVector(sn);
  819. break;
  820. case "maxmotorforce":
  821. ((PrismaticJoint)joint).MaxMotorForce = float.Parse(sn.Value);
  822. break;
  823. case "motorspeed":
  824. ((PrismaticJoint)joint).MotorSpeed = float.Parse(sn.Value);
  825. break;
  826. case "lowertranslation":
  827. ((PrismaticJoint)joint).LowerLimit = float.Parse(sn.Value);
  828. break;
  829. case "uppertranslation":
  830. ((PrismaticJoint)joint).UpperLimit = float.Parse(sn.Value);
  831. break;
  832. case "referenceangle":
  833. ((PrismaticJoint)joint).ReferenceAngle = float.Parse(sn.Value);
  834. break;
  835. }
  836. }
  837. break;
  838. case JointType.Pulley:
  839. {
  840. switch (sn.Name.ToLower())
  841. {
  842. case "groundanchora":
  843. ((PulleyJoint)joint).GroundAnchorA = ReadVector(sn);
  844. break;
  845. case "groundanchorb":
  846. ((PulleyJoint)joint).GroundAnchorB = ReadVector(sn);
  847. break;
  848. case "lengtha":
  849. ((PulleyJoint)joint).LengthA = float.Parse(sn.Value);
  850. break;
  851. case "lengthb":
  852. ((PulleyJoint)joint).LengthB = float.Parse(sn.Value);
  853. break;
  854. case "localanchora":
  855. ((PulleyJoint)joint).LocalAnchorA = ReadVector(sn);
  856. break;
  857. case "localanchorb":
  858. ((PulleyJoint)joint).LocalAnchorB = ReadVector(sn);
  859. break;
  860. case "maxlengtha":
  861. ((PulleyJoint)joint).MaxLengthA = float.Parse(sn.Value);
  862. break;
  863. case "maxlengthb":
  864. ((PulleyJoint)joint).MaxLengthB = float.Parse(sn.Value);
  865. break;
  866. case "ratio":
  867. ((PulleyJoint)joint).Ratio = float.Parse(sn.Value);
  868. break;
  869. }
  870. }
  871. break;
  872. case JointType.Revolute:
  873. {
  874. switch (sn.Name.ToLower())
  875. {
  876. case "enablelimit":
  877. ((RevoluteJoint)joint).LimitEnabled = bool.Parse(sn.Value);
  878. break;
  879. case "enablemotor":
  880. ((RevoluteJoint)joint).MotorEnabled = bool.Parse(sn.Value);
  881. break;
  882. case "localanchora":
  883. ((RevoluteJoint)joint).LocalAnchorA = ReadVector(sn);
  884. break;
  885. case "localanchorb":
  886. ((RevoluteJoint)joint).LocalAnchorB = ReadVector(sn);
  887. break;
  888. case "maxmotortorque":
  889. ((RevoluteJoint)joint).MaxMotorTorque = float.Parse(sn.Value);
  890. break;
  891. case "motorspeed":
  892. ((RevoluteJoint)joint).MotorSpeed = float.Parse(sn.Value);
  893. break;
  894. case "lowerangle":
  895. ((RevoluteJoint)joint).LowerLimit = float.Parse(sn.Value);
  896. break;
  897. case "upperangle":
  898. ((RevoluteJoint)joint).UpperLimit = float.Parse(sn.Value);
  899. break;
  900. case "referenceangle":
  901. ((RevoluteJoint)joint).ReferenceAngle = float.Parse(sn.Value);
  902. break;
  903. }
  904. }
  905. break;
  906. case JointType.Weld:
  907. {
  908. switch (sn.Name.ToLower())
  909. {
  910. case "localanchora":
  911. ((WeldJoint)joint).LocalAnchorA = ReadVector(sn);
  912. break;
  913. case "localanchorb":
  914. ((WeldJoint)joint).LocalAnchorB = ReadVector(sn);
  915. break;
  916. }
  917. }
  918. break;
  919. case JointType.Rope:
  920. {
  921. switch (sn.Name.ToLower())
  922. {
  923. case "localanchora":
  924. ((RopeJoint)joint).LocalAnchorA = ReadVector(sn);
  925. break;
  926. case "localanchorb":
  927. ((RopeJoint)joint).LocalAnchorB = ReadVector(sn);
  928. break;
  929. case "maxlength":
  930. ((RopeJoint)joint).MaxLength = float.Parse(sn.Value);
  931. break;
  932. }
  933. }
  934. break;
  935. case JointType.Gear:
  936. throw new Exception("Gear joint is unsupported");
  937. case JointType.Angle:
  938. {
  939. switch (sn.Name.ToLower())
  940. {
  941. case "biasfactor":
  942. ((AngleJoint)joint).BiasFactor = float.Parse(sn.Value);
  943. break;
  944. case "maximpulse":
  945. ((AngleJoint)joint).MaxImpulse = float.Parse(sn.Value);
  946. break;
  947. case "softness":
  948. ((AngleJoint)joint).Softness = float.Parse(sn.Value);
  949. break;
  950. case "targetangle":
  951. ((AngleJoint)joint).TargetAngle = float.Parse(sn.Value);
  952. break;
  953. }
  954. }
  955. break;
  956. case JointType.Slider:
  957. {
  958. switch (sn.Name.ToLower())
  959. {
  960. case "dampingratio":
  961. ((SliderJoint)joint).DampingRatio = float.Parse(sn.Value);
  962. break;
  963. case "frequencyhz":
  964. ((SliderJoint)joint).Frequency = float.Parse(sn.Value);
  965. break;
  966. case "maxlength":
  967. ((SliderJoint)joint).MaxLength = float.Parse(sn.Value);
  968. break;
  969. case "minlength":
  970. ((SliderJoint)joint).MinLength = float.Parse(sn.Value);
  971. break;
  972. case "localanchora":
  973. ((SliderJoint)joint).LocalAnchorA = ReadVector(sn);
  974. break;
  975. case "localanchorb":
  976. ((SliderJoint)joint).LocalAnchorB = ReadVector(sn);
  977. break;
  978. }
  979. }
  980. break;
  981. }
  982. }
  983. }
  984. }
  985. }
  986. }
  987. private Vector2 ReadVector(XMLFragmentElement node)
  988. {
  989. string[] values = node.Value.Split(' ');
  990. return new Vector2(float.Parse(values[0]), float.Parse(values[1]));
  991. }
  992. private object ReadSimpleType(XMLFragmentElement node, Type type, bool outer)
  993. {
  994. if (type == null)
  995. return ReadSimpleType(node.Elements[1], Type.GetType(node.Elements[0].Value), outer);
  996. XmlSerializer serializer = new XmlSerializer(type);
  997. XmlSerializerNamespaces xmlnsEmpty = new XmlSerializerNamespaces();
  998. xmlnsEmpty.Add("", "");
  999. using (MemoryStream stream = new MemoryStream())
  1000. {
  1001. StreamWriter writer = new StreamWriter(stream);
  1002. {
  1003. writer.Write((outer) ? node.OuterXml : node.InnerXml);
  1004. writer.Flush();
  1005. stream.Position = 0;
  1006. }
  1007. XmlReaderSettings settings = new XmlReaderSettings();
  1008. settings.ConformanceLevel = ConformanceLevel.Fragment;
  1009. return serializer.Deserialize(XmlReader.Create(stream, settings));
  1010. }
  1011. }
  1012. }
  1013. #region XMLFragment
  1014. public class XMLFragmentAttribute
  1015. {
  1016. public string Name { get; set; }
  1017. public string Value { get; set; }
  1018. }
  1019. public class XMLFragmentElement
  1020. {
  1021. private List<XMLFragmentAttribute> _attributes = new List<XMLFragmentAttribute>();
  1022. private List<XMLFragmentElement> _elements = new List<XMLFragmentElement>();
  1023. public IList<XMLFragmentElement> Elements
  1024. {
  1025. get { return _elements; }
  1026. }
  1027. public IList<XMLFragmentAttribute> Attributes
  1028. {
  1029. get { return _attributes; }
  1030. }
  1031. public string Name { get; set; }
  1032. public string Value { get; set; }
  1033. public string OuterXml { get; set; }
  1034. public string InnerXml { get; set; }
  1035. }
  1036. public class XMLFragmentException : Exception
  1037. {
  1038. public XMLFragmentException()
  1039. {
  1040. }
  1041. public XMLFragmentException(string message)
  1042. : base(message)
  1043. {
  1044. }
  1045. public XMLFragmentException(string message, Exception inner)
  1046. : base(message, inner)
  1047. {
  1048. }
  1049. }
  1050. public class FileBuffer
  1051. {
  1052. public FileBuffer(Stream stream)
  1053. {
  1054. using (StreamReader sr = new StreamReader(stream))
  1055. Buffer = sr.ReadToEnd();
  1056. Position = 0;
  1057. }
  1058. public string Buffer { get; set; }
  1059. public int Position { get; set; }
  1060. public int Length
  1061. {
  1062. get { return Buffer.Length; }
  1063. }
  1064. public char Next
  1065. {
  1066. get
  1067. {
  1068. char c = Buffer[Position];
  1069. Position++;
  1070. return c;
  1071. }
  1072. }
  1073. public char Peek
  1074. {
  1075. get { return Buffer[Position]; }
  1076. }
  1077. public bool EndOfBuffer
  1078. {
  1079. get { return Position == Length; }
  1080. }
  1081. }
  1082. public class XMLFragmentParser
  1083. {
  1084. private static List<char> _punctuation = new List<char> { '/', '<', '>', '=' };
  1085. private FileBuffer _buffer;
  1086. private XMLFragmentElement _rootNode;
  1087. public XMLFragmentParser(Stream stream)
  1088. {
  1089. Load(stream);
  1090. }
  1091. public XMLFragmentParser(string fileName)
  1092. {
  1093. using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read))
  1094. Load(fs);
  1095. }
  1096. public XMLFragmentElement RootNode
  1097. {
  1098. get { return _rootNode; }
  1099. }
  1100. public void Load(Stream stream)
  1101. {
  1102. _buffer = new FileBuffer(stream);
  1103. }
  1104. public static XMLFragmentElement LoadFromFile(string fileName)
  1105. {
  1106. XMLFragmentParser x = new XMLFragmentParser(fileName);
  1107. x.Parse();
  1108. return x.RootNode;
  1109. }
  1110. public static XMLFragmentElement LoadFromStream(Stream stream)
  1111. {
  1112. XMLFragmentParser x = new XMLFragmentParser(stream);
  1113. x.Parse();
  1114. return x.RootNode;
  1115. }
  1116. private string NextToken()
  1117. {
  1118. string str = "";
  1119. bool _done = false;
  1120. while (true)
  1121. {
  1122. char c = _buffer.Next;
  1123. if (_punctuation.Contains(c))
  1124. {
  1125. if (str != "")
  1126. {
  1127. _buffer.Position--;
  1128. break;
  1129. }
  1130. _done = true;
  1131. }
  1132. else if (char.IsWhiteSpace(c))
  1133. {
  1134. if (str != "")
  1135. break;
  1136. else
  1137. continue;
  1138. }
  1139. str += c;
  1140. if (_done)
  1141. break;
  1142. }
  1143. str = TrimControl(str);
  1144. // Trim quotes from start and end
  1145. if (str[0] == '\"')
  1146. str = str.Remove(0, 1);
  1147. if (str[str.Length - 1] == '\"')
  1148. str = str.Remove(str.Length - 1, 1);
  1149. return str;
  1150. }
  1151. private string PeekToken()
  1152. {
  1153. int oldPos = _buffer.Position;
  1154. string str = NextToken();
  1155. _buffer.Position = oldPos;
  1156. return str;
  1157. }
  1158. private string ReadUntil(char c)
  1159. {
  1160. string str = "";
  1161. while (true)
  1162. {
  1163. char ch = _buffer.Next;
  1164. if (ch == c)
  1165. {
  1166. _buffer.Position--;
  1167. break;
  1168. }
  1169. str += ch;
  1170. }
  1171. // Trim quotes from start and end
  1172. if (str[0] == '\"')
  1173. str = str.Remove(0, 1);
  1174. if (str[str.Length - 1] == '\"')
  1175. str = str.Remove(str.Length - 1, 1);
  1176. return str;
  1177. }
  1178. private string TrimControl(string str)
  1179. {
  1180. string newStr = str;
  1181. // Trim control characters
  1182. int i = 0;
  1183. while (true)
  1184. {
  1185. if (i == newStr.Length)
  1186. break;
  1187. if (char.IsControl(newStr[i]))
  1188. newStr = newStr.Remove(i, 1);
  1189. else
  1190. i++;
  1191. }
  1192. return newStr;
  1193. }
  1194. private string TrimTags(string outer)
  1195. {
  1196. int start = outer.IndexOf('>') + 1;
  1197. int end = outer.LastIndexOf('<');
  1198. return TrimControl(outer.Substring(start, end - start));
  1199. }
  1200. public XMLFragmentElement TryParseNode()
  1201. {
  1202. if (_buffer.EndOfBuffer)
  1203. return null;
  1204. int startOuterXml = _buffer.Position;
  1205. string token = NextToken();
  1206. if (token != "<")
  1207. throw new XMLFragmentException("Expected \"<\", got " + token);
  1208. XMLFragmentElement element = new XMLFragmentElement();
  1209. element.Name = NextToken();
  1210. while (true)
  1211. {
  1212. token = NextToken();
  1213. if (token == ">")
  1214. break;
  1215. else if (token == "/") // quick-exit case
  1216. {
  1217. NextToken();
  1218. element.OuterXml =
  1219. TrimControl(_buffer.Buffer.Substring(startOuterXml, _buffer.Position - startOuterXml)).Trim();
  1220. element.InnerXml = "";
  1221. return element;
  1222. }
  1223. else
  1224. {
  1225. XMLFragmentAttribute attribute = new XMLFragmentAttribute();
  1226. attribute.Name = token;
  1227. if ((token = NextToken()) != "=")
  1228. throw new XMLFragmentException("Expected \"=\", got " + token);
  1229. attribute.Value = NextToken();
  1230. element.Attributes.Add(attribute);
  1231. }
  1232. }
  1233. while (true)
  1234. {
  1235. int oldPos = _buffer.Position; // for restoration below
  1236. token = NextToken();
  1237. if (token == "<")
  1238. {
  1239. token = PeekToken();
  1240. if (token == "/") // finish element
  1241. {
  1242. NextToken(); // skip the / again
  1243. token = NextToken();
  1244. NextToken(); // skip >
  1245. element.OuterXml =
  1246. TrimControl(_buffer.Buffer.Substring(startOuterXml, _buffer.Position - startOuterXml)).Trim();
  1247. element.InnerXml = TrimTags(element.OuterXml);
  1248. if (token != element.Name)
  1249. throw new XMLFragmentException("Mismatched element pairs: \"" + element.Name + "\" vs \"" +
  1250. token + "\"");
  1251. break;
  1252. }
  1253. else
  1254. {
  1255. _buffer.Position = oldPos;
  1256. element.Elements.Add(TryParseNode());
  1257. }
  1258. }
  1259. else
  1260. {
  1261. // value, probably
  1262. _buffer.Position = oldPos;
  1263. element.Value = ReadUntil('<');
  1264. }
  1265. }
  1266. return element;
  1267. }
  1268. public void Parse()
  1269. {
  1270. _rootNode = TryParseNode();
  1271. if (_rootNode == null)
  1272. throw new XMLFragmentException("Unable to load root node");
  1273. }
  1274. }
  1275. #endregion
  1276. }