AngularJS项目步骤6——双向数据绑定+步骤7——XHRs与依赖注入

工作区切换到步骤6

在这一步骤中,我们将为程序添加一个特性使得可以对手机的显示排序。这还需要我们为手机数据模型中添加一些新信息作为排序的依据,并以此写出转换器的处理让数据起作用使得程序达到预期效果。

程序现在除了有一个搜索框还有一个下拉选择框,它允许选择数据排序的要求。

下面看一下我们的模板发生了上面样的变化

phone-list.template.html

<div class="container-fluid">
  <div class="row">
    <div class="col-md-2">
      <!--Sidebar content-->

      <p>
        Search:
        <input ng-model="$ctrl.query" />
      </p>

      <p>
        Sort by:
        <select ng-model="$ctrl.orderProp">
          <option value="name">Alphabetical</option>
          <option value="age">Newest</option>
        </select>
      </p>

    </div>
    <div class="col-md-10">
      <!--Body content-->

      <ul class="phones">
        <li ng-repeat="phone in $ctrl.phones | filter:$ctrl.query | orderBy:$ctrl.orderProp">
          <span>{{phone.name}}</span>
          <p>{{phone.snippet}}</p>
        </li>
      </ul>

    </div>
  </div>
</div>

首先,我们添加了一个<select>的html元素,并命名为orderProp,这样我们就有了两个排序依据选择

tutorial_04.png

然后我们链接了经过filter转换器过滤后的数据作为orderBy转换器处理的输入。orderBy转换器会依据规则对输入的数据(数组)排序,即输出一个排序后的新数据(数组形式)。

在这里,Angular在select元素和orderProp数据模型间创建了一个双向数据绑定,然后orderProp被用于orderBy转换器(过滤器)。

正如我们在步骤3中看到的,因为这样的数据绑定使得任何的数据变化都可以及时的反映到输出结果中(这里是通过下拉菜单对排序条件的改变),而且这一过程中没有特别指定复杂的DOM操作处理(没有专门写这方面的代码)就自动实现了效果,这就是Angular数据绑定的威力所在。

控制器

phone-list.component.js

'use strict';

// Register `phoneList` component, along with its associated controller and template
angular.
  module('phoneList').
  component('phoneList', {
    templateUrl: 'phone-list/phone-list.template.html',
    controller: function PhoneListController() {
      this.phones = [
        {
          name: 'Nexus S',
          snippet: 'Fast just got faster with Nexus S.',
          age: 1
        }, {
          name: 'Motorola XOOM™ with Wi-Fi',
          snippet: 'The Next, Next Generation tablet.',
          age: 2
        }, {
          name: 'MOTOROLA XOOM™',
          snippet: 'The Next, Next Generation tablet.',
          age: 3
        }
      ];

      this.orderProp = 'age';
    }
  });
  • 我们编辑了phones的数据模型,即手机数组,在这个结构对每条记录中增加了age元素项,其可用于按手机推出时间排序。

  • 我们在控制器中设置了默认的值作为排序依据,即设置了orderProp的值为 age。如果不在这里设置这个值,则orderBy转换器(过滤器)是没有初始化的,直到我们在页面下拉菜单中进行了选择为止。

现在可以好好来谈谈双向数据绑定了。注意,当浏览器加载完成程序后下拉菜单的Newest项目是被选中的,这就是因为我们在控制器中设置了orderProp的排序依据是age,所以发生了数据模型向UI的绑定,现在我们在下拉菜单选择Alphabetically项,则数据模型马上会自动更新,让手机列表的排序依据要求发生变化,这时的数据绑定方向与前面恰好相反,是UI向数据模型的。

步骤7——XHRs与依赖注入

显然,一个足够强大的程序仅仅依靠硬编码的3条手机数据设置是不行的。这里我们将利用Angular内置的叫做$http的服务功能,从web服务器获取一个巨大的数据表。我们还将使用到Angular的依赖注入(dependency injection)向PhoneListCtrl控制器提供服务。

现在有20条手机信息了,而且这些信息是通过服务器加载的。

工作区切换到步骤7

数据

在项目app/phones/phones.json文件中有一个巨大的列表,,是采用JSON格式组织的手机信息数据。文件中,它们大概是如下的形式:

[
 {
  "age": 13,
  "id": "motorola-defy-with-motoblur",
  "name": "Motorola DEFY\u2122 with MOTOBLUR\u2122",
  "snippet": "Are you ready for everything life throws your way?"
  ...
 },
...
]

控制器

这里的控制器中,我们使用了Angular的$http服务,利用一个HTTP请求,从服务器中获取到app/phones/phones.json文件数据。$http是Angular内建web程序通用服务(功能)中的一个,Angular会在程序需要时自动注入这些服务功能。

这些服务由Angular依赖注入子系统进行管理。依赖注入帮助你的程序在好的结构(例如独立的数据、控制和表现/展示)和松耦合(组件间解耦,组件之间的依赖关系不由组件自身确定,而由依赖管理子系统协调)。phone-list.component.js:

'use strict';

// Register `phoneList` component, along with its associated controller and template
angular.
  module('phoneList').
  component('phoneList', {
    templateUrl: 'phone-list/phone-list.template.html',
    controller: function PhoneListController($http) {
      var self = this;
      self.orderProp = 'age';

      $http.get('phones/phones.json').then(function(response) {
        self.phones = response.data;
      });
    }
  });

$http发起一个HTTP GET请求,其内容是请求web服务器端的phones/phones.json文件(这里的URL是相对于index.html的相对路径)。服务器端响应这个请求,提供了json文件的内容(响应其实是一个后台服务中动态处理的反馈结果,这对于浏览器或者我们的程序来说这看起来是相同的/透明的。这个教程中为了简单起便,直接是一个json数据文件了。)

$http服务得到了一个有success方法的promise object,然后我们就可以调用方法来处理异步响应和分配手机数据来构建我们控制器中作用范围中的phones数据了。注意,这里Angular自动检测了json类型响应,并分析结构化了数据。

为了使用Angular服务,你只用在控制器中需要的地方简单把调用名字作为构造函数的参数,例如:

phonecatApp.controller('PhoneListCtrl', function ($scope, $http) {...}

Angular的依赖注入管理会在控制器初始化时自动的提供声明的功能,而且依赖注入管理还自动的处理相应的层次依赖关系(通常一个服务功能还取决于其他服务功能,这些问题Angular都会自动处理)。

注意参数的名字很重要(不能随意变动),因为依赖管理会用到这些名字进行查找来解决依赖关系并进行注入。

tutorial_05.png

$前缀名的约定

你可以创建自己的服务,事实上我们在步骤11中就会这样做。作为一个名称方面的约定,Angular内置的服务,作用范围方法和一些其它的Angular的API有一个$前缀。

这样有$前缀的都被用于了Angular预定服务,为了防止名称方面的冲突,你在定义服务和数据模型时最好都不要用$作为前缀。

在检查代码时你可能会注意到有些作用范围内的数据定义是$$来开始的,这意味这这些内容是私人的(该受保护的),你不该在外部进行访问或修改。

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

推荐阅读更多精彩内容