Source: sketch.js

  1. /*******************************************************************************
  2. *
  3. * @file sketch.js
  4. * @brief A quick sketch showing different methods of generating a convexhull
  5. *
  6. * @author <a href='mailto:omareq08@gmail.com'> Omar Essilfie-Quaye </a>
  7. * @version 1.0
  8. * @date 27-July-2019
  9. *
  10. *******************************************************************************/
  11. let resetButton;
  12. /**
  13. * Variable that store how far away from the side edges new points can spawn
  14. *
  15. * @type {Integer}
  16. */
  17. let xBuffer;
  18. /**
  19. * Variable that store how far away from the top and bottom edges new points
  20. * can spawn
  21. *
  22. * @type {Integer}
  23. */
  24. let yBuffer;
  25. /**
  26. * Enumeration of the different algorithm types
  27. *
  28. * @enum {String}
  29. */
  30. const algorithms = Object.freeze({
  31. MONOTONE: "Monotone Chain",
  32. JARVIS: "Jarvis March",
  33. DIVIDE: "Divide & Conquer",
  34. GRAHAM: "Graham Scan"
  35. });
  36. /**
  37. * Variable that stores the current algorithm tha is being executed
  38. *
  39. * @type {Enum<String>}
  40. */
  41. let algorithm = algorithms.JARVIS;
  42. /**
  43. * Pointer for the number of points slider in the DOM
  44. *
  45. * @type {p5.Element}
  46. */
  47. let pointsSlider;
  48. /**
  49. * Number of points to be bound by the generated convex hull
  50. *
  51. * @type {Integer}
  52. */
  53. let numPoints = 45;
  54. /**
  55. * Pointer for the paragraph which shows the number of points in the DOM
  56. *
  57. * @type {p5.Element}
  58. */
  59. let pointsDisplay;
  60. /**
  61. * Size of the points drawn on the canvas
  62. *
  63. * @type {Integer}
  64. */
  65. let pointRadius = 10;
  66. /**
  67. * Pointer to select field that selects the executing algorithm
  68. *
  69. * @type {p5.Element}
  70. */
  71. let selectAlgorithm;
  72. /**
  73. * Pointer to select field that selects the method that is used to split the
  74. * points when divide and conquer is running
  75. *
  76. * @type {p5.Element}
  77. */
  78. let selectSplitMethod;
  79. /**
  80. * Pointer to select field that selects the direction that the Jarvis march
  81. * algorithm finds theconvex hull
  82. *
  83. * @type {p5.Element}
  84. */
  85. let selectAngularDirection;
  86. /**
  87. * Reset all algorithm variables to initial values
  88. */
  89. function reset() {
  90. hull = [];
  91. jarvisStep = 0;
  92. leftPointIndex = 0;;
  93. currentIndex = -1;
  94. nextIndex = 1;
  95. index = 2;
  96. time = 0;
  97. internalHulls = [[], [], []];
  98. finalHull = [];
  99. finalPoints = [];
  100. lower = [];
  101. upper = [];
  102. divideStep = divideSteps.SPLIT;
  103. jarvisStep = jarvisSteps.LEFT;
  104. monotoneStep = monotoneSteps.SORT;
  105. grahamStep = grahamSteps.SORT;
  106. points = [];
  107. for(let i = 0; i < numPoints; ++i) {
  108. const x = random(xBuffer, width - xBuffer);
  109. const y = random(yBuffer, height - yBuffer);
  110. const newPoint = createVector(x, y);
  111. points.push(newPoint);
  112. }
  113. }
  114. /**
  115. * Changes the algorithm depending on what the value of the selectAlgorithm
  116. * element is
  117. */
  118. function algorithmSelectEvent() {
  119. let selectVal = selectAlgorithm.value();
  120. if(selectVal != algorithm) {
  121. algorithm = selectVal;
  122. }
  123. if(algorithm == algorithms.DIVIDE) {
  124. selectSplitMethod.show();
  125. } else {
  126. selectSplitMethod.hide();
  127. }
  128. if(algorithm == algorithms.JARVIS) {
  129. selectAngularDirection.show();
  130. } else {
  131. selectAngularDirection.hide();
  132. }
  133. if(algorithm == algorithms.MONOTONE) {
  134. // show monotone chain options
  135. } else {
  136. // hide monotone chain options
  137. }
  138. if(algorithm == algorithms.GRAHAM) {
  139. // show graham scan options
  140. } else {
  141. // hide graham scan options
  142. }
  143. reset();
  144. }
  145. /**
  146. * Changes the method for splitting the points in divide and conquer depending
  147. * on the value of the selectSplitMethod element
  148. */
  149. function splitSelectEvent() {
  150. selectVal = selectSplitMethod.value();
  151. if(selectVal != splitMethod) {
  152. splitMethod = selectVal;
  153. reset();
  154. }
  155. }
  156. /**
  157. * Changes the direction the jarvis march algorithm selects new points for the
  158. * convex hull depending on the value of the selectAngularDirection element
  159. */
  160. function angularSelectEvent() {
  161. selectVal = selectAngularDirection.value();
  162. if(selectVal != angularDirection) {
  163. angularDirection = selectVal;
  164. reset();
  165. }
  166. }
  167. /**
  168. * Draws a convex hull with a given colour
  169. *
  170. * @param hullArray {Array<p5.Vector>} Array of points that amke up the
  171. * verticies of the hull.
  172. *
  173. * @param colour {Integer} Value of the hue of the given hull from 0 to 100
  174. */
  175. function drawHull(hullArray, colour) {
  176. push();
  177. colorMode(HSB, 100);
  178. fill(colour, 100, 100, 50);
  179. beginShape();
  180. for (let i = hullArray.length - 1; i >= 0; i--) {
  181. push();
  182. stroke(colour, 100,100);
  183. strokeWeight(0.3 * pointRadius);
  184. ellipse(hullArray[i].x, hullArray[i].y, pointRadius, pointRadius);
  185. vertex(hullArray[i].x, hullArray[i].y);
  186. pop();
  187. }
  188. endShape(CLOSE);
  189. colorMode(RGB);
  190. pop();
  191. }
  192. /**
  193. * Draws the edges of a convex hull with a given colour
  194. *
  195. * @param hullArray {Array<p5.Vector>} Array of points that amke up the
  196. * verticies of the hull.
  197. *
  198. * @param colour {Integer} Value of the hue of the given hull from 0 to 100
  199. */
  200. function drawEdges(hullArray, colour) {
  201. push();
  202. colorMode(HSB, 100);
  203. stroke(colour, 100, 100, 50);
  204. strokeWeight(0.3 * pointRadius);
  205. noFill();
  206. beginShape();
  207. for (let i = hullArray.length - 1; i >= 0; i--) {
  208. ellipse(hullArray[i].x, hullArray[i].y, pointRadius, pointRadius);
  209. vertex(hullArray[i].x, hullArray[i].y);
  210. }
  211. endShape();
  212. colorMode(RGB);
  213. pop();
  214. }
  215. /**
  216. * p5.js setup function, creates canvas.
  217. */
  218. function setup() {
  219. let cnvSize;
  220. if(windowWidth > windowHeight) {
  221. cnvSize = 0.95 * windowHeight;
  222. } else {
  223. cnvSize = 0.6 * windowWidth;
  224. }
  225. let cnv = createCanvas(cnvSize, 0.7 * cnvSize);
  226. cnv.parent("sketch");
  227. pointsSlider = createSlider(12, 100, numPoints, 1);
  228. pointsSlider.parent("num-points");
  229. pointsDisplay = createP(numPoints);
  230. pointsDisplay.parent("points-val");
  231. selectAlgorithm = createSelect();
  232. selectAlgorithm.parent("algorithm");
  233. selectAlgorithm.option(algorithms.JARVIS);
  234. selectAlgorithm.option(algorithms.DIVIDE);
  235. selectAlgorithm.option(algorithms.MONOTONE);
  236. selectAlgorithm.option(algorithms.GRAHAM);
  237. selectAlgorithm.changed(algorithmSelectEvent);
  238. selectAngularDirection = createSelect();
  239. selectAngularDirection.parent("algorithm-options");
  240. selectAngularDirection.option(direction.CLOCKWISE);
  241. selectAngularDirection.option(direction.ANTICLOCKWISE);
  242. selectAngularDirection.changed(angularSelectEvent);
  243. selectAngularDirection.hide();
  244. selectSplitMethod = createSelect();
  245. selectSplitMethod.parent("algorithm-options");
  246. selectSplitMethod.option(splitMethods.HORIZONTAL);
  247. selectSplitMethod.option(splitMethods.VERTICAL);
  248. selectSplitMethod.option(splitMethods.RADIAL);
  249. selectSplitMethod.changed(splitSelectEvent);
  250. selectSplitMethod.hide();
  251. //TODO (omar: omareq08@gmail.com): ANGULAR split method for dnc notworking
  252. // selectSplitMethod.option(splitMethods.ANGULAR);
  253. resetButton = createButton("Reset", "value");
  254. resetButton.parent("reset-button");
  255. resetButton.mousePressed(reset);
  256. xBuffer = 0.05 * width;
  257. yBuffer = 0.05 * height;
  258. for(let i = 0; i < numPoints; ++i) {
  259. const x = random(xBuffer, width - xBuffer);
  260. const y = random(yBuffer, height - yBuffer);
  261. const newPoint = createVector(x, y);
  262. points.push(newPoint);
  263. }
  264. }
  265. /**
  266. * p5.js draw function, is run every frame to create the desired animation
  267. */
  268. function draw() {
  269. background(0);
  270. fill(255);
  271. textSize(height * 0.03);
  272. textAlign(LEFT, TOP);
  273. noStroke();
  274. text(algorithm, 0, 0);
  275. let sliderVal = pointsSlider.value();
  276. if(sliderVal != numPoints) {
  277. numPoints = sliderVal;
  278. pointsDisplay.elt.innerText = "Number of Points: " + str(numPoints);
  279. reset();
  280. }
  281. for (var i = points.length - 1; i >= 0; i--) {
  282. ellipse(points[i].x, points[i].y, pointRadius, pointRadius);
  283. }
  284. switch(algorithm) {
  285. case algorithms.JARVIS:
  286. jarvisMarch();
  287. break;
  288. case algorithms.MONOTONE:
  289. monotoneChain();
  290. break;
  291. case algorithms.DIVIDE:
  292. divideAndConquer();
  293. break;
  294. case algorithms.GRAHAM:
  295. grahamScan();
  296. break;
  297. default:
  298. background(255);
  299. break;
  300. }
  301. }

Documentation generated by JSDoc 3.6.3 on Sun Jun 05 2022 20:10:36 GMT+0100 (BST)