手写vue中v-bind:style效果的自定义指令

自定义指令

  1. 什么是自定义指令

    以 v- 为前缀,然后加上自己定义好的名字组成的一个指令就是自定义指令。为什么要有自定义指令呢?在有些时候,你仍然需要对普通的DOM元素进行底层的操作,这个时候就可以用到自定义指令。

  2. 自定义指令的语法
    1. 全局自定义指令

      // 注册一个全局自定义指令 `v-focus`
      Vue.directive('focus', {
        // 当被绑定的元素插入到 DOM 中时……
        inserted: function (el) {
          // 聚焦元素
          el.focus()
        }
      })
    2. 局部自定义指令
      directives: {
        focus: {
          // 指令的定义
          inserted: function (el) {
            el.focus()
          }
        }
      }
  3. 钩子函数

    看了上述的代码,如果你从来没接触过这类内容,你可能会很生疏,下面我给大家讲讲其每一步所需要掌握的东西

    首先是钩子函数:

    • bind :只会调用一次的函数,表示指令第一次绑定元素时调用
    • inserted :被绑定元素插入到父节点时调用(仅保证父节点存在,但不一定已被插入文档中)。
    • update :所在组件的 VNode 更新时调用,但是可能发生在其子 VNode 更新之前。指令的值可能发生了改变,也可能没有。
    • componentUpdated :指令所在组件的 VNode 及其子 VNode 全部更新后调用。
    • unbind :只调用一次,指令与元素解绑时调用。

    然后我们看看钩子函数中的参数列表

    • el:指令所绑定的元素,可以用来直接操作 DOM 。
    • binding:一个对象,包含以下属性:
      • name:指令名,不包括 v- 前缀。
      • value:指令的绑定值,例如:v-my-directive="1 + 1" 中,绑定值为 2
      • oldValue:指令绑定的前一个值,仅在 updatecomponentUpdated 钩子中可用。无论值是否改变都可用。
      • expression:字符串形式的指令表达式。例如 v-my-directive="1 + 1" 中,表达式为 "1 + 1"
      • arg:传给指令的参数,可选。例如 v-my-directive:foo 中,参数为 "foo"
      • modifiers:一个包含修饰符的对象。例如:v-my-directive.foo.bar 中,修饰符对象为 { foo: true, bar: true }
    • vnode:Vue 编译生成的虚拟节点。
    • oldVnode:上一个虚拟节点,仅在 updatecomponentUpdated 钩子中可用。

    除了 el 之外,其它参数都应该是只读的,切勿进行修改。如果需要在钩子之间共享数据,建议通过元素的 dataset 来进行。

    下面我们来分析几个简单的钩子函数,及其参数

    代码如下:(看完代码我再将其)

    <div id="app">
      <span v-mmm='{color:color, fontSize:"20px"}'>bind的对象形式</span>
      <br>
      <button @click='changeStyle'>改变颜色</button>
    </div>
    
    <script>
      const vm = new Vue({
        el: '#app',
        data: {
          color: 'cyan',
          style1: {color: 'lightblue'},
          style2: {fontSize:"20px"}
        },
        directives: {
          mmm: {
            bind(el,binding) {
              //  binding.value = {color:color, fontSize:"20px"}
              if(binding.value.constructor === Object) {
                Object.keys(binding.value).forEach(key => {
                  el.style[key] = binding.value[key];
                })
              }
            }
          }
        },
        methods: {
          changeStyle() {
            this.color = 'lightpink'
          }
        },
      })
    </script>

    效果图:

    下面我们换成 update 钩子函数:

    update(el,binding) {
      //  binding.value = {color:color, fontSize:"20px"}
      if(binding.value.constructor === Object) {
        Object.keys(binding.value).forEach(key => {
          el.style[key] = binding.value[key];
        })
      }
    }

    效果图:

    当我们点击按钮后:

    我们发现他会随着数据改变而更新,但是他刚开始不会调用它,只有当数据发生改变之后才会调用该钩子函数

    如果我们想要要刚开始就调用,并且会跟随数据改变而改变,那么我们就要同时调用 bindupdate 这两个钩子函数,但是两个钩子函数中的内容又是一样的,那么书写起来就很麻烦。那么我们可以这样写:

    mmm: function(el , binding) {
      if(binding.value.constructor === Object) {
        Object.keys(binding.value).forEach(key => {
          el.style[key] = binding.value[key];
        })
      }
    },

    这样之后我们就可以达到那样的效果了。

  4. 书写一个类似于 v-bind:style 的效果的自定义指令
    <div id="app">
      <span v-mystyle='{color:color, fontSize:"20px"}'>v-mystyle的对象形式</span>
    
      <p v-mystyle='[style1 , style2]'>我使用的是v-mystyle的数组形式</p>
    
      <button @click='changeStyle'>改变颜色</button>
    </div>
    
    <script>
      const vm = new Vue({
        el: '#app',
        data: {
          color: 'cyan',
          style1: {color: 'lightblue'},
          style2: {fontSize:"20px"}
        },
        directives: {
          mystyle: function(el , binding) {
            if(binding.value.constructor === Object) {
              Object.keys(binding.value).forEach(key => {
                el.style[key] = binding.value[key];
              })
            } else if(binding.value.constructor === Array) {
              for(item of binding.value) {
                for(key in item) {
                  el.style[key] = item[key];
                }
              }
            }
          },
        },
        methods: {
          changeStyle() {
            this.color = 'lightpink'
          }
        },
      })
    </script>

    效果图:

原文地址:https://www.cnblogs.com/liuyilong/p/12219159.html

时间: 2024-08-26 19:23:55

手写vue中v-bind:style效果的自定义指令的相关文章

12 vue中的样式-style

vue中的样式 是动态绑定style , 对象 是无需键值对的集合 1. <h1 :style="{ color:'red' ,fontSize:'32px'}">这是最基础的一个</h1> 直接在对象中写键值对 2. <h1 :style="styleObj2">这是第一个</h1> 将键值对对象写在data中.直接动态绑定 3. <h1 :style="[ styleObj1, styleObj2

手写Vue (1) 对象劫持

1.引入我们手写的Vue 拿到配置数据 import Vue from '../source/src/index'; let vm = new Vue({ el: '#app', data() { return { msg: 'hello', school: { name: 'zf', age: 10 }, arr: [1, 2, 3] } }, computed: { }, watch: { } }) console.log(vm) // setTimeout(() => { // vm.ar

TensorFlow实现Softmax Regression识别手写数字中&quot;TimeoutError: [WinError 10060] 由于连接方在一段时间后没有正确答复或连接的主机没有反应,连接尝试失败”问题

出现问题: 在使用TensorFlow实现MNIST手写数字识别时,出现"TimeoutError: [WinError 10060] 由于连接方在一段时间后没有正确答复或连接的主机没有反应,连接尝试失败"问题. 截图如下: 问题原因: 出现该问题的原因可能是由于自身网络问题或者mnist数据集下载网页连接不成功导致. 解决办法: 进入MNIST数据下载网页:http://yann.lecun.com/exdb/mnist/ 将数据下载到相应的路径下,参考如图: 代码中显示将数据下载至

vue - 计算属性、监听、自定义指令

1.计算属性 <div id="app"> <input type="text" v-model="x"> <input type="text" v-model="y"> {{get}} </div> <script> var app=new Vue({ el:"#app", data:{ x:"", y:&

Vue中Class与Style绑定

操作元素的class列表和内联样式是数据绑定的一个常见需求.因为它们都是属性,所以我们可以使用v-bind处理它们:只需要通过表达式计算出字符串结果即可.不过拼接字符串比较麻烦,因此在v-bind用于class和style时,vue做了专门的增强,表达式结果的类型除了字符串之外,还可以是对象和数组. 绑定HTML Class (1) 对象语法可以给v-bind:class一个对象,以动态切换class: <div v-bind:class="{active: isActive}"

手写vue双向绑定数据

来一张原理图: 实现思路: (1)绑定data 种的数据,为每个数据添加指令.通过Object,defineProperty() 来通知属性是否更改 (2) 找到每个DOM节点的指令.绑定事件.并绑定watcher (3)  实现DOM事件改变之后, 响应data数据,实现视图更新 <!DocType> <html> <title>vue 的双向绑定事件</title> <body id="app"> <input ty

学习手写vue,理解原理

class Compiler{ constructor(el,vm){ // 判断el属性 是不是 一个元素, 如果不是就获取 this.el = this.isElementNode(el)?el:document.querySelector(el); // console.log(this.el); this.vm = vm; // 把当前节点放到内存中 let fragment = this.node2fragment(this.el); // console.log(fragment,"

vue中,class与style绑定

<template> <div> <p v-bind:class="{active:isActive,'demo':Demo}">嘿嘿</p> <p v-bind:class="styleObj">哈哈</p> <p v-bind:class="styleObj2">咯吱咯吱</p> <p v-bind:class="[active1]

手写Vue (1) 准备工作

1.安装插件 "devDependencies": { "html-webpack-plugin": "^4.0.4", "webpack": "^4.42.1", "webpack-cli": "^3.3.11", "webpack-dev-server": "^3.10.3" } 2.配置 项目 的基本结果 (1)根目录下新