vue富文本编辑器插件:VUE3(二十六)基于wangeditor自定义富文本插件
项目中需要使用到wangeditor,这里封装一个自定义组件,便于后期使用。 首先,官方文档: https://www.wangeditor.com/ 1:安装 npm i wangeditor --
顺晟科技
2022-09-13 11:21:18
110
一、在vue-cli的框架下的main.js(或者main.ts)中引入高德插件,代码如下:
import Vue from \'vue\'
import VueAMap from \'vue-amap\'
import ElementUI from \'element-ui\'
import App from \'./App.vue\'
import router from \'./router\'
import store from \'./store\'
import \'./registerServiceWorker\'
Vue.use(VueAMap)
Vue.use(ElementUI)
VueAMap.initAMapApiLoader({
// 高德的key
key: \'你的高德key\',
// 插件集合
plugin: [
\'AMap.Autocomplete\',
\'AMap.PlaceSearch\',
\'AMap.Scale\',
\'AMap.OverView\',
\'AMap.ToolBar\',
\'AMap.MapType\',
\'AMap.PolyEditor\',
\'AMap.CircleEditor\',
\'AMap.Geocoder\',
\'AMap.Geolocation\'
],
// 高德 sdk 版本,默认为 1.4.4
v: \'1.4.10\'
})
Vue.config.productionTip = false
new Vue({
router,
store,
render: h => h(App)
}).$mount(\'#app\')
第三种画多边形的效果图:
注意:1、这种画多边形,开始就需要一个初始的多边形;
2、所以,输入要画多边形范围的地名,点击搜索,地图会跳转到搜索的地方,同时得到经纬度;
3、点“范围绘制”时,我再方法里根据第2步的经纬度,初始了一个多边形;
****隐藏彩蛋****
下图的 “请输入经纬度” 可以输入一大组的经纬度,按回车键,也可以画出多边形,在按“范围绘制”也可以更改;(格式如下:)
这个格式就是复制的地图上显示的经纬度坐标
106.2246 , 29.59258 106.225064 , 29.593287 106.226137 , 29.593558 106.22692 , 29.593083

二、第一种画化:使用Geolocation画多边形(效果是在地图点了,才会形成多边形)
// 新增 编辑 查看
<template>
<div class="point">
<el-header></el-header>
<div class="action-bar">
<el-form class="inline-form" :rules="rules" ref="formData" size="small" :model="formData">
<el-form-item label label-width="220" prop="location">
<el-input
:disabled="!ifFalg"
class="name-input"
clearable
v-model="formData.location"
placeholder="名称"
maxlength="30"
></el-input>
</el-form-item>
<el-form-item label prop="longitude">
<el-input
:disabled="!ifFalg"
class="my-input"
clearable
v-model.number="formData.longitude"
placeholder="经度 "
></el-input>
</el-form-item>
<el-form-item label prop="latitude">
<el-input
:disabled="!ifFalg"
class="my-input"
clearable
v-model.number="formData.latitude"
placeholder="纬度"
></el-input>
</el-form-item>
<el-button class="my-button" v-if="ifFalg" type="primary" @click="save" size="small">保存</el-button>
<el-button class="my-button" size="small" @click="close">关闭</el-button>
</el-form>
</div>
<div class="map-box">
<div class="map-tool">
<div v-if="ifFalg">
<el-checkbox v-model="enterType">地图上描点</el-checkbox>
</div>
<!-- <el-checkbox @change="checkbox" v-model="enterType">地图上描点</el-checkbox> -->
<div class="longlat">
<ul>
<li v-for="(item, index) in lnglatpoints" :key="index">
{{item.longitude}} , {{item.latitude}}
<i
v-if="ifFalg"
class="el-icon-close"
@click="deletes(item)"
></i>
</li>
</ul>
<el-input
v-if="ifFalg"
class="my-input"
size="small"
clearable
v-model="lngLat"
@keyup.enter.native="submitEnter"
placeholder="请输入经纬度"
></el-input>
<el-button v-if="ifFalg" size="small" @click="clear" type="primary" class="claer">清除</el-button>
</div>
</div>
<div class="map" id="map">
<el-amap
ref="map"
bubble
:plugin="plugin"
:zoom="map.zoom"
:center="map.center"
:events="events"
id="amap"
>
<el-amap-polygon
:events="plugin.events"
:path="path"
:draggable="draggable"
fillColor="#2b83f9"
fillOpacity="0.5"
strokeWeight="0"
strokeColor="#2b83f9"
strokeOpacity="0.5"
></el-amap-polygon>
<!-- <el-amap-marker :position="marker.position" :events="plugin.events"></el-amap-marker> -->
<el-amap-marker v-if="formData.type === 1" :position="map.center" :label="label"></el-amap-marker>
</el-amap>
</div>
</div>
</div>
</template>
<script lang="ts">
import * as api from \'@/utils/api/index\'
import { Component, Vue } from \'vue-property-decorator\'
import eHeader from \'@/components/header.vue\'
import { constants } from \'http2\'
import * as util from \'@/utils/util.ts\'
const testLongitude = (rule: any, value: string, callback: Function) => {
if (util.regExp.longitudeRegExp.test(value)) {
return callback()
} else {
return callback(new Error(\'请输入正确的经度\'))
}
}
const testLatitude = (rule: any, value: string, callback: Function) => {
if (util.regExp.latitudeRegExp.test(value)) {
return callback()
} else {
return callback(new Error(\'请输入正确的纬度\'))
}
}
@Component({
components: {
\'el-header\': eHeader
}
})
export default class point extends Vue {
private breadcrumbId = 0
private id = \'\'
private lngLat = \'\'
private ifFalg = true
private map = {
zoom: 15,
center: [106.55073, 29.56471]
}
private path: any = []
private draggable = false
private lnglatpoints: any = []
private enterType = false // 录入坐标 | 地图上描点
private cities = []
private formData = {
location: \'\',
longitude: \'\',
latitude: \'\'
}
plugin = {
pName: \'Geolocation\',
events: {}
}
events = {}
private test = 1
private rules = {
location: [
{ required: true, message: \'请输入接送点名称\', trigger: \'blur\' }
],
longitude: [{ validator: testLongitude, trigger: \'blur\' }],
latitude: [{ validator: testLatitude, trigger: \'blur\' }]
}
mounted() {
this.id = this.$route.params.id
this.breadcrumbId = Number(this.$route.query.breadcrumbId)
if (this.breadcrumbId === 2) {
this.ifFalg = false
}
if (this.id !== \'-1\') {
this.details()
}
// this.city()
let _this: any = this
// 地图点击事件
_this.events = {
click: (e: any) => {
if (this.enterType) {
this.path = []
console.log(e.lnglat)
let lnglat = e.lnglat
this.lnglatpoints.push({
latitude: lnglat.lat,
longitude: lnglat.lng
})
console.log(this.lnglatpoints)
this.lnglatpoints.map((val: any, index: number) => {
console.log(index)
if (index === 0) {
this.map.center = [val.longitude, val.latitude]
}
let arr = [val.longitude, val.latitude]
this.path.push(arr)
})
// this.setFitView()
}
}
}
// 多边形点击事件
_this.plugin.events = {
click: (e: any) => {
if (this.enterType) {
this.path = []
console.log(e.lnglat)
let lnglat = e.lnglat
this.lnglatpoints.push({
latitude: lnglat.lat,
longitude: lnglat.lng
})
console.log(this.lnglatpoints)
this.lnglatpoints.map((val: any, index: number) => {
console.log(index)
if (index === 0) {
this.map.center = [val.longitude, val.latitude]
}
let arr = [val.longitude, val.latitude]
this.path.push(arr)
})
// this.setFitView()
}
}
}
}// 获取接送范围集合
details() {
const loading = this.$loading({
lock: true,
text: \'加载中...\'
})
api.main.boss_line_point__get({ params: {param: this.id}}).then((res: any) => {
if (res.data.success) {
const response = res.data.data
this.formData = response
let points = res.data.data.points
if (points != null) {
for (let i = 0; i < points.length; i++) {
points[i].id = i
}
this.lnglatpoints = points
this.lnglatpoints.map((val: any, index: number) => {
if (index === 0) {
this.map.center = [val.longitude, val.latitude]
}
let arr = [val.longitude, val.latitude]
this.path.push(arr)
})
} else {
this.map.center = [
Number(this.formData.longitude),
Number(this.formData.latitude)
]
this.label.content = this.formData.location
}
setTimeout(this.setFitView, 0)
} else {
this.$message.error(res.data.message)
}
loading.close()
})
}
// 移除经纬度
deletes(data: any) {
let e: any = this
this.path = []
for (let i = 0; i < e.lnglatpoints.length; i++) {
if (
data.latitude === e.lnglatpoints[i].latitude &&
data.longitude === e.lnglatpoints[i].longitude
) {
e.lnglatpoints.splice(i, 1)
}
}
console.log(e.path)
this.lnglatpoints.map((val: any, index: number) => {
let arr = [val.longitude, val.latitude]
this.path.push(arr)
if (index === 0) {
this.map.center = [val.longitude, val.latitude]
}
console.log(this.path)
})
}
clear() {
this.$confirm(\'确认删除绘制的接送区域?\', \'删除\', {
confirmButtonText: \'确定\',
cancelButtonText: \'取消\',
type: \'warning\'
})
.then(() => {
let self: any = this
this.path = []
this.lnglatpoints = []
// this.map.center = [106.5507300000, 29.5647100000]
this.lngLat = \'\'
self.formData.points = []
})
.catch(() => {})
}
// 输入经纬度
submitEnter() {
// eslint-disable-next-line
const illegalRegExp = /^(\D|\d*\.?\d*,*\s)|[^\d\s,\.]|^\d*\.?\d*$|(,\.|\.,)+|(\d*\.*\d*,){2,}|(\d*\.){2,}|(\d*\s){2,}|(\s\d*\.?\d*|\D)$/g
const replaceWhiteSpaceRegExp = /(?<=(,|\.|\s))\s+|\s+(?=(,|\.))|^\s|\s+$/g
this.lngLat = this.lngLat.replace(replaceWhiteSpaceRegExp, \'\')
if (illegalRegExp.test(this.lngLat)) {
return this.$message.error(\'经纬度格式错误!\')
}
const lnglatArray = this.lngLat.split(\' \')
lnglatArray.forEach(lnglatString => {
const lnglatObject = {
longitude: lnglatString.split(\',\')[0],
latitude: lnglatString.split(\',\')[1]
}
this.lnglatpoints.push(lnglatObject)
})
this.path = []
this.lnglatpoints.map((val: any, index: number) => {
let arr = [val.longitude, val.latitude]
this.path.push(arr)
this.lngLat = \'\'
if (index === 0) {
this.map.center = [val.longitude, val.latitude]
}
})
}
setFitView() {
const vm: any = this
let map = vm.$refs.map.$$getInstance()
map.setFitView()
}
close() {
this.$router.push({
name: \'pointList\'
})
}
save() {
let e: any = this
let params: any = {}
if (this.id !== \'-1\') {
// 编辑
e.formData.id = this.id
params.id = this.id
}
e.formData.points = this.lnglatpoints
if (e.formData.location === \'\' || e.formData.location === null) {
this.$message.warning(\'名称不能为空!\')
return
}
if (this.lnglatpoints.length < 3 && e.formData.type === 2) {
this.$message.warning(\'经纬度不能小于三组!\')
return
}
params.points = this.lnglatpoints
params.location = this.formData.location
params.longitude = this.formData.longitude
params.latitude = this.formData.latitude
if (this.id !== \'-1\') {
api.main.boss_line_point_update_post({ data: params }).then((res: any) => {
if (res.data.success) {
this.$message.success(\'保存成功!\')
this.$router.push({
name: \'pointList\'
})
} else {
this.$message.error(res.data.message)
}
})
} else {
api.main
.boss_line_point_addAndBindLine_post({ data: params })
.then((res: any) => {
if (res.data.success) {
this.$message.success(\'保存成功!\')
this.$router.push({
name: \'pointList\'
})
} else {
this.$message.error(res.data.message)
}
})
}
}
}
</script>
<style lang="scss" scoped>
ul,
li {
list-style: none;
margin: 0;
padding: 0;
}
.inline-form {
display: flex;
display: -webkit-flex;
flex-direction: row;
flex-wrap: wrap;
.el-form-item {
margin-bottom: 10px;
margin-left: 15px;
display: flex;
}
.el-button {
margin-left: 15px;
height: 32px;
}
}
.action-bar {
box-sizing: border-box;
padding: 10px;
padding-bottom: 0;
border: {
top: 1px solid #ddd;
bottom: 1px solid #ddd;
}
.my-input {
width: 150px;
}
.name-input {
width: 260px;
}
}
.el-select-dropdown__item {
background-color: white;
text-indent: 10px;
}
.claer {
margin-top: 15px;
float: right;
}
$map_height: calc(100vh - 55px - 50px - 75px - 15px);
.map-box {
position: relative;
height: $map_height;
.map-tool {
position: absolute;
width: 220px;
z-index: 170;
top: 0;
left: 0;
max-height: 100%;
box-sizing: border-box;
padding: 10px;
overflow-y: auto;
background-color: #fff;
box-shadow: 2px 4px 7px 1px #dedede;
}
.map {
transition: all 0.6s;
position: absolute;
top: 0;
right: 0;
bottom: 0;
left: 0;
}
}
.swiper-box {
position: relative;
z-index: 161;
display: flex;
align-items: center;
flex-direction: row;
justify-content: center;
width: 100%;
transition: transform ease-in 0.6s;
transform: translateX(0);
white-space: nowrap;
.swiper-item {
width: 100%;
height: $map_height;
}
}
.hide-text-area {
transform: translateX(-100%);
}
.gray-map {
filter: grayscale(90%);
}
.longlat {
margin-top: 15px;
padding-bottom: 15px;
ul {
li {
padding: 6px;
background-color: #ddd;
border-radius: 4px;
margin-bottom: 15px;
font-size: 14px;
color: #666;
position: relative;
}
}
}
.el-icon-close {
display: inline-block;
position: absolute;
right: 10px;
color: #000 !important;
cursor: pointer;
}
.my-button {
margin-bottom: 10px;
}
</style>
三、第二种画化:使用AMap.MouseTool画多边形(效果是:多边形随鼠标左键点击,多边形直接跟着变化)
// 新增 编辑 查看
<template>
<div class="point">
<el-header></el-header>
<div class="action-bar">
<el-form class="inline-form" :rules="rules" ref="formData" size="small" :model="formData">
<el-form-item label prop="location">
<el-input
:disabled="!ifFalg"
class="name-input"
clearable
v-model="formData.location"
placeholder="名称"
maxlength="30"
></el-input>
</el-form-item>
<el-form-item label prop="longitude">
<el-input
:disabled="!ifFalg"
class="my-input"
clearable
v-model.number="formData.longitude"
placeholder="经度 "
></el-input>
</el-form-item>
<el-form-item label prop="latitude">
<el-input
:disabled="!ifFalg"
class="my-input"
clearable
v-model.number="formData.latitude"
placeholder="纬度"
></el-input>
</el-form-item>
<el-button class="my-button" v-if="ifFalg" type="primary" @click="save" size="small">保存</el-button>
<el-button class="my-button" size="small" @click="close">关闭</el-button>
</el-form>
</div>
<div class="map-box">
<div class="map-tool">
<div v-if="ifFalg">
<el-checkbox >地图上描点</el-checkbox>
</div>
<div class="longlat">
<ul><li v-for="(item, index) in lnglatpoints" :key="index">
{{item.longitude}} , {{item.latitude}}
<i
v-if="ifFalg"
class="el-icon-close"
@click="deletes(item)"
></i>
</li>
</ul>
<br>
<div>
<span >输入范围经纬度:</span>
<el-input
type="textarea"
autosize
placeholder="请输入内容"
v-model="lnglatpointsString">
</el-input>
</div>
<el-button v-if="ifFalg" size="small" @click="clear1" type="primary" class="claer1">确定</el-button>
<el-button v-if="ifFalg" size="small" @click="clear" type="primary" class="claer">清除</el-button>
</div>
</div>
<div class="map" id="map">
<el-amap
ref="map"
bubble
:zoom="map.zoom"
:center="map.center"
:events="mapEvents"
id="amap"
>
<el-amap-polygon
:events="plugin.events"
:path="path"
fillColor="#2b83f9"
fillOpacity="0.5"
strokeWeight="0"
strokeColor="#2b83f9"
strokeOpacity="0.5"
></el-amap-polygon>
<el-amap-marker v-if="formData.type === 1" :position="map.center" :label="label"></el-amap-marker>
</el-amap>
</div>
<div class="my-tools">
<el-row>
<el-button type="primary" v-if="ifFalg" @click="drawPolygon()">鼠标绘制</el-button>
<el-button type="primary" v-if="ifFalg" @click="polygonEditor.close()">结束编辑</el-button>
</el-row>
</div>
</div>
</div>
</template>
<script lang="ts">同上
/**
* 绘制多边形
*/
private drawPolygon () {
let vm: any = this
let map = vm.$refs.map.$$getInstance()
map.plugin([\'AMap.MouseTool\'], function () {
var mouseTool = new AMap.MouseTool(map)
var drawPolygon = mouseTool.polygon()
AMap.event.addListener(mouseTool, \'draw\', function (e: any) {
e.obj.Je.visible = false
let path = e.obj.getPath()
vm.drawPolygonsToMap(path)
path.forEach((point:any) => {
vm.lnglatpoints.push({
latitude: point.lat,
longitude: point.lng
})
});
// vm.mapDates =path
// e.obj.hide()
mouseTool.close()
})
})
}
同上
}
</script>
<style lang="scss" scoped>
和上面一样
</style>
三、第三种画化:使用AMap.Polygon和AMap.PolyEditor画多边形(推荐,效果是:https://lbs.amap.com/api/javascript-api/example/overlayers/polygon-draw-and-edit)
注意哦:1、以为这种画多边形,先需要3个点来确定初始的多边形,所以添加了一个功能:搜索 (功能:点击搜索名称的经纬度;);
2、然后我再 ‘范围绘制’ 的方法里根据“搜索”得来的经纬度,手动的弄了3个经纬度数组。
3、然后就可以快乐的画图了。(这画图是真的方便,特别是画范围很复杂的)
// 新增 编辑 查看
<template>
<div class="point">
<el-header></el-header>
<div class="action-bar">
<el-form class="inline-form" :rules="rules" ref="formData" size="small" :model="formData">
<el-form-item label prop="location">
<el-input
:disabled="!ifFalg"
class="name-input"
clearable
v-model="formData.location"
placeholder="名称"
maxlength="30"
></el-input>
</el-form-item>
<el-button class="my-button" type="info" @click="getLocation" size="small">搜索</el-button>
<el-form-item label prop="longitude">
<el-input
:disabled="!ifFalg"
class="my-input"
clearable
v-model.number="formData.longitude"
placeholder="经度 "
></el-input>
</el-form-item>
<el-form-item label prop="latitude">
<el-input
:disabled="!ifFalg"
class="my-input"
clearable
v-model.number="formData.latitude"
placeholder="纬度"
></el-input>
</el-form-item>
<el-button class="my-button" v-if="ifFalg" type="primary" @click="save" size="small">保存</el-button>
<el-button class="my-button" size="small" @click="close">关闭</el-button>
</el-form>
</div>
<div class="map-box">
<div class="map-tool">
<div v-if="ifFalg">
<el-checkbox >地图上描点</el-checkbox>
</div>
<div class="longlat">
<ul>
<li v-for="(item, index) in lnglatpoints" :key="index">
{{item.longitude}} , {{item.latitude}}
<i
v-if="ifFalg"
class="el-icon-close"
@click="deletes(item)"
></i>
</li>
</ul>
<br>
<div>
<span >输入范围经纬度:</span>
<el-input
type="textarea"
autosize
placeholder="请输入内容"
v-model="lnglatpointsString">
</el-input>
</div>
<el-button v-if="ifFalg" size="small" @click="clear1" type="primary" class="claer1">确定</el-button>
<el-button v-if="ifFalg" size="small" @click="clear" type="primary" class="claer">清除</el-button>
</div>
</div>
同上
<div class="my-tools">
<el-row>
<el-button type="primary" v-if="ifFalg" @click="drawPolygon()">鼠标绘制</el-button>
<el-button type="primary" v-if="ifFalg" @click="polygonEditor.close()">结束编辑</el-button>
</el-row>
</div>
</div>
</div>
</template>
<script lang="ts">
同上
//画多边形
private drawPolygon(){
let vm: any = this
if (vm.formData.location === \'\' || vm.formData.location === null) {
this.$message.warning(\'请先输入名称,才能开始画范围!\')
return
}
let map = new AMap.Map("map", {
center:[106.55073, 29.56471],
zoom: 15
});
// 多边形覆盖物节点坐标数组
let polygonArr:any = []
let lng = Number(this.formData.longitude)
let lat = Number(this.formData.latitude)
if(vm.path.length > 0){
polygonArr = vm.path
}else{
polygonArr.push([lng, lat])
polygonArr.push([lng, lat - 0.001])
polygonArr.push([lng - 0.001, lat - 0.001])
}
//使用 AMap.Polygon构建多边形
let polygon = new AMap.Polygon({
path:polygonArr,
strokeColor: "#FF33FF",
strokeWeight: 6,
strokeOpacity: 0.2,
fillOpacity: 0.4,
fillColor: \'#1791fc\',
zIndex: 50,
})
//将多边形增加到地图上
map.add(polygon)
// 缩放地图到合适的视野级别
map.setFitView([ polygon ])
//构造折线编辑对象,并开启折线的编辑状态
map.plugin(["AMap.PolyEditor"],function(){
let polygonEditor = new AMap.PolyEditor(map,polygon);
vm.polygonEditor =polygonEditor
polygonEditor.open();
//关闭多边形编辑polygonEditor.close()触发该方法;
polygonEditor.on(\'end\', function(event:any) {
// event.target 即为编辑后的多边形对象,event.target.getPath()得到编辑完成后的点数组
let pointArr = event.target.getPath()
vm.lnglatpoints = []
pointArr.forEach((point:any)=>{
vm.lnglatpoints.push({latitude: point.lat,longitude: point.lng})
})
vm.path = []
vm.lnglatpoints.map((val: any, index: number) => {
let arr = [val.longitude, val.latitude]
vm.path.push(arr)
if (index === 0) {
vm.map.center = [val.longitude, val.latitude]
}
})
})
});
}
/**
* 地理编码(地址 -> 坐标)
*/
private getLocation () {
let loc = this.formData.location
AMap.plugin(\'AMap.Geocoder\', () => {
let geocoder = new AMap.Geocoder()
geocoder.getLocation(loc, (status: string, result: any) => {
if (status === \'complete\' && result.info === \'OK\') {
let { lat, lng } = result.geocodes[0].location
if (lat && lng) {
this.map.center = [lng, lat]
this.formData.longitude=lng
this.formData.latitude=lat
}
}
})
})
}
同上 } </script> <style lang="scss" scoped> 和上面一样 </style>
123
23
2022-09
23
2022-09
13
2022-09
13
2022-09
13
2022-09
13
2022-09