xxxxxxxxxx
113
// Daniel Shiffman
// https://thecodingtrain.com/CodingChallenges/148-gift-wrapping.html
// https://youtu.be/YNyULRrydVI
// https://editor.p5js.org/codingtrain/sketches/IVE9CxBOF
// Gift Wrapping Algorithm
const points = [];
const hull = [];
let leftMost;
let currentVertex;
let index;
let nextIndex = -1;
let nextVertex;
function setup() {
createCanvas(500, 500);
let buffer = 20;
for (let i = 0; i < 50; i++) {
points.push(createVector(random(buffer, width - buffer), random(buffer, height - buffer)));
}
points.sort((a, b) => a.x - b.x);
leftMost = points[0];
currentVertex = leftMost;
hull.push(currentVertex);
nextVertex = points[1];
index = 2;
}
const LEFTMOST = 0;
const NEXTPOINT = 1;
const ADDVERTEX = 2;
function* alg() {
for (;;) {
let checking = points[index];
yield ['checking', {checking}];
const a = p5.Vector.sub(nextVertex, currentVertex);
const b = p5.Vector.sub(checking, currentVertex);
const cross = a.cross(b);
if (cross.z < 0) {
nextVertex = checking;
nextIndex = index;
}
index = index + 1;
if (index == points.length) {
if (nextVertex == leftMost) {
return;
} else {
hull.push(nextVertex);
currentVertex = nextVertex;
index = 0;
//points.splice(nextIndex, 1);
nextVertex = leftMost;
}
}
yield ['step', {}];
}
}
let gen = alg();
function draw() {
background(0);
stroke(255);
strokeWeight(8);
for (let p of points) {
point(p.x, p.y);
}
stroke(0, 0, 255);
fill(0, 0, 255, 50);
beginShape();
for (let p of hull) {
vertex(p.x, p.y);
}
endShape(CLOSE);
stroke(0, 255, 0);
strokeWeight(32);
point(leftMost.x, leftMost.y);
stroke(200, 0, 255);
strokeWeight(32);
point(currentVertex.x, currentVertex.y);
stroke(0, 255, 0);
strokeWeight(2);
line(currentVertex.x, currentVertex.y, nextVertex.x, nextVertex.y);
for (;;) {
let {value, done} = gen.next();
if (done) {
console.log('done');
noLoop();
return;
}
let [step, data] = value;
if (step === 'checking') {
let {checking} = data;
stroke(255);
line(currentVertex.x, currentVertex.y, checking.x, checking.y);
} else if (step === 'step') {
break;
}
}
}