PosApi.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Runtime.InteropServices;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. using DevExpress.Drawing.Internal.Fonts.Interop;
  8. using DevExpress.XtraBars.Docking2010.Views.Widget;
  9. using XdCxRhDW.App.Model;
  10. using static System.Windows.Forms.VisualStyles.VisualStyleElement.TextBox;
  11. namespace XdCxRhDW.App.Api
  12. {
  13. /// <summary>
  14. ///TODO 定位算法接口.在这里调用罗博士的算法库
  15. /// </summary>
  16. public static class PosApi
  17. {
  18. #region cpp dll Interop
  19. //一星一地测向带参定位
  20. private const string XDCX = @"AddIns\DLL_DTO_DOA_DW.dll";
  21. //两星一地带参定位
  22. private const string X2D1 = @"AddIns\DLL_SC_2X1D_DW.dll";
  23. //三星双时差带参、三星双时差无参、三星双频差带参、双星时频差带参、两星一地无参
  24. private const string OtherPos = @"AddIns\DLL_11J.dll";
  25. [DllImport(XDCX, EntryPoint = "XD_CX_DW", CallingConvention = CallingConvention.Cdecl)]//一星一地测向带参
  26. private extern static void X1D1_Pos20240305_Core(double[] mainSat, double[] satStation, double[] cdbStation, double[] cxStation, double[] refStation, double[] zone, double theta, double tarDto, double refDto, double[] res);
  27. [DllImport(X2D1, EntryPoint = "SC_2X1D_DW", CallingConvention = CallingConvention.Cdecl)]//两星一地带参
  28. private extern static void X2D1_Pos20240305_Core(double[] mainSat, double[] adjaSat, double[] cdbStation, double[] satStation1, double[] satStation2, double[] satStation3, double[] satStation4,
  29. double[] satStation5, double[] refStation, double[] zone, double tarSxDto, double tarXdDto, double samp_main_dto, double samp_neigh_dto, double[] res);
  30. [DllImport(OtherPos, EntryPoint = "XingDi_2X1D_DW_NoRef", CallingConvention = CallingConvention.Cdecl)]//两星一地无参
  31. private extern static void X2D1_PosNoRef20240305_Core(double[] mainSat, double[] adjaSat, double[] cdbStation, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  32. , double[] zone, double tarSxDto, double tarXdDto, double[] res);
  33. [DllImport(OtherPos, EntryPoint = "SanXing_DW", CallingConvention = CallingConvention.Cdecl)]//三星双时差带参
  34. private extern static void X3_Pos20240305_Core(double[] mainSat, double[] adjaSat1, double[] adjaSat2, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  35. , double[] mainSatRefStation, double[] adjaSat1RefStation, double[] adjaSat2RefStation, double[] refStation, double[] zone, double tarDto1, double tarDto2, double refDto1, double refDto2, double[] res);
  36. [DllImport(OtherPos, EntryPoint = "SanXing_DW_NoRef", CallingConvention = CallingConvention.Cdecl)]//三星双时差无参
  37. private extern static void X3_PosNoRef20240305_Core(double[] mainSat, double[] adjaSat1, double[] adjaSat2, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  38. , double[] zone, double tarDto1, double tarDto2, double[] res);
  39. [DllImport(OtherPos, EntryPoint = "TriStar_2DFO_DW", CallingConvention = CallingConvention.Cdecl)]//三星双频差带参
  40. private extern static void X3_PosTwoDfo20240305_Core(double[] mainSat, double[] adjaSat1, double[] adjaSat2, double[] mainSatTarStation, double[] adjaSat1TarStation, double[] adjaSat2TarStation
  41. , double[] refStation, double[] zone, double tarDfo1, double tarDfo2, double refDfo1, double refDfo2, double fu1, double fd1, double fu2, double fd2, double[] res);
  42. [DllImport(OtherPos, EntryPoint = "TwoStar_DTFO_DW", CallingConvention = CallingConvention.Cdecl)]//双星时频差带参
  43. private extern static void X2_Pos20240305_Core(double[] mainSat, double[] adjaSat, double[] mainSatTarStation, double[] adjaSatTarStation, double[] refStation, double[] zone
  44. , double tarDto, double tarDfo, double refDto, double refDfo, double fu1, double fd1, double fu2, double fd2, double[] res);
  45. #endregion
  46. /// <summary>
  47. /// 一星一地带参,返返回经度纬度高度及镜像点,数组长度为6
  48. /// </summary>
  49. /// <param name="cgRes">参估结果</param>
  50. /// <param name="sRes">站点信息</param>
  51. /// <param name="cxRes">测向结果</param>
  52. /// <returns></returns>
  53. public static double[] X1D1_Pos(CgRes cgRes, StationRes sRes, CxRes cxRes)
  54. {
  55. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  56. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  57. double[] cdbStation = new double[3] { sRes.CdbTxLon.Value, sRes.CdbTxLat.Value, 0 };
  58. double[] cxStation = new double[3] { sRes.CxLon.Value, sRes.CxLat.Value, 0 };
  59. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  60. double dtoCdb = cgRes.DtoCdb.Value / 1e6;
  61. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  62. double theta = cxRes.Fx;//单位°
  63. double[] res = new double[6];
  64. var ybDto1 = cgRes.YbMainDto.Value / 1e6;
  65. X1D1_Pos20240305_Core(mainSat, satStation, cdbStation, cxStation, refStation, zone, theta, dtoCdb, ybDto1, res);
  66. return ConvertToGeoPoint(res);
  67. }
  68. /// <summary>
  69. /// 两星一地带参,返回经度纬度高度及镜像点,数组长度为6
  70. /// </summary>
  71. /// <param name="cgRes">参估结果</param>
  72. /// <param name="sRes">站点信息</param>
  73. /// <returns></returns>
  74. public static double[] X2D1_Pos(CgRes cgRes, StationRes sRes)
  75. {
  76. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  77. double[] adjaSat = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  78. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  79. double[] cdbStation = new double[3] { sRes.CdbTxLon.Value, sRes.CdbTxLat.Value, 0 };
  80. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  81. double dto1 = cgRes.Dto1.Value / 1e6;
  82. double dtoCdb = cgRes.DtoCdb.Value / 1e6;
  83. double ybDto1 = cgRes.YbMainDto.Value / 1e6;
  84. double ybDto2 = cgRes.YbAdja1Dto.Value / 1e6;
  85. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  86. double[] res = new double[6];
  87. X2D1_Pos20240305_Core(mainSat, adjaSat, cdbStation, satStation, satStation, satStation, satStation, satStation, refStation, zone, dto1, dtoCdb, ybDto1, ybDto2, res);
  88. return ConvertToGeoPoint(res);
  89. }
  90. /// <summary>
  91. /// 两星一地无参,返回经度纬度高度及镜像点,数组长度为6
  92. /// </summary>
  93. /// <param name="cgRes">参估结果</param>
  94. /// <param name="sRes">站点信息</param>
  95. /// <returns></returns>
  96. public static double[] X2D1_PosNoRef(CgRes cgRes, StationRes sRes)
  97. {
  98. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  99. double[] adjaSat = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  100. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  101. double[] cdbStation = new double[3] { sRes.CdbTxLon.Value, sRes.CdbTxLat.Value, 0 };
  102. double dto1 = cgRes.Dto1.Value / 1e6;
  103. double dtoCdb = cgRes.DtoCdb.Value / 1e6;
  104. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  105. double[] res = new double[6];
  106. X2D1_PosNoRef20240305_Core(mainSat, adjaSat, cdbStation, satStation, satStation, satStation, zone, dto1, dtoCdb, res);
  107. return ConvertToGeoPoint(res);
  108. }
  109. /// <summary>
  110. /// 融合定位带参,返回经度纬度高度及镜像点,数组长度为6
  111. /// </summary>
  112. /// <param name="cgRes">参估结果</param>
  113. /// <param name="sRes">站点信息</param>
  114. /// <param name="cxRes">测向结果</param>
  115. /// <returns></returns>
  116. public static double[] RH_Pos(CgRes cgRes, StationRes sRes, CxRes cxRes)
  117. {
  118. var res1 = X1D1_Pos(cgRes, sRes, cxRes);
  119. var res2 = X2D1_Pos(cgRes, sRes);
  120. double[] res = new double[] { 999, 999, 0, 999, 999, 0 };
  121. var p1 = res1.Take(3).ToArray();
  122. var p2 = res1.Skip(3).ToArray();
  123. var p3 = res2.Take(3).ToArray();
  124. var p4 = res2.Skip(3).ToArray();
  125. if (IsGeoPoint(p1) && IsGeoPoint(p2) && IsGeoPoint(p3) && IsGeoPoint(p4))
  126. {
  127. res = new double[6] {
  128. (p1[0] + p3[0]) / 2 + 0.003,
  129. (p1[1] + p3[1]) / 2 - 0.002,
  130. 0,
  131. (p2[0] + p4[0]) / 2 + 0.003,
  132. (p2[1] + p4[1]) / 2 - 0.002,
  133. 0,
  134. };
  135. }
  136. else if (IsGeoPoint(p1) && IsGeoPoint(p3))
  137. {
  138. res = new double[6] {
  139. (p1[0] + p3[0]) / 2 + 0.003,
  140. (p1[1] + p3[1]) / 2 - 0.002,
  141. 0,
  142. 999,
  143. 999,
  144. 0,
  145. };
  146. }
  147. else if (IsGeoPoint(p1) && IsGeoPoint(p2))
  148. {
  149. res = new double[6]
  150. {
  151. p1[0] + 0.003,
  152. p1[1] - 0.002,
  153. 0,
  154. p2[0] + 0.003,
  155. p2[1] - 0.002,
  156. 0,
  157. };
  158. }
  159. else if (IsGeoPoint(p3) && IsGeoPoint(p4))
  160. {
  161. res = new double[6]
  162. {
  163. p3[0] + 0.003,
  164. p3[1] - 0.002,
  165. 0,
  166. p4[0] + 0.003,
  167. p4[1] - 0.002,
  168. 0,
  169. };
  170. }
  171. return res;
  172. }
  173. /// <summary>
  174. /// 三星双时差带参,返回经度纬度高度及镜像点,数组长度为6
  175. /// </summary>
  176. /// <param name="cgRes">参估结果</param>
  177. /// <param name="sRes">站点信息</param>
  178. /// <returns></returns>
  179. public static double[] X3_Pos(CgRes cgRes, StationRes sRes)
  180. {
  181. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  182. double[] adjaSat1 = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  183. double[] adjaSat2 = new double[3] { cgRes.Adja2X.Value, cgRes.Adja2Y.Value, cgRes.Adja2Z.Value };
  184. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  185. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  186. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  187. var tarDto1 = cgRes.Dto1.Value / 1e6;
  188. var tarDto2 = cgRes.Dto2.Value / 1e6;
  189. var refDto1 = (cgRes.YbMainDto.Value - cgRes.YbAdja1Dto.Value) / 1e6;
  190. var refDto2 = (cgRes.YbMainDto.Value - cgRes.YbAdja2Dto.Value) / 1e6;
  191. double[] res = new double[6];
  192. X3_Pos20240305_Core(mainSat, adjaSat1, adjaSat2, satStation, satStation, satStation, satStation,
  193. satStation, satStation, refStation, zone, tarDto1, tarDto2, refDto1, refDto2, res);
  194. return ConvertToGeoPoint(res);
  195. }
  196. /// <summary>
  197. /// 三星双时差无参,返回经度纬度高度及镜像点,数组长度为6
  198. /// </summary>
  199. /// <param name="cgRes">参估结果</param>
  200. /// <param name="sRes">站点信息</param>
  201. /// <returns></returns>
  202. public static double[] X3_PosNoRef(CgRes cgRes, StationRes sRes)
  203. {
  204. double[] mainSat = new double[3] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value };
  205. double[] adjaSat1 = new double[3] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value };
  206. double[] adjaSat2 = new double[3] { cgRes.Adja2X.Value, cgRes.Adja2Y.Value, cgRes.Adja2Z.Value };
  207. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  208. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  209. var tarDto1 = cgRes.Dto1.Value / 1e6;
  210. var tarDto2 = cgRes.Dto2.Value / 1e6;
  211. double[] res = new double[6];
  212. X3_PosNoRef20240305_Core(mainSat, adjaSat1, adjaSat2, satStation, satStation, satStation, zone, tarDto1, tarDto2, res);
  213. return ConvertToGeoPoint(res);
  214. }
  215. /// <summary>
  216. /// 三星双频差带参,返回经度纬度高度及镜像点,数组长度为6
  217. /// </summary>
  218. /// <param name="cgRes">参估结果</param>
  219. /// <param name="sRes">站点信息</param>
  220. /// <returns></returns>
  221. public static double[] X3_PosTwoDfo(CgRes cgRes, StationRes sRes)
  222. {
  223. double[] mainSat = new double[6] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value, cgRes.MainVx.Value, cgRes.MainVy.Value, cgRes.MainVz.Value };
  224. double[] adjaSat1 = new double[6] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value, cgRes.Adja1Vx.Value, cgRes.Adja1Vy.Value, cgRes.Adja1Vz.Value };
  225. double[] adjaSat2 = new double[6] { cgRes.Adja2X.Value, cgRes.Adja2Y.Value, cgRes.Adja2Z.Value, cgRes.Adja2Vx.Value, cgRes.Adja2Vy.Value, cgRes.Adja2Vz.Value };
  226. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  227. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  228. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  229. var tarDfo1 = cgRes.Dfo1.Value;
  230. var tarDfo2 = cgRes.Dfo2.Value;
  231. var refDfo1 = cgRes.YbMainDfo.Value - cgRes.YbAdja1Dfo.Value;
  232. var refDfo2 = cgRes.YbMainDfo.Value - cgRes.YbAdja2Dfo.Value;
  233. double fu1 = cgRes.TarFreqUp.Value;
  234. double fd1 = cgRes.TarFreqDown.Value;
  235. double fu2 = cgRes.RefFreqUp.Value;
  236. double fd2 = cgRes.RefFreqDown.Value;
  237. double[] res = new double[6];
  238. X3_PosTwoDfo20240305_Core(mainSat, adjaSat1, adjaSat2, satStation, satStation, satStation,
  239. refStation, zone, tarDfo1, tarDfo2, refDfo1, refDfo2, fu1, fd1, fu2, fd2, res);
  240. return ConvertToGeoPoint(res);
  241. }
  242. /// <summary>
  243. /// 双星时频差带参,返回经度纬度高度及镜像点,数组长度为6
  244. /// </summary>
  245. /// <param name="cgRes">参估结果</param>
  246. /// <param name="sRes">站点信息</param>
  247. /// <returns></returns>
  248. public static double[] X2_PosDtoDfo(CgRes cgRes, StationRes sRes)
  249. {
  250. double[] mainSat = new double[6] { cgRes.MainX.Value, cgRes.MainY.Value, cgRes.MainZ.Value, cgRes.MainVx.Value, cgRes.MainVy.Value, cgRes.MainVz.Value };
  251. double[] adjaSat = new double[6] { cgRes.Adja1X.Value, cgRes.Adja1Y.Value, cgRes.Adja1Z.Value, cgRes.Adja1Vx.Value, cgRes.Adja1Vy.Value, cgRes.Adja1Vz.Value };
  252. double[] satStation = new double[3] { sRes.SatTxLon, sRes.SatTxLat, 0 };
  253. double[] refStation = new double[3] { sRes.RefLon.Value, sRes.RefLat.Value, 0 };
  254. double[] zone = new double[] { -85, 85, -180, 180 }; //定位区域
  255. var tarDto = cgRes.Dto1.Value / 1e6;
  256. var tarDfo = cgRes.Dfo1.Value;
  257. var refDto = (cgRes.YbMainDto.Value - cgRes.YbAdja1Dto.Value) / 1e6;
  258. var refDfo = cgRes.YbMainDfo.Value - cgRes.YbAdja1Dfo.Value;
  259. double fu1 = cgRes.TarFreqUp.Value;
  260. double fd1 = cgRes.TarFreqDown.Value;
  261. double fu2 = cgRes.RefFreqUp.Value;
  262. double fd2 = cgRes.RefFreqDown.Value;
  263. double[] res = new double[6];
  264. X2_Pos20240305_Core(mainSat, adjaSat, satStation, satStation, refStation, zone, tarDto, tarDfo, refDto, refDfo, fu1, fd1, fu2, fd2, res);
  265. return ConvertToGeoPoint(res);
  266. }
  267. private static double[] ConvertToGeoPoint(double[] res)
  268. {
  269. if (res == null || res.Length == 0)
  270. {
  271. return new double[6] { 999, 999, 0, 999, 999, 0 };
  272. }
  273. else if (res.Length != 3 && res.Length != 6)
  274. {
  275. throw new Exception("定位结果解析异常,长度不是3或6");
  276. }
  277. var list = res.Select(p => Math.Round(p, 4)).ToArray();
  278. if (list.Length == 3)
  279. {
  280. list = list.Concat(new double[3] { 999, 999, 0 }).ToArray();
  281. }
  282. list[2] = list[5] = 0;//高度
  283. if (list[0] < -180 || list[0] > 180)
  284. {
  285. list[0] = list[1] = 999;
  286. }
  287. else if (list[1] < -90 || list[1] > 90)
  288. {
  289. list[0] = list[1] = 999;
  290. }
  291. if (list[3] < -180 || list[3] > 180)
  292. {
  293. list[3] = list[4] = 999;
  294. }
  295. else if (list[4] < -90 || list[4] > 90)
  296. {
  297. list[3] = list[4] = 999;
  298. }
  299. else if (double.IsNaN(list[3]) || double.IsNaN(list[4]))
  300. {
  301. list[3] = list[4] = 999;
  302. }
  303. else if (double.IsNaN(list[0]) || double.IsNaN(list[1]))
  304. {
  305. list[0] = list[1] = 999;
  306. }
  307. return list;
  308. }
  309. private static bool IsGeoPoint(double[] res)
  310. {
  311. if (res.Length != 3) return false;
  312. if (res[0] < -180 || res[0] > 180)
  313. {
  314. return false;
  315. }
  316. else if (res[1] < -90 || res[1] > 90)
  317. {
  318. return false;
  319. }
  320. return true;
  321. }
  322. }
  323. }