gcj02Mecator.js 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. // 导入proj控件,使用其方法注入gcj02坐标系
  2. import * as proj from 'ol/proj';
  3. var forEachPoint = function (func) {
  4. return function (input, opt_output, opt_dimension) {
  5. var len = input.length;
  6. var dimension = opt_dimension ? opt_dimension : 2;
  7. var output;
  8. if (opt_output) {
  9. output = opt_output;
  10. } else {
  11. if (dimension !== 2) {
  12. output = input.slice();
  13. } else {
  14. output = new Array(len);
  15. }
  16. }
  17. for (var offset = 0; offset < len; offset += dimension) {
  18. func(input, output, offset);
  19. }
  20. return output;
  21. };
  22. };
  23. var gcj02 = {};
  24. var i = 0;
  25. var PI = Math.PI;
  26. var AXIS = 6378245.0;
  27. var OFFSET = 0.00669342162296594323; // (a^2 - b^2) / a^2
  28. function delta(wgLon, wgLat) {
  29. var dLat = transformLat(wgLon - 105.0, wgLat - 35.0);
  30. var dLon = transformLon(wgLon - 105.0, wgLat - 35.0);
  31. var radLat = (wgLat / 180.0) * PI;
  32. var magic = Math.sin(radLat);
  33. magic = 1 - OFFSET * magic * magic;
  34. var sqrtMagic = Math.sqrt(magic);
  35. dLat = (dLat * 180.0) / (((AXIS * (1 - OFFSET)) / (magic * sqrtMagic)) * PI);
  36. dLon = (dLon * 180.0) / ((AXIS / sqrtMagic) * Math.cos(radLat) * PI);
  37. return [dLon, dLat];
  38. }
  39. function outOfChina(lon, lat) {
  40. if (lon < 72.004 || lon > 137.8347) {
  41. return true;
  42. }
  43. if (lat < 0.8293 || lat > 55.8271) {
  44. return true;
  45. }
  46. return false;
  47. }
  48. function transformLat(x, y) {
  49. var ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y + 0.2 * Math.sqrt(Math.abs(x));
  50. ret += ((20.0 * Math.sin(6.0 * x * PI) + 20.0 * Math.sin(2.0 * x * PI)) * 2.0) / 3.0;
  51. ret += ((20.0 * Math.sin(y * PI) + 40.0 * Math.sin((y / 3.0) * PI)) * 2.0) / 3.0;
  52. ret += ((160.0 * Math.sin((y / 12.0) * PI) + 320 * Math.sin((y * PI) / 30.0)) * 2.0) / 3.0;
  53. return ret;
  54. }
  55. function transformLon(x, y) {
  56. var ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1 * Math.sqrt(Math.abs(x));
  57. ret += ((20.0 * Math.sin(6.0 * x * PI) + 20.0 * Math.sin(2.0 * x * PI)) * 2.0) / 3.0;
  58. ret += ((20.0 * Math.sin(x * PI) + 40.0 * Math.sin((x / 3.0) * PI)) * 2.0) / 3.0;
  59. ret += ((150.0 * Math.sin((x / 12.0) * PI) + 300.0 * Math.sin((x / 30.0) * PI)) * 2.0) / 3.0;
  60. return ret;
  61. }
  62. gcj02.toWGS84 = forEachPoint(function (input, output, offset) {
  63. var lng = input[offset];
  64. var lat = input[offset + 1];
  65. if (!outOfChina(lng, lat)) {
  66. var deltaD = delta(lng, lat);
  67. lng = lng - deltaD[0];
  68. lat = lat - deltaD[1];
  69. }
  70. output[offset] = lng;
  71. output[offset + 1] = lat;
  72. });
  73. gcj02.fromWGS84 = forEachPoint(function (input, output, offset) {
  74. var lng = input[offset];
  75. var lat = input[offset + 1];
  76. if (!outOfChina(lng, lat)) {
  77. var deltaD = delta(lng, lat);
  78. lng = lng + deltaD[0];
  79. lat = lat + deltaD[1];
  80. }
  81. output[offset] = lng;
  82. output[offset + 1] = lat;
  83. });
  84. var sphericalMercator = {};
  85. var RADIUS = 6378137;
  86. var MAX_LATITUDE = 85.0511287798;
  87. var RAD_PER_DEG = Math.PI / 180;
  88. sphericalMercator.forward = forEachPoint(function (input, output, offset) {
  89. var lat = Math.max(Math.min(MAX_LATITUDE, input[offset + 1]), -MAX_LATITUDE);
  90. var sin = Math.sin(lat * RAD_PER_DEG);
  91. output[offset] = RADIUS * input[offset] * RAD_PER_DEG;
  92. output[offset + 1] = (RADIUS * Math.log((1 + sin) / (1 - sin))) / 2;
  93. });
  94. sphericalMercator.inverse = forEachPoint(function (input, output, offset) {
  95. output[offset] = input[offset] / RADIUS / RAD_PER_DEG;
  96. output[offset + 1] = (2 * Math.atan(Math.exp(input[offset + 1] / RADIUS)) - Math.PI / 2) / RAD_PER_DEG;
  97. });
  98. var projzh = {};
  99. projzh.ll2gmerc = function (input, opt_output, opt_dimension) {
  100. let output = gcj02.fromWGS84(input, opt_output, opt_dimension);
  101. return projzh.ll2smerc(output, output, opt_dimension);
  102. };
  103. projzh.gmerc2ll = function (input, opt_output, opt_dimension) {
  104. let output = projzh.smerc2ll(input, input, opt_dimension);
  105. return gcj02.toWGS84(output, opt_output, opt_dimension);
  106. };
  107. projzh.smerc2gmerc = function (input, opt_output, opt_dimension) {
  108. let output = projzh.smerc2ll(input, input, opt_dimension);
  109. output = gcj02.fromWGS84(output, output, opt_dimension);
  110. return projzh.ll2smerc(output, output, opt_dimension);
  111. };
  112. projzh.gmerc2smerc = function (input, opt_output, opt_dimension) {
  113. let output = projzh.smerc2ll(input, input, opt_dimension);
  114. output = gcj02.toWGS84(output, output, opt_dimension);
  115. return projzh.ll2smerc(output, output, opt_dimension);
  116. };
  117. projzh.ll2smerc = sphericalMercator.forward;
  118. projzh.smerc2ll = sphericalMercator.inverse;
  119. // 定义GCJ02
  120. const gcj02Extent = [-20037508.342789244, -20037508.342789244, 20037508.342789244, 20037508.342789244];
  121. const gcj02Mecator = new proj.Projection({
  122. code: 'GCJ-02',
  123. extent: gcj02Extent,
  124. units: 'm',
  125. });
  126. proj.addProjection(gcj02Mecator);
  127. // 将4326/3857转为gcj02坐标的方法定义
  128. proj.addCoordinateTransforms('EPSG:4326', gcj02Mecator, projzh.ll2gmerc, projzh.gmerc2ll);
  129. proj.addCoordinateTransforms('EPSG:3857', gcj02Mecator, projzh.smerc2gmerc, projzh.gmerc2smerc);
  130. // 我使用的react,所以这里需要导出定义的gcj02Mecator,提供给外部使用
  131. export default gcj02Mecator;