block by fil 497d327919606ba6024cb13b4bee38a4

Raster Reprojection of panorama images (360°) with WebGL

Full Screen

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.

index.html

<!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>