+ Metadata mMetadata;\r
+ char[,] mGrid;\r
+ List<RawEntity> mEntities;\r
+\r
+ public Modal(Metadata metadata, char[,] grid, List<RawEntity> entities)\r
+ {\r
+ Debug.Assert(metadata != null);\r
+ Debug.Assert(grid != null);\r
+ Debug.Assert(entities != null);\r
+ Debug.Assert(metadata.GridWidth * metadata.GridHeight == grid.Length);\r
+\r
+ mMetadata = metadata;\r
+ mGrid = grid;\r
+ mEntities = entities;\r
+\r
+#if DEBUG\r
+ Console.WriteLine("Loaded map {0} of type {1} written by {2}.",\r
+ metadata.Name,\r
+ metadata.Type,\r
+ metadata.Author);\r
+#endif\r
+ }\r
+\r
+\r
+ public Metadata Metadata { get { return mMetadata; } }\r
+ public List<RawEntity> Entities { get { return mEntities; } }\r
+\r
+\r
+ public bool IsCellOpen(int x, int y)\r
+ {\r
+ // TODO: Still need to define characters for types of scenery.\r
+ return mGrid[x, y] == ' ';\r
+ }\r
+\r
+\r
+ public List<object> GetAllEntities()\r
+ {\r
+ List<object> list = new List<object>();\r
+\r
+ foreach (RawEntity raw in mEntities)\r
+ {\r
+ if (raw.Attributes.ContainsKey("type"))\r
+ {\r
+ string typename = raw.Attributes["type"];\r
+\r
+ object[] args = new object[3];\r
+ args[0] = raw.Id;\r
+ args[1] = raw.Position;\r
+ args[2] = raw.Attributes;\r
+\r
+ try\r
+ {\r
+\r
+ object entity = Activator.CreateInstance(System.Type.GetType("CarFire." + typename), args);\r
+ if (entity != null) list.Add(entity);\r
+ else throw new RuntimeException();\r
+ }\r
+#pragma warning disable 0168\r
+ catch (System.Exception ex)\r
+#pragma warning restore 0168\r
+ {\r
+ throw new RuntimeException("Entity of type " + typename + " not loaded because an entity class can't be found.");\r
+ }\r
+ }\r
+ else\r
+ {\r
+ Console.WriteLine("Ignoring entity with identifier " + raw.Id + " since it has no type key.");\r
+ }\r
+ }\r
+\r
+ return list;\r
+ }\r
+\r
+ public List<T> GetEntities<T>()\r
+ {\r
+ System.Type type = typeof(T);\r
+ List<T> list = new List<T>();\r
+\r
+ string typename = typeof(T).Name;\r
+ foreach (RawEntity raw in mEntities)\r
+ {\r
+ if (raw.Attributes.ContainsKey("type") && typename == raw.Attributes["type"])\r
+ {\r
+ object[] args = new object[3];\r
+ args[0] = raw.Id;\r
+ args[1] = raw.Position;\r
+ args[2] = raw.Attributes;\r
+\r
+ T entity = (T)Activator.CreateInstance(type, args);\r
+ if (entity != null) list.Add(entity);\r
+ else throw new RuntimeException("Entity of type " + typename + " not loaded because an entity class can't be found.");\r
+ }\r
+ }\r
+\r
+ return list;\r
+ }\r
+ }\r
+\r
+ class View\r
+ {\r
+ Modal mData;\r
+\r
+ public Vector2 CenterCell;\r
+ Viewport mViewport;\r
+\r
+\r
+ public View(Modal data)\r
+ {\r
+ Debug.Assert(data != null);\r
+ mData = data;\r
+ }\r
+\r
+ public void Draw(SpriteBatch spriteBatch)\r
+ {\r
+ mViewport = spriteBatch.GraphicsDevice.Viewport;\r
+\r
+ // TODO: There is no culling yet, but it runs so fast that it probably won't ever need it.\r
+ for (int y = 0; y < mData.Metadata.GridHeight; y++)\r
+ {\r
+ for (int x = 0; x < mData.Metadata.GridWidth; x++)\r
+ {\r
+ if (mData.IsCellOpen(x, y))\r
+ {\r
+ spriteBatch.Draw(Map.DefaultTile, GetRectangleFromCoordinates(x, y), Color.White);\r
+ }\r
+ else\r
+ {\r
+ spriteBatch.Draw(Map.DefaultTile, GetRectangleFromCoordinates(x, y), Color.DarkBlue);\r
+ }\r
+ }\r
+ }\r
+ }\r
+\r
+ /// <summary>\r
+ /// Get a matrix to transform a point from grid-space to screen coordinates. This\r
+ /// method uses the viewport to bound the edges of the map such that the camera\r
+ /// will not show anything outside of the grid.\r
+ /// </summary>\r
+ /// <param name="center">The point to put in the center.</param>\r
+ /// <returns>The transformation matrix.</returns>\r
+ Matrix GetTransformation(Vector2 center)\r
+ {\r
+ float halfRatio = PixelsToUnitSquares * 0.5f;\r
+ Matrix transform = Matrix.CreateTranslation(-center.X, -center.Y, 0.0f);\r
+ transform *= Matrix.CreateScale(PixelsToUnitSquares);\r
+ transform *= Matrix.CreateTranslation(mViewport.Width * 0.5f - halfRatio,\r
+ mViewport.Height * 0.5f - halfRatio, 0.0f);\r
+\r
+ Vector2 topLeft = Vector2.Transform(new Vector2(0.0f, 0.0f), transform);\r
+ topLeft.X = Math.Max(mViewport.X, topLeft.X);\r
+ topLeft.Y = Math.Max(mViewport.Y, topLeft.Y);\r
+ transform *= Matrix.CreateTranslation(-topLeft.X, -topLeft.Y, 0.0f);\r
+\r
+ Vector2 bottomRight = Vector2.Transform(new Vector2((float)mData.Metadata.GridWidth,\r
+ (float)mData.Metadata.GridHeight), transform);\r
+ float right = mViewport.X + mViewport.Width;\r
+ float bottom = mViewport.Y + mViewport.Height;\r
+ bottomRight.X = Math.Min(right, bottomRight.X) - right;\r
+ bottomRight.Y = Math.Min(bottom, bottomRight.Y) - bottom;\r
+ transform *= Matrix.CreateTranslation(-bottomRight.X, -bottomRight.Y, 0.0f);\r
+\r
+ return transform;\r
+ }\r
+\r
+\r
+ public Point GetPointFromCoordinates(float x, float y)\r
+ {\r
+ Matrix transform = GetTransformation(CenterCell);\r
+ Vector2 point = Vector2.Transform(new Vector2(x, y), transform);\r
+\r
+ return new Point((int)point.X, (int)point.Y);\r
+ }\r
+\r
+ public Rectangle GetRectangleFromCoordinates(float x, float y)\r
+ {\r
+ Matrix transform = GetTransformation(CenterCell);\r
+ Vector2 point = Vector2.Transform(new Vector2(x, y), transform);\r
+ \r
+ return new Rectangle((int)point.X, (int)point.Y, (int)PixelsToUnitSquares, (int)PixelsToUnitSquares);\r
+ }\r