Out: Friday Sep 21, 2018
Due: Friday Oct 5, 2018 at 11:59pm
Work in groups of up to 4.
In this assignment, you will build a web application that mimics the basic behavior of Google Street View:
Your application will take a number of spherical panorama images taken around the Cornell campus, and display them as if the user is standing at the center of the sphere. When the user clicks and drags, the camera will pan and tilt according to their mouse movements. When the user scrolls, the camera will zoom in and out accordingly. Your program will also display semi-opaque icons that indicate the direction towards other panoramas. When the user clicks on a badge, the panorama currently being viewed will be replaced with the panorama at the neighboring location.
First, download the starter code. We recommend creating a git repository for development, but please don't share your code publicly.
Inside the directory, open index.html
, and you should see a web page with only the page's
header, a WebGL canvas, a short bit of JavaScript code, and some text asking you to list your team members.
Please modify index.html
so that it displays the panoramas in a way similar to what is shown
in the video. You may also add other code to the directory. However, do not use any 3D rendering
framework such as three.js. Linear algebra libraries such as
glMatrix are fine. Our solution code, however, does not use any
extra libraries besides what is already in the ZIP file.
Once you're done writing your solution, ZIP the cs4621/ppa02_student
directory, and submit
the ZIP file to CMS.
There are a few different pieces to this assignment. We will describe how they should behave in your program below, but we leave most of the implementation details (e.g., data management, the JavaScript/GLSL split, how many and what kind of attribute, varying, and uniform variables to use, etc.) up to you.
We provide a number of JPEG images in the img/
directory. Each of these is a spherical panorama at
a particular location on campus as indicated by the name of the image. The user will view one of these panoramas at a time.
Each of these images has their pixels laid out according to spherical coordinates, where rows of pixels correspond to
lines of latitude, and columns of pixels correspond to lines of longitude.
To view the image correctly, each pixel on the canvas must calculate spherical coordinates, and then translate these to texture coordinates, which will then be used to sample the image. You can calculate each pixel's spherical coordinates from a camera model that is very similar to the model from the previous assignment. The camera's position doesn't change as the user looks around the panorama, so the region of the image displayed to the canvas depends only on the camera's viewing direction and its field of view.
Keep in mind there are many edge cases to account for here. What happens when the user views the seam where edges of the image meet? What happens when the user looks at (or looks near) the poles? Is it true that each row of pixels on the canvas has the same latitude? If you see some form of distortion in your images, we recommend testing this piece of the program by coloring each pixel according to its texture coordinate.
When the user clicks and drags, the camera should pan and tilt accordingly. When the user is looking out towards the horizon, the spot on the panorama that the user clicks should nearly follow the cursor as they drag it around the canvas. When the user is looking near the poles, however, this requirement is relaxed. You should prevent the user from tilting upwards when they are looking straight up or tilting downwards when they are looking straight down, and you shouldn't allow the user to flip the camera upside down or otherwise modify the camera's roll. In short, the camera rotation behavior should be similar to that of Google Street View. As a side note, you might find it useful to keep the camera rotating as the user drags beyond the bounds of the canvas, though this is not required.
When a user scrolls on the canvas, you should prevent the page from scrolling as it normally would. Instead, you should zoom the camera in when the user scrolls up, or zoom the camera out when the user scrolls down. You can zoom by modifying the camera's field of view. You should clamp the field of view to a range of reasonable values (e.g., between 1 and 80 degrees).
Each panorama image contains metadata that describes (among other things) where the photo was taken and the orientation of the image. To extract this metadata, we use the exif-js library (included in the framework). This library takes raw image data and extracts metadata in the form of EXIF, IPTC, and XMP tags. However, it is a little inconvenient to use when you are extracting data from several images at once.
To ease this process, we have provided a function called getImageMetadata(queue, callback)
in
index.html
. This function takes as arguments a PreloadJS LoadQueue
object (in which all
images have previously been loaded), and a callback function. The getImageMetadata
function returns immediately,
but the images will be processed asynchronously. When the metadata is ready, the input callback function will be called
with a single argument, which is an array of objects that is the same length as the input queue. Each object in this array
contains four fields:
id
: A string which is the id associated with the image in the PreloadJS LoadQueue
.latitude:
An array (length 3) of numbers, containing the degrees, minutes, and seconds of latitude north.longitude:
An array (length 3) of numbers, containing the degrees, minutes, and seconds of longitude west.heading:
An integer in the range [0, 359], representing the direction in the center of the panorama. This is specified as a compass heading; that is, 0 degrees is north, and degrees increase as you turn clockwise.Once you have the location of each image, you should determine which locations should be accessible from each of the others. That is, you should construct a graph between the locations where edges indicate that it is possible to move from one location to the other. In order to make sure there are no dead ends, I recommend using the following criteria to create edges:
Once you have this graph, you should display icons on the panorama showing the directions towards neighboring locations. This icon has
the path img/chevron.png
. This image contains transparency data to separate foreground from background, which you should
use when displaying it on the canvas. In addition, you should modify the foreground of the image so it is semi-opaque, i.e., the
panorama should still be slightly visible underneath the icon. When the user hovers over an icon, create some visual indication that
the icon is selected; for instance, make the icon brighter or more opaque. When the user clicks on an icon, they should be taken to the
appropriate location, i.e., the current panorama should be exchanged for the one in the direction of the icon.
The icon should be placed at a reasonable location on the canvas (i.e., pointing in a direction that is consistent with the current panorama and the relative direction towards the neighboring panorama), and should not be distorted. The icon should be considered clicked if and only if the user clicks on a non-transparent pixel on the icon. It is okay if the icon does not change size as the user zooms in or out, but make sure that the clickable area of the icon is consistent with this behavior. If the user clicks on a location where two icons overlap, you may break ties arbitrarily, but this should be consistent with whichever icon is highlighted.
In order to receive full credit, you must implement at least one other feature beyond what is specified above. The amount of points you will receive depends on the effort spent on extra features. Implementing many features, or implementing an especially impressive feature, may earn extra credit. Some ideas for extra features are listed below.
If you would like to make your own spherical panorama images, you can do so with a modern smartphone. The Google Street View app allows you to capture panoramas by taking several photos with your smartphone camera, which it then stitches together into a single image. You can then transfer captured images to your computer. (The app also allows you to download spherical panoramas that others have captured and shared publicly, but these have low resolution).