使用Angular编写TodoMVC Vol 5

序言

本章我们将学习如果利用service在controller之间共享、传递数据。还将调整我们在不同状态下显示不同状态的任务数据清单的函数,将其从Controller层使用函数过滤改为View层使用Filter过滤器进行加工。

1. Service

1.1 什么是Service

首先我们要简单的说明一下什么是Service。Service是Angular中主要的组件之一,Service被认为一种可注入的对象。而根据注入的形式与方法不同,在Angualr1.x中Service主要以下几种形式出现:


Angular Service结构

而一般来说我们最常用的两种形式应该是

  1. service
  2. factory
    简单来说service和factory的区别在于,service注入的是一个构造函数,而factory注入的是一个方法。所有的service都会以单例的模式存在整个angular上下文中。而根据单例整个特性,我们便可以在不同的controller操作service中同一数据副本。

1.2 交互原理

使用Service在Controller之间共享数据

我们将通过新增一个TaskService来封装原先vm.tasks中的数据,在不同的controller中进行共享。

2. 使用TaskService来封装任务数据

2.1 新建TaskService

我们创建一个TaskService用来保存所有的tasks列表信息。

//services/task.service.sj
(function ()
{
    'use strict';

    angular
        .module('todomvc')
        .service('TaskService', TaskService);

    TaskService.$inject = [];
    function TaskService()
    {
      var service = {
        tasks:[
        {
            title: "第一个任务",
            completed: true
        },
        {
            title: "第二个任务",
            completed: false
        }]
      }
        
      return service;
    }
})();

我们为TaskService中增加一个属性字段tasks用于保存任务列表数据。

2.2 向TodoController中注入TasksService

原先我们的TodoController只注入了我们需要的$routeParam服务来获取ng-route框架传递的路径status参数。

    TodoController.$inject = ['$routeParams'];
    function TodoController($routeParams){
      //...
   }

现在我们额外想TodoController中注入TaskService。我们通过操作$inject数组来告知实际注入的服务,然后再在TodoController的声明签名部分增加对应的内部变量。
一定注意顺序要完全一致!

    TodoController.$inject = ['$routeParams','TaskService'];
    function TodoController($routeParams,TaskService){
    //...
    }

2.3 绑定Service中的tasks数组

原有的任务的列表绑定于controller中的tasks变量,现在我们调整赋值逻辑,让controller中的vm.tasks获取来自于TaskService中的tasks变量。这样便可以让不同的Controller都可以操作同一份数据副本。

        function init(){
            // var task = [...];
            vm.status = $routeParams.status||"";
            // vm.tasks = _filterDataByStatus(tasks,vm.status);
            vm.tasks = _filterDataByStatus(TaskService.tasks,vm.status)
        }
Paste_Image.png

2.4 测试应用

我们就先调整着几行代码,然后刷新页面试一下。
我们添加一个新的任务,然后再使用底部状态过滤便签进行过滤。


添加新的任务

切换状态标签,我们新添加的任务也在任务清单中。


显示待完成的清单

虽然我们就只改了几行代码,但整个应用已经变得与预期的一样的。
但是现在还有一个小小的bug,便是当我们在completed页面对一个“已经被处理过的”tasks数组做操作的时候,newTask虽然是一个completed:false的状态但是还是会被增加到当前视图绑定的vm.tasks中。
如果继续使用Controller的数据过滤方案,则需要每次在不同页面对vm.tasks数组操作后重新激活过滤的条件的数组。
所以我们将使用Angular提供的Filter过滤器方案做前端视图的过滤。

3. Filter过滤器

3.1 什么是Filter

Angular中的Filter是一种对数据进行加工的组件,在View中通过"|"运算符进行操作,其本质是对左侧的输出结果做进一步的操作。
如对vm.tasks | filter:{completed:true}的表达式便是对vm.tasks中使用属性的过滤器,只显示 tasks中completed属性为true的条目。

3.2 根据状态对vm.tasks做过滤显示

首先,既然是做View层的过滤器方案,便需要将我们当前的路由获取的status与status的过滤器具体值传递给前端。我们调整下之前的_filterDataByStatus函数,将其调整会根据不同的status返回不同的过滤条件对象

        function _filterDataByStatus(status){
            if (status === 'active'){
                return {completed:false};
            }else if (status === 'completed'){
                return {completed:true};
            }else{
                return {};
            }
        }

顺便将函数的名称变为更加贴近实际操作的_filterByStatus

        function _filterByStatus(status){
        //....
        }

然后,调整init函数,将获取的statusFilter传递给View层

        function init(){
            vm.status =  $routeParams.status||"";
            vm.statusFilter =_filterByStatus(vm.status);
            vm.tasks = TaskService.tasks;
        }

我们新增了一个vm.statusFilter用于保存对应的过滤条件:

  • 显示所有数据,则值为{}即不做任何过滤;
  • 显示对应状态的数据,则值为{completed: true/false}则会对vm.tasks对应的completed字段做出对应的过滤操作。

最后,我们调整View中的vm,tasks的ng-repeat指令,在其之后增加过滤器操作,在其做出渲染前对数据几个做出加工即只显示对应过滤条件的数据。

<ul id="todo-list">
            {{}}
            <li ng-repeat="task in vm.tasks|filter:vm.statusFilter " ng-class="{completed: task.completed, editing: task === vm.editedTask}">
            //....
            </li>
</ul>

到这里,Todos应用的基本功能都完成了。可是唯一的问题是,现在的Todos一旦进行了刷新操作,则数据就会全部消失了,缺少了所谓“持久化”的操作。
下一章节中,我们将使用Html5中本地存储接口将任务数据存储到浏览器的localStorage中,使其就算关闭了浏览器,数据也不会消失的效果。

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,639评论 18 139
  • AngularJS是什么?AngularJs(后面就简称ng了)是一个用于设计动态web应用的结构框架。首先,它是...
    200813阅读 1,592评论 0 3
  • 序言 本章将通过angular默认的ng-route组件来复用视图与前端的路由控制。 上一个章节的完成图 本章目的...
    AkiraPan阅读 325评论 0 2
  • 1、angularjs的几大特性是什么? 双向数据绑定、依赖注入、模板、指令、MVC/MVVM 2、列举几种常见的...
    2e9a10d418ab阅读 1,265评论 0 10
  • Angular面试题 一、ng-show/ng-hide与ng-if的区别? 第一点区别是,ng-if在后面表达式...
    w_zhuan阅读 5,521评论 0 26