DisplayTerrain.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using CommonAI.RTS;
  5. using CommonLang.Vector;
  6. using System.Drawing;
  7. using CommonAI.Zone;
  8. using CommonLang;
  9. using System.Drawing.Drawing2D;
  10. using System.Windows.Forms;
  11. using CommonFroms;
  12. using CommonLang.Property;
  13. using CommonAI.Zone.Helper;
  14. using CommonAI.RTS.Manhattan;
  15. using System.Reflection;
  16. namespace GameEditorPlugin.Win32
  17. {
  18. public class EditorZoneManhattanMap : ZoneManhattanMap
  19. {
  20. public EditorZoneManhattanMap(ZoneInfo info, TerrainDefinitionMap define)
  21. : base(info, define)
  22. {
  23. }
  24. public override AstarManhattan.MMapNode CreateMapNode()
  25. {
  26. return new AstarManhattan.MMapNode();
  27. }
  28. }
  29. //-----------------------------------------------------------------------------------------------
  30. abstract public class DisplayObject
  31. {
  32. public abstract bool Visible { get; }
  33. public abstract float X { get; }
  34. public abstract float Y { get; }
  35. public abstract float Z { get; }
  36. public abstract RectangleF LocalBounds { get; }
  37. public abstract void render(Graphics g);
  38. public virtual void render_begin(Graphics g) { }
  39. public virtual void render_end(Graphics g) { }
  40. public virtual void renderHP(Graphics g) { }
  41. public virtual void renderName(Graphics g, Font font, Brush brush) { }
  42. }
  43. //-----------------------------------------------------------------------------------------------
  44. abstract public class DisplayTerrain : IDisposable
  45. {
  46. //-----------------------------------------------------------------------------------------------------------------
  47. #region ShowTags
  48. [Desc("显示Z高度", "show")]
  49. public bool ShowZ = true;
  50. [Desc("显示日志", "show")]
  51. public bool ShowLog = false;
  52. [Desc("显示地形", "show")]
  53. public bool ShowTerrain = true;
  54. public void GenShowItems(ToolStripDropDownButton drop)
  55. {
  56. List<ToolStripMenuItem> list = new List<ToolStripMenuItem>();
  57. foreach (var field in GetType().GetFields())
  58. {
  59. DescAttribute desc = PropertyUtil.GetAttribute<DescAttribute>(field);
  60. if (desc != null && field.FieldType == typeof(bool) && desc.Category == "show")
  61. {
  62. ToolStripMenuItem chk = new ToolStripMenuItem();
  63. chk.Size = new System.Drawing.Size(152, 22);
  64. chk.CheckOnClick = true;
  65. chk.Checked = (bool)field.GetValue(this);
  66. chk.Text = desc.Desc;
  67. chk.Tag = field;
  68. chk.Click += new System.EventHandler(this.chk_Click);
  69. list.Add(chk);
  70. }
  71. }
  72. drop.DropDownItems.AddRange(list.ToArray());
  73. }
  74. private void chk_Click(object sender, EventArgs e)
  75. {
  76. var chk = sender as ToolStripMenuItem;
  77. var field = chk.Tag as FieldInfo;
  78. field.SetValue(this, chk.Checked);
  79. }
  80. #endregion
  81. //-----------------------------------------------------------------------------------------------------------------
  82. private ZoneInfo mZoneInfo;
  83. private List<DisplayLog> logs = new List<DisplayLog>();
  84. //-----------------------------------------------------------------------------------------------------------------
  85. public abstract void Dispose();
  86. //-----------------------------------------------------------------------------------------------------------------
  87. public ZoneInfo Terrain { get { return mZoneInfo; } }
  88. protected virtual void InitTerrain(ZoneInfo zoneInfo)
  89. {
  90. this.mZoneInfo = zoneInfo;
  91. }
  92. public int Width { get { if (mZoneInfo != null) return mZoneInfo.TotalWidth; return 1; } }
  93. public int Height { get { if (mZoneInfo != null) return mZoneInfo.TotalHeight; return 1; } }
  94. public int CellW { get { if (mZoneInfo != null) return mZoneInfo.GridCellW; return 1; } }
  95. public int CellH { get { if (mZoneInfo != null) return mZoneInfo.GridCellH; return 1; } }
  96. public int XCount { get { if (mZoneInfo != null) return mZoneInfo.XCount; return 1; } }
  97. public int YCount { get { if (mZoneInfo != null) return mZoneInfo.YCount; return 1; } }
  98. //-----------------------------------------------------------------------------------------------------------------
  99. private QueryKey queryKey;
  100. public void BindMessageFilter(PictureBox control)
  101. {
  102. if (queryKey == null)
  103. {
  104. this.queryKey = new QueryKey(control);
  105. }
  106. }
  107. public void UnbindMessageFilter()
  108. {
  109. if (queryKey != null)
  110. {
  111. queryKey.Dispose();
  112. queryKey = null;
  113. }
  114. }
  115. public bool IsMouseDown(MouseButtons btn)
  116. {
  117. if (queryKey != null)
  118. {
  119. return queryKey.IsMouseDown(btn);
  120. }
  121. return false;
  122. }
  123. public bool IsMouseUp(MouseButtons btn)
  124. {
  125. if (queryKey != null)
  126. {
  127. return queryKey.IsMouseUp(btn);
  128. }
  129. return false;
  130. }
  131. public bool IsMouseHold(MouseButtons btn)
  132. {
  133. if (queryKey != null)
  134. {
  135. return queryKey.IsMouseHold(btn);
  136. }
  137. return false;
  138. }
  139. //-----------------------------------------------------------------------------------------------------------------
  140. // windows
  141. private RectangleF mWindow = new RectangleF(0, 0, 512, 512);
  142. private float mouseX = 0;
  143. private float mouseY = 0;
  144. private float cameraX = 0;
  145. private float cameraY = 0;
  146. private float cameraScaleX = 1;
  147. private float cameraScaleY = 1;
  148. public float MouseX { get { return mouseX; } }
  149. public float MouseY { get { return mouseY; } }
  150. public float CameraX { get { return cameraX; } }
  151. public float CameraY { get { return cameraY; } }
  152. public float WindowW { get { return mWindow.Width; } }
  153. public float WindowH { get { return mWindow.Height; } }
  154. public RectangleF CameraScrollRect
  155. {
  156. get
  157. {
  158. return new RectangleF(
  159. screenToWorldX(0),
  160. screenToWorldY(0),
  161. screenToWorldSizeX(mWindow.Width),
  162. screenToWorldSizeY(mWindow.Height));
  163. }
  164. }
  165. public void setWindow(RectangleF window)
  166. {
  167. this.mWindow.X = window.X;
  168. this.mWindow.Y = window.Y;
  169. this.mWindow.Width = window.Width;
  170. this.mWindow.Height = window.Height;
  171. }
  172. public float screenToWorldSizeX(float s)
  173. {
  174. return s / cameraScaleX;
  175. }
  176. public float screenToWorldSizeY(float s)
  177. {
  178. return s / cameraScaleY;
  179. }
  180. public float worldToScreenSizeX(float s)
  181. {
  182. return s * cameraScaleX;
  183. }
  184. public float worldToScreenSizeY(float s)
  185. {
  186. return s * cameraScaleY;
  187. }
  188. public float screenToWorldX(float x)
  189. {
  190. return (x - mWindow.Width / 2) / cameraScaleX + cameraX;
  191. }
  192. public float screenToWorldY(float y)
  193. {
  194. return (y - mWindow.Height / 2) / cameraScaleY + cameraY;
  195. }
  196. public float worldToScreenX(float x)
  197. {
  198. return (x - cameraX) * cameraScaleX + mWindow.Width / 2;
  199. }
  200. public float worldToScreenY(float y)
  201. {
  202. return (y - cameraY) * cameraScaleY + mWindow.Height / 2;
  203. }
  204. public void setCamera(float x, float y)
  205. {
  206. this.cameraX = x;
  207. this.cameraY = y;
  208. }
  209. public void setCameraScale(float scaleX, float scaleY)
  210. {
  211. cameraScaleX = scaleX;
  212. cameraScaleX = Math.Max(cameraScaleX, 0.01f);
  213. cameraScaleY = scaleY;
  214. cameraScaleY = Math.Max(cameraScaleY, 0.01f);
  215. }
  216. public float getCameraScaleX()
  217. {
  218. return cameraScaleX;
  219. }
  220. public float getCameraScaleY()
  221. {
  222. return cameraScaleY;
  223. }
  224. public float getCameraScale()
  225. {
  226. return Math.Max(cameraScaleX, cameraScaleY);
  227. }
  228. //-----------------------------------------------------------------------------------------------------------------
  229. virtual public void update(int intervalMS)
  230. {
  231. }
  232. virtual public void update_flush()
  233. {
  234. if (queryKey != null)
  235. {
  236. queryKey.Update();
  237. this.mouseX = queryKey.MouseX;
  238. this.mouseY = queryKey.MouseY;
  239. }
  240. for (int i = logs.Count - 1; i >= 0; --i)
  241. {
  242. DisplayLog u = logs[i];
  243. u.update();
  244. if (!u.Alive)
  245. {
  246. logs.RemoveAt(i);
  247. }
  248. }
  249. }
  250. //-----------------------------------------------------------------------------------------------------------------
  251. #region _render_
  252. public static Font font = new Font(@"微软雅黑", 9, FontStyle.Regular);
  253. private float begin_pen_scale = 1f;
  254. public float PenScale { get { return begin_pen_scale; } }
  255. protected readonly SolidBrush brush_cell = new SolidBrush(Color.FromArgb(0xff, 0x20, 0x40, 0x20));
  256. protected readonly Pen pen_z = new Pen(Color.White);
  257. protected readonly Pen pen_grid = new Pen(Color.FromArgb(0x10, 0xff, 0xff, 0xff));
  258. virtual public void render(Graphics g)
  259. {
  260. RectangleF worldBounds = new RectangleF(
  261. screenToWorldX(0),
  262. screenToWorldY(0),
  263. screenToWorldSizeX(mWindow.Width),
  264. screenToWorldSizeY(mWindow.Height));
  265. GraphicsState gs = g.Save();
  266. {
  267. this.begin_pen_scale = 1f / getCameraScale();
  268. g.TranslateTransform(
  269. mWindow.Width / 2,
  270. mWindow.Height / 2);
  271. g.ScaleTransform(
  272. cameraScaleX,
  273. cameraScaleY);
  274. g.TranslateTransform(
  275. -cameraX,
  276. -cameraY);
  277. renderTerrain(g, worldBounds);
  278. renderObjects(g, worldBounds);
  279. }
  280. g.Restore(gs);
  281. renderScreen(g, worldBounds);
  282. if (ShowLog) renderLogs(g);
  283. }
  284. private void renderLogs(Graphics g)
  285. {
  286. for (int i = logs.Count - 1; i >= 0; --i)
  287. {
  288. DisplayLog u = logs[i];
  289. GraphicsState state = g.Save();
  290. g.TranslateTransform(
  291. worldToScreenX(u.pos.X),
  292. worldToScreenY(u.pos.Y));
  293. u.render(g);
  294. g.Restore(state);
  295. }
  296. }
  297. protected abstract void renderObjects(Graphics g, RectangleF worldBounds);
  298. protected abstract void renderScreen(Graphics g, RectangleF worldBounds);
  299. protected virtual void renderTerrain(Graphics g, RectangleF rect)
  300. {
  301. if (ShowTerrain && mZoneInfo != null)
  302. {
  303. int sx = (int)(rect.X / CellW);
  304. int sy = (int)(rect.Y / CellH);
  305. int sw = (int)(rect.Width / CellW) + 1;
  306. int sh = (int)(rect.Height / CellH) + 1;
  307. int lw = (int)(rect.Width + CellW * 2);
  308. int lh = (int)(rect.Height + CellH * 2);
  309. for (int x = sx + sw; x >= sx; --x)
  310. {
  311. if (x < mZoneInfo.XCount && x >= 0)
  312. {
  313. for (int y = sy + sh; y >= sy; --y)
  314. {
  315. if (y < mZoneInfo.YCount && y >= 0)
  316. {
  317. int flag = mZoneInfo.TerrainMatrix[x, y];
  318. renderCell(g, x, y, flag);
  319. }
  320. }
  321. }
  322. }
  323. }
  324. }
  325. protected virtual void renderCell(Graphics g, int bx, int by, int flag)
  326. {
  327. if (flag != 0)
  328. {
  329. brush_cell.Color = Color.FromArgb(flag);
  330. g.FillRectangle(brush_cell,
  331. bx * CellW,
  332. by * CellH,
  333. CellW,
  334. CellH);
  335. }
  336. }
  337. //----------------------------------------------------------------------------------------------------
  338. public delegate void DrawAction(Graphics g, RectangleF cameraBounds);
  339. public void drawInWorldSpace(Graphics g, DrawAction action)
  340. {
  341. RectangleF worldBounds = new RectangleF(
  342. screenToWorldX(0),
  343. screenToWorldY(0),
  344. screenToWorldSizeX(mWindow.Width),
  345. screenToWorldSizeY(mWindow.Height));
  346. GraphicsState gs = g.Save();
  347. {
  348. this.begin_pen_scale = 1f / getCameraScale();
  349. g.TranslateTransform(
  350. mWindow.Width / 2,
  351. mWindow.Height / 2);
  352. g.ScaleTransform(
  353. cameraScaleX,
  354. cameraScaleY);
  355. g.TranslateTransform(
  356. -cameraX,
  357. -cameraY);
  358. action(g, worldBounds);
  359. }
  360. g.Restore(gs);
  361. }
  362. public void drawGrid(Graphics g, RectangleF rect, int cellW, int cellH)
  363. {
  364. pen_grid.Width = 1f / getCameraScale();
  365. int sx = (int)(rect.X / cellW);
  366. int sy = (int)(rect.Y / cellH);
  367. int sw = (int)(rect.Width / cellW) + 1;
  368. int sh = (int)(rect.Height / cellH) + 1;
  369. int lw = (int)(rect.Width + cellW * 2);
  370. int lh = (int)(rect.Height + cellH * 2);
  371. for (int x = sx + sw; x >= sx; --x)
  372. {
  373. g.DrawLine(pen_grid, x * cellW, sy * cellH, x * cellW, (sy * cellH + lh));
  374. }
  375. for (int y = sy + sh; y >= sy; --y)
  376. {
  377. g.DrawLine(pen_grid, sx * cellW, y * cellH, (sx * cellW + lw), y * cellH);
  378. }
  379. }
  380. protected void drawObjectes<T>(Graphics g, RectangleF rect, ICollection<T> units)
  381. where T : DisplayObject
  382. {
  383. float penscale = 1f / getCameraScale();
  384. foreach (DisplayObject u in units)
  385. {
  386. if (u.Visible && CMath.intersectRect2(
  387. rect.X,
  388. rect.Y,
  389. rect.Width,
  390. rect.Height,
  391. u.X + u.LocalBounds.X,
  392. u.Y + u.LocalBounds.Y,
  393. u.LocalBounds.Width,
  394. u.LocalBounds.Height
  395. ))
  396. {
  397. GraphicsState state = g.Save();
  398. try
  399. {
  400. g.TranslateTransform(u.X, u.Y);
  401. if (ShowZ && u.Z != 0)
  402. {
  403. pen_z.Width = penscale;
  404. g.DrawLine(pen_z, 0, 0, 0, -u.Z);
  405. g.ScaleTransform(1 + u.Z, 1 + u.Z);
  406. }
  407. u.render_begin(g);
  408. u.render(g);
  409. u.render_end(g);
  410. }
  411. catch (Exception err)
  412. {
  413. Console.WriteLine(err.Message + "\n" + err.StackTrace);
  414. }
  415. finally
  416. {
  417. g.Restore(state);
  418. }
  419. }
  420. }
  421. }
  422. #endregion
  423. //-----------------------------------------------------------------------------------------------------------------
  424. //-----------------------------------------------------------------------------------------------------------------
  425. public void showLog(string text, float x, float y)
  426. {
  427. if (ShowLog)
  428. logs.Add(new DisplayLog(text, x, y, Color.White));
  429. }
  430. public void showLog(string text, float x, float y, Color color)
  431. {
  432. if (ShowLog)
  433. logs.Add(new DisplayLog(text, x, y, color));
  434. }
  435. public class DisplayLog
  436. {
  437. public readonly Vector2 pos = new Vector2();
  438. private Brush brush;
  439. private int time = 20;
  440. private readonly string text;
  441. internal DisplayLog(string text, float x, float y, Color color)
  442. {
  443. this.text = text;
  444. this.time = 20;
  445. this.pos.SetX(x);
  446. this.pos.SetY(y);
  447. this.brush = new SolidBrush(color);
  448. }
  449. internal void update()
  450. {
  451. time--;
  452. }
  453. internal void render(Graphics g)
  454. {
  455. SizeF size = g.MeasureString(text, font);
  456. g.DrawString(
  457. text,
  458. font,
  459. brush,
  460. -size.Width / 2,
  461. -size.Height / 2 - 20 + time,
  462. StringFormat.GenericDefault);
  463. }
  464. internal bool Alive
  465. {
  466. get { return time > 0; }
  467. }
  468. }
  469. }
  470. //-----------------------------------------------------------------------------------------------
  471. abstract public class DisplayManhattanTerrain : DisplayTerrain
  472. {
  473. [Desc("显示地形边界", "show")]
  474. public bool ShowTerrainMesh = false;
  475. [Desc("显示空间分割", "show")]
  476. public bool ShowSpaceDiv = false;
  477. [Desc("显示Area值", "show")]
  478. public bool ShowAreaValue = false;
  479. public abstract AstarManhattan PathFinder { get; }
  480. public abstract int SpaceDivSize { get; }
  481. protected readonly Pen pen_node_mesh = new Pen(Color.FromArgb(0xFF, 0xff, 0xff, 0xff));
  482. protected readonly SolidBrush pen_area_value = new SolidBrush(Color.White);
  483. protected override void renderTerrain(Graphics g, RectangleF rect)
  484. {
  485. base.renderTerrain(g, rect);
  486. this.renderSpaceDiv(g, rect);
  487. this.renderTerrainMesh(g, rect);
  488. }
  489. protected virtual void renderTerrainMesh(Graphics g, RectangleF rect)
  490. {
  491. pen_node_mesh.Width = PenScale;
  492. if (ShowTerrainMesh || ShowAreaValue)
  493. {
  494. if (PathFinder != null)
  495. {
  496. int sx = (int)(rect.X / CellW);
  497. int sy = (int)(rect.Y / CellH);
  498. int sw = (int)(rect.Width / CellW) + 1;
  499. int sh = (int)(rect.Height / CellH) + 1;
  500. int lw = (int)(rect.Width + CellW * 2);
  501. int lh = (int)(rect.Height + CellH * 2);
  502. for (int bx = sx + sw; bx >= sx; --bx)
  503. {
  504. for (int by = sy + sh; by >= sy; --by)
  505. {
  506. var tn = PathFinder.GetMapNode(bx, by);
  507. if (tn != null)
  508. {
  509. if (ShowTerrainMesh) { renderMeshCell(g, tn); }
  510. if (ShowAreaValue) { renderAreaValue(g, tn); }
  511. }
  512. }
  513. }
  514. }
  515. }
  516. }
  517. protected virtual void renderMeshCell(Graphics g, AstarManhattan.MMapNode tn)
  518. {
  519. if (tn.Mesh != null)
  520. {
  521. using (var list = ListObjectPool<CommonLang.Geometry.Line2>.AllocAutoRelease())
  522. {
  523. tn.Mesh.ToLines(list);
  524. foreach (var line in list)
  525. {
  526. g.DrawLine(pen_node_mesh, line.P.X, line.P.Y, line.Q.X, line.Q.Y);
  527. }
  528. }
  529. }
  530. }
  531. protected virtual void renderAreaValue(Graphics g, AstarManhattan.MMapNode tn)
  532. {
  533. if (!tn.Blocked)
  534. {
  535. var text = tn.AreaValue.ToString();
  536. var save = g.Save();
  537. try
  538. {
  539. g.TranslateTransform(tn.BX * CellW, tn.BY * CellH);
  540. g.ScaleTransform(PenScale, PenScale);
  541. g.DrawString(
  542. text,
  543. font,
  544. pen_area_value,
  545. 0, 0,
  546. StringFormat.GenericDefault);
  547. }
  548. finally
  549. {
  550. g.Restore(save);
  551. }
  552. }
  553. }
  554. protected virtual void renderSpaceDiv(Graphics g, RectangleF rect)
  555. {
  556. if (ShowSpaceDiv && SpaceDivSize > 0)
  557. {
  558. drawGrid(g, rect, SpaceDivSize * CellW, SpaceDivSize * CellH);
  559. }
  560. }
  561. protected void drawPath(Graphics g, Vector2 path_begin, AstarManhattan.MWayPoint path, Pen pen_line, Pen pen_begin = null, Brush pen_brush = null)
  562. {
  563. if (path_begin != null && pen_begin != null)
  564. {
  565. g.DrawLine(pen_begin, path_begin.X, path_begin.Y, path.PosX, path.PosY);
  566. }
  567. if (path != null)
  568. {
  569. foreach (AstarManhattan.MWayPoint wp in path)
  570. {
  571. if (pen_brush != null)
  572. {
  573. g.FillRectangle(pen_brush, wp.BX * CellW, wp.BY * CellH, CellW, CellH);
  574. }
  575. if (wp.Next != null)
  576. {
  577. g.DrawLine(pen_line, wp.PosX, wp.PosY, wp.Next.PosX, wp.Next.PosY);
  578. }
  579. }
  580. }
  581. }
  582. }
  583. //-----------------------------------------------------------------------------------------------
  584. }