DisplayLayerObject.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. using System;
  2. using CommonAI.ZoneClient;
  3. using System.Drawing;
  4. using CommonAI.Zone;
  5. using CommonLang;
  6. using CommonAI.Zone.Helper;
  7. using CommonLang.Vector;
  8. using CommonFroms.Drawing;
  9. namespace GameEditorPlugin.Win32.BattleClient
  10. {
  11. public abstract class DisplayLayerObject : DisplayObject
  12. {
  13. readonly private ZoneObject mObj;
  14. readonly public DisplayLayerWorld world;
  15. protected Pen pen;
  16. protected SolidBrush brush;
  17. protected RectangleF mLocalBounds;
  18. public DisplayLayerObject(DisplayLayerWorld wd, ZoneObject obj)
  19. {
  20. this.world = wd;
  21. this.mObj = obj;
  22. this.pen = new Pen(Color.Green);
  23. this.brush = new SolidBrush(Color.Green);
  24. }
  25. public override bool Visible { get { return true; } }
  26. public override float X { get { return mObj.X; } }
  27. public override float Y { get { return mObj.Y; } }
  28. public override float Z { get { return mObj.Z; } }
  29. public override RectangleF LocalBounds { get { return mLocalBounds; } }
  30. public string Name { get { return mObj.Name; } }
  31. public uint ObjectID { get { return mObj.ObjectID; } }
  32. public ZoneObject ZoneObjectData { get { return mObj; } }
  33. public override void renderHP(Graphics g) { }
  34. public override void renderName(Graphics g, Font font, Brush brush)
  35. {
  36. SizeF fsize = g.MeasureString(this.Name, font);
  37. g.DrawString(mObj.DisplayName, font, brush, -fsize.Width / 2, -fsize.Height);
  38. }
  39. //-------------------------------------------------------------------------------------------------
  40. #region Drawing
  41. public void RenderDirection(Graphics g, Pen pen, float direction, float range)
  42. {
  43. g.DrawLine(pen, 0, 0,
  44. (float)Math.Cos(direction) * range,
  45. (float)Math.Sin(direction) * range);
  46. }
  47. public void RenderAttackShape(
  48. Graphics g,
  49. Pen pen,
  50. AttackShape shape,
  51. float direction,
  52. float size,
  53. float distance,
  54. float angle,
  55. float strip_wide,
  56. float start_offset,
  57. ZoneObject target)
  58. {
  59. System.Drawing.Drawing2D.GraphicsState gs = g.Save();
  60. try
  61. {
  62. if (start_offset != 0)
  63. {
  64. float dx = 0;
  65. float dy = 0;
  66. MathVector.movePolar(ref dx, ref dy, direction, start_offset);
  67. g.TranslateTransform(dx, dy);
  68. }
  69. switch (shape)
  70. {
  71. case AttackShape.Round:
  72. g.DrawEllipse(pen, new RectangleF(-size, -size, size * 2, size * 2));
  73. break;
  74. case AttackShape.Circle:
  75. g.DrawEllipse(pen, new RectangleF(-size, -size, size * 2, size * 2));
  76. float sr = size - strip_wide;
  77. g.DrawEllipse(pen, new RectangleF(-sr, -sr, sr * 2, sr * 2));
  78. break;
  79. case AttackShape.Fan:
  80. DrawingUtils.DrawFan(g, pen, direction, size, angle);
  81. break;
  82. case AttackShape.Strip:
  83. {
  84. float d_width = strip_wide / 2f;
  85. //float d_angle = direction + CMath.PI_DIV_2;
  86. float d_distance = distance / 2f;
  87. Vector2 p0 = new Vector2(0, 0);
  88. Vector2 p1 = new Vector2(0, 0);
  89. MathVector.movePolar(p0, direction, -d_distance);
  90. MathVector.movePolar(p1, direction, +d_distance);
  91. DrawingUtils.DrawLineRoundRect(g, pen, p0.X, p0.Y, p1.X, p1.Y, d_width);
  92. }
  93. break;
  94. case AttackShape.StripRay:
  95. case AttackShape.StripRayTouchEnd:
  96. {
  97. float d_width = strip_wide / 2f;
  98. Vector2 p1 = new Vector2(0, 0);
  99. MathVector.movePolar(p1, direction, distance);
  100. DrawingUtils.DrawLineRoundRect(g, pen, 0, 0, p1.X, p1.Y, d_width);
  101. }
  102. break;
  103. case AttackShape.RectStrip:
  104. {
  105. float d_width = strip_wide / 2f;
  106. //float d_angle = direction + CMath.PI_DIV_2;
  107. float d_distance = distance / 2f;
  108. Vector2 p0 = new Vector2(0, 0);
  109. Vector2 p1 = new Vector2(0, 0);
  110. MathVector.movePolar(p0, direction, -d_distance);
  111. MathVector.movePolar(p1, direction, +d_distance);
  112. DrawingUtils.DrawLineRect(g, pen, p0.X, p0.Y, p1.X, p1.Y, d_width);
  113. }
  114. break;
  115. case AttackShape.RectStripRay:
  116. {
  117. float d_width = strip_wide / 2f;
  118. Vector2 p1 = new Vector2(0, 0);
  119. MathVector.movePolar(p1, direction, distance);
  120. DrawingUtils.DrawLineRect(g, pen, 0, 0, p1.X, p1.Y, d_width);
  121. }
  122. break;
  123. case AttackShape.WideStrip:
  124. {
  125. float d_width = strip_wide / 2f;
  126. float d_angle = direction + CMath.PI_DIV_2;
  127. float d_distance = distance / 2f;
  128. Vector2 p0 = new Vector2(0, 0);
  129. Vector2 p1 = new Vector2(0, 0);
  130. MathVector.movePolar(p0, d_angle, -d_width);
  131. MathVector.movePolar(p1, d_angle, +d_width);
  132. DrawingUtils.DrawLineRoundRect(g, pen, p0.X, p0.Y, p1.X, p1.Y, d_distance);
  133. }
  134. break;
  135. case AttackShape.LineToTarget:
  136. case AttackShape.LineToStart:
  137. {
  138. Vector2 p1 = new Vector2(0, 0);
  139. MathVector.movePolar(p1, direction, distance);
  140. g.DrawLine(pen, 0, 0, p1.X, p1.Y);
  141. g.DrawEllipse(pen, new RectangleF(-size, -size, size * 2, size * 2));
  142. }
  143. break;
  144. }
  145. }
  146. finally
  147. {
  148. g.Restore(gs);
  149. }
  150. }
  151. #endregion
  152. }
  153. public class DisplayGameUnit : DisplayLayerObject
  154. {
  155. readonly public ZoneUnit Data;
  156. readonly public UnitInfo Info;
  157. private static HashMap<UnitActionStatus, SolidBrush> status_bursh = new HashMap<UnitActionStatus, SolidBrush>();
  158. static DisplayGameUnit()
  159. {
  160. status_bursh[UnitActionStatus.Spawn] = new SolidBrush(Color.FromArgb(0xff, 0x40, 0x40, 0x40));
  161. status_bursh[UnitActionStatus.Idle] = new SolidBrush(Color.FromArgb(0, 0, 0, 0));
  162. status_bursh[UnitActionStatus.Damage] = new SolidBrush(Color.FromArgb(0x80, 0xFF, 0, 0));
  163. status_bursh[UnitActionStatus.Dead] = new SolidBrush(Color.FromArgb(0x80, 0x40, 0x40, 0x40));
  164. status_bursh[UnitActionStatus.Move] = new SolidBrush(Color.FromArgb(0x80, 0, 0, 0xFF));
  165. status_bursh[UnitActionStatus.Skill] = new SolidBrush(Color.FromArgb(0x80, 0xFF, 0xFF, 0));
  166. status_bursh[UnitActionStatus.Stun] = new SolidBrush(Color.FromArgb(0x80, 0xFF, 0xFF, 0xFF));
  167. status_bursh[UnitActionStatus.Chaos] = new SolidBrush(Color.FromArgb(0x80, 0xFF, 0x80, 0x80));
  168. status_bursh[UnitActionStatus.Escape] = new SolidBrush(Color.FromArgb(0x80, 0xFF, 0x80, 0x80));
  169. status_bursh[UnitActionStatus.Pause] = new SolidBrush(Color.FromArgb(0x80, 0x80, 0x80, 0x80));
  170. status_bursh[UnitActionStatus.Pick] = new SolidBrush(Color.FromArgb(0x80, 0, 0xFF, 0xFF));
  171. }
  172. protected Brush brush_dead = new SolidBrush(Color.FromArgb(128, 255, 255, 255));
  173. protected Pen pen_attack = new Pen(Color.FromArgb(0x80, 0xff, 0x00, 0x00));
  174. protected Pen pen_guard = new Pen(Color.FromArgb(0x80, 0xff, 0xff, 0x00));
  175. protected Pen pen_path = new Pen(Color.FromArgb(0xff, 0x80, 0xff, 0x80));
  176. protected Pen pen_hit = new Pen(Color.FromArgb(0xff, 0x80, 0xff, 0x80));
  177. protected Brush brush_hp = new SolidBrush(Color.FromArgb(0xff, 0, 0xff, 0));
  178. protected Brush brush_mp = new SolidBrush(Color.FromArgb(0xff, 0x80, 0x80, 0xff));
  179. protected Brush brush_black = new SolidBrush(Color.FromArgb(0xff, 0, 0, 0));
  180. protected Brush brush_attack = new SolidBrush(Color.FromArgb(0x80, 0xff, 0x00, 0x00));
  181. private SkillTemplate mBaseSkill;
  182. Pen[] pens = new Pen[]
  183. {
  184. new Pen(Color.LightBlue),
  185. new Pen(Color.Red),
  186. new Pen(Color.Green),
  187. new Pen(Color.Magenta),
  188. };
  189. public DisplayGameUnit(DisplayLayerWorld wd, ZoneUnit unit)
  190. : base(wd, unit)
  191. {
  192. this.Data = unit;
  193. this.Info = unit.Info;
  194. this.brush = new SolidBrush(Color.FromArgb(0x80, 0x80, 0x80, 0x80));
  195. if (unit.Force == 0)
  196. {
  197. this.pen = new Pen(Color.Blue);
  198. this.brush = new SolidBrush(Color.Blue);
  199. }
  200. else
  201. {
  202. this.pen = new Pen(Color.Red);
  203. this.brush = new SolidBrush(Color.Red);
  204. }
  205. if (Info != null)
  206. {
  207. this.mLocalBounds = new RectangleF(
  208. -Info.BodySize,
  209. -Info.BodySize,
  210. Info.BodySize * 2,
  211. Info.BodySize * 2);
  212. }
  213. else
  214. {
  215. this.mLocalBounds = new RectangleF(-1, -1, 2, 2);
  216. }
  217. this.pen = CUtils.GetMinOrMax<Pen>(pens, Data.Force);
  218. if (Info.BaseSkillID != null)
  219. {
  220. ZoneUnit.SkillState ss = unit.GetSkillState(Info.BaseSkillID.SkillID);
  221. if (ss != null)
  222. {
  223. mBaseSkill = ss.Data;
  224. }
  225. }
  226. this.Data.OnDoEvent += doEvent;
  227. }
  228. private void doEvent(ZoneObject obj, ObjectEvent e)
  229. {
  230. if (e is UnitLaunchSkillEvent)
  231. {
  232. UnitLaunchSkillEvent me = (UnitLaunchSkillEvent)e;
  233. }
  234. else if (e is UnitHitEvent)
  235. {
  236. UnitHitEvent me = (UnitHitEvent)e;
  237. world.showLog(me.hp.ToString(), X, Y);
  238. }
  239. else if (e is UnitDeadEvent)
  240. {
  241. UnitDeadEvent me = (UnitDeadEvent)e;
  242. }
  243. }
  244. public override void render(Graphics g)
  245. {
  246. float penscale = 1f / world.getCameraScale();
  247. pen.Width = penscale;
  248. pen_path.Width = penscale;
  249. pen_attack.Width = penscale;
  250. pen_guard.Width = penscale;
  251. pen_hit.Width = penscale;
  252. DisplayLayerWorld.pen_write.Width = penscale;
  253. float bs = Info.BodySize;
  254. if (bs > 0)
  255. {
  256. SolidBrush sbursh;
  257. if (status_bursh.TryGetValue(Data.CurrentState, out sbursh))
  258. {
  259. g.FillPie(sbursh, -bs, -bs, bs * 2, bs * 2, 0, 360);
  260. }
  261. // render body
  262. g.DrawArc(pen, -bs, -bs, bs * 2, bs * 2, 0, 360);
  263. // render direction
  264. RenderDirection(g, pen, Data.Direction, bs);
  265. }
  266. // render hit body
  267. if (world.ShowDamageRange)
  268. {
  269. if (Info.BodyHitSize > 0)
  270. {
  271. g.DrawArc(pen_hit, -Info.BodyHitSize, -Info.BodyHitSize, Info.BodyHitSize * 2, Info.BodyHitSize * 2, 0, 360);
  272. }
  273. }
  274. // render attack angle
  275. if (world.ShowAttackRange && mBaseSkill != null)
  276. {
  277. float rg = Data.GetSkillAttackRange(mBaseSkill);
  278. DrawingUtils.DrawFan(g, pen_attack, Data.Direction, rg, mBaseSkill.AttackAngle);
  279. if (Data is ZoneActor)
  280. {
  281. ZoneActor actor = Data as ZoneActor;
  282. if (actor.CurrentSkillActionData != null && actor.CurrentSkillActionData.OverrideAttackShape != null)
  283. {
  284. UnitActionData.AttackShape shape = actor.CurrentSkillActionData.OverrideAttackShape;
  285. RenderAttackShape(g, pen_attack,
  286. (AttackShape)shape.AShape,
  287. Data.Direction,
  288. shape.AttackRange,
  289. shape.AttackRange,
  290. shape.AttackAngle,
  291. shape.StripWide,
  292. shape.OffsetRadius,
  293. null);
  294. }
  295. }
  296. }
  297. if (world.ShowGuardRange)
  298. {
  299. if (Info.GuardRange > 0)
  300. g.DrawArc(pen_guard, -Info.GuardRange, -Info.GuardRange, Info.GuardRange * 2, Info.GuardRange * 2, 0, 360);
  301. if (Info.GuardRangeLimit > 0)
  302. g.DrawArc(pen_guard, -Info.GuardRangeLimit, -Info.GuardRangeLimit, Info.GuardRangeLimit * 2, Info.GuardRangeLimit * 2, 0, 360);
  303. }
  304. if (world.SelectedObject == this)
  305. {
  306. g.DrawRectangle(DisplayLayerWorld.pen_write, -bs, -bs, bs * 2, bs * 2);
  307. }
  308. if (OnRender != null)
  309. {
  310. OnRender.Invoke(g, this);
  311. }
  312. }
  313. public override void renderHP(Graphics g)
  314. {
  315. float r = 20;
  316. float sx = -r - 2;
  317. float sy = -r - 8;
  318. float sw = r * 2 + 4;
  319. float sh = 6;
  320. g.FillRectangle(brush_black, sx, sy, sw, sh);
  321. g.FillRectangle(brush_hp, sx + 1, sy + 1,
  322. (sw - 2) * Data.HP / Data.MaxHP, sh - 2);
  323. g.FillRectangle(brush_black, sx, sy + sh, sw, sh);
  324. g.FillRectangle(brush_mp, sx + 1, sy + 1 + sh,
  325. (sw - 2) * Data.MP / Data.MaxMP, sh - 2);
  326. if (Data.GetDeadTimeCD() > 0)
  327. {
  328. g.FillRectangle(brush_dead, sx + 1, sy + 1, (sw - 2) * Data.GetDeadTimeCD(), sh - 2);
  329. }
  330. if (OnRenderHP != null)
  331. {
  332. OnRenderHP.Invoke(g, this);
  333. }
  334. }
  335. public delegate void OnRenderHandler(Graphics g, DisplayGameUnit unit);
  336. public delegate void OnRenderHPHandler(Graphics g, DisplayGameUnit unit);
  337. public event OnRenderHandler OnRender;
  338. public event OnRenderHPHandler OnRenderHP;
  339. }
  340. public class DisplayGameSpell : DisplayLayerObject
  341. {
  342. readonly public ZoneSpell Data;
  343. readonly public SpellTemplate Info;
  344. private float mSize;
  345. public DisplayGameSpell(DisplayLayerWorld wd, ZoneSpell spell)
  346. : base(wd, spell)
  347. {
  348. this.Data = spell;
  349. this.Info = spell.Info;
  350. this.pen = new Pen(Color.Yellow);
  351. this.brush = new SolidBrush(Color.Yellow);
  352. if (Info != null)
  353. {
  354. this.mLocalBounds = new RectangleF(
  355. -Info.BodySize,
  356. -Info.BodySize,
  357. Info.BodySize * 2,
  358. Info.BodySize * 2);
  359. this.mSize = Info.BodySize;
  360. }
  361. else
  362. {
  363. this.mLocalBounds = new RectangleF(-1, -1, 2, 2);
  364. }
  365. }
  366. public override void render(Graphics g)
  367. {
  368. float penscale = 1f / world.getCameraScale();
  369. pen.Width = penscale;
  370. RenderAttackShape(g, pen, (AttackShape)Info.BodyShape, Data.Direction, Data.BodySize, Data.Distance, Data.Info.FanAngle, Data.Info.RectWide, 0, Data.Target);
  371. //
  372. // switch (Info.BodyShape)
  373. // {
  374. // case SpellTemplate.Shape.Round:
  375. // {
  376. // g.DrawLine(pen, 0, 0, (float)Math.Cos(Data.Direction) * mSize, (float)Math.Sin(Data.Direction) * mSize);
  377. // g.DrawEllipse(pen, new RectangleF(-mSize, -mSize, mSize * 2, mSize * 2));
  378. // }
  379. // break;
  380. // case SpellTemplate.Shape.Fan:
  381. // {
  382. // RenderFan(g, pen, Data.Direction, mSize, Data.Info.FanAngle);
  383. // }
  384. // break;
  385. // case SpellTemplate.Shape.Strip:
  386. // {
  387. // float d_angle = Data.Direction + CMath.PI_DIV_2;
  388. // float d_width = Info.RectWide / 2f;
  389. // float d_distance = Data.Distance / 2f;
  390. // Vector2 p0 = new Vector2(0, 0);
  391. // Vector2 p1 = new Vector2(0, 0);
  392. // MathVector.movePolar(p0, d_angle, -d_distance);
  393. // MathVector.movePolar(p1, d_angle, d_distance);
  394. // RenderLineRoundRect(g, pen, p0.x, p0.y, p1.x, p1.y, Info.RectWide);
  395. // }
  396. // break;
  397. // case SpellTemplate.Shape.StripRay:
  398. // case SpellTemplate.Shape.StripRayTouchEnd:
  399. // {
  400. // float d_width = Info.RectWide / 2f;
  401. // Vector2 p1 = new Vector2(0, 0);
  402. // MathVector.movePolar(p1, Data.Direction, Data.Distance);
  403. // RenderLineRoundRect(g, pen, 0, 0, p1.x, p1.y, Info.RectWide);
  404. // }
  405. // break;
  406. // case SpellTemplate.Shape.LineToTarget:
  407. // if (Data.Target != null && CMath.includeRoundPoint(Data.X, Data.Y, Data.Distance, Data.Target.X, Data.Target.Y))
  408. // {
  409. // g.DrawLine(pen, 0, 0, Data.Target.X - Data.X, Data.Target.Y - Data.Y);
  410. // }
  411. // break;
  412. // }
  413. }
  414. public override void renderName(Graphics g, Font font, Brush brush)
  415. {
  416. //base.renderName(g, font, brush);
  417. }
  418. }
  419. public class DisplayGameItem : DisplayLayerObject
  420. {
  421. readonly public ZoneItem Data;
  422. readonly public ItemTemplate Info;
  423. public DisplayGameItem(DisplayLayerWorld wd, ZoneItem spell)
  424. : base(wd, spell)
  425. {
  426. this.Data = spell;
  427. this.Info = spell.Info;
  428. this.pen = new Pen(Color.Red);
  429. this.pen.DashStyle = System.Drawing.Drawing2D.DashStyle.Custom;
  430. this.pen.DashPattern = new float[] { Info.BodySize / 8, Info.BodySize / 8 };
  431. this.pen.LineJoin = System.Drawing.Drawing2D.LineJoin.Round;
  432. this.brush = new SolidBrush(Color.Red);
  433. if (Info != null)
  434. {
  435. this.mLocalBounds = new RectangleF(
  436. -Info.BodySize,
  437. -Info.BodySize,
  438. Info.BodySize * 2,
  439. Info.BodySize * 2);
  440. }
  441. else
  442. {
  443. this.mLocalBounds = new RectangleF(-1, -1, 2, 2);
  444. }
  445. }
  446. public override void render(Graphics g)
  447. {
  448. if (world.Layer.Actor != null)
  449. {
  450. if (world.Layer.IsPickableItem(world.Layer.Actor, this.Data))
  451. {
  452. SolidBrush brush = new SolidBrush(Color.FromArgb(0x80, 0xFF, 0xFF, 0));
  453. g.FillEllipse(brush, mLocalBounds);
  454. }
  455. }
  456. float penscale = 1f / world.getCameraScale();
  457. pen.Width = penscale;
  458. g.DrawEllipse(pen, mLocalBounds);
  459. // render direction
  460. RenderDirection(g, pen, Data.Direction, Data.Info.BodySize);
  461. }
  462. }
  463. }