1 |
|
---|
2 | \documentclass[a4paper,10pt]{article}
|
---|
3 |
|
---|
4 | \usepackage{hyperref}
|
---|
5 | \usepackage{mathpazo}
|
---|
6 | \usepackage{amsmath}
|
---|
7 |
|
---|
8 | \usepackage{tikz}
|
---|
9 | \usetikzlibrary{calc}
|
---|
10 |
|
---|
11 | \begin{document}
|
---|
12 |
|
---|
13 | \section{Implement formation motion as a separate (scripted) component}
|
---|
14 |
|
---|
15 | A formation needs different paths than a regular unit. Formations can go over small obstacles, and let their units walk around it. While formations should avoid to walk right next to big obstacles, which causes half of their . This proposal aims to fullfill the special formation movement needs.
|
---|
16 |
|
---|
17 | \subsection{The aim of formation pathfinding}
|
---|
18 |
|
---|
19 | A formation is basically a rectangle, with a depth and a width. Not all terrain under a formation should be passable, but our aim is to keep as much terrain passable, so most units in formations can walk without bumping into obstructions.
|
---|
20 |
|
---|
21 | Next to that, turning formations is also a difficult case. The turn should be taken slowly to not let units run around like mads.
|
---|
22 |
|
---|
23 | The formation path should however make sure that there's at least one way all units in the formation can use.
|
---|
24 |
|
---|
25 | \subsection{Getting the path}
|
---|
26 |
|
---|
27 | The formation should get a long-range path from the pathfinder. This happens by calling the ComputePathASync method of the pathfinder, and listening to the PathResult message. Both need to be made available for JS, which isn't hard (a path can easily be encoded as a simple array of waypoints).
|
---|
28 |
|
---|
29 | A path will ony contain the waypoints where an entity has to take a turn.
|
---|
30 |
|
---|
31 | \subsubsection{Generalisation}
|
---|
32 |
|
---|
33 | After the path is received, it needs to be generalised to get rid of unneeded waypoints. If waypoints are closer to eash other than the formation size (probably best expressed as the minimum of the depth and width), then the average position of both waypoints should taken (unless one of the two was an endpoint, then the endpoint is taken). This can be recursively executed until all waypoints are at least the formation size away.
|
---|
34 |
|
---|
35 | This way of generalisation is cheap, makes sure there's still a valid path nearby (if the original path was valid), and makes sure turns don't happen very often.
|
---|
36 |
|
---|
37 | So an ideal formation path consists of long straight lines, and a few turns.
|
---|
38 |
|
---|
39 | \subsection{Following the path}
|
---|
40 | \subsubsection{Following a straight line}
|
---|
41 |
|
---|
42 | When following a straight line, the overall direction is kept and the formation keeps turned towards that direction. For each turn along the line, three possible positions are compared:
|
---|
43 |
|
---|
44 | \begin{enumerate}
|
---|
45 | \item For the walking distance of one turn forward along the direction
|
---|
46 | \item For the walking distance of one turn sideways along the direction $+45^\circ$
|
---|
47 | \item For the walking distance of one turn sideways along the direction $-45^\circ$
|
---|
48 | \end{enumerate}
|
---|
49 |
|
---|
50 | For both outer two positions, the distance to the original (generalised) path is compared. If the distance is bigger than the formation width, than the position is ignored.
|
---|
51 |
|
---|
52 | For each of the remaining positions, the pathfinder is requested to do an obstructed-tile check in the formation rectangle. This check is similar to the foundation-placement check. But here it should return the number of obstructed tiles under the formation. If possible, the pathfinder should also take units (that don't belong to our formation) into account. The position with the least obstructed tiles, will be the target for the next turn. If two positions are equal, the path closest to the original path will be taken. If it's one of the sideway positions, then the formation as a whole will not turn, but units will turn in their position.
|
---|
53 |
|
---|
54 | \subsubsection{Detecting a corner}
|
---|
55 |
|
---|
56 | On every turn, you also need to detect whether you're starting a corner. This can be done with a simple projection of the formation position onto the straight line, and comparing if the projected point is still between the waypoints.
|
---|
57 |
|
---|
58 | \subsubsection{Taking a corner}
|
---|
59 |
|
---|
60 | When taking a corner, the centre of the formation will quite often not be on the corner point, but shifted a bit. The formation should turn as a block around the waypoint of the generalised path. Thus if the centre isn't at the waypoint, it should turn in an arc around it, rotating bit by bit.
|
---|
61 |
|
---|
62 | The rotation speed should depend on the run speed of units, and the distance they have to cover (so it depends on how far from the centre it turns). When rotating, the formation shouldn't go forward.
|
---|
63 |
|
---|
64 | \subsubsection{Finalising the path}
|
---|
65 |
|
---|
66 | When the distance to the last waypoint is smaller than the distance to the straight line you're following, then you're finalising your path. From that point onwards, you should go in a straight line to the last waypoint (while keeping the formation still oriented to the path direction), and stop comparing obstructions.
|
---|
67 |
|
---|
68 | \subsection{Schematic drawing}
|
---|
69 |
|
---|
70 | The image shows a path result from the long-range pathfinder (the solid line) passing by obstacles (the grey rectangles), together with the actual path a formation will follow (the dashed path), and how the formation will be rotated (the dashed rectangles).
|
---|
71 |
|
---|
72 | You can clearly see that the path isn't optimal (you can clearly see the path deviating from the optimal path), but this isn't neccesary, as long as there's a good path nearby.
|
---|
73 |
|
---|
74 | \begin{tikzpicture}[>=stealth,scale=0.8]
|
---|
75 | \node (s) at (-5,-5) [label=below:{Start}] {};
|
---|
76 | \node (wp1) at (0,0) [label=below:{}] {};
|
---|
77 | \node (wp2) at (5,0) [label=below:{}] {};
|
---|
78 | \node (t) at (7,-2) [label=below:{Target}] {};
|
---|
79 |
|
---|
80 | \draw (wp1) circle (0.1cm);
|
---|
81 | \draw (wp2) circle (0.1cm);
|
---|
82 | \draw (s) circle (0.1cm);
|
---|
83 | \draw (t) circle (0.1cm);
|
---|
84 |
|
---|
85 |
|
---|
86 | \draw[->] (s) -- (wp1) -- (wp2) -- (t);
|
---|
87 |
|
---|
88 | % Obstruction squares
|
---|
89 | \path[draw=none,fill=black,opacity=0.2]
|
---|
90 | (2, -0.5) --
|
---|
91 | (4.5,-0.5) --
|
---|
92 | (4.5,-4.5) --
|
---|
93 | (2,-4.5) --
|
---|
94 | cycle;
|
---|
95 |
|
---|
96 | \path[draw=none,fill=black,opacity=0.2]
|
---|
97 | (-6,-2.5) --
|
---|
98 | (-6,-1) --
|
---|
99 | (-3,-1) --
|
---|
100 | (-3,-2.5) --
|
---|
101 | cycle;
|
---|
102 |
|
---|
103 | % formation path
|
---|
104 | \draw[->,dashed] (s) --
|
---|
105 | (-3.5,-3.5) --
|
---|
106 | (-2.5,-3.5) --
|
---|
107 | (-2,-3) --
|
---|
108 | (-2,-2) --
|
---|
109 | (0,0) --
|
---|
110 | (1.5, 0) --
|
---|
111 | (2, 0.5) --
|
---|
112 | (5, 0.5) arc[radius = 5mm, start angle= 90, end angle= 45] --
|
---|
113 | (5.85, -0.15) --
|
---|
114 | (5.85, -0.85) --
|
---|
115 | (t);
|
---|
116 |
|
---|
117 | %formation representations
|
---|
118 | \draw[rotate around={-45:(-3.5,-3.5)},dashed] (-4.5,-3) rectangle (-2.5,-4);
|
---|
119 | \draw[rotate around={-45:(-2.5,-3.5)},dashed] (-3.5,-3) rectangle (-1.5,-4);
|
---|
120 | %\draw[rotate around={-45:(-2,-3)},dashed] (-3,-2.5) rectangle (-1,-3.5);
|
---|
121 | \draw[rotate around={-45:(-2,-2)},dashed] (-3,-1.5) rectangle (-1,-2.5);
|
---|
122 | \draw[rotate around={-45:(0,0)},dashed] (-1,0.5) rectangle (1,-0.5);
|
---|
123 | \draw[dashed] (-0.5,1) rectangle (0.5,-1);
|
---|
124 | \draw[dashed] (1,1) rectangle (2,-1);
|
---|
125 | \draw[dashed] (1.5,1.5) rectangle (2.5,-0.5);
|
---|
126 | \draw[dashed] (4.5,1.5) rectangle (5.5,-0.5);
|
---|
127 | \draw[rotate around={-45:(5,0)},dashed] (4.5,1.5) rectangle (5.5,-0.5);
|
---|
128 | \end{tikzpicture}
|
---|
129 |
|
---|
130 | \end{document}
|
---|