>웹 프론트엔드 >H5 튜토리얼 >Amap + 캔버스 드로잉을 결합하여 작은 프로젝트를 구현합니다.

Amap + 캔버스 드로잉을 결합하여 작은 프로젝트를 구현합니다.

巴扎黑
巴扎黑원래의
2017-06-23 11:12:094595검색

예전에 친구가 스타트업 프로젝트를 의뢰했는데, 그 때 마침 시간이 좀 있어서 도와달라고 해서 흔쾌히 수락했어요.

제가 팀에 합류했을 때, 프로젝트의 프론트엔드와 백엔드가 분리되어 있다는 것을 알게 되었습니다. 이미 인터페이스를 주로 구현하는 백엔드 엔지니어가 있었고, IOS 측 엔지니어도 있었습니다. .아직 웹 프론트 엔드 엔지니어가 누락되었습니다. 예전에 js와 css를 좀 써봤고, 약간의 기술도 있었지만, 아직 프론트엔드 엔지니어가 되기에는 멀었습니다. 친구에게 상황을 설명한 후, 친구가 대담하게도 한 번 해보라고 했습니다. 가장 큰 이유는 사람을 찾을 수 없었기 때문입니다(현재 프론트엔드 엔지니어에 대한 견적이 너무 비싸기 때문일 수도 있습니다. 사업을 시작하면 가능하다면 돈을 절약할 수 있습니다. 이해합니다...), 다른 방법은 없습니다. 한 번에 한 단계씩 진행하세요.

다른 관리 페이지는 괜찮습니다. 메인 대시보드는 경도, 위도, 반경, 각도 등을 기준으로 스프링클러 기계의 실시간 위치와 스프링클러 스프링클러의 부채꼴 영역을 그려야 합니다.

저는 Amap을 사용해본 적도 없고 그림을 그려본 적도 없어서 처음 이 프로젝트를 받았을 때 정말 혼란스러울 수밖에 없었어요. Amap의 API를 공부하다가 그 안에 그림이 있다는 걸 알게 됐거든요. 폴리라인 도면, 폴리곤 및 기타 API. 곧 Amap에서 제공하는 공식 데모를 기반으로 다음 코드를 신속하게 작성했습니다.

 1 //开始绑定 2                 for (var m = 0; m < deviceList.length; m++) { 3                     var device = deviceList[m]; 4                     var point = new AMap.LngLat(device.longitude, device.latitude);// 圆心位置; 5  6                     var circle = new AMap.Circle({ 7                         center: point,// 圆心位置 8                         radius: device.radius, //半径 9                         strokeColor: "white", //线颜色10                         strokeOpacity: 1, //线透明度11                         strokeWeight: 1, //线粗细度12                         fillColor: "#6e97ce", //填充颜色13                         fillOpacity: 0.9//填充透明度14                     });15 16 17                     circle.setMap(map);18 19                     var marker = new AMap.Marker({20                         map: map,21                         position: [device.longitude, device.latitude],22                     });23 24                     //注册点击事件25                     addClickHandler(circle, device);26 27                 }
코드 보기

다음 그래픽이 구현되었습니다. 그런데 폴리라인과 섹터를 그릴 때 지도상의 API가 완벽하게 구현되지 못하고, 결과 섹터가 항상 조금씩 변형되는 현상을 발견했습니다. 아직 제가 원하는 효과와는 조금 거리가 있습니다.

Amap의 API를 계속 사용할 수밖에 없습니다...

레이어에서 이미지 레이어를 봤는데, 내 요구 사항을 충족할 수 있을 것 같지만 순수 js이고 동적으로 생성하고 싶습니다. 그림을 지도에 바인딩하는 것이 너무 복잡해 보입니다. . . 내 프론트엔드 실력이 너무 뛰어난 것일 수도 있다.

 좋아, 포기하고 계속 연구해봐...

 커스텀 레이어가 캔버스()를 사용하여 만들어진 것을 발견하고 눈이 반짝반짝 빛이 났습니다. 그러면 캔버스를 사용하여 그림을 그린 다음 지도에 붙여넣을 수 있습니다. 나는 조금 흥분했다. . .

그런데 생각해보니 저는 캔버스를 사용해본 적이 없어서 어쩔 수 없이 계속 읽어보게 되었어요...

학습자료를 많이 찾아보다가 저랑 좀 닮은 사진을 발견했어요. 위대한 예술가들이 캔버스를 이용해 그린 시계. 아직도 움직일 수 있고, 새로운 문이 열린 것 같은 느낌이 듭니다. . .

대가님들의 코드를 잔뜩 참고했습니다(원래는 링크를 하나씩 다 올리고 싶었는데 북마크하는 걸 깜빡해서 지금은 링크를 찾을 수가 없네요). 단 몇 마디로, 하지만 며칠 밤을 씹은 끝에 마침내 그래픽을 그렸습니다. . .

말도 안 돼요. 코드부터 시작해 보겠습니다.

  1  <div>  2         <canvas id="pie" width="300px" height="300px"></canvas>  3     </div>  4     <script>  5         var dom = document.getElementById("pie");  6         var ctx = dom.getContext("2d");  7         var width = ctx.canvas.width;  8         var height = ctx.canvas.height;  9         var r = width / 2; 10         var rem = width / 200; 11  12  13         function drawBackground() { 14             ctx.save(); 15             ctx.translate(r, r);//重新定义圆点到中心 16             ctx.beginPath(); 17             ctx.lineWidth = rem; 18             ctx.fillStyle = "#00AEE8"; 19             ctx.strokeStyle = "#fff"; 20             ctx.arc(0, 0, r, 0, Math.PI * 2, false);//圆点坐标,起始角0,结束角2π,顺时针 21             ctx.stroke(); 22             ctx.fill(); 23         } 24  25         function drawsector(sDeg,eDeg) { 26             //画扇形 27             ctx.beginPath(); 28             //定义起点 29             ctx.moveTo(0, 0); 30             ctx.fillStyle = "#0A73B1"; 31             //以起点为圆心,画一个半径为100的圆弧 32             ctx.arc(0, 0, r, sDeg * Math.PI / 180, eDeg * Math.PI / 180); 33             ctx.closePath(); 34             //ctx.stroke(); 35             ctx.fill(); 36  37         } 38  39         function drawtext(PDeg) { 40             //写文字 41             ctx.font = "18px Arial"; 42             ctx.textAlign = "center"; 43             ctx.textBaseline = "middle"; 44             ctx.strokeStyle = "black"; 45             ctx.fillStyle = "black"; 46             var rad = 90 * Math.PI / 180;//弧度 47             var x = (r - 30 * rem) * Math.cos(rad); 48             var y = (r - 30 * rem) * Math.sin(rad); 49             ctx.rotate((PDeg-90) * Math.PI / 180); 50             ctx.strokeText("<", x, y); 51             ctx.fillText("<", x, y); 52  53         } 54  55         function drawStart(rDeg) {//起始位置 56             ctx.save(); 57             ctx.beginPath(); 58             var rad = rDeg * Math.PI / 180;//弧度 59             var x = (r) * Math.cos(rad); 60             var y = (r) * Math.sin(rad); 61  62             ctx.strokeStyle = "black"; 63             ctx.lineWidth = 2*rem; 64             ctx.moveTo(0, 0); 65             ctx.lineTo(x, y); 66             ctx.lineCap = "round"; 67             ctx.stroke(); 68             ctx.restore(); 69         } 70         function drawPosition(PDeg) {//实时位置 71             ctx.save(); 72             ctx.beginPath(); 73             var rad = PDeg * Math.PI / 180 ; 74             //ctx.rotate(rad); 75             var x = (r) * Math.cos(rad); 76             var y = (r) * Math.sin(rad); 77  78             ctx.strokeStyle = "#fff"; 79  80             ctx.lineWidth = 3 * rem; 81             ctx.moveTo(0, 0); 82             ctx.lineTo(x, y); 83             ctx.lineCap = "round"; 84             ctx.stroke(); 85  86             ctx.restore(); 87         } 88  89         function drawPause() {//暂停 90             ctx.save(); 91             ctx.beginPath(); 92             var rad = 120 * Math.PI / 180; 93             //ctx.rotate(rad); 94             var x = (r) * Math.cos(rad); 95             var y = (r) * Math.sin(rad); 96  97             ctx.strokeStyle = "#fff"; 98  99             ctx.lineWidth = 10 * rem;100             ctx.moveTo(x+30, -y+80);101             ctx.lineTo(x+30, y-80);102             ctx.lineCap = "round";103             ctx.stroke();104 105             ctx.restore();106             107             108             ctx.save();109             ctx.beginPath();110             var rad = 60 * Math.PI / 180;111            112             var x2 = (r) * Math.cos(rad);113             var y2 = (r) * Math.sin(rad);114 115             ctx.strokeStyle = "#fff";116 117             ctx.lineWidth = 10 * rem;118             ctx.moveTo(x2-30, -y2+80);119             ctx.lineTo(x2-30, y2-80);120             ctx.lineCap = "round";121             ctx.stroke();122 123             ctx.restore();124         }125         function draw() {126             ctx.clearRect(0, 0, width, height);127             drawBackground();//背景128             drawsector(50, 180);129             130             //drawPause();131             132             drawStart(50);133             drawPosition(100);134             drawtext(110);135             ctx.restore();136         }137        138 139         draw();140     </script>
코드 보기

그래픽은 다음과 같습니다.

이 그림에서 제가 더 문제라고 생각하는 점은 주목할 가치가 있습니다. 작은 검은색 화살표는 회전을 사용합니다. 반복 테스트를 통해 0~360도에서 원의 중심을 따라 회전하는 것으로 나타났습니다. 아래 그림의 검은색 화살표는 80° 방향으로 회전합니다. 사실 이 위치는 +90°와 같습니다. 일관성이 있으므로 안심하십시오(즉, 빨간색 화살표). 이 기능을 파악하여 화살표를 원 방향으로 회전시키는 문제를 해결했습니다. .

그림을 다 완성하고 나니 다시 기하학적 도형을 복습하는 기분이더라구요... 역시 수학, 물리, 화학 잘 배우면 세계여행도 두렵지 않겠죠 ㅎㅎ .

이제 캔버스 지도가 기본적으로 완성되었으니 이를 어떻게 가오더 지도에 통합하고 지도의 비율에 따라 크기를 조정할 것인가가 나의 다음 과제가 되었습니다...

                         알겠습니다. 저는 이번이 처음이에요 블로그를 쓰다보니, (-__-)b 그냥 제가 겪은 생각이나 어려움을 기록하는 것 뿐이에요.

다음 글에서는 Gaode 지도에 통합하는 방법과 몇 가지 공통점에 대해 집중적으로 살펴보겠습니다.

위 내용은 Amap + 캔버스 드로잉을 결합하여 작은 프로젝트를 구현합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.