diff --git a/package.json b/package.json index 690db19..0baffd1 100644 --- a/package.json +++ b/package.json @@ -42,6 +42,7 @@ "clipboard": "2.0.8", "core-js": "3.25.3", "echarts": "5.4.0", + "element-china-area-data": "^6.1.0", "element-ui": "2.15.14", "file-saver": "2.0.5", "fuse.js": "6.4.3", diff --git a/src/api/ss/mchApply.js b/src/api/ss/mchApply.js new file mode 100644 index 0000000..a7cdbdd --- /dev/null +++ b/src/api/ss/mchApply.js @@ -0,0 +1,44 @@ +import request from '@/utils/request' + +// 查询商家合作申请列表 +export function listMchApply(query) { + return request({ + url: '/ss/mchApply/list', + method: 'get', + params: query + }) +} + +// 查询商家合作申请详细 +export function getMchApply(applyId) { + return request({ + url: '/ss/mchApply/' + applyId, + method: 'get' + }) +} + +// 新增商家合作申请 +export function addMchApply(data) { + return request({ + url: '/ss/mchApply', + method: 'post', + data: data + }) +} + +// 修改商家合作申请 +export function updateMchApply(data) { + return request({ + url: '/ss/mchApply', + method: 'put', + data: data + }) +} + +// 删除商家合作申请 +export function delMchApply(applyId) { + return request({ + url: '/ss/mchApply/' + applyId, + method: 'delete' + }) +} diff --git a/src/components/AreaTextSelect/index.vue b/src/components/AreaTextSelect/index.vue new file mode 100644 index 0000000..33c4f7f --- /dev/null +++ b/src/components/AreaTextSelect/index.vue @@ -0,0 +1,107 @@ + + + + + diff --git a/src/components/FormCol/index.vue b/src/components/FormCol/index.vue new file mode 100644 index 0000000..9cac131 --- /dev/null +++ b/src/components/FormCol/index.vue @@ -0,0 +1,27 @@ + + + diff --git a/src/components/Map/PlaceSearch/PlaceSearchDialog.vue b/src/components/Map/PlaceSearch/PlaceSearchDialog.vue index f456b87..25dbc57 100644 --- a/src/components/Map/PlaceSearch/PlaceSearchDialog.vue +++ b/src/components/Map/PlaceSearch/PlaceSearchDialog.vue @@ -1,9 +1,22 @@ @@ -23,16 +36,44 @@ export default { type: Boolean, default: false, }, + initLng: { + type: Number, + default: 120.250452 + }, + initLat: { + type: Number, + default: 27.101745 + }, }, data() { return { - keyword: null, selected: null, } }, methods: { - onSelectChange(data) { - console.log(data); + onMapGeo(data, lat, lng) { + console.log("onMapGeo", data) + let component = data.regeocode.addressComponent; + this.selected = { + address: data.regeocode.formattedAddress, + lng: lat, + lat: lng, + province: component.province, + city: component.city === '' ? '市辖区' : component.city, + county: component.district, + } + }, + onSelectChange(addr) { + console.log("onSelectChange", addr) + let data = addr.selected.data; + this.selected = { + address: data.pname + (data.cityname === data.pname ? '' : data.cityname) + data.adname + data.address + data.name, + lng: data.location.lng, + lat: data.location.lat, + province: data.pname, + city: data.cityname === data.pname ? '市辖区' : data.cityname, + county: data.adname, + } }, // 确定 onSubmit() { @@ -45,6 +86,7 @@ export default { }, // 关闭弹窗 close() { + this.selected = null; this.$emit('update:show', false); }, } diff --git a/src/components/Map/PlaceSearch/PlaceSearchMap.vue b/src/components/Map/PlaceSearch/PlaceSearchMap.vue index 67a4972..34e3817 100644 --- a/src/components/Map/PlaceSearch/PlaceSearchMap.vue +++ b/src/components/Map/PlaceSearch/PlaceSearchMap.vue @@ -1,11 +1,22 @@ @@ -13,9 +24,11 @@ import AMapLoader from "@amap/amap-jsapi-loader"; import globalConfig from '@/utils/config/globalConfig' import { debounce } from '@/utils' +import AreaTextSelect from '@/components/AreaTextSelect/index.vue' export default { name: "PlaceSearchMap", + components: { AreaTextSelect }, props: { width: { type: String, @@ -25,6 +38,14 @@ export default { type: String, default: "100%" }, + initLng: { + type: Number, + default: 120.250452 + }, + initLat: { + type: Number, + default: 27.101745 + }, }, data() { return { @@ -33,27 +54,34 @@ export default { placeSearch: null, // POI查询 geocoder: null, // 逆地理编码 loading: false, - lng: null, - lat: null, keyword: null, - // 返回的结果 - result: { - lat: null, - lng: null, - address: null, - }, markers: [], + area: { + province: '福建省', + city: '宁德市', + } } }, mounted() { this.initAMap(); }, - unmounted() { - this.map?.destroy(); + beforeDestroy() { + this.destroyMap(); }, methods: { + destroyMap() { + this.map?.destroy(); + }, + onChangeArea() { + this.loadPlaceSearch(this.area.province, this.area.city); + if (this.keyword != null) { + this.doPlaceSearch(this.keyword); + } else { + let city = this.area.city === '市辖区' ? this.area.province : this.area.city; + this.doPlaceSearch(city); + } + }, initAMap() { - let _this = this; AMapLoader.load({ key: globalConfig.aMap.key, // 申请好的Web端开发者Key,首次调用 load 时必填 version: "2.0", // 指定要加载的 JSAPI 的版本,缺省时默认为 1.4.15 @@ -63,21 +91,44 @@ export default { this.map = new AMap.Map("container", { // 设置地图容器id viewMode: "3D", // 是否为3D地图模式 - zoom: 11, // 初始化地图级别 - center: [120.250452, 27.101745], // 初始化地图中心点位置 + zoom: 16, // 初始化地图级别 + center: [this.initLng == null ? 120.250452 : this.initLng, this.initLat == null ? 27.101745 : this.initLat], // 初始化地图中心点位置 }); this.map.on('click', this.onClickMap); + // POI - this.loadPlaceSearch(); - if (this.keyword != null) { - this.doPlaceSearch(this.keyword); - } + this.loadPlaceSearch(this.area.province, this.area.city); + // 逆地理编码 this.loadGeoCoder(); + + // 加载初始地址 + this.initMarker(); + }).catch((e) => { console.log(e); }); }, + async initMarker() { + this.removeAllMarker(); + if (this.initLng != null && this.initLat != null) { + this.getGeoAddress(this.initLng, this.initLat).then(res => { + // 标点 + this.removeAllMarker(); + this.addMarker(this.initLng, this.initLat, res.regeocode.formattedAddress); + this.$emit('map-geo', res, this.initLng, this.initLat); + + // 地区 + let component = res.regeocode.addressComponent; + this.area = { + province: component.province, + city: component.city === '' ? '市辖区' : component.city, + } + }).finally(() => { + this.loading = false; + }) + } + }, addMarker(lng, lat, title) { //创建一个 Marker 实例: let marker = new AMap.Marker({ @@ -88,6 +139,7 @@ export default { this.map.add(marker); this.markers.push(marker); }, + // 删除所有的标记点 removeAllMarker() { this.map.remove(this.markers); this.markers = []; @@ -95,22 +147,26 @@ export default { // 点击地图事件 onClickMap(e) { console.log('clickMap', e); - this.lng = e.lnglat.lng; - this.lat = e.lnglat.lat; - this.getGeoAddress(this.lng, this.lat).then(res => { + let lng = e.lnglat.lng; + let lat = e.lnglat.lat; + this.changeMarker(lng, lat); + }, + // 逆地理编码,并标点 + changeMarker(lng, lat) { + this.loading = true; + this.getGeoAddress(lng, lat).then(res => { this.removeAllMarker(); - this.result.address = res.regeocode.formattedAddress; - this.result.lng = this.lng; - this.result.lat = this.lat; - this.addMarker(this.lng, this.lat, this.result.address); + this.addMarker(lng, lat, res.regeocode.formattedAddress); + this.$emit('map-geo', res, lng, lat); + }).finally(() => { + this.loading = false; }) }, + // 获取逆地理编码 getGeoAddress(lng, lat) { return new Promise((resolve, reject) =>{ this.geocoder.getAddress([lng, lat], (status, result) => { - console.log("geoCode", status, result); if (status === 'complete' && result.info === 'OK') { - // result为对应的地理位置详细信息 resolve(result); } else { reject(status); @@ -127,14 +183,20 @@ export default { }) }, // 加载POI - loadPlaceSearch() { + loadPlaceSearch(province, city) { + if (city == null) { + city = '北京市'; + } + if (city === '市辖区') { + city = province; + } AMap.plugin(["AMap.PlaceSearch"], () => { //构造地点查询类 this.placeSearch = new AMap.PlaceSearch({ pageSize: 5, // 单页显示结果条数 pageIndex: 1, // 页码 - city: "010", // 兴趣点城市 - citylimit: false, //是否强制限制在设置的城市内搜索 + city: city, // 兴趣点城市 + citylimit: true, //是否强制限制在设置的城市内搜索 map: this.map, // 展现结果的地图实例 panel: "panel", // 结果列表将在此容器中进行展示。 autoFitView: true, // 是否自动调整地图视野使绘制的 Marker点都处于视口的可见范围 @@ -145,21 +207,33 @@ export default { this.$emit('select-changed', data); }) }, - // 周边查询 - doSearchNearBy(keyword, lng, lat) { - console.log('searchNearBy', keyword, lng, lat) - this.placeSearch.searchNearBy(keyword, [lng, lat], 200, (status, result) => { - console.log("POI", status, result); - }); - }, //关键字查询 doPlaceSearch(keyword) { + if (keyword == null) { + return this.$message.warning("请输入关键词"); + } this.loading = true; this.placeSearch.search(keyword, (status, result) => { this.loading = false; console.log("POI", status, result); }); }, + // 周边搜索 + doSearchNearBy(keyword, lng, lat, radius = 200) { + if (keyword == null) { + return this.$message.warning("请输入关键词"); + } + this.loading = true; + this.placeSearch.searchNearBy(keyword, [lng, lat], radius, (status, result) => { + this.loading = false; + console.log("POI NearBy", status, result); + }) + }, + // 根据地图中心点查询周边 + doSearchNearByCenter(keyword) { + let center = this.map.getCenter(); + this.doSearchNearBy(keyword, center.lng, center.lat, 1000); + }, // 监听输入事件 onInputSearch: debounce(function () { this.doPlaceSearch(this.keyword) @@ -167,19 +241,29 @@ export default { }, }; - diff --git a/src/main.js b/src/main.js index 51a9083..cca4da1 100644 --- a/src/main.js +++ b/src/main.js @@ -37,6 +37,8 @@ import DictTag from '@/components/DictTag' import VueMeta from 'vue-meta' // 字典数据组件 import DictData from '@/components/DictData' +// 行内表单组件 +import FormCol from '@/components/FormCol/index.vue' // 过滤器 import filter from "@/utils/filter"; filter(Vue); @@ -61,6 +63,7 @@ Vue.component('Editor', Editor) Vue.component('FileUpload', FileUpload) Vue.component('ImageUpload', ImageUpload) Vue.component('ImagePreview', ImagePreview) +Vue.component('FormCol', FormCol) Vue.use(directive) Vue.use(plugins) diff --git a/src/utils/config/globalConfig.js b/src/utils/config/globalConfig.js index 9f6554a..aed8478 100644 --- a/src/utils/config/globalConfig.js +++ b/src/utils/config/globalConfig.js @@ -2,12 +2,12 @@ export default { /** * 高德地图配置 */ - // aMap: { - // key: '11da89fddf9340d0a69d4fff53c0ec4b', - // secret: '32dca5ef246f3b96234cd8ef891e4d59' - // } aMap: { - key: "8cd2f6263fa49537ab9e9fbec328f1c4", - secret: "4bb470e4c380afc28218fa36cdd43ab0", + key: '11da89fddf9340d0a69d4fff53c0ec4b', + secret: '32dca5ef246f3b96234cd8ef891e4d59' } + // aMap: { + // key: "8cd2f6263fa49537ab9e9fbec328f1c4", + // secret: "4bb470e4c380afc28218fa36cdd43ab0", + // } } diff --git a/src/views/ss/mchApply/index.vue b/src/views/ss/mchApply/index.vue new file mode 100644 index 0000000..9122ef4 --- /dev/null +++ b/src/views/ss/mchApply/index.vue @@ -0,0 +1,319 @@ + + + diff --git a/src/views/ss/store/index.vue b/src/views/ss/store/index.vue index 9405723..03c72f3 100644 --- a/src/views/ss/store/index.vue +++ b/src/views/ss/store/index.vue @@ -17,14 +17,6 @@ @keyup.enter.native="handleQuery" /> - - - 搜索 重置 @@ -79,14 +71,24 @@ - - + + + + + + + + @@ -118,20 +120,49 @@ /> - + - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + - + @@ -148,10 +184,13 @@ import { listStore, getStore, delStore, addStore, updateStore } from "@/api/ss/s import SmUserSelect from '@/components/Business/SmUser/smUserSelect.vue' import UserInput from '@/components/Business/SmUser/UserInput.vue' import PlaceSearchDialog from '@/components/Map/PlaceSearch/PlaceSearchDialog.vue' +import AreaTextSelect from '@/components/AreaTextSelect/index.vue' +import { parseTime } from '../../../utils/ruoyi' export default { name: "Store", - components: { PlaceSearchDialog, UserInput, SmUserSelect }, + dicts: ['ss_store_type'], + components: { AreaTextSelect, PlaceSearchDialog, UserInput, SmUserSelect }, data() { return { // 遮罩层 @@ -190,19 +229,49 @@ export default { { required: true, message: "用户不能为空", trigger: "change" } ], name: [ - { required: true, message: "店铺名称不能为空", trigger: "blur" } + { required: true, message: "店铺名称不能为空", trigger: "change" } ], address: [ - { required: true, message: "门店地址不能为空", trigger: "blur" } + { required: true, message: "详细地址不能为空", trigger: "change" } + ], + type: [ + { required: true, message: "详细地址不能为空", trigger: "change" } + ], + businessTimeStart: [ + { required: true, message: "营业时间不允许为空", trigger: "change" } ], }, showPlaceSearchMap: false, + span: 12, }; }, + computed: { + formatBusinessTime: { + set(val) { + this.form.businessTimeStart = val[0]; + this.form.businessTimeEnd = val[1]; + }, + get() { + if (this.form.businessTimeStart == null || this.form.businessTimeEnd == null) { + return null; + } + return [this.form.businessTimeStart, this.form.businessTimeEnd] + } + } + }, created() { this.getList(); }, methods: { + parseTime, + onSubmitAddress(addr) { + this.form.address = addr.address; + this.form.lat = addr.lat; + this.form.lng = addr.lng; + this.form.province = addr.province; + this.form.city = addr.city; + this.form.county = addr.county; + }, /** 查询商户列表列表 */ getList() { this.loading = true; @@ -225,7 +294,13 @@ export default { picture: null, name: null, address: null, - lon: null, + specificAddress: null, + businessTimeStart: "08:00", + businessTimeEnd: "18:00", + province: null, + city: null, + county: null, + lng: null, lat: null, createTime: null, createBy: null, diff --git a/src/views/system/device/index.vue b/src/views/system/device/index.vue index 834f9cb..ba3e311 100644 --- a/src/views/system/device/index.vue +++ b/src/views/system/device/index.vue @@ -1,10 +1,10 @@