Angular实现双向折叠列表组件的示例代码


Posted in Javascript onNovember 21, 2017

最近在做一个双向折叠组件,如下图所示,页面是分为两组,左边页面是Summary Panel,主要是一组列表,右边页面是Detail Panel,展示左边列表中某一项的具体信息,我们把它记作“Middle State”。

Angular实现双向折叠列表组件的示例代码

我们还看到有“<”和“>”两组按钮,这就是我们要做的“双向折叠组件”。点击左边的“<”,Summary Panel折叠起来,Detail Panel铺满整个页面,我们把它记作“Left State”,如下图:

Angular实现双向折叠列表组件的示例代码

在Summary Panel折叠状态下,点击“>”,又回到“Middle State”。点击“>”,Detail Panel折叠起来,Summary Panel铺满整个页面,我们把它记作“Right State”,如下图:

Angular实现双向折叠列表组件的示例代码

我们通过以上的需求分析可知,

1.页面的总体布局是一个Summary的div,两个箭头buttons,一个Detail的div。

2.页面总共有三种state:“Middle”、“Left”、“Right”,有两个button:“<”和“>”,也就是两个button去控制三个state。

因此我们需要定义一个枚举来记录页面的三种状态(注意,定义枚举要用export导出,否则后面会出错)

export enum CollapseExpandState {
  Middle = 1,
  Left,
  Right
}

页面的结构如下,并且通过一个变量_collapseExpandState去控制“left”和“right”两个button,具体为“<”会在页面状态为“Middle”和“Right”的情况下出现,“>”会在页面状态为“Middle”和“Left”状态下出现,从需求图中即可得知:

<div id="container">
  <div id="summary"></div>
  <div id="buttons">
    <div id="left" *ngIf="_collapseExpandState === CollapseExpandState.Middle || CollapseExpandState.Right" (click)="_onHandleLeft($event)">《</div>
    <div id="right"> *ngIf="_collapseExpandState === CollapseExpandState.Middle || CollapseExpandState.Left" (click)="_onHandleLeft($event)">》</div>
  </div>
  <div id="detail"></div>
</div>

这里在angular的template中用到了枚举,遭遇了一些麻烦,如果我们按上述定义了枚举,并且在Angular Component的template中用了枚举,我们会得到以下的错误提示:

TypeError: Cannot read property 'Middle' of undefined

也就是说,在Angular2的template中无法识别定义的枚举类型CollapseExpandState,这是因为你写的Angular Component的template模板的执行环境是你定义的component class,但是在class中并没有关于CollapseExpandState枚举的任何引用,所以Angular在为你的component生成模板的时候认为CollapseExpandState是undefined的。知道了原因,解决方案就很容易了,只需要在component class中加入这个枚举的引用即可:

@component(...)
export class ContainerWidget {
  public CollpaseExpandState: any = CollapseExpandState;
}

我们通过枚举状态来控制了两个buttons是否在恰当的页面状态显示与否,但是即使是同一个buttons,在不同的页面状态下所用到的样式也会不同,这里的样式其实最主要的就是位置了。我们先来考虑如何去控制button的样式,再来考虑如何去正确定位不同页面状态下button的位置。

对于控制button的样式,我们需要控制三个样式:"left button"、"right button"还有“buttons”。能够想到有三种方案:

I、用ng-class

ng-class一般的用法如下:

<some-element [ngClass]="{'first': true, 'second': true, 'third': false}">...</some-element>

因此它需要用boolean去控制,每一种样式需要一个boolean去控制,left和right各有两种样式,buttons有三种样式,这样就需要用5个boolean去控制,略显麻烦。

II、 用ElementRef.nativeElement.className

分别在buttons、left和right上用模板变量,然后在class中定义:

@ViewChild("buttons") buttons: ElementRef;
@ViewChild("left") left: ElementRef;
@ViewChild("right") right: ElementRef;

在处理函数中这样去给class赋值从而改变样式:

this.left.nativeElement.className = "XXXXX";

这样的话,我们需要从元素的角度出发,只需要3个元素变量,从而改变元素上的className即可。但是这样做有个隐患,注意到我们是用的ngIf来控制left和right在不同状态下是否存在,因为每一次事件处理都需要对三个元素的样式进行赋值,但是如果某个页面状态下ngIf为false从而元素不存在,那么就会报“Null Pointer”的错误,所以如果所引用的元素受到了ngIf的控制,不确定是否一定存在的情况下,要慎用该方法为元素赋予样式。

III、 用class="{{}}"

为了II中的尴尬,我们采用在HTML元素上对class进行直接赋值的方式,但是需要借用插值表达式{{}}。我们在css中用class的形式定义好样式,并且在compoennt class中定义三个字符串变量记录className,然后在事件处理函数中把相应的className赋予变量即可。这样我们就不用担心元素是否存在而导致的空指针了。

考虑完如何控制样式,下面我们进入CSS样式的讨论,这里其实主要考虑的就是位置。

我们采用flex布局,从左到右依次排列Summary Panel, buttons和Detail Panel。我们希望Buttons向左移,但是空出的位置被Detail Panel来填充。首先来看一下不设样式的效果图:

Angular实现双向折叠列表组件的示例代码

显然buttons是占据了文档流的位置的,如果这时候我们用relative定位buttpms,并且设置left的值为-74px(注意到left为负数就会把元素往左推):

.buttons{
  display: flex;
  position: relative;
  margin-top: 23px;
  left: -74px;
}

效果图为:

Angular实现双向折叠列表组件的示例代码

发现如果用left的话,buttons原来的文档流位置依然存在,只是buttons相对于原来的位置移动了一定的位移。

如果我们用margin-left来设置呢:

.buttons{
  display: flex;
  position: relative;
  margin-top: 23px;
  margin-left: -74px;
}

效果图为:

Angular实现双向折叠列表组件的示例代码

它和left不同之处在于,left会留住原来的文档流位置,但是用margin-left原来的文档流位置会消失,而由后面的元素补充过来,而我们想要的效果,正好是用margin-left来实现的。

所以用CSS定位的时候,要明白left和margin-left的区别,从而选择正确的方式来定位。

总结一下,从这个案例中我们学习到了:

  1. 双向折叠可以用“3种页面状态去控制2个按钮”来实现
  2. enum在Angular Component的template中用到时,需要在compnent class中添加它的引用
  3. 控制元素样式有很多方法,要选择合适的方法
  4. CSS定位中left和margin-left虽然都能把元素推向左边,但是left保留原来文档流位置,margin-left不保留原来文档流位置。

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

Javascript 相关文章推荐
jquery ajax 局部无刷新更新数据的实现案例
Feb 08 Javascript
前端学习笔记style,currentStyle,getComputedStyle的用法与区别
May 28 Javascript
React组件的三种写法总结
Jan 12 Javascript
javascript将url解析为json格式的两种方法
Aug 18 Javascript
基于ES6作用域和解构赋值详解
Nov 03 Javascript
AngularJS实现的锚点楼层跳转功能示例
Jan 02 Javascript
Webpack 之 babel-loader文件预处理器详解
Mar 23 Javascript
快速解决angularJS中用post方法时后台拿不到值的问题
Aug 14 Javascript
使用jQuery mobile NuGet让你的网站在移动设备上同样精彩
Jun 18 jQuery
vue实现滑动超出指定距离回顶部功能
Jul 31 Javascript
js实现GIF动图分解成多帧图片上传
Oct 24 Javascript
javaScript代码飘红报错看不懂?读完这篇文章再试试
Aug 19 Javascript
bootstrap表格内容过长时用省略号表示的解决方法
Nov 21 #Javascript
bootstrap treeview 扩展addNode方法动态添加子节点的方法
Nov 21 #Javascript
Angular4自制一个市县二级联动组件示例
Nov 21 #Javascript
微信小程序实现跟随菜单效果和循环嵌套加载数据
Nov 21 #Javascript
基于bootstrap写的一点localStorage本地储存
Nov 21 #Javascript
微信小程序实现下拉刷新和轮播图效果
Nov 21 #Javascript
基于 Vue.js 之 iView UI 框架非工程化实践记录(推荐)
Nov 21 #Javascript
You might like
用PHP连mysql和oracle数据库性能比较
2006/10/09 PHP
一个查看session内容的函数
2006/10/09 PHP
php 数学运算验证码实现代码
2009/10/11 PHP
如何批量清理系统临时文件(语言:C#、 C/C++、 php 、python 、java )
2016/02/01 PHP
php项目中类的自动加载实例讲解
2019/09/12 PHP
用js计算页面执行时间的函数
2006/12/07 Javascript
符合标准的js表单提交的代码
2007/09/13 Javascript
javascript 密码强度验证规则、打分、验证(给出前端代码,后端代码可根据强度规则翻译)
2010/05/18 Javascript
js arguments对象应用介绍
2012/11/28 Javascript
让checkbox不选中即将选中的checkbox不选中
2014/07/11 Javascript
浅谈JS日期(Date)处理函数
2014/12/07 Javascript
详解jQuery中的元素的属性和相关操作
2015/08/14 Javascript
jQuery实现鼠标悬停背景翻转的黑色导航菜单代码
2015/09/14 Javascript
jQuery和CSS仿京东仿淘宝列表导航菜单
2017/01/04 Javascript
ES6新特性之Symbol类型用法分析
2017/03/31 Javascript
JavaScript设计模式之单例模式详解
2017/06/09 Javascript
[07:39]第一届亚洲邀请赛回顾视频
2017/02/14 DOTA
Python排序搜索基本算法之希尔排序实例分析
2017/12/09 Python
Python贪心算法实例小结
2018/04/22 Python
python通过伪装头部数据抵抗反爬虫的实例
2018/05/07 Python
selenium获取当前页面的url、源码、title的方法
2019/06/12 Python
Django在admin后台集成TinyMCE富文本编辑器的例子
2019/08/09 Python
python实现的多任务版udp聊天器功能案例
2019/11/13 Python
Python爬取YY评级分数并保存数据实现过程解析
2020/06/01 Python
HTML5 Canvas的事件处理介绍
2015/04/24 HTML / CSS
墨西哥运动服饰和鞋网上商店:Netshoes墨西哥
2016/07/28 全球购物
澳大利亚Rockwear官网:女子瑜伽、健身和运动服
2021/01/26 全球购物
求职自荐书范文
2013/12/04 职场文书
《陈涉世家》教学反思
2014/04/12 职场文书
土地租赁协议书
2015/01/29 职场文书
单位考核聘任报告
2015/03/02 职场文书
导游词之嵊泗列岛
2019/10/30 职场文书
python实现三次密码验证的示例
2021/04/29 Python
SpringBoot 整合mongoDB并自定义连接池的示例代码
2022/02/28 MongoDB
SpringBoot2零基础到精通之数据库专项精讲
2022/03/22 Java/Android
SQLyog的下载、安装、破解、配置教程(MySQL可视化工具安装)
2022/09/23 MySQL