landxml/lib/projection.js
2020-12-10 16:54:02 +01:00

168 lines
5.5 KiB
JavaScript

import proj4 from 'proj4';
// Development tools
// Map application with coordinates projection
// https://dominoc925-pages.appspot.com/mapplets/cs_ecef.html
// The World coordinates converter
// https://twcc.fr/#
export const Projections = [
{
title: "RGF93 (Lambert-93)",
url: "http://spatialreference.org/ref/epsg/rgf93-lambert-93/",
code: "EPSG:2154",
proj4: "+proj=lcc +lat_1=49 +lat_2=44 +lat_0=46.5 +lon_0=3 +x_0=700000 +y_0=6600000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC50)",
code: "EPSG:3950",
url: "http://spatialreference.org/ref/epsg/3950/",
proj4: "+proj=lcc +lat_1=49.25 +lat_2=50.75 +lat_0=50 +lon_0=3 +x_0=1700000 +y_0=9200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC49)",
code: "EPSG:3949",
url: "http://spatialreference.org/ref/epsg/3949/",
proj4: "+proj=lcc +lat_1=48.25 +lat_2=49.75 +lat_0=49 +lon_0=3 +x_0=1700000 +y_0=8200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC48)",
code: "EPSG:3948",
url: "http://spatialreference.org/ref/epsg/3948/",
proj4: "+proj=lcc +lat_1=47.25 +lat_2=48.75 +lat_0=48 +lon_0=3 +x_0=1700000 +y_0=7200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC47)",
code: "EPSG:3947",
url: "http://spatialreference.org/ref/epsg/3947/",
proj4: "+proj=lcc +lat_1=46.25 +lat_2=47.75 +lat_0=47 +lon_0=3 +x_0=1700000 +y_0=6200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC46)",
code: "EPSG:3946",
url: "http://spatialreference.org/ref/epsg/3946/",
proj4: "+proj=lcc +lat_1=45.25 +lat_2=46.75 +lat_0=46 +lon_0=3 +x_0=1700000 +y_0=5200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC45)",
code: "EPSG:3945",
url: "http://spatialreference.org/ref/epsg/3945/",
proj4: "+proj=lcc +lat_1=44.25 +lat_2=45.75 +lat_0=45 +lon_0=3 +x_0=1700000 +y_0=4200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "RGF93 (CC44)",
code: "EPSG:3944",
url: "http://spatialreference.org/ref/epsg/3944/",
proj4: "+proj=lcc +lat_1=43.25 +lat_2=44.75 +lat_0=44 +lon_0=3 +x_0=1700000 +y_0=3200000 +ellps=GRS80 +towgs84=0,0,0,0,0,0,0 +units=m +no_defs"
},
{
title: "NTF (Lambert II)",
code: "EPSG:27572",
url: "http://spatialreference.org/ref/epsg/27572/",
proj4: "+proj=lcc +lat_1=46.8 +lat_0=46.8 +lon_0=0 +k_0=0.99987742 +x_0=600000 +y_0=2200000 +a=6378249.2 +b=6356515 +towgs84=-168,-60,320,0,0,0,0 +pm=paris +units=m +no_defs"
},
{
title: "NTF (Lambert III)",
code: "EPSG:27573",
url: "http://spatialreference.org/ref/epsg/27573/",
proj4: "+proj=lcc +lat_1=44.10000000000001 +lat_0=44.10000000000001 +lon_0=0 +k_0=0.999877499 +x_0=600000 +y_0=3200000 +a=6378249.2 +b=6356515 +towgs84=-168,-60,320,0,0,0,0 +pm=paris +units=m +no_defs"
},
{
title: "NTF",
code: "EPSG:4275",
url: "http://spatialreference.org/ref/epsg/4275/",
proj4: "+proj=longlat +a=6378249.2 +b=6356515 +towgs84=-168,-60,320,0,0,0,0 +no_defs"
},
{
title: "WGS84",
code: "EPSG:4326",
url: "https://epsg.io/4326",
proj4: "+proj=longlat +datum=WGS84 +no_defs"
}
];
export const Proj4Defs = Projections.map(p => [p.code, p.proj4]);
proj4.defs(Proj4Defs);
export function project(from, to, y, x) {
console.log(`Projecting coordinates '${x},${y}' from '${from}' to '${to}'`);
const coords = proj4(from, to, [x, y]);
return { x: coords[0], y: coords[1] };
};
// Voir https://geodesie.ign.fr/contenu/fichiers/documentation/pedagogiques/TransformationsCoordonneesGeodesiques.pdf
export const WGS84Ellipsoid = createEllipsoid(6378137, 1/298.257223563);
function createEllipsoid(a, f) {
const b = a * (1 - f);
const asqr = a * a;
const bsqr = b * b;
const e = Math.sqrt((asqr-bsqr)/asqr);
const eprime = Math.sqrt((asqr-bsqr)/bsqr);
return { a, f, b, e, bsqr, asqr, eprime };
}
function degrees(angle) {
return angle * (180 / Math.PI);
}
export function XYZtoLLA(x, y, z, ellipsoid) {
const p = Math.sqrt(x*x + y*y);
const theta = Math.atan((z*ellipsoid.a)/(p*ellipsoid.b));
const sintheta = Math.sin(theta);
const costheta = Math.cos(theta);
const num = z + ellipsoid.eprime * ellipsoid.eprime * ellipsoid.b * sintheta * sintheta * sintheta;
const denom = p - ellipsoid.e * ellipsoid.e * ellipsoid.a * costheta * costheta * costheta;
const lat = Math.atan(num/denom);
let lon = Math.atan(y/x);
const n = getN(lat, ellipsoid);
const alt = (p / Math.cos(lat)) - n;
if (x < 0 && y < 0) {
lon = lon - Math.PI;
}
if (x < 0 && y > 0) {
lon = lon + Math.PI;
}
return { latitude: degrees(lat), longitude: degrees(lon), altitude: alt };
}
function getN(lat, ellipsoid) {
const sinlat = Math.sin(lat);
const denom = Math.sqrt(1-ellipsoid.e*ellipsoid.e*sinlat*sinlat);
const n = ellipsoid.a / denom;
return n;
}
export function LLAtoXYZ(latitude, longitude, altitude, ellipsoid) {
const cosLat = Math.cos(latitude * Math.PI / 180.0);
const sinLat = Math.sin(latitude * Math.PI / 180.0);
const cosLon = Math.cos(longitude * Math.PI / 180.0);
const sinLon = Math.sin(longitude * Math.PI / 180.0);
const rad = ellipsoid.a;
const f = ellipsoid.f;
const C = 1.0 / Math.sqrt(cosLat * cosLat + (1 - f) * (1 - f) * sinLat * sinLat);
const S = (1.0 - f) * (1.0 - f) * C;
const h = altitude;
const x = (rad * C + h) * cosLat * cosLon;
const y = (rad * C + h) * cosLat * sinLon;
const z = (rad * S + h) * sinLat;
return { x, y, z };
}