Windy.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  1. /****
  2. *风场类
  3. ****/
  4. var Cesium = require("cesium/Cesium");
  5. var CanvasWindy = function (json,params) {
  6. //风场json数据
  7. this.windData = json;
  8. //可配置参数
  9. this.viewer = params.viewer;
  10. this.canvas = params.canvas;
  11. this.extent = params.extent || [];//风场绘制时的地图范围,范围不应该大于风场数据的范围,顺序:west/east/south/north,有正负区分,如:[110,120,30,36]
  12. this.canvasContext = params.canvas.getContext("2d");//canvas上下文
  13. this.canvasWidth = params.canvasWidth || 300;//画板宽度
  14. this.canvasHeight = params.canvasHeight || 180;//画板高度
  15. this.speedRate = params.speedRate || 100;//风前进速率,意思是将当前风场横向纵向分成100份,再乘以风速就能得到移动位置,无论地图缩放到哪一级别都是一样的速度,可以用该数值控制线流动的快慢,值越大,越慢,
  16. this.particlesNumber = params.particlesNumber || 20000;//初始粒子总数,根据实际需要进行调节
  17. this.maxAge = params.maxAge || 120;//每个粒子的最大生存周期
  18. this.frameTime = 1000/(params.frameRate || 10);//每秒刷新次数,因为requestAnimationFrame固定每秒60次的渲染,所以如果不想这么快,就把该数值调小一些
  19. this.color = params.color || '#ffffff';//线颜色,提供几个示例颜色['#14208e','#3ac32b','#e0761a']
  20. this.lineWidth = params.lineWidth || 1;//线宽度
  21. //内置参数
  22. this.initExtent = [];//风场初始范围
  23. this.calc_speedRate = [0,0];//根据speedRate参数计算经纬度步进长度
  24. this.windField = null;
  25. this.particles = [];
  26. this.animateFrame = null;//requestAnimationFrame事件句柄,用来清除操作
  27. this.isdistory = false;//是否销毁,进行删除操作
  28. this._init();
  29. };
  30. CanvasWindy.prototype = {
  31. constructor: CanvasWindy,
  32. _init: function () {
  33. var self = this;
  34. // 创建风场网格
  35. this.windField = this.createField();
  36. this.initExtent = [this.windField.west-180,this.windField.east-180,this.windField.south,this.windField.north];
  37. //如果风场创建时,传入的参数有extent,就根据给定的extent,让随机生成的粒子落在extent范围内
  38. if(this.extent.length!=0){
  39. this.extent = [
  40. Math.max(this.initExtent[0],this.extent[0]),
  41. Math.min(this.initExtent[1],this.extent[1]),
  42. Math.max(this.initExtent[2],this.extent[2]),
  43. Math.min(this.initExtent[3],this.extent[3])
  44. ];
  45. }
  46. // console.log(this.extent);
  47. this._calcStep();
  48. // 创建风场粒子
  49. for (var i = 0; i < this.particlesNumber; i++) {
  50. this.particles.push(this.randomParticle(new CanvasParticle()));
  51. }
  52. this.canvasContext.fillStyle = "rgba(0, 0, 0, 0.97)";
  53. this.canvasContext.globalAlpha = 0.6;
  54. this.animate();
  55. var then = Date.now();
  56. (function frame() {
  57. if(!self.isdistory){
  58. self.animateFrame = requestAnimationFrame(frame);
  59. var now = Date.now();
  60. var delta = now - then;
  61. if (delta > self.frameTime) {
  62. then = now - delta % self.frameTime;
  63. self.animate();
  64. }
  65. }else{
  66. self.removeLines();
  67. }
  68. })();
  69. },
  70. //计算经纬度步进长度
  71. _calcStep:function(){
  72. var isextent = (this.extent.length!=0);
  73. var calcExtent = isextent?this.extent:this.initExtent;
  74. var calcSpeed = this.speedRate;
  75. this.calc_speedRate = [(calcExtent[1]-calcExtent[0])/calcSpeed,(calcExtent[3]-calcExtent[2])/calcSpeed];
  76. },
  77. //根据现有参数重新生成风场
  78. redraw:function(){
  79. window.cancelAnimationFrame(this.animateFrame);
  80. this.particles = [];
  81. this._init();
  82. },
  83. createField: function () {
  84. var data = this._parseWindJson();
  85. return new CanvasWindField(data);
  86. },
  87. animate: function () {
  88. var self = this,
  89. field = self.windField;
  90. var nextLng = null,
  91. nextLat = null,
  92. uv = null;
  93. self.particles.forEach(function (particle) {
  94. if (particle.age <= 0) {
  95. self.randomParticle(particle);
  96. }
  97. if (particle.age > 0) {
  98. var x = particle.x,
  99. y = particle.y,
  100. tlng = particle.tlng,
  101. tlat = particle.tlat;
  102. var gridpos = self._togrid(tlng,tlat);
  103. var tx = gridpos[0];
  104. var ty = gridpos[1];
  105. if (!self.isInExtent(tlng,tlat)) {
  106. particle.age = 0;
  107. } else {
  108. uv = field.getIn(tx, ty);
  109. nextLng = tlng + self.calc_speedRate[0] * uv[0];
  110. nextLat = tlat + self.calc_speedRate[1] * uv[1];
  111. particle.lng = tlng;
  112. particle.lat = tlat;
  113. particle.x = tx;
  114. particle.y = ty;
  115. particle.tlng = nextLng;
  116. particle.tlat = nextLat;
  117. particle.age--;
  118. }
  119. }
  120. });
  121. if (self.particles.length <= 0) this.removeLines();
  122. self._drawLines();
  123. },
  124. //粒子是否在地图范围内
  125. isInExtent:function(lng,lat){
  126. var calcExtent = this.initExtent;
  127. if((lng>=calcExtent[0] && lng<=calcExtent[1]) && (lat>=calcExtent[2] && lat<=calcExtent[3])) return true;
  128. return false;
  129. },
  130. _resize:function(width,height){
  131. this.canvasWidth = width;
  132. this.canvasHeight = height;
  133. },
  134. _parseWindJson: function () {
  135. var uComponent = null,
  136. vComponent = null,
  137. header = null;
  138. this.windData.forEach(function (record) {
  139. var type = record.header.parameterCategory + "," + record.header.parameterNumber;
  140. switch (type) {
  141. case "2,2":
  142. uComponent = record['data'];
  143. header = record['header'];
  144. break;
  145. case "2,3":
  146. vComponent = record['data'];
  147. break;
  148. default:
  149. break;
  150. }
  151. });
  152. return {
  153. header: header,
  154. uComponent: uComponent,
  155. vComponent: vComponent
  156. };
  157. },
  158. removeLines: function () {
  159. window.cancelAnimationFrame(this.animateFrame);
  160. this.isdistory = true;
  161. this.canvas.width = 1;
  162. // document.getElementById('content').removeChild(this.canvas);
  163. },
  164. //根据粒子当前所处的位置(棋盘网格位置),计算经纬度,在根据经纬度返回屏幕坐标
  165. _tomap: function (lng,lat,particle) {
  166. var ct3 = Cesium.Cartesian3.fromDegrees(lng,lat,0);
  167. // 判断当前点是否在地球可见端
  168. var isVisible = new Cesium.EllipsoidalOccluder(Cesium.Ellipsoid.WGS84, this.viewer.camera.position).isPointVisible(ct3);
  169. var pos = Cesium.SceneTransforms.wgs84ToWindowCoordinates(this.viewer.scene, ct3);
  170. if(!isVisible){
  171. particle.age = 0;
  172. }
  173. // console.log(pos);
  174. return pos?[pos.x,pos.y]:null;
  175. },
  176. //根据经纬度,算出棋盘格位置
  177. _togrid: function (lng,lat) {
  178. var field = this.windField;
  179. var x = (lng-this.initExtent[0])/(this.initExtent[1]-this.initExtent[0])*(field.cols-1);
  180. var y = (this.initExtent[3]-lat)/(this.initExtent[3]-this.initExtent[2])*(field.rows-1);
  181. return [x,y];
  182. },
  183. _drawLines: function () {
  184. var self = this;
  185. var particles = this.particles;
  186. this.canvasContext.lineWidth = self.lineWidth;
  187. //后绘制的图形和前绘制的图形如果发生遮挡的话,只显示后绘制的图形跟前一个绘制的图形重合的前绘制的图形部分,示例:https://www.w3school.com.cn/tiy/t.asp?f=html5_canvas_globalcompop_all
  188. this.canvasContext.globalCompositeOperation = "destination-in";
  189. this.canvasContext.fillRect(0,0,this.canvasWidth,this.canvasHeight);
  190. this.canvasContext.globalCompositeOperation = "lighter";//重叠部分的颜色会被重新计算
  191. this.canvasContext.globalAlpha = 0.9;
  192. this.canvasContext.beginPath();
  193. this.canvasContext.strokeStyle = this.color;
  194. particles.forEach(function (particle) {
  195. var movetopos = self._tomap(particle.lng, particle.lat,particle);
  196. var linetopos = self._tomap(particle.tlng, particle.tlat,particle);
  197. // console.log(movetopos,linetopos);
  198. if(movetopos!=null && linetopos!=null){
  199. self.canvasContext.moveTo(movetopos[0],movetopos[1]);
  200. self.canvasContext.lineTo(linetopos[0],linetopos[1]);
  201. }
  202. });
  203. this.canvasContext.stroke();
  204. },
  205. //随机数生成器(小数)
  206. fRandomByfloat:function(under, over){
  207. return under+Math.random()*(over-under);
  208. },
  209. //随机数生成器(整数)
  210. fRandomBy:function(under, over){
  211. switch(arguments.length){
  212. case 1: return parseInt(Math.random()*under+1);
  213. case 2: return parseInt(Math.random()*(over-under+1) + under);
  214. default: return 0;
  215. }
  216. },
  217. //根据当前风场extent随机生成粒子
  218. randomParticle: function (particle) {
  219. var safe = 30,x=-1, y=-1,lng=null,lat=null;
  220. var hasextent = this.extent.length!=0;
  221. var calc_extent = hasextent?this.extent:this.initExtent;
  222. do {
  223. try{
  224. if(hasextent){
  225. var pos_x = this.fRandomBy(0,this.canvasWidth);
  226. var pos_y = this.fRandomBy(0,this.canvasHeight);
  227. var cartesian = this.viewer.camera.pickEllipsoid(new Cesium.Cartesian2(pos_x, pos_y), this.viewer.scene.globe.ellipsoid);
  228. var cartographic = this.viewer.scene.globe.ellipsoid.cartesianToCartographic(cartesian);
  229. if(cartographic){
  230. //将弧度转为度的十进制度表示
  231. lng = Cesium.Math.toDegrees(cartographic.longitude);
  232. lat = Cesium.Math.toDegrees(cartographic.latitude);
  233. }
  234. }else{
  235. lng = this.fRandomByfloat(calc_extent[0],calc_extent[1]);
  236. lat = this.fRandomByfloat(calc_extent[2],calc_extent[3]);
  237. }
  238. }catch(e){
  239. }
  240. if(lng){
  241. var gridpos = this._togrid(lng,lat);
  242. x = gridpos[0];
  243. y = gridpos[1];
  244. }
  245. } while (this.windField.getIn(x, y)[2] <= 0 && safe++ < 30);
  246. var field = this.windField;
  247. var uv = field.getIn(x, y);
  248. var nextLng = lng + this.calc_speedRate[0] * uv[0];
  249. var nextLat = lat + this.calc_speedRate[1] * uv[1];
  250. particle.lng = lng;
  251. particle.lat = lat;
  252. particle.x = x;
  253. particle.y = y;
  254. particle.tlng = nextLng;
  255. particle.tlat = nextLat;
  256. particle.speed = uv[2];
  257. particle.age = Math.round(Math.random() * this.maxAge);//每一次生成都不一样
  258. return particle;
  259. }
  260. };
  261. /****
  262. *棋盘类
  263. *根据风场数据生产风场棋盘网格
  264. ****/
  265. var CanvasWindField = function (obj) {
  266. this.west = null;
  267. this.east = null;
  268. this.south = null;
  269. this.north = null;
  270. this.rows = null;
  271. this.cols = null;
  272. this.dx = null;
  273. this.dy = null;
  274. this.unit = null;
  275. this.date = null;
  276. this.grid = null;
  277. this._init(obj);
  278. };
  279. CanvasWindField.prototype = {
  280. constructor: CanvasWindField,
  281. _init: function (obj) {
  282. var header = obj.header,
  283. uComponent = obj['uComponent'],
  284. vComponent = obj['vComponent'];
  285. this.west = +header['lo1'];
  286. this.east = +header['lo2'];
  287. this.south = +header['la2'];
  288. this.north = +header['la1'];
  289. this.rows = +header['ny'];
  290. this.cols = +header['nx'];
  291. this.dx = +header['dx'];
  292. this.dy = +header['dy'];
  293. this.unit = header['parameterUnit'];
  294. this.date = header['refTime'];
  295. this.grid = [];
  296. var k = 0,
  297. rows = null,
  298. uv = null;
  299. for (var j = 0; j < this.rows; j++) {
  300. rows = [];
  301. for (var i = 0; i < this.cols; i++, k++) {
  302. uv = this._calcUV(uComponent[k], vComponent[k]);
  303. rows.push(uv);
  304. }
  305. this.grid.push(rows);
  306. }
  307. },
  308. _calcUV: function (u, v) {
  309. return [+u, +v, Math.sqrt(u * u + v * v)];
  310. },
  311. //二分差值算法计算给定节点的速度
  312. _bilinearInterpolation: function (x, y, g00, g10, g01, g11) {
  313. var rx = (1 - x);
  314. var ry = (1 - y);
  315. var a = rx * ry, b = x * ry, c = rx * y, d = x * y;
  316. var u = g00[0] * a + g10[0] * b + g01[0] * c + g11[0] * d;
  317. var v = g00[1] * a + g10[1] * b + g01[1] * c + g11[1] * d;
  318. return this._calcUV(u, v);
  319. },
  320. getIn: function (x, y) {
  321. if(x<0 || x>=359 || y>=180){
  322. return [0,0,0];
  323. }
  324. var x0 = Math.floor(x),
  325. y0 = Math.floor(y),
  326. x1, y1;
  327. if (x0 === x && y0 === y) return this.grid[y][x];
  328. x1 = x0 + 1;
  329. y1 = y0 + 1;
  330. var g00 = this.getIn(x0, y0),
  331. g10 = this.getIn(x1, y0),
  332. g01 = this.getIn(x0, y1),
  333. g11 = this.getIn(x1, y1);
  334. var result = null;
  335. try{
  336. result = this._bilinearInterpolation(x - x0, y - y0, g00, g10, g01, g11);
  337. }catch(e){
  338. console.log(x,y);
  339. }
  340. return result;
  341. },
  342. isInBound: function (x, y) {
  343. if ((x >= 0 && x < this.cols-1) && (y >= 0 && y < this.rows-1)) return true;
  344. return false;
  345. }
  346. };
  347. /****
  348. *粒子对象
  349. ****/
  350. var CanvasParticle = function () {
  351. this.lng = null;//粒子初始经度
  352. this.lat = null;//粒子初始纬度
  353. this.x = null;//粒子初始x位置(相对于棋盘网格,比如x方向有360个格,x取值就是0-360,这个是初始化时随机生成的)
  354. this.y = null;//粒子初始y位置(同上)
  355. this.tlng = null;//粒子下一步将要移动的经度,这个需要计算得来
  356. this.tlat = null;//粒子下一步将要移动的y纬度,这个需要计算得来
  357. this.age = null;//粒子生命周期计时器,每次-1
  358. this.speed = null;//粒子移动速度,可以根据速度渲染不同颜色
  359. };
  360. export default CanvasWindy;