219 lines
7.0 KiB
JavaScript
219 lines
7.0 KiB
JavaScript
|
|
function mathTool() {
|
|
this.PI = 3.141592653589793238;
|
|
this.a = 6378137.0;
|
|
this.b = 6356752.3142
|
|
this.f = (this.a - this.b) / this.a;
|
|
this.e_sq = this.f * (2.0 - this.f);
|
|
this.ee = 0.00669437999013;
|
|
this.WGSF = 1 / 298.257223563;
|
|
this.WGSe2 = this.WGSF * (2 - this.WGSF);
|
|
this.WGSa = 6378137.00000;
|
|
this.EPSILON = 1.0e-12;
|
|
}
|
|
|
|
/*
|
|
*解算坐标
|
|
*@point 原点
|
|
*@azimuth 方位角
|
|
*@elevation 仰角
|
|
*@distance 距离
|
|
*/
|
|
mathTool.prototype.CalculateCoordinates = function (point, azimuth, elevation, distance) {
|
|
let vertical_height = distance * Math.sin(2 * this.PI / 360 * elevation);//垂直高度
|
|
let horizontal_distance = distance * Math.cos(2 * this.PI / 360 * elevation);//水平距离
|
|
if (azimuth > 360) azimuth = azimuth % 360;
|
|
if (azimuth < 0) azimuth = 360 + (azimuth % 360);
|
|
|
|
let point = this.lonLat2WebMercator(point);
|
|
let lnglat = null;
|
|
|
|
if (azimuth <= 90) {//第四象限
|
|
let x_length = horizontal_distance * Math.cos(2 * this.PI / 360 * azimuth);
|
|
let y_length = horizontal_distance * Math.sin(2 * this.PI / 360 * azimuth);
|
|
lnglat = {
|
|
x: point.x + x_length,
|
|
y: point.y - y_length
|
|
}
|
|
} else if (azimuth > 90 && azimuth <= 180) {//第三象限
|
|
let x_length = horizontal_distance * Math.sin(2 * this.PI / 360 * (azimuth - 90));
|
|
let y_length = horizontal_distance * Math.cos(2 * this.PI / 360 * (azimuth - 90));
|
|
lnglat = {
|
|
x: point.x - x_length,
|
|
y: point.y - y_length
|
|
}
|
|
} else if (azimuth > 180 && azimuth <= 270) {//第二象限
|
|
let x_length = horizontal_distance * Math.cos(2 * this.PI / 360 * (azimuth - 180));
|
|
let y_length = horizontal_distance * Math.sin(2 * this.PI / 360 * (azimuth - 180));
|
|
lnglat = {
|
|
x: point.x - x_length,
|
|
y: point.y + y_length
|
|
}
|
|
} else {//第一象限
|
|
let x_length = horizontal_distance * Math.sin(2 * this.PI / 360 * (azimuth - 270));
|
|
let y_length = horizontal_distance * Math.cos(2 * this.PI / 360 * (azimuth - 270));
|
|
lnglat = {
|
|
x: point.x + x_length,
|
|
y: point.y + y_length
|
|
}
|
|
}
|
|
lnglat = this.webMercator2LonLat(lnglat);
|
|
return {
|
|
lng: lnglat.x,
|
|
lat: lnglat.y,
|
|
height: vertical_height
|
|
}
|
|
}
|
|
/*
|
|
*经纬度转Web墨卡托
|
|
*@lonLat 经纬度
|
|
*/
|
|
mathTool.prototype.lonLat2WebMercator = function (lonLat) {
|
|
let x = lonLat.x * this.a / 180;
|
|
let y = Math.log(Math.tan((90 + lonLat.y) * this.PI / 360)) / (this.PI / 180);
|
|
y = y * this.a / 180;
|
|
return {
|
|
x: x,
|
|
y: y
|
|
}
|
|
}
|
|
/*
|
|
*Web墨卡托转经纬度
|
|
*@mercator 平面坐标
|
|
*/
|
|
mathTool.prototype.webMercator2LonLat = function (mercator) {
|
|
let x = mercator.x / this.a * 180;
|
|
let y = mercator.y / this.a * 180;
|
|
y = 180 / this.PI * (2 * (Math.exp(y * this.PI / 180)) - this.PI / 2);
|
|
return {
|
|
x: x,
|
|
y: y
|
|
}
|
|
}
|
|
|
|
mathTool.prototype.get_atan = function (z, y) {
|
|
let x;
|
|
if (z == 0) {
|
|
x = this.PI / 2;
|
|
} else {
|
|
if (y == 0) {
|
|
x = this.PI;
|
|
} else {
|
|
x = Math.atan(Math.abs(y / z));
|
|
if ((y > 0) && (z < 0)) {
|
|
x = this.PI - x;
|
|
} else if ((y < 0) && (z < 0)) {
|
|
x = this.PI + x;
|
|
} else if ((y < 0) && (z > 0)) {
|
|
x = 2 * this.M_PI - x;
|
|
}
|
|
}
|
|
}
|
|
return x;
|
|
}
|
|
//WGS84转ECEF坐标系
|
|
mathTool.prototype.ConvertLLAToXYZ = function (LLACoor) {
|
|
let lon = this.PI / 180 * LLACoor.longitude;
|
|
let lat = this.PI / 180 * LLACoor.latitude;
|
|
let H = LLACoor.altitude;
|
|
let N0 = this.a / Math.sqrt(1.0 - this.ee * Math.sin(lat) * Math.sin(lat));
|
|
let x = (N0 + H) * Math.cos(lat) * Math.cos(lon);
|
|
let y = (N0 + H) * Math.cos(lat) * Math.sin(lon);
|
|
let z = (N0 * (1.0 - this.ee) + H) * Math.sin(lat);
|
|
return {
|
|
x: x,
|
|
y: y,
|
|
z: z
|
|
}
|
|
}
|
|
|
|
//ECEF坐标系转WGS84
|
|
mathTool.prototype.ConvertXYZToLLA = function (XYZCoor) {
|
|
let longitude = this.get_atan(XYZCoor.x, XYZCoor.y);
|
|
if (longitude < 0) {
|
|
longitude = longitude + this.PI;
|
|
}
|
|
let latitude = this.get_atan(Math.sqrt(XYZCoor.x * XYZCoor.x + XYZCoor.y * XYZCoor.y), XYZCoor.z);
|
|
|
|
let W = Math.sqrt(1 - this.WGSe2 * Math.sin(latitude) * Math.sin(latitude));
|
|
let N = this.WGSa / W;
|
|
let B1;
|
|
do {
|
|
B1 = latitude;
|
|
W = Math.sqrt(1 - this.WGSe2 * Math.sin(B1) * Math.sin(B1));
|
|
N = this.WGSa / W;
|
|
latitude = this.get_atan(Math.sqrt(XYZCoor.x * XYZCoor.x + XYZCoor.y * XYZCoor.y), (XYZCoor.z + N * this.WGSe2 * Math.sin(B1)));
|
|
}
|
|
while (Math.abs(latitude - B1) > this.EPSILON);
|
|
|
|
altitude = Math.sqrt(XYZCoor.x * XYZCoor.x + XYZCoor.y * XYZCoor.y) / Math.cos(latitude) - this.WGSa / Math.sqrt(1 - this.WGSe2 * Math.sin(latitude) * Math.sin(latitude));
|
|
|
|
return {
|
|
longitude: longitude * 180 / this.PI,
|
|
latitude: latitude * 180 / this.PI,
|
|
altitude: altitude
|
|
}
|
|
}
|
|
/*北东天坐标系转WGS84
|
|
@ a A点坐标
|
|
@ p 相对参数,距离、方位角、仰角
|
|
*/
|
|
mathTool.prototype.enu_to_ecef = function (a, p) {
|
|
//距离
|
|
let distance = p.distance;
|
|
//方位角
|
|
let azimuth = p.azimuth;
|
|
//仰角
|
|
let elevation = p.elevation;
|
|
|
|
let zUp = elevation >= 0 ? distance * Math.sin(this.PI / 180 * elevation) : (-1) * distance * Math.sin(this.PI / 180 * Math, abs(elevation));
|
|
|
|
let d = distance * Math.cos(this.PI / 180 * Math.abs(elevation));
|
|
let xEast;
|
|
let yNorth;
|
|
if (azimuth <= 90) {
|
|
xEast = d * Math.sin(this.PI / 180 * azimuth);
|
|
yNorth = d * Math.cos(this.PI / 180 * azimuth);
|
|
} else if (azimuth > 90 && azimuth < 180) {
|
|
xEast = d * Math.cos(this.PI / 180 * (azimuth - 90));
|
|
yNorth = (-1)*d * Math.sin(this.PI / 180 * (azimuth - 90));
|
|
} else if (azimuth > 180 && azimuth < 270) {
|
|
xEast = (-1) * d * Math.sin(this.PI / 180 * (azimuth - 180));
|
|
yNorth =(-1) * d * Math.cos(this.PI / 180 * (azimuth - 180));
|
|
} else {
|
|
xEast = (-1) * d * Math.sin(this.PI / 180 * (360 - azimuth));
|
|
yNorth = d * Math.cos(this.PI / 180 * (360 - azimuth));
|
|
}
|
|
|
|
let lamb = this.radians(a.latitude);
|
|
let phi = this.radians(a.longitude);
|
|
let h0 = a.altitude;
|
|
|
|
let s = Math.sin(lamb);
|
|
let N = this.a / Math.sqrt(1.0 - this.e_sq * s * s);
|
|
|
|
let sin_lambda = Math.sin(lamb);
|
|
let cos_lambda = Math.cos(lamb);
|
|
|
|
let sin_phi = Math.sin(phi);
|
|
let cos_phi = Math.cos(phi);
|
|
|
|
let x0 = (h0 + N) * cos_lambda * cos_phi;
|
|
let y0 = (h0 + N) * cos_lambda * sin_phi;
|
|
let z0 = (h0 + (1 - this.e_sq) * N) * sin_lambda;
|
|
|
|
let t = cos_lambda * zUp - sin_lambda * yNorth;
|
|
|
|
let zd = sin_lambda * zUp + cos_lambda * yNorth;
|
|
let xd = cos_phi * t - sin_phi * xEast;
|
|
let yd = sin_phi * t + cos_phi * xEast;
|
|
|
|
return this.ConvertXYZToLLA({
|
|
x: xd + x0,
|
|
y: yd + y0,
|
|
z: zd + z0
|
|
})
|
|
}
|
|
mathTool.prototype.radians = function (degree) {
|
|
return this.PI / 180 * degree;
|
|
} |