Vue.js备忘记录(四) vue过滤器文本格式化, 插槽 sl

2020-11-03  本文已影响0人  熊爸天下_56c7

一. vue过滤器 //文本格式化

Vue.js 允许你自定义过滤器,可被用于一些常见的文本格式化。

过滤器可以用在两个地方:双花括号插值和v-bind表达式

过滤器应该被添加在 JavaScript 表达式的尾部,由“管道”符号指示:

<!-- 在双花括号中 -->
{{ message | capitalize }}

<!-- 在 `v-bind` 中 -->
<div v-bind:id="rawId | formatId"></div>

<!-- 过滤器可以串联 -->
{{ message | filterA | filterB }}

你可以在一个组件的选项中定义本地的过滤器:

filters: {
  capitalize: function (value) {
    if (!value) return ''
    value = value.toString()
    return value.charAt(0).toUpperCase() + value.slice(1)
  }
}

或者在创建 Vue 实例之前全局定义过滤器:

Vue.filter('capitalize', function (value) {
  if (!value) return ''
  value = value.toString()
  return value.charAt(0).toUpperCase() + value.slice(1)
})

new Vue({
  // ...
})

注意:过滤器的第一个参数已经被规定死了,永远都是 过滤器 管道符号前面的那个数据

二. 数组中的那些方法是响应式的?

我们知道,我们可以用v-for遍历数组, 那么数组中那些方法是改变自身的?只有改变自身的方法才不用重新赋值吧

1.以下方法是响应式的

let arr =['a','b','c','d']
//1 push数组最后面添加元素
arr.push('e')

//2.pop删除最后一个元素
arr.pop()

//3.shift删除第一个元素
arr.shift()

//4.unshift 数组最前面添加元素
arr.unshift('e') 

//5.splice 删除/插入/替换元素
arr.splice(2,1) //从2开始删除1个
arr.splice(2,0,'f') //从2之后插入元素
arr.splice(2,1,'f') //替换2之后的元素(从2删除1个元素,再添加一个元素)

//6.sort排序
arr.sort()

//7. reverse()反序
arr.reverse()

2.以下方法不是响应式的

通过索引值改变数组内容不是响应式的

this.arr[2]='f'

如何解决?

this.arr.splice(2,1,'f')
//或者
Vue.set(this.arr,2,'f')     //vue方式

三. 插槽 slot

插槽的目的是提高组件的拓展性, 就像电脑的USB插槽.

解决子组件模板太过固化不灵活的问题,这时候我们给模板预备插槽

比如:

下面的例子,不同的导航栏存在区别,同时又有共性,

我们可以定义一个组件,这个组件里有三个插槽,每个插槽里都可以传入不同的东西,这使得这个导航栏组件可以应用的更广

这里我们的思想是: 抽取共性,保留插槽

1.插槽的基本使用

定义插槽<slot>默认值</slot>

    let template=`
    <div>
      <p>下面让你自由发挥,你给我啥我显示啥</p>
      <slot></slot>    <--这就是个插槽,里面的内容由调用者决定  -->
    </div>

这样子组件就有插槽了,父组件调用时可以往里面插入东西了

<子组件> 把要插入的内容写在这,可以是任何标签 </子组件> 

2.具名插槽

<slot  name="left"></slot>  
<--这个插槽名叫left  -->  <slot  name="center"></slot>

调用时: 要在插入的标签中使用 slot="插槽名" 来插入

<子组件> <插入的标签 slot="left"></插入的标签> </子组件>

例:神还原上面 的京东截图 😂


<!DOCTYPE html>
<html lang='en'>

<head>
  <meta charset='UTF-8'>
  <meta name='viewport' content='width=device-width, initial-scale=1.0'>
  <title>Document</title>
</head>

<body>
  <div id='app'>
    <cpn>
      <span slot="left">三</span>
      <span slot="center">JD<input type="text" placeholder="🔍这是一个搜索框"></span>
      <button slot="right">登录</button>
    </cpn>
    <cpn><input slot="center" type="text" placeholder="🔎这是一个搜索框"></cpn>
    <cpn><strong slot="center">"购物车"</strong></cpn>
    <cpn>
      <table slot="center" style="display: inline-block;">
        <tr>
          <td> 📍商品 </td>
          <td> 评价 </td>
          <td> 详情 </td>
          <td> 推荐 </td>
        </tr>
      </table>
    </cpn>
    <cpn></cpn>
  </div>
  <script src='https://cdn.jsdelivr.net/npm/vue/dist/vue.js'></script>
  <script>

    let template = `
    <div>
      <slot name="left"><</slot>
      <slot name="center"><span>默认显示一个span</span></slot>
      <slot name="right">...</slot>
    </div>
    `
    const cpn = {
      template,
    }
    const vm = new Vue({
      el: '#app',
      components: {
        cpn
      }
    })
  </script>
</body>

</html>

3. 作用域插槽

父组件通过作用域插槽拿到子组件数据,但是按照父组件自己的方式展示

要做到这一点,我们要在子组件模板中声明slot时 v-bind动态绑定数据源

在父组件调用时,标签中加入slot-scope='slot' 来获取slot标签

然后用slot.data获取数据,用来展示

<!DOCTYPE html>
<html lang='en'>

<head>
  <meta charset='UTF-8'>
  <meta name='viewport' content='width=device-width, initial-scale=1.0'>
  <title>Document</title>
</head>

<body>
  <div id='app'>
    <cpn></cpn>
    <cpn>
      <div slot-scope='slot'>
        <span v-for="item in slot.data" >{{item}} - </span>       父组件的展示
      </div>
    </cpn>
  </div>
  <script src='https://cdn.jsdelivr.net/npm/vue/dist/vue.js'></script>
  <script>

    let template = `
    <div>   
    <slot :data='languages'>    //子组件用列表来展示
      <ul>
        <li v-for="language in languages" :key="language">{{language}}</li>
      </ul>
    </slot> 
    </div> 
    `
    const cpn = {
      template,
      data() {
        return {
          languages: ['c', "python", 'javascript', 'c#', 'DART', 'JAVA']
        }
      },
    }
    const vm = new Vue({
      el: '#app',
      components: {
        cpn
      }
    })
  </script>
</body>

</html>

4. 不要在插槽上直接设置class

<slot name="item-text" :class="{acticv:isActive}"></slot>
这种写法不会起作用,
因为插槽是直接被替换成别的标签了,
那个标签替换它的时候不会关注到它里面的class

正确的写法是:
<div :class="{acticv:isActive}">
  <slot name="item-text"></slot>
</div>
上一篇 下一篇

猜你喜欢

热点阅读