summaryrefslogtreecommitdiff
blob: e7d44322d54587aec045dfea56e8f1a58eefac52 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
-module(shr_location).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% TYPES %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-type type() :: ({non_neg_integer(), non_neg_integer()} | 'nowhere').

-export_type([type/0]).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% EXPORTS %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-export
(
   [
      decode/1,
      encode/1,
      get_nowhere/0
   ]
).

-export
(
   [
      generate_neighborhood/4,
      apply_direction/2,
      dist/2
   ]
).

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% LOCAL FUNCTIONS %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-spec validate ({integer(), integer()}) -> type().
validate ({X, Y}) ->
   if
      (X < 0) -> nowhere;
      (Y < 0) -> nowhere;
      true -> {X, Y}
   end.

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% EXPORTED FUNCTIONS %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
-spec get_nowhere () -> type().
get_nowhere () -> nowhere.

-spec apply_direction (shr_direction:enum(), type()) -> type().
apply_direction (left, {X, Y}) ->
   validate({(X - 1), Y});
apply_direction (right, {X, Y}) ->
   validate({(X + 1), Y});
apply_direction (up, {X, Y}) ->
   validate({X, (Y - 1)});
apply_direction (down, {X, Y}) ->
   validate({X, (Y + 1)});
apply_direction (_, nowhere) ->
   error("Trying to move from 'nowhere'."),
   nowhere.

-spec dist(type(), type()) -> non_neg_integer().
dist ({OX, OY}, {DX, DY}) ->
   (abs(DY - OY) + abs(DX - OX));
dist (_, _) ->
   error("Trying to measure distance to 'nowhere'"),
   999.

-spec encode (type()) -> {list(any())}.
encode ({X, Y}) ->
   {
      [
         {<<"x">>, X},
         {<<"y">>, Y}
      ]
   };
encode (nowhere) ->
   {
      [
         {<<"x">>, -1},
         {<<"y">>, -1}
      ]
   }.

-spec decode (map()) -> type().
decode (Map) ->
   X = maps:get(<<"x">>, Map),
   Y = maps:get(<<"y">>, Map),

   true = (is_integer(X) and is_integer(Y)),

   validate({X, Y}).

-spec generate_neighborhood
   (
      non_neg_integer(),
      non_neg_integer(),
      non_neg_integer(),
      type()
   )
   -> ordsets:ordset(type()).
generate_neighborhood(_, _, _, nowhere) -> ordsets:new();
generate_neighborhood(MapWidth, MapHeight, Dist, {StartingX, StartingY}) ->
   lists:foldl
   (
      fun (CurrentYMod, CurrentYResult) ->
         CurrentY = (StartingY + CurrentYMod),
         case ((CurrentY < 0) or (CurrentY >= MapHeight)) of
            true -> CurrentYResult;
            false ->
               XDistRange = Dist - abs(CurrentYMod),
               lists:foldl
               (
                  fun (CurrentXMod, CurrentResult) ->
                     CurrentX = (StartingX + CurrentXMod),
                     case ((CurrentX < 0) or (CurrentX >= MapWidth)) of
                        true -> CurrentResult;
                        false ->
                           ordsets:add_element
                           (
                              {CurrentX, CurrentY},
                              CurrentResult
                           )
                     end
                  end,
                  CurrentYResult,
                  lists:seq(-XDistRange, XDistRange)
               )
         end
      end,
      ordsets:new(),
      lists:seq(-Dist, Dist)
   ).