zoukankan      html  css  js  c++  java
  • Vue文本模糊匹配功能

    模糊匹配功能在下拉菜单的组件中用的非常多,于是打算写几个demo看看细节上是如何实现的。

    一、最简单的模糊匹配:计算属性

     1 <!DOCTYPE html>
     2 <html lang="zh-CN">
     3 <head>
     4     <meta charset="UTF-8">
     5     <meta name="viewport" content="width=device-width, initial-scale=1.0">
     6     <meta http-equiv="X-UA-Compatible" content="ie=edge">
     7     <title>Document</title>
     8 </head>
     9 <body>
    10     <div id="app">
    11         <input type="text" v-model="message">
    12         <ul>
    13             <li v-for="(option, index) in matchedOptions" :key="index">{{ option }}</li>
    14         </ul>
    15     </div>
    16     <script src="./vue.js"></script>
    17     <script>
    18         new Vue({
    19             el: '#app',
    20             data: {
    21                 message: '',
    22                 options: ['html', 'css', 'javascript']
    23             },
    24             computed: {
    25                 matchedOptions() {
    26                     if (this.message !== '') {
    27                         return this.options.filter(option => option.includes(this.message))
    28                     }
    29                     return this.options
    30                 }
    31             }
    32         })
    33     </script>
    34 </body>
    35 </html>

    在上面的例子中,计算属性matchedOptions会在文本框内容message变化时筛选options里的数据,效果图如下所示:

    二、使用作用域插槽实现

    使用插槽主要是为了使该功能组件化:在select组件中插入option,然后option通过作用域插槽从select中获取文本值:

     1 <!DOCTYPE html>
     2 <html lang="zh-CN">
     3 <head>
     4     <meta charset="UTF-8">
     5     <meta name="viewport" content="width=device-width, initial-scale=1.0">
     6     <meta http-equiv="X-UA-Compatible" content="ie=edge">
     7     <title>Document</title>
     8 </head>
     9 <body>
    10     <div id="app">
    11         <my-select>
    12             <template #default="{ message }">
    13                 <ul>
    14                     <li v-for="(option, index) in options" :key="index" v-show="option.includes(message)">{{ option }}</li>
    15                 </ul>
    16             </template>
    17         </my-select>
    18     </div>
    19     <script src="./vue.js"></script>
    20     <script>
    21         Vue.component('my-select', {
    22             template: `
    23                 <div class="my-select">
    24                     <input type="text" v-model="message">
    25                     <slot :message="message"></slot>
    26                 </div>
    27             `,
    28             data() {
    29                 return {
    30                     message: ''
    31                 }
    32             }
    33         })
    34         new Vue({
    35             el: '#app',
    36             data: {
    37                 options: ['html', 'css', 'javascript']
    38             }
    39         })
    40     </script>
    41 </body>
    42 </html>

    全局注册了my-select组件后,可以删除app里的message数据,使用v-show来控制选项的显示,运行效果和计算属性方式相同。缺点就是无法单文件化(刚学vue没多久,不知道怎么在单文件里使用作用域插槽,试过直接把template里的东西封装成my-option好像并不管用)

    三、混入广播和派发方法在独立组件中实现模糊匹配

    首先需要一个emitter文件:

     1 /**
     2  * 子组件广播事件
     3  * @param {string} componentName 子组件名
     4  * @param {string} eventName 事件名
     5  * @param {...any} params 事件参数
     6  */
     7 function _broadcast(componentName, eventName, ...params) {
     8     this.$children.forEach(child => {
     9         if (child.$options.name === componentName) {
    10             child.$emit(eventName, ...params)
    11         }
    12         _broadcast.call(child, componentName, eventName, ...params)
    13     })
    14 }
    15 
    16 /**
    17  * 父组件派发事件
    18  * @param {string} componentName 父组件名
    19  * @param {string} eventName 事件名
    20  * @param {...any} params 事件参数
    21  */
    22 function _dispatch(componentName, eventName, ...params) {
    23     if (this.$parent) {
    24         if (this.$parent.$options.name === componentName) {
    25             this.$parent.$emit(eventName, ...params)
    26         }
    27         _dispatch.call(this.$parent, componentName, eventName, ...params)
    28     }
    29 }
    30 
    31 /**
    32  * mixin
    33  */
    34 export default {
    35     methods: {
    36         broadcast(componentName, eventName, ...params) {
    37             _broadcast.call(this, componentName, eventName, ...params)
    38         },
    39         dispatch(componentName, eventName, ...params) {
    40             _dispatch.call(this, componentName, eventName, ...params)
    41         }
    42     }
    43 }

    注意,这里的$children和$parent都是指具有dom父子关系的vue组件。

    最后,通过设置查询条件来控制子组件的显示与隐藏即可实现实时模糊搜索。

  • 相关阅读:
    ORA-01940: cannot drop a user that is currently connected 问题解析
    Oracle11g数据库导入Oracle10g操作成功
    固态硬盘
    Oracle数据库默认的data pump dir在哪
    navicat 关于orcale新建表空间,用户和权限分配
    oracle 11g 完全卸载方法
    完全卸载oracle11g步骤
    架构设计:负载均衡层设计方案(4)——LVS原理
    C++中使用REST操作
    在C#中实现视频播放器
  • 原文地址:https://www.cnblogs.com/viewts/p/11163307.html
Copyright © 2011-2022 走看看