行政区划
This commit is contained in:
parent
14c63a176f
commit
0391204510
287
src/views/data/city/edit.vue
Normal file
287
src/views/data/city/edit.vue
Normal file
@ -0,0 +1,287 @@
|
|||||||
|
<template>
|
||||||
|
<basicModal
|
||||||
|
@register="modalRegister"
|
||||||
|
ref="modalRef"
|
||||||
|
class="basicModal basicFormModal"
|
||||||
|
@on-ok="handleSubmit"
|
||||||
|
@on-close="handleClose"
|
||||||
|
>
|
||||||
|
<template #default>
|
||||||
|
<n-form ref="formRef" :model="formData" label-width="85px" label-placement="left">
|
||||||
|
<div class="flex">
|
||||||
|
<n-form-item
|
||||||
|
label="上级城市"
|
||||||
|
path="pid"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ type: 'number', required: true, message: '请选择上级城市', trigger: 'change' }"
|
||||||
|
>
|
||||||
|
<n-tree-select
|
||||||
|
v-model:value="formData.pid"
|
||||||
|
:options="cityOptions"
|
||||||
|
:on-load="loadTree"
|
||||||
|
@update:value="handleNodeClick"
|
||||||
|
label-field="name"
|
||||||
|
key-field="id"
|
||||||
|
children-field="children"
|
||||||
|
:disabled="parentData?.areaCode || cityId ? true : false"
|
||||||
|
placeholder="上级城市"
|
||||||
|
/>
|
||||||
|
</n-form-item>
|
||||||
|
<n-form-item label="城市级别" path="level" class="flex-1">
|
||||||
|
<n-select
|
||||||
|
v-model:value="formData.level"
|
||||||
|
clearable
|
||||||
|
placeholder="请选择城市级别"
|
||||||
|
:options="optionData.cityTypeList"
|
||||||
|
label-field="name"
|
||||||
|
value-field="id"
|
||||||
|
/>
|
||||||
|
</n-form-item>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<n-form-item
|
||||||
|
label="城市名称"
|
||||||
|
path="name"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ required: true, message: '请输入城市名称', trigger: 'blur' }"
|
||||||
|
>
|
||||||
|
<n-input v-model:value="formData.name" placeholder="请输入城市名称" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
<n-form-item
|
||||||
|
label="城市简称"
|
||||||
|
path="shortName"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ required: true, message: '城市简称称', trigger: 'blur' }"
|
||||||
|
>
|
||||||
|
<n-input v-model:value="formData.shortName" placeholder="请输入城市简称称" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<n-form-item
|
||||||
|
label="城市区号"
|
||||||
|
path="cityCode"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ required: true, message: '城市区号', trigger: 'blur' }"
|
||||||
|
>
|
||||||
|
<n-input v-model:value="formData.cityCode" placeholder="请输入城市区号" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
<n-form-item
|
||||||
|
label="行政编码"
|
||||||
|
path="areaCode"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ required: true, message: '行政编码', trigger: 'blur' }"
|
||||||
|
>
|
||||||
|
<n-input v-model:value="formData.areaCode" placeholder="请输入行政编码" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<n-form-item
|
||||||
|
label="城市经度"
|
||||||
|
path="lng"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ required: true, message: '城市经度', trigger: 'blur' }"
|
||||||
|
>
|
||||||
|
<n-input v-model:value="formData.lng" placeholder="请输入城市经度" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
<n-form-item
|
||||||
|
label="城市纬度"
|
||||||
|
path="lat"
|
||||||
|
class="flex-1"
|
||||||
|
:rule="{ required: true, message: '城市纬度', trigger: 'blur' }"
|
||||||
|
>
|
||||||
|
<n-input v-model:value="formData.lat" placeholder="请输入城市纬度" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
</div>
|
||||||
|
<div class="flex">
|
||||||
|
<n-form-item label="邮政编码" path="zipCode" class="flex-1">
|
||||||
|
<n-input v-model:value="formData.zipCode" placeholder="请输入邮政编码" clearable />
|
||||||
|
</n-form-item>
|
||||||
|
</div>
|
||||||
|
</n-form>
|
||||||
|
</template>
|
||||||
|
</basicModal>
|
||||||
|
</template>
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { cityAdd, cityUpdate, getCityDetail } from '@/api/data/city';
|
||||||
|
import { onMounted, reactive, ref, shallowRef, nextTick } from 'vue';
|
||||||
|
import { getCityByList } from '@/api/system/user';
|
||||||
|
import { useMessage, useDialog } from 'naive-ui';
|
||||||
|
import { useModal } from '@/components/Modal';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 定义参数变量
|
||||||
|
*/
|
||||||
|
const emit = defineEmits(['success', 'update:visible']);
|
||||||
|
const formRef = ref();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 定义接收的参数
|
||||||
|
*/
|
||||||
|
const props = defineProps({
|
||||||
|
visible: {
|
||||||
|
type: Boolean,
|
||||||
|
required: true,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
cityId: {
|
||||||
|
type: Number,
|
||||||
|
required: true,
|
||||||
|
default: 0,
|
||||||
|
},
|
||||||
|
parentData: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {},
|
||||||
|
},
|
||||||
|
itemData: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 定义选项数据
|
||||||
|
*/
|
||||||
|
const optionData = reactive({
|
||||||
|
cityTypeList: [
|
||||||
|
{
|
||||||
|
id: 0,
|
||||||
|
name: '省份',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 1,
|
||||||
|
name: '城市',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 2,
|
||||||
|
name: '县区',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 3,
|
||||||
|
name: '街道',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 4,
|
||||||
|
name: '居委会',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 定义表单参数
|
||||||
|
*/
|
||||||
|
const formData = reactive({
|
||||||
|
id: '',
|
||||||
|
pid: 0,
|
||||||
|
level: '',
|
||||||
|
name: '',
|
||||||
|
cityCode: '',
|
||||||
|
areaCode: '',
|
||||||
|
parentCode: '',
|
||||||
|
zipCode: '',
|
||||||
|
shortName: '',
|
||||||
|
lng: '',
|
||||||
|
lat: '',
|
||||||
|
});
|
||||||
|
const [modalRegister, { openModal, setSubLoading }] = useModal({
|
||||||
|
title: props.cityId ? '编辑' : '添加',
|
||||||
|
subBtuText: '确定',
|
||||||
|
width: 700,
|
||||||
|
});
|
||||||
|
const message = useMessage();
|
||||||
|
const parentCode = ref({});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 关闭窗体
|
||||||
|
*/
|
||||||
|
const handleClose = () => {
|
||||||
|
formData.pid = 0;
|
||||||
|
emit('update:visible', false);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 加载树结构
|
||||||
|
*/
|
||||||
|
const cityOptions = ref([{ id: 0, name: '根目录', areaCode: 0, isLeaf: false }]);
|
||||||
|
const loadTree = async (treeNode) => {
|
||||||
|
console.log(treeNode);
|
||||||
|
return new Promise<void>(async (resolve) => {
|
||||||
|
const data = await getCityByList(treeNode.areaCode);
|
||||||
|
data.map((item) => {
|
||||||
|
item.isLeaf = item.level == 3 ? true : false;
|
||||||
|
});
|
||||||
|
treeNode.children = data;
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 执行节点点击事件
|
||||||
|
*/
|
||||||
|
const handleNodeClick = (data, node, extra) => {
|
||||||
|
parentCode.value = node;
|
||||||
|
formData.parentCode = node.areaCode;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 执行提交表单
|
||||||
|
*/
|
||||||
|
const handleSubmit = async () => {
|
||||||
|
formRef.value
|
||||||
|
.validate()
|
||||||
|
.then(async () => {
|
||||||
|
if (formData.pid == 0) {
|
||||||
|
formData.parentCode = '0';
|
||||||
|
}
|
||||||
|
props.cityId ? await cityUpdate(formData) : await cityAdd(formData);
|
||||||
|
message.success('操作成功');
|
||||||
|
setSubLoading(false);
|
||||||
|
emit('update:visible', false);
|
||||||
|
emit('success', parentCode.value);
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
setSubLoading(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置表单数据
|
||||||
|
* @param data 参数
|
||||||
|
*/
|
||||||
|
const setFormData = (data: Record<any, any>) => {
|
||||||
|
for (const key in formData) {
|
||||||
|
if (data[key] != null && data[key] != undefined) {
|
||||||
|
formData[key] = data[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
parentCode.value = { areaCode: formData.parentCode };
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取详情
|
||||||
|
*/
|
||||||
|
const getDetail = async () => {
|
||||||
|
const data = await getCityDetail(props.cityId);
|
||||||
|
setFormData(data);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 钩子函数
|
||||||
|
*/
|
||||||
|
onMounted(() => {
|
||||||
|
if (props.parentData && Object.keys(props.parentData).length !== 0) {
|
||||||
|
cityOptions.value[0].children = [];
|
||||||
|
cityOptions.value[0].children.push(props.parentData);
|
||||||
|
}
|
||||||
|
if (props.cityId) {
|
||||||
|
getDetail();
|
||||||
|
formData.pid = props.itemData.pid;
|
||||||
|
} else if (props.itemData) {
|
||||||
|
formData.pid = props.itemData.id;
|
||||||
|
formData.parentCode = props.itemData.areaCode;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
//导出方法
|
||||||
|
defineExpose({
|
||||||
|
openModal,
|
||||||
|
});
|
||||||
|
</script>
|
308
src/views/data/city/index.vue
Normal file
308
src/views/data/city/index.vue
Normal file
@ -0,0 +1,308 @@
|
|||||||
|
<template>
|
||||||
|
<div class="menu-index">
|
||||||
|
<n-card :bordered="false" class="pt-3 mb-3 proCard">
|
||||||
|
<div>
|
||||||
|
<n-button type="primary" @click="handleAdd()" v-perm="['sys:city:add']">
|
||||||
|
<template #icon>
|
||||||
|
<PlusOutlined />
|
||||||
|
</template>
|
||||||
|
新增
|
||||||
|
</n-button>
|
||||||
|
</div>
|
||||||
|
</n-card>
|
||||||
|
<n-card :bordered="false" class="pt-3 mb-3 proCard">
|
||||||
|
<n-spin :show="!showTable">
|
||||||
|
<n-data-table
|
||||||
|
:columns="columns"
|
||||||
|
@load="getChild"
|
||||||
|
:paginate-single-page="false"
|
||||||
|
:data="lists"
|
||||||
|
v-if="showTable"
|
||||||
|
:row-key="(row) => row.areaCode"
|
||||||
|
>
|
||||||
|
<template #description> 加载中 </template>
|
||||||
|
</n-data-table>
|
||||||
|
</n-spin>
|
||||||
|
</n-card>
|
||||||
|
<editDialog
|
||||||
|
ref="createModalRef"
|
||||||
|
v-if="editVisible"
|
||||||
|
:cityId="cityId"
|
||||||
|
:parentData="parentData"
|
||||||
|
:itemData="itemData"
|
||||||
|
v-model:visible="editVisible"
|
||||||
|
@success="refreshDataList"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup name="menu">
|
||||||
|
import { defineAsyncComponent, nextTick, onMounted, ref, reactive, h } from 'vue';
|
||||||
|
import { PlusOutlined, EditOutlined, DeleteOutlined, FormOutlined } from '@vicons/antd';
|
||||||
|
import { cityDelete } from '@/api/data/city';
|
||||||
|
import { getCityByList } from '@/api/system/user';
|
||||||
|
import editDialog from './edit.vue';
|
||||||
|
import { TableAction } from '@/components/Table';
|
||||||
|
import { renderIcon } from '@/utils';
|
||||||
|
import { useMessage, useDialog } from 'naive-ui';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 定义参数变量
|
||||||
|
*/
|
||||||
|
const columns = [
|
||||||
|
{
|
||||||
|
title: '城市名称',
|
||||||
|
key: 'name',
|
||||||
|
width: 100,
|
||||||
|
align: 'left',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '城市拼音',
|
||||||
|
key: 'pinyin',
|
||||||
|
width: 100,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '城市级别',
|
||||||
|
key: 'level',
|
||||||
|
width: 100,
|
||||||
|
customRender({ record }) {
|
||||||
|
let levelText = '';
|
||||||
|
switch (record.level) {
|
||||||
|
case 0:
|
||||||
|
levelText = '省份';
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
levelText = '城市';
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
levelText = '县区';
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
levelText = '街道';
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
levelText = '居委会';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return h('span', levelText || '-');
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '城市区号',
|
||||||
|
key: 'cityCode',
|
||||||
|
width: 100,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '行政编码',
|
||||||
|
key: 'areaCode',
|
||||||
|
width: 100,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '城市邮编',
|
||||||
|
key: 'zipCode',
|
||||||
|
width: 100,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: '操作',
|
||||||
|
fixed: 'right',
|
||||||
|
key: 'action',
|
||||||
|
width: 100,
|
||||||
|
render(record) {
|
||||||
|
return h(TableAction as any, {
|
||||||
|
style: 'button',
|
||||||
|
actions: [
|
||||||
|
{
|
||||||
|
label: '新增',
|
||||||
|
type: 'info',
|
||||||
|
icon: renderIcon(PlusOutlined),
|
||||||
|
auth: ['sys:city:add'],
|
||||||
|
onclick: handleAdd.bind(null, record),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '编辑',
|
||||||
|
type: 'warning',
|
||||||
|
icon: renderIcon(FormOutlined),
|
||||||
|
auth: ['sys:city:update'],
|
||||||
|
onclick: handleEdit.bind(null, record),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '删除',
|
||||||
|
type: 'error',
|
||||||
|
icon: renderIcon(DeleteOutlined),
|
||||||
|
auth: ['sys:city:delete'],
|
||||||
|
onclick: handleDelete.bind(null, record),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const message = useMessage();
|
||||||
|
const dialog = useDialog();
|
||||||
|
const fwbHeight = document.body.clientHeight - 350;
|
||||||
|
const loading = ref(false);
|
||||||
|
const createModalRef = ref();
|
||||||
|
const editVisible = ref(false);
|
||||||
|
const showTable = ref(true);
|
||||||
|
const cityId = ref(0);
|
||||||
|
const parentData = ref();
|
||||||
|
const itemData = ref({});
|
||||||
|
const lists = ref([]);
|
||||||
|
const expandKeys = ref([]);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取行政区划列表
|
||||||
|
* @param code 参数
|
||||||
|
*/
|
||||||
|
const loadDataTable = async (code: any) => {
|
||||||
|
let res = await getCityByList(0);
|
||||||
|
let data = res.length > 0 ? res : [];
|
||||||
|
data.map((item) => {
|
||||||
|
item.isLeaf = false;
|
||||||
|
});
|
||||||
|
lists.value = data;
|
||||||
|
};
|
||||||
|
/**
|
||||||
|
* 刷新数据列表
|
||||||
|
*/
|
||||||
|
const refreshDataList = async (code) => {
|
||||||
|
showTable.value = false;
|
||||||
|
let res = await getCityByList(0);
|
||||||
|
let data = res.length > 0 ? res : [];
|
||||||
|
data.map((item) => {
|
||||||
|
item.isLeaf = false;
|
||||||
|
});
|
||||||
|
lists.value = data;
|
||||||
|
|
||||||
|
showTable.value = true;
|
||||||
|
};
|
||||||
|
const actionColumn = reactive({
|
||||||
|
width: 220,
|
||||||
|
title: '操作',
|
||||||
|
align: 'center',
|
||||||
|
key: 'action',
|
||||||
|
fixed: 'right',
|
||||||
|
render(record) {
|
||||||
|
return h(TableAction as any, {
|
||||||
|
style: 'button',
|
||||||
|
actions: [
|
||||||
|
{
|
||||||
|
label: '新增',
|
||||||
|
type: 'info',
|
||||||
|
icon: renderIcon(PlusOutlined),
|
||||||
|
auth: ['sys:city:add'],
|
||||||
|
onclick: handleAdd.bind(null, record),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '编辑',
|
||||||
|
type: 'warning',
|
||||||
|
icon: renderIcon(FormOutlined),
|
||||||
|
auth: ['sys:city:update'],
|
||||||
|
onclick: handleEdit.bind(null, record),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '删除',
|
||||||
|
type: 'error',
|
||||||
|
icon: renderIcon(DeleteOutlined),
|
||||||
|
auth: ['sys:city:delete'],
|
||||||
|
onclick: handleDelete.bind(null, record),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 执行添加
|
||||||
|
* @param data 参数
|
||||||
|
*/
|
||||||
|
const handleAdd = async (data: any) => {
|
||||||
|
cityId.value = 0;
|
||||||
|
if (data) {
|
||||||
|
itemData.value = data;
|
||||||
|
let datas = findParentNode(lists.value, data.pid);
|
||||||
|
if (datas) {
|
||||||
|
parentData.value = datas;
|
||||||
|
} else {
|
||||||
|
parentData.value = data;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
parentData.value = {};
|
||||||
|
}
|
||||||
|
editVisible.value = true;
|
||||||
|
await nextTick();
|
||||||
|
createModalRef.value.openModal();
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 执行编辑
|
||||||
|
* @param data 参数
|
||||||
|
*/
|
||||||
|
const handleEdit = async (data: any) => {
|
||||||
|
cityId.value = data.id;
|
||||||
|
parentData.value = findParentNode(lists.value, data.pid);
|
||||||
|
editVisible.value = true;
|
||||||
|
await nextTick();
|
||||||
|
createModalRef.value.openModal();
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 执行删除
|
||||||
|
* @param row 参数
|
||||||
|
*/
|
||||||
|
const handleDelete = async (row: any) => {
|
||||||
|
dialog.warning({
|
||||||
|
title: '提示',
|
||||||
|
content: '确定要删除?',
|
||||||
|
positiveText: '确定',
|
||||||
|
negativeText: '取消',
|
||||||
|
onPositiveClick: async () => {
|
||||||
|
await cityDelete(row.id);
|
||||||
|
message.success('删除成功');
|
||||||
|
refreshDataList();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const getChild = async (row: Record<string, unknown>) => {
|
||||||
|
expandKeys.value.push(row.areaCode);
|
||||||
|
return new Promise<void>(async (resolve) => {
|
||||||
|
const data = await getCityByList(row.areaCode);
|
||||||
|
data.map((item) => {
|
||||||
|
item.isLeaf = item.level == 3 ? true : false;
|
||||||
|
});
|
||||||
|
row.children = data;
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据子节点找到父节点
|
||||||
|
* @param nodes 节点
|
||||||
|
* @param targetId 目标节点ID
|
||||||
|
*/
|
||||||
|
const findParentNode = (nodes, targetId) => {
|
||||||
|
for (let node of nodes) {
|
||||||
|
if (node.id == targetId) {
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
if (node.children && node.children.length > 0) {
|
||||||
|
const result = findParentNode(node.children, targetId);
|
||||||
|
if (result) {
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 钩子函数
|
||||||
|
*/
|
||||||
|
onMounted(() => {
|
||||||
|
loadDataTable();
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped></style>
|
Loading…
Reference in New Issue
Block a user