class Arcball {
constructor(canvas) {
this.canvas = canvas;
this.center = [canvas.width / 2, canvas.height / 2];
this.radius = (Math.min(canvas.width, canvas.height) / 2) * 0.9;
this.prev = [];
this.transform = mat4.create();
this.mouseIsDown = false;
}
ballPos(p) {
let u = vec2.sub([], p, this.center);
if (vec2.len(u) >= this.radius) return vec3.normalize([], [...u, 0]);
let z = Math.sqrt(this.radius ** 2 - u[0] ** 2 - u[1] ** 2);
return vec3.normalize([], [...u, z]);
}
mousedown(x, y) {
this.mouseIsDown = true;
this.prev = [x, y];
}
mousemove(x, y) {
if (!this.mouseIsDown) return this.transform;
let mouse = [x, y];
let u = this.ballPos(this.prev);
let v = this.ballPos(mouse);
this.prev = mouse;
let axis = vec3.cross([], v, u);
let ang = Math.asin(vec3.len(axis));
let rot = mat4.fromRotation([], ang, vec3.normalize([], axis));
[this.u, this.v, this.ang, this.axis] = [u, v, ang, axis];
return mat4.mul(this.transform, this.transform, rot);
}
mouseup(x, y) {
this.mouseIsDown = false;
}
}