javascript模拟select,jselect的方法实现


Posted in Javascript onNovember 08, 2012

由于主流浏览器对select元素渲染不同,所以在每种浏览器下显示也不一样,最主要的是默认情况下UI太粗糙,即使通过css加以美化也不能达到很美观的效果。这对于我们这些专注于UX的前端开发人员是无法容忍的。于是在项目不太忙的时候,就计划写一个模拟的select控件出来。接下来就把实现的细节、遇到的问题以及如何使用和大家分享一下。
1. 实现细节
init: function(context) {
//获取指定上下文所有select元素
var elems = squid.getElementsByTagName('select', context)
this.globalEvent()
this.initView(elems)
}
在一个用户注册的应用场景,有多个select元素。模拟的select控件(以下简称jselect)初始化方法会获取页面上所有select元素,然后绑定全局事件globalEvent,初始化页面显示initView。globalEvent方法如下:

globalEvent: function() { 
//document 添加click事件,用户处理每个jselect元素展开关闭 
var target, 
className, 
elem, 
wrapper, 
status, 
that = this; squid.on(document, 'click', function(event) { 
target = event.target, 
className = target.className; 
switch(className) { 
case 'select-icon': 
case 'select-default unselectable': 
elem = target.tagName.toLowerCase() === 'div' ? target : target.previousSibling 
wrapper = elem.nextSibling.nextSibling 
//firefox 鼠标右键会触发click事件 
//鼠标左键点击执行 
if(event.button === 0) { 
//初始化选中元素 
that.initSelected(elem) 
if(squid.isHidden(wrapper)) { 
status = 'block' 
//关闭所有展开jselect 
that.closeSelect() 
}else{ 
status = 'none' 
} 
wrapper.style.display = status 
elem.focus() 
}else if(event.button === 2){ 
wrapper.style.display = 'none' 
} 
that.zIndex(wrapper) 
break 
case 'select-option': 
case 'select-option selected': 
if(event.button === 0) { 
that.fireSelected(target, target.parentNode.parentNode.previousSibling.previousSibling) 
wrapper.style.display = 'none' 
} 
break 
default: 
while(target && target.nodeType !== 9) { 
if(target.nodeType === 1) { 
if(target.className === 'select-wrapper') { 
return 
} 
} 
target = target.parentNode 
} 
that.closeSelect() 
break 
} 
}) 
}

globalEvent实现了在document绑定click事件,然后在页面上触发点击事件的时候通过事件代理来判断当前点击元素是否是需要进行处理的目标元素,判断条件是通过元素的class,代码中语句的分支分别是:展开当前点击的jselect元素下拉、选中点击列表项、判断是否需要关闭jselect。

initView方法如下:

initView: function(elems) { 
var i = 0, 
elem, 
length = elems.length, 
enabled; for(; i < length; i++) { 
elem = elems[i] 
enabled = elem.getAttribute('data-enabled') 
//使用系统select 
if(!enabled || enabled === 'true') 
continue 
if(squid.isVisible(elem)) 
elem.style.display = 'none' 
this.create(elem) 
} 
}

initView实现了将需要使用jselect替换的select元素先隐藏然后调用create方法,生成单个jselect的整体结构并插入到页面并替代默认select位置。

create方法如下:

create: function(elem) { 
var data = [], 
i = 0, 
length, 
option, 
options, 
value, 
text, 
obj, 
lis, 
ul, 
_default, 
icon, 
selectedText, 
selectedValue, 
div, 
wrapper, 
position, 
left, 
top, 
cssText; options = elem.getElementsByTagName('option') 
length = options.length 
for(; i < length; i++) { 
option = options[i] 
value = option.value 
text = option.innerText || option.textContent 
obj = { 
value: value, 
text: text 
} 
if(option.selected) { 
selectedValue = value 
selectedText = text 
obj['selected'] = true 
} 
data.push(obj) 
} 
lis = this.render(this.tmpl, data) 
ul = '<ul class="select-item">' + lis + '</ul>' 
// 
div = document.createElement('div') 
div.style.display = 'none' 
div.className = 'select-wrapper' 
//已选元素 
_default = document.createElement('div') 
_default.className = 'select-default unselectable' 
_default.unselectable = 'on' 
//让div元素能够获取焦点 
_default.setAttribute('tabindex', '1') 
_default.setAttribute('data-value', selectedValue) 
_default.setAttribute('hidefocus', true) 
_default.innerHTML = selectedText 
div.appendChild(_default) 
//选择icon 
icon = document.createElement('span') 
icon.className = 'select-icon' 
div.appendChild(icon) 
//下拉列表 
wrapper = document.createElement('div') 
wrapper.className = 'select-list hide' 
wrapper.innerHTML = ul 
//生成新的元素 
div.appendChild(wrapper) 
//插入到select元素后面 
elem.parentNode.insertBefore(div, null) 
//获取select元素left top值 
//先设置select显示,取完left, top值后重新隐藏 
elem.style.display = 'block' 
//事件绑定 
this.sysEvent(div) 
position = squid.position(elem) 
elem.style.display = 'none' 
left = position.left 
top = position.top 
cssText = 'left: ' + left + 'px; top: ' + top + 'px; display: block;' 
div.style.cssText = cssText 
}

create方法实现了将系统select数据拷贝到jselect下拉列表,jselect的层级关系是最外层有一个class为select-wrapper的元素包裹,里面有class为select-default的元素用于存放已选的元素,class为select-icon的元素用户告诉用户这是一个下拉列表,class为select-list的div元素里面包含了一个ul元素里面是从系统select拷贝的option的文本和值分别存放在li元素的文本和data-value属性。sysEvent方法是为jselect添加点击展开关闭下拉列表事件以及键盘上下选择下拉元素回车选中下拉元素事件。squid.position方法用于获取系统select元素相对于其offsetParent的位置,这里与获取系统select元素的offset是有区别。其实就是获取自己的offset得到top,left值然后分别减去offsetParent获取的offset的top,left值。最后是把jselect插入到系统select元素后面,显示到页面。

jselect创建的基本流程就是上面描述的这样,剩下就是细节地方的实现,比如说:点击展开下拉显示上次已选择的元素,具体实现该功能的是initSelected方法如下

initSelected: function(elem) { 
var curText = elem.innerText || elem.textContent, 
curValue = elem.getAttribute('data-value'), 
wrapper = elem.nextSibling.nextSibling, 
n = wrapper.firstChild.firstChild, 
text, 
value, 
dir, 
min = 0, 
max, 
hidden = false; for(; n; n = n.nextSibling) { 
text = n.innerText || n.textContent 
value = n.getAttribute('data-value') 
if(curText === text && curValue === value) { 
//显示已选中元素 
if(squid.isHidden(wrapper)) { 
wrapper.style.display = 'block' 
hidden = true 
} 
max = wrapper.scrollHeight 
if(n.offsetTop > (max / 2)) { 
if(wrapper.clientHeight + wrapper.scrollTop === max) 
dir = 'up' 
else 
dir = 'down' 
}else{ 
if(wrapper.scrollTop === min) 
dir = 'down' 
else 
dir = 'up' 
} 
this.inView(n, wrapper, dir) 
if(hidden) 
wrapper.style.display = 'none' 
this.activate(n) 
break 
} 
} 
}

该方法接收class为select-default的div元素即用于存放用户已选择内容的元素,具体实现方式是先遍历所有选项获取class有selected的li元素,通过activate方法标示为当前已选中的元素。这里有一个需要计算的地方,就是每次展开下拉列表都要将已选中的元素滚动到页面可视区。因为有可能下来列表内容很多,但是下拉列表的外层select-list会有一个最大的高度,超过最大高度会出现滚动条,默认不做计算的话有可能已选中的元素会在滚动条下面或者是滚动条上面,所以需要通过计算来重置容器滚动条的位置。具体是已选中内容显示到滚动条的上面还是下面需要根据已选中元素的offsetTop值是否大于外层容器select-list的实际高度一半,把已选中元素显示到可视区的方式是inView方法。inView方法如下
inView: function(elem, wrapper, dir) { 
var scrollTop = wrapper.scrollTop, 
//已选中元素offsetTop 
offsetTop = elem.offsetTop, 
top; if(dir === 'up') { 
if(offsetTop === 0) { 
//滚动条置顶 
wrapper.scrollTop = offsetTop; 
}else if(offsetTop < scrollTop) { 
top = offsetTop - scrollTop 
//滚动条滚动到top值 
this.scrollInView(wrapper, top) 
} 
}else{ 
var clientHeight = wrapper.clientHeight; 
if(offsetTop + elem.offsetHeight === wrapper.scrollHeight) { 
wrapper.scrollTop = wrapper.scrollHeight - wrapper.clientHeight 
}else if(offsetTop + elem.offsetHeight > clientHeight + scrollTop) { 
top = (offsetTop + elem.offsetHeight) - (scrollTop + clientHeight) 
this.scrollInView(wrapper, top) 
} 
} 
}

inView方法需要判断是向上滚动还是向下滚动,scrollInView方法代码很简单就是把下拉列表外层容器的scrollTop设置为指定的值。方法实现如下
scrollInView: function(elem, top) { 
setTimeout(function() { 
elem.scrollTop += top 
}, 10) 
}

这个方法实现放到了setTimeout里面做了一个延迟添加到javascript执行队列里面,主要解决的是IE8下展开下拉列表滚动条会最终滚动到顶部,忽略代码设置的scrollTop(从表现上来看好像对scrollTop的设置也能生效,但是最后会重置滚动条到顶部,不知道IE8为什么会有这个问题。),不能把已选中的元素显示到可视区范围,其他浏览器下不会有这个问题。
整个的实现细节大致就这么多,键盘上下键回车键,关闭下拉列表逻辑都很简单。
遇到的问题
如何让div获取焦点来响应键盘keydown, keyup, keypress事件,到谷歌(非常时期谷歌都不好用了,没办法谁让这是咱的特色呢)查找一些资料最后发现需要为div元素设置tabindex属性,这样就可以让div元素获取焦点,来响应用户的操作。因为浏览器在默认情况下双击或者是点击太频繁的话会选中当前区域,为了取消这个默认操作给用户一个好的体验需要为div元素添加一个属性unselectable,不过这个属性只能适用于IE浏览器,其他浏览器下可以通过添加一个class名字是unselectable来避免这个问题。其他的问题都是逻辑上的控制,还有一些位置的计算了,这里就不再说了。
使用方法
首先是在页面模板把希望通过jselect替换的元素隐藏或者不做任何处理,默认情况下jselect会获取页面所有select依次替换,如果不希望jselect替换的select元素
需要添加自定义属性data-enabled="true"。当然添加data-enabled="false"和没有这个自定义属性一样都会被jselect替换。在使用的过程中可能对于布局结构比较复杂的页面还会有其他的问题,因为我测试的页面结构很简单,所以可能没有测试出来。
使用jselect需要先引入squid.js,然后引入jselect-1.0.js, jselect-1.0.css文件,在需要调用jselect的地方通过如下的调用方式来初始化jselect:squid.swing.jselect();
注:jselect源码以及demo可以通过这里下载。
Javascript 相关文章推荐
Javascript 修改String 对象 增加去除空格功能(示例代码)
Nov 30 Javascript
php+ajax+jquery实现点击加载更多内容
May 03 Javascript
js简单倒计时实现代码
Apr 30 Javascript
Javascript 实现微信分享(QQ、朋友圈、分享给朋友)
Oct 21 Javascript
JavaScript日期选择功能示例
Jan 16 Javascript
原生js实现打字动画游戏
Feb 04 Javascript
JavaScript无阻塞加载和defer、async详解
Feb 26 Javascript
JavaScript 完成注册页面表单校验的实例
Aug 19 Javascript
vue基于Element构建自定义树的示例代码
Sep 19 Javascript
详解Vue内部怎样处理props选项的多种写法
Nov 06 Javascript
JavaScript动态添加数据到表单并提交的几种方式
Jun 26 Javascript
Element Popover 弹出框的使用示例
Jul 26 Javascript
js实现图片放大缩小功能后进行复杂排序的方法
Nov 08 #Javascript
jquery的ajax()函数传值中文乱码解决方法介绍
Nov 08 #Javascript
表头固定(利用jquery实现原理介绍)
Nov 08 #Javascript
Javascript继承(上)——对象构建介绍
Nov 08 #Javascript
异步javascript的原理和实现技巧介绍
Nov 08 #Javascript
找出字符串中出现次数最多的字母和出现次数精简版
Nov 07 #Javascript
jquery 如何动态添加、删除class样式方法介绍
Nov 07 #Javascript
You might like
PHP基于socket实现的简单客户端和服务端通讯功能示例
2017/07/10 PHP
PHP多进程通信-消息队列使用
2019/03/08 PHP
javascript new 需不需要继续使用
2009/07/02 Javascript
javascript event 事件解析
2011/01/31 Javascript
JavaScript高级程序设计 阅读笔记(十二) js内置对象Math
2012/08/14 Javascript
js判断IE浏览器版本过低示例代码
2013/11/22 Javascript
javascript 通用loading动画效果实例代码
2014/01/14 Javascript
js时间日期格式化封装函数
2014/12/02 Javascript
轻量级javascript 框架Backbone使用指南
2015/07/24 Javascript
js实现添加删除表格(两种方法)
2017/04/27 Javascript
HTML5开发Kinect体感游戏的实例应用
2017/09/18 Javascript
jQuery实现表单动态添加与删除数据操作示例
2018/07/03 jQuery
vue、react等单页面项目部署到服务器的方法及vue和react的区别
2018/09/29 Javascript
微信小程序如何再次获取用户授权的方法
2019/05/10 Javascript
vue实现购物车功能(商品分类)
2020/04/20 Javascript
vue中解决chrome浏览器自动播放音频和MP3语音打包到线上的实现方法
2020/10/09 Javascript
Python多进程机制实例详解
2015/07/02 Python
python采用django框架实现支付宝即时到帐接口
2016/05/17 Python
详解supervisor使用教程
2017/11/21 Python
python使用selenium登录QQ邮箱(附带滑动解锁)
2019/01/23 Python
centos6.5安装python3.7.1之后无法使用pip的解决方案
2019/02/14 Python
win10子系统python开发环境准备及kenlm和nltk的使用教程
2019/10/14 Python
使用python 将图片复制到系统剪贴中
2019/12/13 Python
python 实用工具状态机transitions
2020/11/21 Python
HTML5仿手机微信聊天界面
2016/03/18 HTML / CSS
Smallable意大利家庭概念店:设计师童装及家居装饰
2018/01/08 全球购物
西班牙购买隐形眼镜、眼镜和太阳镜网站:Lentiamo.es
2020/06/11 全球购物
工程管理专业个人求职信范文
2013/12/07 职场文书
优秀志愿者事迹材料
2014/02/03 职场文书
葛优非诚勿扰搞笑征婚台词
2014/03/17 职场文书
遗嘱继承公证书
2014/04/09 职场文书
防灾减灾活动总结
2014/08/30 职场文书
群众路线教育实践活动批评与自我批评
2014/09/15 职场文书
政风行风评议工作总结
2014/10/21 职场文书
优秀护士事迹材料
2014/12/25 职场文书
焦点访谈观后感
2015/06/11 职场文书