class Lens {
constructor(center, axis, focusLen, size, mirror = false) {
vec2.normalize(axis, axis);
Object.assign(this, { center, axis, focusLen, size });
this.reverse = mirror ? -1 : 1;
this.ray = new Ray(this.center, [-axis[1], axis[0]]);
}
hit(incomingRay, tMin = 0) {
const [t, u] = incomingRay.intersectRay(this.ray);
if (t < tMin || Math.abs(u) > this.size / 2) return false;
return [t, u];
}
refract(incomingRay) {
let { axis, ray, focusLen, center, size, reverse } = this;
if (vec2.dot(incomingRay.v, axis) * reverse < 0)
axis = vec2.scale([], axis, -1);
const tu = this.hit(incomingRay);
if (!tu) return false;
const [t, u] = tu;
const p = incomingRay.at(t);
const v1 = vec2.orthoProj([], incomingRay.v, axis);
const v2 = vec2.sub([], incomingRay.v, v1);
const sgn = Math.sign(vec2.dot(vec2.sub([], p, center), v2));
const m = vec2.len(v2) / vec2.len(v1);
const h = vec2.dist(p, center);
const sgn2 = Math.sign(vec2.dot(v2, ray.v));
const mprime = sgn2 * (m - (sgn * h) / focusLen);
const r = vec2.normalize([], vec2.scaleAndAdd([], axis, ray.v, mprime));
const porig = new Ray(p, r).at(-t);
return { t, ray: new Ray(porig, r) };
}
draw(ctx) {
let pts =
this.focusLen < 0
? [
...arrowHead(this.ray.translated(this.size / 2).rotated(Math.PI)),
...arrowHead(
this.ray
.scaled(-1)
.translated(this.size / 2)
.rotated(Math.PI)
)
]
: [
...arrowHead(this.ray.translated(this.size / 2)),
...arrowHead(this.ray.scaled(-1).translated(this.size / 2))
];
ctx.beginPath();
for (let p of pts) ctx.lineTo(...p);
ctx.fill();
ctx.stroke();
ctx.beginPath();
ctx.arc(
...vec2.scaleAndAdd([], this.center, this.axis, this.focusLen),
5,
0,
Math.PI * 2
);
ctx.fill();
}
}