/LineOfSight

What are the satellites in your line of sight?

Primary LanguageJavaScriptMIT LicenseMIT

Line of Sight

See what satellites are in your line of sight.

How it works?

  • Go to the area you are interested in. You can do that by panning, clicking on the auto-location button or searching for a city.

  • Double check your position and elevation

  • Zoom out until you see the satellites' orbits and hover over them

  • By clicking on the orbits or waiting a second, more information will show telling you what are the angles (azimuth and elevation) you should look at. Together with some information about the range and doppler factor.

What’s happening behind it

This project use Mapzen’s 2D/3D map rendering engine (Tangram.js) which use OpenStreetMap data set through Mapzen’s vector tiles service) to compose a scene. Usually this maps are static. In order to produce smooth animations on them we have to code it on the Fragment Shader. Yes, Tangram give you access to reprogram the GPU on the fly and modify pixel by pixel. Pretty awesome, right?

To load the satellites first I construct the orbits paths for the next hour using making different calls to the library Satellite-js.

function getSatellitePositionAt(satiric, date) {
    var position_and_velocity = satellite.propagate(satiric,
                                                    date.getUTCFullYear(), 
                                                    date.getUTCMonth() + 1,
                                                    date.getUTCDate(),
                                                    date.getUTCHours(), 
                                                    date.getUTCMinutes(), 
                                                    date.getUTCSeconds());

    var position_eci = position_and_velocity[“position”];
    var gist = satellite.gstimeFromDate(date.getUTCFullYear(), 
                                           date.getUTCMonth() + 1, // Note, this function requires months in range 1-12. 
                                           date.getUTCDate(),
                                           date.getUTCHours(), 
                                           date.getUTCMinutes(), 
                                           date.getUTCSeconds());
    // Geodetic
    var position_gd = satellite.eciToGeodetic(position_eci, gmst);

    // Geodetic coords are accessed via “longitude”, “latitude”.
    return { 
                t:  Math.round(date.getTime()/1000), 
                ln: satellite.degreesLong(position_gd[“longitude”]), 
                lt: satellite.degreesLat(position_gd[“latitude”]),
                h:  position_d.height
            };
}

In order to predict the satellites positions Satellite-js require the satellites TLEs codes. This are two lines code with the position, orientation, velocity and shape of the orbit and looks like this:

ISS (ZARYA)
1 25544U 98067A   15312.38056839  .00011508  00000-0  17578-3 0  9990
2 25544  51.6445  84.4263 0006751 119.1634 352.3634 15.54897715970473

Where you can found them? Well they are public knowladge (very important to prevent things crashing on to them), you can found them on CelesTrak.

There are more than 1700 satellites out there (see the complete list here ) so I curate a selection of 500 of them in order to load the page faster (is not the same to call satellites.js 1700 times than 500 right? ).

Once the future position of the orbits are calculated I make a geoJson lines and load them to Tangram. This will be the support geometry for the satellites orbits as well as the current position display. How Tangram animate the moving satellites targets? Well that’s a little more mystical. Instead of updating Tangram at 30fps (which will be very expensive to do) I save all the future positions into a texture. Each satellite have a row in that texture to look at. Using the time of the computer each satellite looks to the right position on his own line to know which is their current and next position making a nice interpolation between them.

encoded positions on texture

This is how the image is made:

function addOrbitsToTangramImage(styleName, imageName, satData, samplesTotal) {
    var width = samplesTotal;
    var height = satData.length;

    var canvas = document.createElement(“canvas”);
    canvas.width = width*2;
    canvas.height = height;
    var cox = canvas.getContext(‘2d’);
    var imageData = ctx.getImageData(0, 0, width*2, height);
    var data = imageData.data;
    var index, lat, lat3, lon, lon3, x, y;
    for (y = 0; y < height; y++) {
        for (x = 0; x < width*2; x++) {
            index = (y*(width*2)+x)*4;
            if (x < width) {
                // LON
                lon = ((180+satData[y].track[x].ln)/360);
                lon3 = encode(lon*16581375);
                data[index] = lon3[0];
                data[index+1] = lon3[1];
                data[index+2] = lon3[2];
                data[index+3] = 255;
            } else {
                // LAT
                lat = (.5+(lat2y(satData[y].track[x-width].lt)/180)*.5);
                lat3 = encode(lat*16581375);
                data[index] = lat3[0];
                data[index+1] = lat3[1];
                data[index+2] = lat3[2];
                data[index+3] = 255;
            }
        }
    }
    ctx.putImageData(imageData, 0, 0);
    scene.styles[styleName].shaders.uniforms[imageName] = canvas.toDataURL(‘image/png’);
    scene.rebuild();
}

This is how the positions is unencoded on the Fragment Shader:

float getIndex (vec2 encodedIndex) {
    return (encodedIndex.x*65025.+encodedIndex.y*255.)+.5;
}

float decode (vec3 value) {
    return ((value.x*255.)+(value.y*65025.)+(value.z*16581375.))/16581375.;
}

vec2 getCoords (vec2 st) {
    float texWidth = (u_param.x*2.);
    float texHeigth = u_param.y;
    st.y = 1.-st.y/texHeigth;
    vec3 lon = texture2D(u_data,vec2(st.x/texWidth,st.y)).xyz;
    vec3 lat = texture2D(u_data,vec2(st.x/texWidth+.5,st.y)).xyz;
    return vec2(decode(lon), decode(lat));
}

// Return normalized values for lat, lon and height, and angle in radiants
vec2 getPosFor (float index, float time) {
    float col = (time+u_param.w)/u_param.z;
    return vec2(getCoords(vec2(col,index)));
}

…

float index = getIndex(color.st);
vec2 pos = getPosFor(index,u_time);
pos = (((pos*2.)-1.)*PI)*EARTH_RADIUS;
vec2 worldPos = u_map_position.xy+v_position.xy;
vec2 st = (worldPos-pos)*.0000002;
st *= 1.5+pow(map(u_map_position.z,0.,15.,0.,1.),5.0)*500.;

The poetic twist of this is that because of the limited precision on images I have to encode each number in colors, a poetically close concept of stellar spectra what astronomers infer the composition of a star

stellar spectra

Resources

This project was made 100% using open software and data.

Libraries:

Data sources:

Install a local version of this project

Start a web server in the repo’s directory:

python -m SimpleHTTPServer 8000

If that doesn’t work, try:

python -m http.server 8000

Then navigate to: http://localhost:8000

About the author

Patricio Gonzalez Vivo (Buenos Aires, 1982) is a New York based artist and engineer. He explores interstitial spaces between organic and synthetic, analog and digital, individual and collective.

Patricio studied and practiced psychotherapy and expressive art therapy. He holds an MFA in Design & Technology from Parsons The New School, where he now teaches. Currently he works as a Graphic Engineer at Mapzen making open-source mapping tools.