Vue自定义键盘信息,自定义指令

例:自定义键盘信息
 Vue.directive('on').keyCodes.ctrl=17;
        Vue.directive('on').keyCodes.myenter=13;
        window.onload = function(){
            var vm = new Vue({
                el:'#box',
                data:{},
                methods:{
                    show:function(){
                        alert(111);
                    }
                }
            });
        }

<input type="text" @keydown.ctrl="show">
<hr>
<input type="text" @keydown.myenter="show | debounce 2000">

自定义指令

1、Vue.directive(id,definition)注册一个全局自定义指令,接收两个参数,指令ID以及定义对象

2、钩子函数:将作用域与DOM进行链接,链接函数用来创建可以操作DOM的指令

bind - 只调用一次,在指令第一次绑定到元素上时候调用
update - 在bind之后立即以初始值为参数第一次调用,之后绑定值变化的时候,参数为新值与旧值
unbind - 只调用一次,在指令从元素上解绑的时候调用

Vue.directive('my-directive',{
    bind : function(){
        //准备工作
        //例如,添加事件处理器或只需要运行一次的高耗任务
    },
    update : function(){
        //值更新时的工作
        //也会以初始值为参数调用一次
    },
    unzind : function(){
        //清理工作
        //例如,删除bind()添加的事件监听器
    }
})
//调用
<div v-my-directive="someValue"></div>
//只需要update函数,可以传一个函数替代定义对象
Vue.directive('my-directive',function(value){})
<!-- 如果指令需要多个值,可以传入一个js对象 -->
<!-- 指令可以使用合法的js表达式 -->
<body id="example">
    <div id="demo" v-demo="{color : 'white',text : 'hello!'}"></div>
</body>
<script>
    Vue.directive('demo',function(value){
        console.info(value.color); //white
        console.info(value.text) // hello!
    })
    var demo = new Vue({
        el : '#demo'
    })
</script>

当指令使用字面修饰符,值将按普通字符串处理并传递给update方法,update方法只调用一次,因为普通字符串不能影响数据变化

<body id="example">
    <div id="demo" v-demo.literal="foo bar baz"></div>
</body>
<script>
    Vue.directive('demo',function(value){
        console.info(value); //foo bar baz
    })
    var demo = new Vue({
        el : '#demo'
    })
</script>

3、以属性的形式使用自定义属性

<body id="demo">
    <my-directive class="hello" name="hi"></my-directive>
</body>
<script type="text/javascript">
    Vue.elementDirective('my-directive',{
        //api同普通指令一致
        bind : function(){
            console.info(this.el.className);
            console.info(this.el.getAttribute('name'))
        }
    })

    var demo = new Vue({
        el : '#demo'
    })
</script>

4、自定义属性用在对象上,对象内部属性变化的时候触发update,在指令定义对象中指定deep:true

<body id="demo">
    <div v-my-directive="a"></div>
    <button @click="change">change</button>{{a,b,c}}
</body>
<script>
    Vue.directive('my-directive',function(){
        deep : true,

        update : function(obj){
            //当obj的嵌套属性变化时候调用
            console.info(obj.b.c);
        }
    })
    var demoVM = new Vue({
        el : '#demo',

        data : {
            a : {b : {c : 2}}
        },

        method : {
            change : function(){
                demoVM.a.b.c = 4;
            }
        }
    })
</script>

5、acceptStatement让自定义指令接受内联语句

<body id="demo">
    <div v-my-directive="a++"></div>
    {{a}}
</body>
<script>
    Vue.directive('my-directive',function(){
        acceptStatement : true,

        update : function(fn){
            //当obj的嵌套属性变化时候调用
            console.info(fn.toString())
            fu();
        }
    })
    var demoVM = new Vue({
        el : '#demo',

        data : {
            a : 5
        }
    })
</script>

6、v-on绑定多个事件时,要是是不同的事件类型,例如点击,focus,change,会按照业务需求进行

选择,要是绑定了2个甚至多个click事件,那么v-on只会绑定第一个click事件

附自定义拖拽指令
   <body>
    <div id="box">
        <div v-drag :style="{width:'100px', height:'100px', background:'blue', position:'absolute', right:0, top:0}"></div>
        <div v-drag :style="{width:'100px', height:'100px', background:'red', position:'absolute', left:0, top:0}"></div>
    </div>

</body>
    <script>
        window.onload=function(){
              Vue.directive('drag',{
                inserted: function (el) {
                    var oDiv=el;
                    oDiv.onmousedown=function(ev){
                        var disX=ev.clientX-oDiv.offsetLeft;
                        var disY=ev.clientY-oDiv.offsetTop;
                        document.onmousemove=function(ev){
                            var l=ev.clientX-disX;
                            var t=ev.clientY-disY;
                            oDiv.style.left=l+'px';
                            oDiv.style.top=t+'px';
                        };
                        document.onmouseup=function(){
                            document.onmousemove=null;
                            document.onmouseup=null;
                        };
                    };
                }
            });
            var vm=new Vue({
                el:'#box',
                data:{
                    msg:''
                }
            });
        };

    </script>

字面指令

如果在创建自定义指令的时候传入 isLiteral: true ,那么特性值就会被看成直接字符串,并被赋值给该指令的 expression。字面指令不会试图建立数据监视。
例子:

<div v-literal-dir="foo"></div>
Vue.directive('literal-dir', {
  isLiteral: true,
  bind: function () {
    console.log(this.expression) // 'foo'
  }
})

动态字面指令

然而,在字面指令含有 Mustache 标签的情形下,指令的行为如下:

指令实例会有一个属性,this._isDynamicLiteral被设为true;
如果没有提供update函数,Mustache 表达式只会被求值一次,并将该值赋给this.expression。不会对表达式进行数据监视。
如果提供了update函数,指令将会为表达式建立一个数据监视,并且在计算结果变化的时候调用update。

双向指令

如果你的指令想向 Vue 实例写回数据,你需要传入 twoWay: true 。该选项允许在指令中使用
this.set(value)。

Vue.directive('example', {
  twoWay: true,
  bind: function () {
    this.handler = function () {
      // 把数据写回 vm
      // 如果指令这样绑定 v-example="a.b.c",
      // 这里将会给 `vm.a.b.c` 赋值
      this.set(this.el.value)
    }.bind(this)
    this.el.addEventListener('input', this.handler)
  },
  unbind: function () {
    this.el.removeEventListener('input', this.handler)
  }
})

元素指令

有时候,我们可能想要我们的指令可以以自定义元素的形式被使用,而不是作为一个特性。这与 Angular 的 E 类指令的概念非常相似。元素指令可以看做是一个轻量的自定义组件(后面会讲到)。你可以像下面这样注册一个自定义的元素指令:

Vue.elementDirective('my-directive', {
  // 和普通指令的 API 一致
  bind: function () {
    // 对 this.el 进行操作...
  }
})
使用时我们不再用这样的写法:

<div v-my-directive></div>
而是写成:

<my-directive></my-directive>

元素指令不能接受参数或表达式,但是它可以读取元素的特性,来决定它的行为。与通常的指令有个很大的不同,元素指令是终结性的,这意味着,一旦 Vue 遇到一个元素指令,它将跳过对该元素和其子元素的编译 - 即只有该元素指令本身可以操作该元素及其子元素。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 222,104评论 6 515
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,816评论 3 399
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 168,697评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,836评论 1 298
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,851评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,441评论 1 310
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,992评论 3 421
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,899评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,457评论 1 318
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,529评论 3 341
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,664评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,346评论 5 350
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 42,025评论 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,511评论 0 24
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,611评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 49,081评论 3 377
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,675评论 2 359

推荐阅读更多精彩内容