1.目的

  看了很多element-ui的源码,决定自己实现一个简单的select组件,遇到的几个难点,便记录下来.

2.难点一

  element-ui中的select组件通过v-model可以绑定数据,但在我平时用v-model只是在input中使用过,只知道v-model可以双向绑定数据,但并不清楚其中的实现过程,所以 需要清晰的了解v-model是什么,如下.

<input v-model="test"/>  

<input :value="test" @input="test = $event.target.value"/> // 第一行和第二行的性质是一样的,v-model是一个vue的语法糖
 

  以上是input输入框中的v-model,input标签在输入的时候默认会触发'input'事件, 但是自定义的组件并不会,所以需要我们自己手动发送一个'input'事件,其次是,使用了v-model指令以后会默认动态绑定一个属性值value,因此我们在自定义组件中可以在props接收value,并绑定到组件当中,从而实现了双向绑定,具体可以看参考完整代码.

3.难点二

  当select组件显示选择框时,合理的逻辑是是点击空白或者点击自身都要将选择框关闭, 起初实现是在document中绑定一个click事件用于关闭选择框,当然select点击得阻止事件冒泡,这样的实现方式是在一个页面只有一个select组件是没有问题的,但是当出现多个select组件就会出现一个bug,点击完一个select以后点击另外一个是无法关闭前一个select框的选择框的,问题出在因为每个select框都被阻止了事件的冒泡,自然不会触发document的click事件,从而无法关闭,知晓原因,解决方案如下:

// 显示选择框
showSel(){
this.show = true;
addEvent(document, 'click',this.hideSel, true);
} // 隐藏选择框
hideSel(e){
this.show = false;
  // 如果是子元素,则阻止事件捕获
if(this.$refs.sel && this.$refs.sel.contains(e.target)){ stopEvent(e);
}
removeEvent(document,'click',this.hideSel,true);
} // 显示或隐藏
toggle(){
this.show && this.hideSel() || this.showSel();
} // 注意:其中addEvent,removeEvent,stopEvent是为了兼容处理而自定义的方法

  以上就是这次编写select组件的所得,附上完整实例代码.

<template>
<div class="select" @click="toggle" ref="sel">
<div class="input">
<input
type="text"
:placeholder="placeholder"
readonly
:value = 'value'
@blur="handle">
<img src="../images/drop.svg">
</div>
<ul
class="content"
:class="{'bottom' : position == 'bottom', 'top' : position == 'top'}"
v-show="show && values.length"
ref="content"> <li v-for="item in values">{{item}}</li>
</ul> </div>
</template> <script>
import { addEvent, removeEvent, stopEvent } from '../service/utli.js';
export default { name : 'comSelect',
data(){
return{
val : '',
show : false,
position : 'bottom'
}
},
props : {
values : {
type : Array,
default(){
return []
}
},
value : { },
placeholder:{
type : String,
default : '请选择'
},
},
mounted(){
this.computePos();
},
methods:{
getElementTop(element){
var actualTop = element.offsetTop;
     var current = element.offsetParent;
     while (current !== null){
actualTop += current.offsetTop;
current = current.offsetParent;
     }
     return actualTop;
}, // 计算选择框是往上弹出还是往下弹出
computePos(){ let elHeight = this.$refs.sel.offsetHeight;
let absPos = this.getElementTop(this.$refs.sel);
let contentHeight = this.values.length*40; let docScrollHei = document.body.scrollTop
|| document.documentElement.scrollTop || 0; let docHeight = document.documentElement.clientHeight
|| document.body.clientHeight || 0; if((elHeight+absPos+contentHeight-docScrollHei)>docHeight){
this.position = 'top';
}else{
this.position = 'bottom';
}
},
setVal(item){
this.$emit('input',item);
},
handle(){
this.$emit('blur');
},
showSel(){
this.show = true;
addEvent(document, 'click',this.hideSel, true);
},
hideSel(e){
this.show = false;
console.log(this.$refs.sel.contains(e.target));
if(this.$refs.sel && this.$refs.sel.contains(e.target)){
// 如果是子元素则阻止事件捕获
stopEvent(e);
          this.setVal(e.target.innerHtml);
}
removeEvent(document,'click',this.hideSel,true);
},
toggle(){
this.show && this.hideSel() || this.showSel();
}
}
}
</script> <style scoped lang="scss">
@import '../style/mixin.scss'; .select{
width: 100%;
height: 100%;
position: relative;
cursor: pointer;
}
.input{
width: 100%;
height: 100%;
position: relative;
cursor: pointer; }
.input>input{
width: 100%;
height: 100%;
cursor: pointer; }
.input>img{
right: 0;
top: 50%;
width: 12px;
height: 12px;
position: absolute;
transform: translateY(-50%);
} .content{
width: 100%;
max-height: px(300);
overflow-y: scroll;
border-radius: 10px;
@include padding(4px 0);
position: absolute;
left: 0;
background-color: white;
box-shadow: 0 0 20px 2px #ccc;
@include prix(transform, translateY(5px));
z-index: 2;
}
.content::-webkit-scrollbar {display: none;} .bottom{
top: 100%;
}
.top{
bottom: 125%;
}
.content>li{
height: 40px;
line-height: 40px;
width: 100%;
@include padding(0 0 0 10px);
} .content>li:hover{
color: #409eff;
background-color: rgba(33,33,33,.2);
} </style>

vue自定义select组件的更多相关文章

  1. vue 自定义分页组件

    vue2.5自定义分页组件,可设置每页显示条数,带跳转框直接跳转到相应页面 Pagination.vue 效果如下图: all: small(只显示数字和上一页和下一页): html <temp ...

  2. vue自定义分页组件---切图网

    vue2.5自定义分页组件 Pagination.vue,可设置每页显示条数,带跳转框直接跳转到相应页面,亲测有用.目前很多框架自带有分页组件比如elementUI,不过在面对一个拿到PSD稿,然后重 ...

  3. vue自定义全局组件(自定义插件)

    有时候我们在做开发的时候,就想自己写一个插件然后就可以使用自己的插件,那种成就感很强.博主最近研究element-ui和axios的时候,发现他们是自定义组件,但是唯一有一点不同的是,在用elemen ...

  4. vue 自定义报警组件

    1.自定义报警组件 Alarm.vue <!-- 报警 组件 --> <template> <div class="alarm"> <!- ...

  5. Vue自定义日历组件

    今天给大家介绍Vue的日历组件,可自定义样式.日历类型及支持扩展,可自定义事件回调.Props数据传输. 线上demo效果 示例 Template: <Calendar :sundayStart ...

  6. vue 自定义封装组件 使用 model 选项

    自定义组件的 v-model 一个组件上的 v-model 默认会利用名为 value 的 prop 和名为 input 的事件,但是像单选框.复选框等类型的输入控件可能会将 value 特性用于不同 ...

  7. vue自定义日期组件

    vue-datepicker 基于 vuejs 2.x 可自定义主题的日期组件 github Usage 只需要在项目中加入 calendar.vue,就可以使用了. 向组件传入 prop 即可改变 ...

  8. vue 自定义image组件

    介绍 1:当图片加载失败时,给出错误提示. 2:当图片加载中时,给出加载提示. 3:图片处理模式:等比缩放/裁剪/填充/... 1.图片加载状态处理 通过给图片绑定load事件与error事件处理函数 ...

  9. Vue 自定义事件 && 组件通信

    1 App.vue 2 <template> 3 <!-- 4 组件的自定义事件: 5 1.一种组件间通信的方式,使用于:子组件===>父组件 6 2.使用场景:A是父组件,B ...

随机推荐

  1. nodejs 使用crypto实现sha256\md5加密

    var crypto = require('crypto'); var hash = crypto.createHash('sha256');// sha256或者md5 hash.update('1 ...

  2. [TJOI2018]xor

    题目大意: 有一棵树,根节点为1.每个点有点权.有两种操作. 1. 求节点x所在子树中点权与y异或的最大值.2. 求x到y的路径上点权与z异或的最大值. 解题思路: 可持久化字典树. 对于第一种操作, ...

  3. [luogu2585 ZJOI2006] 三色二叉树 (树形dp)

    传送门 Description Input 输入文件名:TRO.IN 输入文件仅有一行,不超过500000个字符,表示一个二叉树序列. Output 输出文件名:TRO.OUT 输出文件也只有一行,包 ...

  4. 【百度语音识别】JavaAPI方式语音识别示例

    https://ai.baidu.com/forum/topic/show/496730

  5. uva 10003 Cutting Sticks 【区间dp】

    题目:uva 10003 Cutting Sticks 题意:给出一根长度 l 的木棍,要截断从某些点,然后截断的花费是当前木棍的长度,求总的最小花费? 分析:典型的区间dp,事实上和石子归并是一样的 ...

  6. hdu 4037 Development Value(线段树维护数学公式)

    Development Value Time Limit: 5000/3000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Others ...

  7. 2015.05.11,外语,读书笔记-《Word Power Made Easy》 15 “如何谈论事情进展” SESSION 44

    1. not the real McCoy simulate(['simjuleit] v. 假装,冒充,模仿,模拟)来自拉丁simulo,copy的意思.simulo本身派生自拉丁形容词simili ...

  8. BZOJ 4027: [HEOI2015]兔子与樱花 贪心

    4027: [HEOI2015]兔子与樱花 Description 很久很久之前,森林里住着一群兔子.有一天,兔子们突然决定要去看樱花.兔子们所在森林里的樱花树很特殊.樱花树由n个树枝分叉点组成,编号 ...

  9. NFS的搭建(sudo apt-get install nfs-kernel-server),TFTP服务器(sudo apt-get install tftpd-hpa tftp-hpa)

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/Osean_li/article/details/53240705 ***************** ...

  10. ES JVM使用如果超过75%就会GC较多,导致ES索引性能下降

    转自:https://www.elastic.co/guide/en/cloud/current/ec-metrics-memory-pressure.html Scenario: How Does ...