要啥插件!vue3手写省市区街道四级联动!

1. 效果

是骡子是马先拉出来溜溜,我们先看效果:

2.实现思路

1. 获取四级联动的数据文件

首先这个数据必须是经过国家统计局、民政部认证的,不能瞎编乱造,然后数据格式一般是 json 格式。

中国省市区街道json文件下载地址:

bash 复制代码
链接:https://pan.baidu.com/s/1giYramKo5c6R1gIgOouScw?pwd=6cjp 
提取码:6cjp 

2. 数据联动

我们发现这个地址数据格式的特点:只有最里层街道数据是数组,然后外三层都是对象。

那我们首先想到 Object.keys() 方法可以获取所有对象属性的键名(key),也就是最外层的省份数据。

css 复制代码
// 省列表
const provinceList = Object.keys(areaData);

接着我们通过 computed 计算属性进行数据的联动

拿到省份数据之后,我们选择某一个省份,我们就可以通过 areaData[选择的省份] 获取这个省份对应的属性值,然后再用 Object.keys(areaData[选择的省份]) 获取该省份下面所有的市数据。

css 复制代码
import areaData from "../../../assets/json/area.json";

// 市列表
const cityList = computed(() => {
  return area.province ? Object.keys(areaData[area.province]) : [];
});

Object.keys(areaData[选择的省份][选择的市])就是该市下面所有的县区数据

css 复制代码
// 县区列表
const districtList = computed(() => {
  return area.province && area.city
    ? Object.keys(areaData[area.province][area.city])
    : [];
});

areaData[选择的省份][选择的市][选择的县区]就是该县区下面所有的街道数据

css 复制代码
// 街道列表
const streetList = computed(() => {
  return area.province && area.city && area.district
    ? areaData[area.province][area.city][area.district]
    : [];
});

3. 清除关联数据

  • 选择另外一个省份,清除之前的市区街道数据
  • 选择另外的市,清除县区街道数据
  • 选择另外的县区,清除街道数据

这里,我们通过 watch 来监听省份、市、县区的变化

css 复制代码
// 监听省份变化
watch(
  () => area.province,
  (newVal) => {
    area.city = "";
    area.district = "";
    area.street = "";
  }
);
// 监听市区变化
watch(
  () => area.city,
  (newVal) => {
    area.district = "";
    area.street = "";
  }
);
// 监听县区变化
watch(
  () => area.district,
  (newVal) => {
    area.street = "";
  }
);

3.完整代码

css 复制代码
<template>
  <div>
    <el-card class="box-card" style="width: 50%; margin: 20px">
      <template #header>
        <div class="card-header">
          <span>中国省市区街道四级联动</span>
        </div>
      </template>
      <el-form
        class="area"
        ref="areaForm"
        :model="area"
        :rules="rules"
        label-width="100px"
        size="large"
      >
        <el-form-item label="省:" prop="province">
          <el-select
            style="width: 100%"
            v-model="area.province"
            filterable
            clearable
            placeholder="请选择省份"
          >
            <el-option
              v-for="item in provinceList"
              :key="item"
              :label="item"
              :value="item"
            />
          </el-select>
        </el-form-item>
        <el-form-item label="市:" prop="city">
          <el-select
            style="width: 100%"
            v-model="area.city"
            filterable
            clearable
            placeholder="请选择市"
          >
            <el-option v-for="item in cityList" :key="item" :label="item" :value="item" />
          </el-select>
        </el-form-item>
        <el-form-item label="县区:" prop="district">
          <el-select
            style="width: 100%"
            v-model="area.district"
            filterable
            clearable
            placeholder="请选择县区"
          >
            <el-option
              v-for="item in districtList"
              :key="item"
              :label="item"
              :value="item"
            />
          </el-select>
        </el-form-item>
        <el-form-item label="街道:" prop="street">
          <el-select
            style="width: 100%"
            v-model="area.street"
            filterable
            clearable
            placeholder="请选择街道"
          >
            <el-option
              v-for="item in streetList"
              :key="item"
              :label="item"
              :value="item"
            />
          </el-select>
        </el-form-item>
        <el-form-item>
          <el-button size="mini" type="primary" @click="onSubmit">提交</el-button>
          <el-button size="mini">公众号:知否技术</el-button>
        </el-form-item>
      </el-form>
    </el-card>
  </div>
</template>
<script setup>
import { reactive, computed, watch } from "vue";
import areaData from "./assets/json/area.json";
const area = reactive({
  province: "",
  city: "",
  district: "",
  street: "",
});
//------省市县区街道四级联动
// 省列表
const provinceList = Object.keys(areaData);
// 市列表
const cityList = computed(() => {
  return area.province ? Object.keys(areaData[area.province]) : [];
});
// 县区列表
const districtList = computed(() => {
  return area.province && area.city
    ? Object.keys(areaData[area.province][area.city])
    : [];
});
// 街道列表
const streetList = computed(() => {
  return area.province && area.city && area.district
    ? areaData[area.province][area.city][area.district]
    : [];
});
// 监听省份变化
watch(
  () => area.province,
  (newVal) => {
    area.city = "";
    area.district = "";
    area.street = "";
  }
);
// 监听市区变化
watch(
  () => area.city,
  (newVal) => {
    area.district = "";
    area.street = "";
  }
);
// 监听县区变化
watch(
  () => area.district,
  (newVal) => {
    area.street = "";
  }
);
</script>
相关推荐
a程序小傲几秒前
京东Java面试被问:基于Gossip协议的最终一致性实现和收敛时间
java·开发语言·前端·数据库·python·面试·状态模式
小二·3 分钟前
Python Web 开发进阶实战:AI 原生应用商店 —— 在 Flask + Vue 中构建模型即服务(MaaS)与智能体分发平台
前端·人工智能·python
Devlive 开源社区11 分钟前
技术日报|推理RAG文档索引PageIndex登顶日增1374星,React视频工具Remotion二连冠进前二
前端·react.js·前端框架
xkxnq12 分钟前
第三阶段:Vue 路由与状态管理(第 45 天)(路由与状态管理实战:开发一个带登录权限的单页应用)
前端·javascript·vue.js
Irene199118 分钟前
Vue 3 中的具名插槽仍然完全支持,Vue 2 的旧语法 Vue 3 中已废弃
vue.js·slot
方方洛27 分钟前
技术实践总结:schema-bridgion:json、xml、yaml、toml文件相互转换
xml·前端·typescript·node.js·json
object not found1 小时前
基于uniapp开发小程序自定义顶部导航栏状态栏标题栏
前端·javascript·小程序·uni-app
Irene19911 小时前
v-model 在 Vue2 和 Vue3 中的实现对比或异同
vue.js
We་ct1 小时前
LeetCode 28. 找出字符串中第一个匹配项的下标:两种实现与深度解析
前端·算法·leetcode·typescript
xzl041 小时前
小智服务端chat入口工具调用流程
java·服务器·前端