Angular4自制一个市县二级联动组件示例


Posted in Javascript onNovember 21, 2017

最近遇到了不少问题,真的是命运多舛。Angular真是让人又爱又恨的框架,恨的是资料太少,遇到问题无从下手。爱的是许多其他框架难以做到的功能,angular却可以轻松做到。

话不多说,最近遇到了一个旧项目改造的问题。拿到前同事做的页面效果:

Angular4自制一个市县二级联动组件示例

第一眼就看到了这三个下拉框,按捺不住好奇心的我点了点。原来,第一个下拉框可以选择市属和省属,如果选择市属,那么后面就会出现市、县级两个下拉框,如果是省属,那就隐藏了,这个挺容易的。然后就是要选择市之后,区下拉框要有对应区县选项。emmmm,很典型的二级联动,不过既然分析完了思路,那就开始做吧!首先呢,数据肯定要从后端同事那里拿,调用他的接口把数据填充进去。看看数据是什么样子的:

Angular4自制一个市县二级联动组件示例

数据略多,就不全部贴出来了。把实体bean创建一下,

// 市级实体类
export class City {

  // 市级id
  cityId: string;

  // 所属类型(0.市属 1.省属)
  cityType: number;

  // 市级名称(可选属性,若cityType为1时,可不填)
  cityName: string;

  // 所属区县
  counties?: Array<Country>;

}

// 区县级实体类
export class Country {

  // 区县id
  countryId: string;

  // 区县名称
  countryName: string;

}

// 填写市县类
export class CityAndCountry {

  // 市级id
  cityId: string;

  // 县级id
  countryId: string;

  // 市级类型
  cityType: number;

  // 市县级实体构造器 
  constructor() {
    // 给市级id赋予一个真实城市的id初始值
    this.cityId = '***';
    // 同上
    this.countryId = '***';
    // 同上
    this.cityType = 0;
  }
}

实体完成了,开始准备获取数据并填充至实体:

// 二级联动组件
export class CityAreaComponent implements OnInit, OnDestroy {

  // 结果码 (用于页面处理显示标识)
  result_code: number;

  // 市级实体声明
  city: City[];

  // 县区级实体声明
  country: Country[];

  // 市县、区级填写实体声明
  cac: CityAndCountry;

  // 声明订阅对象
  subscript: Subscription;

  /**
  * 构造器
  * @param {CityService} service 注入服务
  */
  constructor (private service: CityService) {
    // 结果码 (-1.网络或其他异常 0.无内容 1.请求成功 2.请等待)
    this.result_code = 2;
    // 初始化填写市区、县级填写实体
    cac = new CityAndCountry();
    // 初始化数组(这步很重要,有很多人说使用数组相关函数会报未定义异常,是因为没有初始化的原因)
    this.city = [];
    this.country = [];
    // 初始化订阅对象
    this.subscript = new Subscription();
  }

  /**
  * 生命周期初始化钩子(生命周期尽量按执行顺序来写,养成好习惯)
  */
  ngOnInit(): void {
    this.getCityArea();
  }

  /** 获取市县数据 */
  getCityArea() {
    /** 将请求交付服务处理(service代码比较简单,就不贴了) */
    this.subscript = this.service.getCityArea().subscribe(res => {
      /** 获取json请求结果 */
      const result = res.json();
      /** 判断结果返回码 */
      switch (result['code']) {
        /** 请求成功,并且有值 */
        case 200:
        /** 改变初始返回码 */
        this.result_code = 1;
        /** 获取并填充数据 */
        this.city = json['city'];
        break;
        /** 其他情况不重复赘述 */
      }
    }, err => {
      /** 显示预设异常信息提示给用户 */
      this.result_code = -1;
      /** 打印log,尽量使用日志产出 */
      console.error(err);
    });
  }

  /** 生命周期销毁钩子 */
  ngOnDestroy(): void {
    /** 取消订阅 */
    this.subscript.unsubscribe();
  }
}

由于此处是单服务请求,为了让代码比较清晰直观,这里我就不做封装处理了。数据获取了之后就该填充到展示界面了:

<!-- 所属类型(此处固定,一般为获取后端数据字典数据) -->
<select class="city_type" [value]="cac.cityType" [(ngModel)]="cac.cityType">
  <!-- 所传内容为整数型 -->
  <option value=0>市属</option>
  <option value=1>省属</option>
</select>

<!-- 市级选择(类型为省属时隐藏) -->
<select class="city" [value]="cac.cityId" [(ngModel)]="cac.cityId" *ngIf="city.cityType==0">
  <!-- 遍历城市数组 -->
  <option *ngFor="let opt of option" [value]="opt.cityId">{{opt.cityName}}</option>
</select>

这时候,我们发现县级获取起来好像并不能直接获取,怎么办呢?我突然想到,我在ts里面声明一个变量获取市级选择的id号,然后再拿id去找下属县区,这样就可以轻松拿到了。既然要实时获取变化,那我们就实现检测变化钩子:

// 二级联动组件
export class CityAreaComponent implements OnInit, OnDestroy, DoCheck{
  // 声明县区级数组
  country: Array<Country>;

  constructor() {
    /** 重复代码不赘述 */
    /** 初始化数组 */
    country = [];
  }  

  /** 生命周期检测变化钩子 */
  ngDoCheck(): void {
    /** 遍历市级数组 */
    for (let i = 0; i < this.city.length; i++) {
      /** 若选择的市级id和市级数组中的id相吻合 */
      if (this.city[i].id == this.cac.countryId) {
        /** 将该索引下的counties数组赋予给区县级数组 */
        this.country = this.city[i].counties;
      }
      /** 我们无法避免直辖市的情况,所以多一重判断 */
      if (this.country.length > 0) {
        /** 为了用户体验,我们要尽量在用户选择市级城市后,默认选择一个区县级城市 */
        this.cac.country.id = this.country[0].id;
      }
    }
  }
}

最后再补上区县级下拉框:

<!-- 区县级下拉框 -->
<select [value]="cac.countryId" [(ngModel)]="cac.countryId" *ngIf="cac.cityType==0 && country.length > 0">
  <option *ngFor="let count of country" [value]="count.id">{{count.name}}</option>
</select>

到此为止,大功告成,再也不用去依赖别人的库了。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
用js查找法实现当前栏目的高亮显示的代码
Nov 24 Javascript
javascript 同时在IE和FireFox获取KeyCode的代码
Feb 07 Javascript
教你使用javascript简单写一个页面模板引擎
May 05 Javascript
js实现文字垂直滚动和鼠标悬停效果
Dec 31 Javascript
jquery div模态窗口的简单实例
May 28 Javascript
jQuery+ajax实现实用的点赞插件代码
Jul 06 Javascript
小发现之浅谈location.search与location.hash的问题
Jun 23 Javascript
Vue项目webpack打包部署到服务器的实例详解
Jul 17 Javascript
JavaScript EventEmitter 背后的秘密 完整版
Mar 29 Javascript
浅析Vue 和微信小程序的区别、比较
Aug 03 Javascript
Vue中jsx不完全应用指南小结
Nov 01 Javascript
vue elementUI表格控制对应列
Apr 13 Vue.js
微信小程序实现跟随菜单效果和循环嵌套加载数据
Nov 21 #Javascript
基于bootstrap写的一点localStorage本地储存
Nov 21 #Javascript
微信小程序实现下拉刷新和轮播图效果
Nov 21 #Javascript
基于 Vue.js 之 iView UI 框架非工程化实践记录(推荐)
Nov 21 #Javascript
微信小程序实现tab和swiper切换结合效果
Jul 17 #Javascript
微信小程序实现tab切换效果
Nov 21 #Javascript
jQuery实现所有验证通过方可提交的表单验证
Nov 21 #jQuery
You might like
从手册去理解分析PHP session机制
2011/07/17 PHP
ThinkPHP字符串函数及常用函数汇总
2014/07/18 PHP
php实现概率性随机抽奖代码
2016/01/02 PHP
CI框架实现创建自定义类库的方法
2018/12/25 PHP
php使用mysqli和pdo扩展,测试对比连接mysql数据库的效率完整示例
2019/05/09 PHP
js获取IP地址的方法小结
2014/07/01 Javascript
jQuery插件datalist实现很好看的input下拉列表
2015/07/14 Javascript
JS+CSS实现下拉列表框美化效果(3款)
2015/08/15 Javascript
基于javascript实现九宫格大转盘效果
2020/05/28 Javascript
JS实现的简单轮播图运动效果示例
2016/12/22 Javascript
angular bootstrap timepicker TypeError提示怎么办
2017/06/13 Javascript
浅谈在node.js进入文件目录的问题
2018/05/13 Javascript
详解Angular6 热加载配置方案
2018/08/18 Javascript
vue+Element实现搜索关键字高亮功能
2019/05/28 Javascript
vue.js实现只能输入数字的输入框
2019/10/19 Javascript
es6中let和const的使用方法详解
2020/02/24 Javascript
JavaScript canvas基于数组生成柱状图代码实例
2020/03/06 Javascript
sharp.js安装过程中遇到的问题总结
2020/04/02 Javascript
vantUI 获得piker选中值的自定义ID操作
2020/11/04 Javascript
不要用强制方法杀掉python线程
2017/02/26 Python
Python3实现爬取指定百度贴吧页面并保存页面数据生成本地文档的方法
2018/04/22 Python
Pyqt清空某一个QTreeewidgetItem下的所有分支方法
2019/06/17 Python
在django中使用apscheduler 执行计划任务的实现方法
2020/02/11 Python
最小二乘法及其python实现详解
2020/02/24 Python
解决python运行启动报错问题
2020/06/01 Python
Pyinstaller加密打包应用的示例代码
2020/06/11 Python
详解Python中的路径问题
2020/09/02 Python
Python基于Socket实现简易多人聊天室的示例代码
2020/11/29 Python
上课睡觉检讨书
2014/01/28 职场文书
法学专业毕业生求职信
2014/06/12 职场文书
出售房屋委托书范本
2014/09/24 职场文书
2015年安全生产目标责任书
2015/01/29 职场文书
改进工作作风心得体会
2016/01/23 职场文书
2016年119消防宣传日活动总结
2016/04/05 职场文书
2019年妇科护士的自我鉴定(3篇)
2019/09/26 职场文书
SQL实战演练之网上商城数据库商品类别数据操作
2021/10/24 MySQL