This “360-degree panorama” source image is in equirectangular coordinates, and is reprojected here using the stereographic projection.
Zoom & pan!
Forked from Mike Bostock’s Milky Way.
This interactive requires WebGL, but see an earlier, slower software implementation of raster reprojection for comparison.
<!DOCTYPE html>
<meta charset="utf-8">
<script src="//d3js.org/d3.v3.min.js"></script>
<style>
body {
margin: 0;
overflow: hidden;
}
canvas {
cursor: move;
}
</style>
<canvas></canvas>
<script id="vertex-shader" type="x-shader/x-vertex">
attribute vec2 a_position;
void main(void) {
gl_Position = vec4(a_position, 0.0, 1.0);
}
</script>
<script id="fragment-shader" type="x-shader/x-fragment">
precision mediump float;
uniform sampler2D u_image;
uniform vec2 u_translate;
uniform float u_scale;
uniform vec2 u_rotate;
const float c_pi = 3.14159265358979323846264;
const float c_halfPi = c_pi * 0.5;
const float c_twoPi = c_pi * 2.0;
float cosphi0 = cos(u_rotate.y);
float sinphi0 = sin(u_rotate.y);
void main(void) {
float x = (gl_FragCoord.x - u_translate.x) / u_scale;
float y = (u_translate.y - gl_FragCoord.y) / u_scale;
// inverse stereographic projection
float rho = sqrt(x * x + y * y);
float c = 2.0 * atan(rho);
float sinc = sin(c);
float cosc = cos(c);
float lambda = atan(x * sinc, rho * cosc);
float phi = asin(y * sinc / rho);
// inverse rotation
float cosphi = cos(phi);
float x1 = cos(lambda) * cosphi;
float y1 = sin(lambda) * cosphi;
float z1 = y * sinc / rho;
lambda = atan(y1, x1 * cosphi0 + z1 * sinphi0) + u_rotate.x;
phi = asin(z1 * cosphi0 - x1 * sinphi0);
gl_FragColor = texture2D(u_image, vec2((lambda + c_pi) / c_twoPi, (phi + c_halfPi) / c_pi));
gl_FragColor[0] = gl_FragColor[0];
gl_FragColor[1] = gl_FragColor[1];
gl_FragColor[2] = gl_FragColor[2];
}
</script>
<script>
// Select the canvas from the document.
var canvas = document.querySelector("canvas");
// Create the WebGL context, with fallback for experimental support.
var context = canvas.getContext("webgl") || canvas.getContext("experimental-webgl");
// Compile the vertex shader.
var vertexShader = context.createShader(context.VERTEX_SHADER);
context.shaderSource(vertexShader, document.querySelector("#vertex-shader").textContent);
context.compileShader(vertexShader);
if (!context.getShaderParameter(vertexShader, context.COMPILE_STATUS)) throw new Error(context.getShaderInfoLog(vertexShader));
// Compile the fragment shader.
var fragmentShader = context.createShader(context.FRAGMENT_SHADER);
context.shaderSource(fragmentShader, document.querySelector("#fragment-shader").textContent);
context.compileShader(fragmentShader);
if (!context.getShaderParameter(fragmentShader, context.COMPILE_STATUS)) throw new Error(context.getShaderInfoLog(fragmentShader));
// Link and use the program.
var program = context.createProgram();
context.attachShader(program, vertexShader);
context.attachShader(program, fragmentShader);
context.linkProgram(program);
if (!context.getProgramParameter(program, context.LINK_STATUS)) throw new Error(context.getProgramInfoLog(program));
context.useProgram(program);
// Define the positions (as vec2) of the square that covers the canvas.
var positionBuffer = context.createBuffer();
context.bindBuffer(context.ARRAY_BUFFER, positionBuffer);
context.bufferData(context.ARRAY_BUFFER, new Float32Array([
-1.0, -1.0,
+1.0, -1.0,
+1.0, +1.0,
-1.0, +1.0
]), context.STATIC_DRAW);
// Bind the position buffer to the position attribute.
var positionAttribute = context.getAttribLocation(program, "a_position");
context.enableVertexAttribArray(positionAttribute);
context.vertexAttribPointer(positionAttribute, 2, context.FLOAT, false, 0, 0);
// Extract the projection parameters.
var translateUniform = context.getUniformLocation(program, "u_translate"),
scaleUniform = context.getUniformLocation(program, "u_scale"),
rotateUniform = context.getUniformLocation(program, "u_rotate");
// Load the reference image.
var image = new Image;
image.src = "image2.jpg";
image.onload = readySoon;
self.onresize = resize;
var width = 960,
height = 500;
// Hack to ensure correct inference of window dimensions.
function readySoon() {
setTimeout(function () {
resize();
ready();
}, 10);
}
function resize() {
width = Math.max(width, self.innerWidth);
height = Math.max(height, self.innerHeight);
canvas.setAttribute("width", width);
canvas.setAttribute("height", height);
context.uniform2f(translateUniform, width / 2, height / 2);
context.viewport(0, 0, width, height);
}
function ready() {
// Create a texture and a mipmap for accurate minification.
var texture = context.createTexture();
context.bindTexture(context.TEXTURE_2D, texture);
context.texParameteri(context.TEXTURE_2D, context.TEXTURE_MAG_FILTER, context.LINEAR);
context.texParameteri(context.TEXTURE_2D, context.TEXTURE_MIN_FILTER, context.LINEAR_MIPMAP_LINEAR);
context.texImage2D(context.TEXTURE_2D, 0, context.RGBA, context.RGBA, context.UNSIGNED_BYTE, image);
context.generateMipmap(context.TEXTURE_2D);
// The current rotation and speed.
var scale = 1300,
translate = [0,0],
speed = [-0.01, 0];
// Rotate and redraw!
function redraw() {
context.uniform1f(scaleUniform, scale);
context.uniform2fv(rotateUniform, [ -5 * translate[0] / scale, 3 * translate[1] / scale ]);
context.bindTexture(context.TEXTURE_2D, texture); // XXX Safari
context.drawArrays(context.TRIANGLE_FAN, 0, 4);
}
interact = 1;
zoom = d3.behavior.zoom()
.scale(scale)
.center([0,0])
.scaleExtent([ scale * .5, scale * 2 ])
.on("zoom.redraw", function () {
scale = d3.event.scale;
var translate0 = translate;
translate = d3.event.translate;
var dx = translate[0] - translate0[0],
dy = translate[1] - translate0[1];
redraw();
if (d3.event.sourceEvent) {
interact = 1;
if (Math.abs(dx) > Math.max(0.005, Math.abs(dy)))
speed = [ -0.05 * dx / Math.abs(dx), 0];
else if (Math.abs(dy) > Math.max(0.005, Math.abs(dx)))
speed = [0, -0.05 * dy / Math.abs(dy)];
else
speed = [0, 0];
}
})
d3.select("canvas")
.call(zoom);
redraw();
var elapsed = null;
function animate(t) {
interact -= 0.1;
if (interact < 0) {
translate[0] += speed[0] * (elapsed - t);
translate[1] += speed[1] * (elapsed - t);
d3.select("canvas")
.call(zoom.translate(translate).event);
redraw();
}
elapsed = t;
requestAnimationFrame(animate);
}
animate();
}
// A polyfill for requestAnimationFrame.
if (!self.requestAnimationFrame) requestAnimationFrame =
self.webkitRequestAnimationFrame || self.mozRequestAnimationFrame || self.msRequestAnimationFrame || self.oRequestAnimationFrame || function (f) {
setTimeout(f, 17);
};
</script>