在 Angular 中实现搜索关键字高亮示例


Posted in Javascript onMarch 21, 2017

在 Angular 中,我们不应该试图直接修改 DOM 的内容,当需要更新 DOM 内容的时候,应该修改的其实是我们的数据模型,也就是 $scope 中的数据,Angular 会帮助我们将修改之后的数据展示在页面中。

但是,在有些情况下,比如我们有一个搜索框,希望将搜索的关键字在正文中高亮出来,这时候就会感觉比较吃力了,filter 可以帮助我们处理这种情况。

实际上,很多时候,我们的数据是不能直接输出到 DOM 中的,典型的比如日期,或者货币等等,通常需要将我们内部的数据格式化之后,再输出到页面中,在 Angular 中,这个工作就是通过 filter 来实现的。

filter 是 module 级别的服务,定义之后,可以在整个 module 内直接使用,有着极高的复用性。

为了说明问题,先回顾一下的 走进AngularJs(七) 过滤器(filter) 说明一下过滤器,然后详细说明如何处理搜索关键字高亮问题。你可以直接从自定义过滤器部分开始。

1. 过滤器的用法

1. 模板中使用

我们可以直接在{{}}中使用filter,跟在表达式后面用 | 分割,语法如下:

{{ expression | filter }}

也可以多个filter连用,上一个filter的输出将作为下一个filter的输入。

{{ expression | filter1 | filter2 | ... }}

filter可以接收参数,参数用 : 进行分隔。

{{ expression | filter:argument1:argument2:... }}

除了对{{}}中的数据进行格式化,我们还可以在指令中使用filter,例如先对数组array进行过滤处理,然后再循环输出:

<span ng-repeat="a in array | filter ">

2. 在 controller 和 service 中使用

我们的js代码中也可以使用过滤器,方式就是我们熟悉的依赖注入,例如我要在controller中使用currency过滤器,只需将它注入到该controller中即可。

app.controller('testC',function($scope,currencyFilter){
  $scope.num = currencyFilter(123534); 
}

在模板中使用{{num}}就可以直接输出 $123,534.00了!在服务中使用filter也是同样的道理。

此时你可能会有疑惑,如果我要在controller中使用多个filter,难道要一个一个注入吗,这岂不太费劲了?小兄弟莫着急~ng提供了一个$filter服务可以来调用所需的filter,你只需注入一个$filter就够了,使用方法如下:

app.controller('testC',function($scope,$filter){
  $scope.num = $filter('currency')(123534);
$scope.date = $filter('date')(new Date()); 
}

2. 内置的 filter

1. currentcy, 使用currency可以将数字格式化为货币,默认是美元符号,你可以自己传入所需的符号

{{num | currency : '¥'}}

2. date, 原生的js对日期的格式化能力有限,ng提供的date过滤器基本可以满足一般的格式化要求。

{{date | date : 'yyyy-MM-dd hh:mm:ss EEEE'}}

3. filter, 注意了,这个过滤器的名字叫 filter,不要搞混了。用来处理一个数组,然后可以过滤出含有某个子串的元素,作为一个子数组来返回。可以是字符串数组,也可以是对象数组。如果是对象数组,可以匹配属性的值。它接收一个参数,用来定义子串的匹配规则。

$scope.childrenArray = [
    {name:'kimi',age:3},
    {name:'cindy',age:4},
    {name:'anglar',age:4},
    {name:'shitou',age:6},
    {name:'tiantian',age:5}
  ];

自定义的辅助函数。

$scope.func = function(e){return e.age>4;}

使用 filter 过滤器

{{ childrenArray | filter : 'a' }} //匹配属性值中含有a的
{{ childrenArray | filter : 4 }} //匹配属性值中含有4的
{{ childrenArray | filter : {name : 'i'} }} //参数是对象,匹配name属性中含有i的
{{childrenArray | filter : func }} //参数是函数,指定返回age>4的

4. json, 格式化 json 对象。json过滤器可以把一个js对象格式化为json字符串。作用就和我们熟悉的JSON.stringify()一样。

{{ jsonTest | json}}

5. limitTo, limitTo过滤器用来截取数组或字符串,接收一个参数用来指定截取的长度。只能从数组或字符串的开头进行截取

{{ childrenArray | limitTo : 2 }} //将会显示数组中的前两项

6. lowercase, 转换为小写。把数据转化为全部小写。

7. uppercase, 转换为大写。

8. number, 格式化数字。number过滤器可以为一个数字加上千位分割,像这样,123,456,789。同时接收一个参数,可以指定小float类型保留几位小数:

{{ num | number : 2 }}

9. orderBy 排序,orderBy过滤器可以将一个数组中的元素进行排序,接收一个参数来指定排序规则,参数可以是一个字符串,表示以该属性名称进行排序。可以是一个函数,定义排序属性。还可以是一个数组,表示依次按数组中的属性值进行排序(若按第一项比较的值相等,再按第二项比较),

<div>{{ childrenArray | orderBy : 'age' }}</div>   //按age属性值进行排序
<div>{{ childrenArray | orderBy : orderFunc }}</div>  //按照函数的返回值进行排序
<div>{{ childrenArray | orderBy : ['age','name'] }}</div> //如果age相同,按照name进行排序

3. 自定义 filter

 我们先定义个没有参数的过滤器。第一个例子来自 Angular 官方的文档。

我们希望检查数据是否为真,如果为真的话,显示为一个 ✓ ,否则的话,显示为  ✘, 在 Unicode 中,\u2713 -> ✓, \u2718 -> ✘,所以,我们需要做的就是检查数据是否为真,然后转化为这两个特殊字符进行输出。

app.filter('checkmark', function() {
 return function(input) {
  return input ? '\u2713' : '\u2718';
 };
});

过滤器还可以有参数,比如内置的过滤器可以格式话货币或者日期。

在过滤器后面,使用冒号 (:) 分隔的就是过滤器的参数,在过滤器中,可以获取这个参数,如果有多个参数,就继续使用冒号 (:) 进行分隔,所以,过滤器可以有多个参数。

下面实现一个截断超长字符串的过滤器。

app.filter("truncate", function(){
  return function(text, length){
    if (text) {
      var ellipsis = text.length > length ? "..." : "";
      return text.slice(0, length) + ellipsis;
    };
    return text;    
  }
});

4. 定义高亮 filter

 我们希望搜索的关键字在正文中高亮出来,正文就是过滤器的第一个参数,第二个参数就是搜索关键字,这样,我们的过滤器将会有两个参数了。

高亮的原理很简单,将需要高亮的内容使用 span 标记隔离出来,再加上一个特殊的 class 进行描述就可以了。

app.filter("highlight", function($sce, $log){

  var fn = function(text, search){
    $log.info("text: " + text);
    $log.info("search: " + search);

    if (!search) {
      return $sce.trustAsHtml(text);
    }
    text = encodeURI(text);
    search = encodeURI(search);

    var regex = new RegExp(search, 'gi')
    var result = text.replace(regex, '<span class="highlightedText">$&</span>');
    result = decodeURI(result);
    $log.info("result: " + result );
    return $sce.trustAsHtml(result);
  };

  return fn;
});

$sce, 和 $log 是 angular 提供的两个服务,其中 $sce 服务需要使用  ngSanitize 模块。关于这个模块,可以参考:angular-ngSanitize模块-$sanitize服务详解

5. 定义服务对象

我们的页面可能很复杂,需要在一个控制器中输入关键字,但是,需要在多个控制器中使用这个关键字进行过滤,怎么处理呢?使用服务可以解决这个问题。

在 Angular 中,服务就是一个单例对象,我们使用 factory 可以直接定义一个服务对象。

app.factory("notifyService", function(){
  var target = {
    search:"key"
  };

  return target;
});

在需要使用这个对象的地方,直接注入就可以获取这个对象了。

6. 定义搜索 Controller

在搜索控制器中,我们希望用户提供搜索关键字。为了方便检查,我们顺便将用户的输入展示出来。

<div ng-controller="tools">
      <div>
        <input type="text" ng-model="notify.search" value=""/>
      </div>
      <div>
        {{notify.search}}
      </div>
    </div>

控制器的实现,我们将服务对象直接注入到控制器中,然后绑定到当前的 $scope 上,以便在当前控制器中实现绑定。

app.controller("tools", function($scope, notifyService){
  $scope.notify = notifyService;
});

现在,我们可以直接输入搜索关键字,这个关键字将被保存到服务对象中,可以在当前的 module 中的各个控制器中访问这个服务对象。

7. 在内容 Controller 中使用 filter

现在,我们已经有了过滤器,也可以直接通过服务对象来获取搜索关键字,现在组合起来就可以使用了。text 中是我们的正文内容。

<div ng-controller="search">
      <div ng-bind-html="text | highlight:notify.search">
      </div>
    </div>

下面再看看 highlight 和 notify.search 来自何方。

app.controller("search", function($scope, $log, notifyService){
  $scope.text = "hello, world. hello, world. hello, world.";
  $scope.notify = notifyService;
})

为了在当前控制器中使用搜索关键字,关键是在搜索关键字发生变化的时候,还需要自动更新,我们将服务对象绑定到当前的 $scope 中,这是引用的名称为 notify.

这样,在复杂页面中,我们可能存在多个控制器,在每个需要高亮的控制器中,我们只需要注入服务对象,就可以直接获取到当前的搜索关键字,配合使用过滤器,就可以直接实现全局高亮了。

8. 总结

结合过滤器和服务对象,我们实现了在复杂页面中,对整个页面中的内容进行关键字高亮。以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
js自带函数备忘 数组
Dec 29 Javascript
jQuery EasyUI API 中文文档 - DataGrid数据表格
Nov 17 Javascript
浅析jQuery(function(){})与(function(){})(jQuery)之间的区别
Jan 09 Javascript
JS模拟并美化的表单控件完整实例
Aug 19 Javascript
jquery获取文档高度和窗口高度汇总
Jan 25 Javascript
图文详解Javascript中的上下文和作用域
Feb 15 Javascript
vue.js树形组件之删除双击增加分支实例代码
Feb 28 Javascript
使用重写url机制实现验证码换一张功能
Aug 01 Javascript
JavaScript实现的文本框placeholder提示文字功能示例
Jul 25 Javascript
swiper在vue项目中loop循环轮播失效的解决方法
Sep 15 Javascript
微信小程序onLaunch异步,首页onLoad先执行?
Sep 20 Javascript
echarts统计x轴区间的数值实例代码详解
Jul 07 Javascript
js省市区级联查询(插件版&amp;无插件版)
Mar 21 #Javascript
Bootstrap学习笔记 轮播(Carousel)插件
Mar 21 #Javascript
详解Vue2+Echarts实现多种图表数据可视化Dashboard(附源码)
Mar 21 #Javascript
node安装--linux下的快速安装教程
Mar 21 #Javascript
Vue 单文件中的数据传递示例
Mar 21 #Javascript
js实现颜色阶梯渐变效果(Gradient算法)
Mar 21 #Javascript
详解Vue 实例中的生命周期钩子
Mar 21 #Javascript
You might like
使用Apache的rewrite技术
2006/06/22 PHP
php下用GD生成生成缩略图的两个选择和区别
2007/04/17 PHP
PHP中的extract的作用分析
2008/04/09 PHP
apache+php完美解决301重定向的两种方法
2011/06/08 PHP
将FCKeditor导入PHP+SMARTY的实现方法
2015/01/15 PHP
php实现微信模板消息推送
2018/03/30 PHP
CI框架教程之优化验证码机制详解【验证码辅助函数】
2019/04/16 PHP
js获取div高度的代码
2008/08/09 Javascript
基于jquery实现的鼠标滑过按钮改变背景图片
2011/07/15 Javascript
jquery中ajax学习笔记4
2011/10/16 Javascript
jQuery中使用Ajax获取JSON格式数据示例代码
2013/11/26 Javascript
jquery实现点击弹出层效果的简单实例
2014/03/03 Javascript
jQuery实现页面滚动时动态加载内容的方法
2015/03/20 Javascript
JS克隆,属性,数组,对象,函数实例分析
2016/11/26 Javascript
微信小程序 scroll-view隐藏滚动条详解
2017/01/16 Javascript
浅析vue component 组件使用
2017/03/06 Javascript
jQuery判断邮箱格式对错实例代码讲解
2017/04/12 jQuery
vue仿淘宝订单状态的tab切换效果
2020/06/23 Javascript
javascript中一些奇葩的日期换算方法总结
2018/11/14 Javascript
JS forEach跳出循环2种实现方法
2020/06/24 Javascript
解决vue-router 嵌套路由没反应的问题
2020/09/22 Javascript
Python读取键盘输入的2种方法
2015/06/16 Python
Python实现分割文件及合并文件的方法
2015/07/10 Python
Python用字典构建多级菜单功能
2019/07/11 Python
Python递归实现打印多重列表代码
2020/02/27 Python
解决django中form表单设置action后无法回到原页面的问题
2020/03/13 Python
Python实现RabbitMQ6种消息模型的示例代码
2020/03/30 Python
python文件读取失败怎么处理
2020/06/23 Python
django ObjectDoesNotExist 和 DoesNotExist的用法
2020/07/09 Python
日本卡普空电视游戏软件公司官方购物网站:e-CAPCOM
2018/07/17 全球购物
Bonprix法国:时尚、鞋子、家居
2020/12/29 全球购物
中层干部考核评语
2015/01/04 职场文书
会计出纳岗位职责
2015/03/31 职场文书
2015年征兵工作总结
2015/07/23 职场文书
温馨祝福晨语:美丽的一天从我的问候开始
2019/11/28 职场文书
MySQL性能指标TPS+QPS+IOPS压测
2022/08/05 MySQL