- 拖动左侧图标到地图上某个位置,在地图上创建该类型的点,存储在list
- 拖拽点移动,重新更新其坐标
- 编辑点的大小及角度,清除之前的重新创建新的点
- 删除点,清除点,但数据不能清除,更改show
- 最后保存list数据并截图保存
- 下次进来渲染点重新编辑
<template>
<div class="index5 w100 h100 white relative">
<canvas
id="myCanvas"
style="display: none; width: 100vw; height: 100vh"
></canvas>
<img
src="./components/img/home.png"
alt=""
class="home pointer"
@click="$router.back()"
/>
<!-- 地图区域 -->
<div id="containerMap" class="w100 h100" ref="htmlImg"></div>
<!-- 左侧菜单 -->
<div
v-show="showMenu"
class="left h80 absolute animate__animated animate__slideInLeft animate__fast"
>
<div class="w100 pdRem-20 h100">
<dv-border-box-8
:color="['rgba(2, 208, 249,.5)', '#0dcff2']"
backgroundColor="rgba(4, 33, 66,.8)"
>
<div class="plrRem-20 ptbRem-10 flex flex-column w100 h100">
<div class="title flex_l mtb-20">
<img src="./components/img/index1-title.png" alt="" />
作战标绘
</div>
<div class="flex flex-wrap w100">
<div
class="item plr-20 text-center mb-20 pointer"
style="width: 50%"
v-for="(item, index) in tab"
:key="index"
>
<!-- ☆☆☆☆☆ 拖拽图片默认会打开窗口预览,换成背景图即可 -->
<!-- <img :src="item.icon" alt="" style="width: 50px" /> -->
<div
draggable="true"
@dragstart="onDragStart"
@dragend="onDragEnd"
:data-index="index"
class="img-bg"
:style="{
backgroundImage: `url(${item.icon})`,
backgroundRepeat: 'no-repeat',
backgroundPosition: 'center',
backgroundSize: 'cover',
backgroundAttachment: 'local',
}"
></div>
<div class="mt-4">{{ item.text }}</div>
</div>
</div>
<img
src="./components/img/index5-btn4.png"
alt=""
style="width: 140px; margin: 60px auto 20px"
@click="issueImg"
class="pointer"
/>
</div>
</dv-border-box-8>
</div>
</div>
<!-- 右下控制 -->
<div
v-show="showMenu && showEdit"
class="right absolute animate__animated animate__slideInLeft animate__fast pdRem-20"
>
<div class="w100 h100 editBox ptRem-14 pbRem-10 plRem-10 prRem-20">
<div class="plrRem-20 ptbRem-10 flex flex-column w100 h100 editContent">
<div class="title flex_l mtb-10">
<img src="./components/img/index1-title.png" alt="" />
标绘属性
</div>
<div class="plrRem-20 mbRem-20 flex flex-column">
<div class="flex1">
<div style="width: 60px; font-size: 0.2rem">缩放:</div>
<div class="flex-1">
<el-slider
:max="10"
:min="0.5"
:step="0.5"
v-model="scale"
@change="editMarker"
></el-slider>
</div>
</div>
<div class="flex1">
<div style="width: 60px; font-size: 0.2rem">角度:</div>
<div class="flex-1">
<el-slider
:max="360"
:step="5"
v-model="rotate"
@change="editMarker"
></el-slider>
</div>
</div>
</div>
<div class="w100 center mb-20">
<!-- <img
src="./components/img/index5-btn1.png"
alt=""
style="width: 140px"
@click="saveImg"
class="pointer"
v-if="isSave"
/> -->
<img
src="./components/img/index5-btn3.png"
alt=""
style="width: 120px"
@click="removeMarker"
class="pointer"
/>
</div>
</div>
</div>
</div>
</div>
</template>
<script>
import html2canvas from "html2canvas";
import "./components/js/flexible.js";
import {
getPoliceDetailApi,
uploadHtmlImg,
dispatchHtmlImg,
} from "./components/js/api";
import AMapLoader from "@amap/amap-jsapi-loader";
import icon1 from "./components/img/index5-1.png";
import icon2 from "./components/img/index5-2.png";
import icon3 from "./components/img/index5-3.png";
import icon4 from "./components/img/index5-4.png";
import icon5 from "./components/img/index5-5.png";
import icon6 from "./components/img/index5-6.png";
import icon7 from "./components/img/index5-7.png";
import icon8 from "./components/img/index5-8.png";
import icon9 from "./components/img/index5-9.png";
window._AMapSecurityConfig = {
securityJsCode: "3e9a9aeb435372339552fc215f8d0918",
};
export default {
data() {
return {
position: [],
policeId: "",
map: null,
showMenu: true,
tab: [
{
id: 1,
icon: icon1,
text: "消防车",
},
{
id: 2,
icon: icon2,
text: "消防员",
},
{
id: 3,
icon: icon3,
text: "水管",
},
{
id: 4,
icon: icon4,
text: "进攻方向",
},
{
id: 5,
icon: icon5,
text: "消防栓",
},
{
id: 6,
icon: icon6,
text: "危化品",
},
{
id: 7,
icon: icon7,
text: "风向",
},
{
id: 8,
icon: icon8,
text: "指北矢标",
},
],
list: [],
draggingEnded: false,
index: null,
showEdit: false,
item: {},
scale: 1,
rotate: 0,
baseUrl: process.env.VUE_APP_BASE_API,
isSave: false,
};
},
mounted() {
this.position = this.$route.query.position.split(",");
this.policeId = this.$route.query.id;
this.initAMap();
},
beforeDestroy() {
this.map.destroy();
},
methods: {
initAMap() {
AMapLoader.load({
key: "130cca3be68a2ff0fd5ebb6de25e4eac",
version: "2.0",
plugins: [
"AMap.Driving",
],
})
.then((AMap) => {
this.map = new AMap.Map("containerMap", {
zoom: 13,
zooms: [2, 20],
center: [119.419251, 32.400703],
mapStyle: "amap://styles/blue",
WebGLParams: {
preserveDrawingBuffer: true,
},
});
this.map.on("complete", () => {
console.log("地图加载完成");
this.isSave = true;
this.createFireMarker();
this.map.on("mouseover", this.onMouseOver);
getPoliceDetailApi(this.policeId).then((res) => {
if (res.data.screenshotParam) {
this.list = JSON.parse(res.data.screenshotParam);
this.showMarker();
}
});
});
this.map.on("click", (e) => {
var lnglat = e.lnglat;
console.log("地图被点击,坐标:" + lnglat.lng + ", " + lnglat.lat);
this.showEdit = false;
});
})
.catch((e) => {
console.log(e);
});
},
createFireMarker() {
var marker = new AMap.Marker({
position: new AMap.LngLat(...this.position),
icon: new AMap.Icon({
image: fireIcon,
imageSize: new AMap.Size(32, 42),
}),
});
this.map.add(marker);
this.map.setCenter(new AMap.LngLat(...this.position));
},
onDragStart(event) {
this.draggingEnded = false;
this.index = event.target.dataset.index * 1;
},
onDragEnd(event) {
this.draggingEnded = true;
},
onMouseOver(event) {
if (this.draggingEnded) {
this.draggingEnded = false;
this.createMarker([event.lnglat.lng, event.lnglat.lat]);
}
},
showMarker() {
this.list.forEach((item, index) => {
item.show = true;
item.marker = null;
item.icon = [icon1, icon2, icon3, icon4, icon5, icon6, icon7, icon8][
item.id * 1
];
var markerContent =
'<div class="custom-content-marker" >' +
'<img style="transform:rotate(' +
item.rotate +
"deg) scale(" +
item.scale +
');width:60px;height:60px" src=" ' +
item.icon +
'">' +
"</div>";
item.marker = new AMap.Marker({
position: new AMap.LngLat(...item.lnglat),
content: markerContent,
offset: new AMap.Pixel(-25, -50),
cursor: "move",
draggable: true,
});
this.map.add(item.marker);
this.list[index].marker.on("click", (e) => {
this.item = this.list[index];
this.scale = this.item.scale;
this.rotate = this.item.rotate;
this.showEdit = true;
console.log("点击:", this.item.id);
});
this.list[index].marker.on("dragend", (e) => {
this.list[index].lnglat = [
this.list[index].marker.getPosition().lng,
this.list[index].marker.getPosition().lat,
];
this.item = this.list[index];
this.scale = this.item.scale;
this.rotate = this.item.rotate;
this.showEdit = true;
console.log(
"最新坐标:",
[
this.list[index].marker.getPosition().lng,
this.list[index].marker.getPosition().lat,
],
this.list,
this.item
);
});
});
console.log("createMarker", this.list);
},
createMarker(lnglat) {
let length = this.list.length;
this.list.push({
id: this.index,
lnglat: lnglat,
icon: this.tab[this.index].icon,
marker: null,
scale: 1,
rotate: 0,
show: true,
});
this.item = this.list[length];
console.log("createMarker", this.index);
var markerContent =
'<div class="custom-content-marker" >' +
'<img style="transform:rotate(' +
this.item.rotate +
"deg) scale(" +
this.item.scale +
');width:60px;height:60px" src=" ' +
this.item.icon +
'">' +
"</div>";
this.item.marker = new AMap.Marker({
position: new AMap.LngLat(...this.item.lnglat),
content: markerContent,
offset: new AMap.Pixel(-25, -50),
cursor: "move",
draggable: true,
});
this.map.add(this.item.marker);
this.showEdit = true;
this.scale = this.item.scale;
this.rotate = this.item.rotate;
console.log("createMarker", this.list, this.item);
this.list[length].marker.on("click", (e) => {
this.item = this.list[length];
this.scale = this.item.scale;
this.rotate = this.item.rotate;
this.showEdit = true;
console.log("点击:", this.item.id);
});
this.list[length].marker.on("dragend", (e) => {
this.list[length].lnglat = [
this.list[length].marker.getPosition().lng,
this.list[length].marker.getPosition().lat,
];
this.item = this.list[length];
this.scale = this.item.scale;
this.rotate = this.item.rotate;
this.showEdit = true;
console.log(
"最新坐标:",
[
this.list[length].marker.getPosition().lng,
this.list[length].marker.getPosition().lat,
],
this.list,
this.item
);
});
},
editMarker() {
var index = this.list.indexOf(this.item);
console.log("编辑", this.rotate, this.scale, this.list[index]);
this.list[index].scale = this.scale;
this.list[index].rotate = this.rotate;
this.map.remove(this.list[index].marker);
this.list[index].marker = null;
var markerContent =
'<div class="custom-content-marker" >' +
'<img style="transform:rotate(' +
this.list[index].rotate +
"deg) scale(" +
this.list[index].scale +
');width:60px;height:60px" src=" ' +
this.list[index].icon +
'">' +
"</div>";
console.log("编辑", markerContent);
this.list[index].marker = new AMap.Marker({
position: new AMap.LngLat(...this.list[index].lnglat),
content: markerContent,
offset: new AMap.Pixel(-25, -50),
cursor: "move",
draggable: true,
});
this.map.add(this.list[index].marker);
this.list[index].marker.on("click", (e) => {
this.item = this.list[index];
this.scale = this.list[index].scale;
this.rotate = this.list[index].rotate;
this.showEdit = true;
console.log("点击11:", this.item.id);
});
this.list[index].marker.on("dragend", (e) => {
this.item = this.list[index];
this.item.lnglat = [
this.list[index].marker.getPosition().lng,
this.list[index].marker.getPosition().lat,
];
this.scale = this.list[index].scale;
this.rotate = this.list[index].rotate;
this.showEdit = true;
});
},
removeMarker() {
var index = this.list.indexOf(this.item);
this.list[index].show = false;
this.map.remove(this.list[index].marker);
this.item = {};
this.showEdit = false;
console.log("删除", this.list, this.item, index);
},
issueImg() {
this.showMenu = false;
let arr = this.list.filter((item) => item.show);
let arr1 = arr.map((item) => {
return {
id: item.id,
lnglat: item.lnglat,
scale: item.scale,
rotate: item.rotate,
};
});
html2canvas(this.$refs.htmlImg, {
useCORS: true,
x: window.pageXOffset,
y: window.pageYOffset,
}).then((canvas) => {
var img = new Image();
var canvas2 = document.getElementById("myCanvas");
var ctx = canvas2.getContext("2d");
ctx.fillStyle = "#FFFFFF";
ctx.lineWidth = "1";
ctx.rect(20, 30, canvas2.width - 40, canvas2.height - 60);
ctx.stroke();
img.src = canvas.toDataURL();
var url = canvas.toDataURL("image/png");
const blobImage = this.dataURLtoBlob(url);
let fileName = `htmlImg_${new Date().getTime()}.jpg`;
const fileOfBlob = new File([blobImage], fileName);
let formData = new FormData();
formData.append("file", fileOfBlob, fileName);
this.isSave = false;
uploadHtmlImg(formData).then((res) => {
this.showMenu = true;
this.isSave = true;
let saveObj = {
policeId: this.$route.query.id,
fileName: res.fileName,
url: res.url,
screenshotParam: JSON.stringify(arr1),
};
dispatchHtmlImg(saveObj).then((res) => {
this.$modal.msgSuccess("标绘下发成功,可在小程序对应警情中查看");
});
console.log("http://192.168.0.19:12020" + res.url);
});
});
},
dataURLtoBlob(dataUrl) {
let arr = dataUrl.split(","),
mime = arr[0].match(/:(.*?);/)[1],
bstr = atob(arr[1]),
n = bstr.length,
u8arr = new Uint8Array(n);
while (n--) {
u8arr[n] = bstr.charCodeAt(n);
}
return new Blob([u8arr], {
type: mime,
});
},
},
};
</script>
<style scoped lang="scss">
@import "./components/css/rem.scss";
.index5 {
.home {
width: 50px;
position: fixed;
right: 20px;
top: 20px;
z-index: 9999;
}
.left {
width: 280px;
left: 0;
top: 10%;
.img-bg {
width: 50px;
height: 50px;
margin: auto;
}
}
.right {
width: 360px;
right: 50px;
bottom: 50px;
.editBox {
background: #0005495d;
background-image: url("./components/img/index5-bg-1.png");
background-repeat: no-repeat;
background-position: center;
background-size: 100% 100%;
background-attachment: local;
.editContent {
}
}
}
::v-deep .el-slider__runway {
background-color: rgba(0, 255, 255, 0.253);
}
::v-deep .el-slider__bar {
background-color: aqua;
}
::v-deep .el-slider__button {
border: 2px solid aqua;
}
::v-deep .amap-marker-label {
background: rgba(0, 0, 0, 0);
color: aqua;
border: none;
}
::v-deep .amap-icon {
width: 50px !important;
height: 50px !important;
}
.title {
font-size: 0.2rem;
font-weight: 700;
text-shadow: 0px 3px 7px #006cbf;
img {
width: 0.4rem;
height: 0.4rem;
}
}
.carInfo {
position: absolute;
border-radius: 10px;
border: 1px solid cyan;
background: rgba(0, 50, 73, 0.8);
background-size: 100% 100%;
padding: 16px 10px 14px 20px;
::v-deep .dv-water-pond-level text {
font-size: 15px;
font-weight: normal !important;
color: #fff !important;
}
}
}
.main {
height: 1600px;
width: 100%;
background: #d4edff;
}
.svg {
border: 1px solid red;
width: 100px;
height: 100px;
}
</style>