R-PAGE
Resistance's Portable-Adventure-Game-Engine
utils.c
Go to the documentation of this file.
1 /* Resistance's Portable-Adventure-Game-Engine (R-PAGE), Copyright (C) 2019 François Gutherz, Resistance.no
2  Released under MIT License, see license.txt for details.
3 */
4 
5 #include <stdio.h>
6 #include <math.h>
7 // #include <exec/types.h>
8 #include "utils.h"
9 
10 #define FP_SHFT 0
11 
12 int qsqr(int i)
13 {
14  int j = 0;
15  while(j * j < i)
16  j++;
17 
18  return j;
19 }
20 
21 void swap(char *x, char *y)
22 {
23  char tmp;
24 
25  tmp = *x;
26  *x = *y;
27  *y = tmp;
28 }
29 
30 short str_find_delimiter(short start, char *str)
31 {
32  short l = (short)strlen(str);
33  while(start < l)
34  {
35  if (str[start] == '\n')
36  return start;
37  start++;
38  }
39 
40  return start;
41 }
42 
43 /* A utility function to reverse a string */
44 void reverse(char str[], int length)
45 {
46  int start = 0;
47  int end = length -1;
48  while (start < end)
49  {
50  swap(str+start, str+end);
51  start++;
52  end--;
53  }
54 }
55 
56 /* Custom Implementation of itoa() */
57 char* citoa(int num, char* str, int base)
58 {
59  int i = 0;
60  BOOL isNegative = FALSE;
61 
62  /* Handle 0 explicitely, otherwise empty string is printed for 0 */
63  if (num == 0)
64  {
65  str[i++] = '0';
66  str[i] = '\0';
67  return str;
68  }
69 
70  // In standard itoa(), negative numbers are handled only with
71  // base 10. Otherwise numbers are considered unsigned.
72  if (num < 0 && base == 10)
73  {
74  isNegative = TRUE;
75  num = -num;
76  }
77 
78  // Process individual digits
79  while (num != 0)
80  {
81  int rem = num % base;
82  str[i++] = (rem > 9)? (rem-10) + 'a' : rem + '0';
83  num = num/base;
84  }
85 
86  // If number is negative, append '-'
87  if (isNegative)
88  str[i++] = '-';
89 
90  str[i] = '\0'; // Append string terminator
91 
92  // Reverse the string
93  reverse(str, i);
94 
95  return str;
96 }
97 
98 int range_adjust(int val, int in_lower, int in_upper, int out_lower, int out_upper)
99 {
100  /* return (val - in_lower) / (in_upper - in_lower) * (out_upper - out_lower) + out_lower */
101  return ((val - in_lower) * (out_upper - out_lower)) / (in_upper - in_lower) + out_lower;
102 }
103 
104 int clamp(int x, int in_lower, int in_upper){
105  return min(max(x, in_lower), in_upper);
106 }
107 
109 {
110  if (pt != NULL)
111  if (pt->x >= r->sx && pt->y >= r->sy)
112  if (pt->x < r->ex && pt->y < r->ey)
113  return TRUE;
114  return FALSE;
115 }
116 
118 {
119  vec2 pt_list[4];
120 
121  pt_list[0].x = pl->p0.x;
122  pt_list[0].y = pl->p0.y;
123  pt_list[1].x = pl->p1.x;
124  pt_list[1].y = pl->p1.y;
125  pt_list[2].x = pl->p2.x;
126  pt_list[2].y = pl->p2.y;
127  pt_list[3].x = pl->p3.x;
128  pt_list[3].y = pl->p3.y;
129 
130  return point_within_polygon(pt, pt_list, 4);
131 }
132 
133 /* Point VS Polygon test, works in integer.
134  Routine by By https://github.com/JustasB */
135 BOOL point_within_polygon(vec2 *pt, vec2 *pt_list, unsigned short n_pt)
136 {
137  short i, j;
138  short pos = 0, neg = 0;
139  short x, y, x1, y1, x2, y2, d;
140  //Check if a triangle or higher n-gon
141  if (n_pt < 3)
142  {
143  // printf("point_within_polygon(), n_pt < 3 !\n");
144  return FALSE;
145  }
146 
147  //n>2 Keep track of cross product sign changes
148 
149  for (i = 0; i < n_pt; i++)
150  {
151  //If point is in the polygon
152  if (pt_list[i].x == pt->x && pt_list[i].y == pt->y)
153  return TRUE;
154 
155  //Form a segment between the i'th point
156  x1 = pt_list[i].x;
157  y1 = pt_list[i].y;
158 
159  //And the i+1'th, or if i is the last, with the first point
160  j = i < (n_pt - 1) ? i + 1 : 0;
161 
162  x2 = pt_list[j].x;
163  y2 = pt_list[j].y;
164 
165  x = pt->x;
166  y = pt->y;
167 
168  //Compute the cross product
169  d = (x - x1)*(y2 - y1) - (y - y1)*(x2 - x1);
170 
171  if (d > 0) pos++;
172  if (d < 0) neg++;
173 
174  //If the sign changes, then point is outside
175  if (pos > 0 && neg > 0)
176  return FALSE;
177  }
178 
179  //If no change in direction, then on same side of all segments, and thus inside
180  return TRUE;
181 }
short ex
Definition: utils.h:69
#define FALSE
Definition: utils.h:41
short sy
Definition: utils.h:69
short y
Definition: utils.h:54
void swap(char *x, char *y)
Definition: utils.c:21
int range_adjust(int val, int in_lower, int in_upper, int out_lower, int out_upper)
Definition: utils.c:98
int qsqr(int i)
Definition: utils.c:12
Definition: utils.h:52
vec2 p2
Definition: utils.h:64
BOOL point_within_polygon(vec2 *pt, vec2 *pt_list, unsigned short n_pt)
Definition: utils.c:135
void reverse(char str[], int length)
Definition: utils.c:44
int BOOL
Definition: utils.h:32
BOOL point_within_rect(vec2 *pt, rect *r)
Definition: utils.c:108
char * citoa(int num, char *str, int base)
Definition: utils.c:57
short str_find_delimiter(short start, char *str)
Definition: utils.c:30
short sx
Definition: utils.h:69
int clamp(int x, int in_lower, int in_upper)
Definition: utils.c:104
vec2 p1
Definition: utils.h:64
short x
Definition: utils.h:54
#define min(x, y)
Definition: utils.h:45
Definition: utils.h:67
short ey
Definition: utils.h:69
#define max(x, y)
Definition: utils.h:49
vec2 p3
Definition: utils.h:64
#define TRUE
Definition: utils.h:37
Definition: utils.h:62
vec2 p0
Definition: utils.h:64
BOOL point_within_quad(vec2 *pt, poly *pl)
Definition: utils.c:117