AngularJS

第1章 介绍

js尽量写到界面的最后

1.0 介绍
    AngularJS是一款由Google公司开发维护的前端MVC框架,其克服了HTML在构建应用上的诸多不足,从而降低了开发成本提升了开发效率。
1.1 特点
    AngularJS与我们之前学习的jQuery是有一定的区别的,jQuery更准确来说只一个类库(类库指的是一系列函数的集合)以DOM做为驱动(核心),
    而AngularJS则一个框架(诸多类库的集合)以数据和逻辑做为驱动(核心)。
    框架对开发的流程和模式做了约束,开发者遵照约束进行开发,更注重的实际的业务逻辑。
    AngularJS有着诸多特性,最为核心的是:模块化、双向数据绑定、语义化标签、依赖注入等。
    与之类似的框架还有BackBone、KnockoutJS、Vue、React等。
1.2 下载
    1、通过AngularJS官网下载,不过由于国内特殊的国情,需要翻墙才能访问。
        https://angularjs.org/
    2、通过npm下载,npm install angular
    3、通过bower下载,bower install angular
1.3 体验AngularJS
    <!-- 引入angularJS框架 -->
    <script src="./libs/angular.min.js"></script>
    <!--  -->
    <body ng-app>
        <!-- 把输入框中的内容放到 msg变量中 -->
        <input type="text" ng-model="msg">
        <!-- 动态的把msg变量的值拿出展示,只要msg的值发生变化就会动态获取 -->
        <h2>{{msg}}</h2>
    </body>
1.4 MVC
    MVC是一种开发模式,由模型(Model)、视图(View)、控制器(Controller)3部分构成,采用这种开发模式为合理组织代码提供了方便、降低了代码间的耦合度、功能结构清晰可见。
        模型(Model)一般用来处理数据(读取/设置),一般指操作数据库。
        视图(View)一般用来展示数据,比如通过HTML展示。
        控制器(Controller)一般用做连接模型和视图的桥梁。
    通过ThinkPHP来演示后端MVC的执行流程,其重点在于理解。
    MVC更多应用在后端开发程序里,后被引入到前端开发中,由于受到前端技术的限制便有了一些细节的调整,进而出现了很多MVC的衍生版(子集)如MVVM、MVW、MVP、MV*等。
    注:做为初学可以不必过于在意这些概念。

第2章 模块化

2.0 介绍
    使用AngularJS构建应用(App)时是以模块化(Module)的方式组织的,即将整个应用划分成若干模块,每个模块都有各自的职责,最终组合成一个整体。
    采用模块化的组织方式,可以最大程度的实现代码的复用,可以像搭积木一样进行开发。
2.1 定义应用View视图
    <!-- 指定模块 -->
    <div class="box" ng-app="App">
        !-- 指定控制器 -->
        <div ng-controller="DemoContoller"></div>
    </div>
    
    一定要先有模块,后有控制器
    一个页可以有多个模块,但是不能互想嵌套
    一般只会有一个
    
2.2 定义模块
    AngularJS提供了一个全局对象angular,在此全局对象下存在若干的方法,其中angular.module()方法用来定义一个模块。
    注:应用(App)其本质也是一个模块(一个比较大的模块)。
    代码:
        var App = angular.module('App', []);
            App就是新创建的模块,这个模块又是一个对象
            在此对象下又有N多方法,可以实现具体业务逻辑
            App 就是div中的ng-app中的内容
    
2.3 定义控制器
    控制器(Controller)作为连接模型(Model)和视图(View)的桥梁存在,所以当我们定义好了控制器以后也就定义好了模型和视图。
    模型(Model)数据是要展示到视图(View)上的,所以需要将控制器(Controller)关联到视图(View)上
    通过为HTML标签添加ng-controller属性并赋值相应的控制器(Controller)的名称,就确立了关联关系。
    代码:
        App.controller('DemoContoller', ['$scope', function ($scope) {}]);
            $scope 是一个空对象{},此对象就是Model
            DemoContoller 是div中的ng-controller属性的值

第3章 指令

3.0 说明
    HTML在构建应用(App)时存在诸多不足之处,AngularJS通过扩展一系列的HTML属性或标签来弥补这些缺陷,
    所谓指令就是AngularJS自定义的HTML属性或标签,这些指令都是以ng-做为前缀的,例如ng-app、ng-controller、ng-repeat等。
3.1 内置指令(view(html)中的ng指令)
    ng-app="Demo" 模块名称,指定应用根元素,至少有一个元素指定了此属性。
    ng-controller="StarsController" 控制器名称,指定控制器
    ng-init="name='shuaige';age=10" 用于初始化属性值,相当于$scope.name='shuaige';$scope.age=10;
    ng-repeat="str in arr" 遍历数组
    ng-switch-when="java" 用来判断某个变量中的值是否是java,跟ng-repeat一起使用可以判断当前对象的值是否是Java。
    ng-switch on 选择判断结构。跟ng-switch-when一起使用。<div ng-switch on="type"><div ng-switch-when="java">当type的值是Java时显示</div></div>
    ng-show="0" 用来显示或隐藏内容(元素实际存在),1显示,0隐藏
    ng-hide="1" 用来显示或隐藏内容(元素实际存在),1隐藏,0显示
    ng-if="1" 用来控制元素是否存在(元素可能不存在),1存在,0不存在
    ng-src="{{path}}" 用来添加src路径。在angular中要使用此指令添加路径,否则会有页面图片找不到到页面从新刷新的小问题。
    ng-href="{{path}}" 用来添加href路径
    ng-class="{red: true}" 用来控制类名。用来添加css中的class样式
        ng-class="{active: type == 'local'}" 用$scope中的type变量的值来控制active这个类是否添加还是移除。
    ng-include 引入模板 <body ng-app="App"><div ng-include="'./header.html'"></div><script>var App = angular.module('App', []);</script></body>
    ng-disabled="1" 表单禁用。用来控制元素是否可用,1不可用,0可用
    ng-readonly="1" 表单只读。用来控制元素是否只读,1只读,0可修改
    ng-checked="1" 单/复选框表单选中。用来控制多选框元素是否选中,1选中,0未选中
    ng-selected="1" 下拉框表单选中。用来控制下拉框元素是否选中,1选中,0未选中
    ng-bind="name" 用来绑定model模型($scope)中的name属性。可以获取name属性的值。跟{{name}}效果一样
    ng-cloak 用来解决页面刷新速度过快导致的闪动问题(可以看到{{}}这个括号),用法<li ng-cloak>{{name}}{{age}}</li>
    ng-bind-template 绑定多个数据,用法<li ng-bind-template="{{name}}{{age}}"></li>
    ng-model="msg" 绑定数据。要实现数据从视图向模型传递需要借助于表单元素,相当于$scope.msg
    ng-click="show()" 单击。用来调用$scope.show = function(){} 这个show方法
        ng-click="show('local')" 调用$scope.switch = function(type){} 这个方法。type就等于local。
    ng-dblclick="double()" 双击。用来调用$scope.double = function(){}这个double方法
    ng-blur="blur()" 失去焦点。用来调用$scope.blur = function(){} 这个blur方法
    ng-mouseout="mouseout()" 鼠标移除事件。用来调用$scope.mouseout = function(){}这个mouseout方法


<h1 ng-cloak>{{name}}</h1>
<h1 ng-bind="name"></h1>
<h1 ng-bind-template="{{name}}"></h1>
<h1>{{aa}}</h1>
<h1>{{bb}}</h1>
<ul>
    <li ng-repeat="str in arrs">{{str}}</li>
</ul>
<ul>
    <li ng-repeat="student in students">{{student.name}}-{{student.age}}</li>
</ul>
<input type="text" ng-if="type"/>
<input type="text" ng-show="type"/>
<input type="text" ng-hide="type"/>
<img ng-src="./img/01.jpg">
<a ng-href="{{path}}">atest</a><br/><br/><br/><br/>
<div ng-include="'./views/one.html'"></div>
<input type="text" ng-disabled="type" />
<input type="text" ng-readonly="type" />
<input type="checkbox" name="sex" value="男" ng-checked="type" />男
<input type="checkbox" name="sex" value="女" ng-checked="type"/>女
<input type="radio" name="gendar" value="男"/>男
<input type="radio" name="gendar" value="女"  ng-checked="type"/>女
<select>
    <option>中国</option>
    <option ng-selected="type">美国</option>
</select>
<input type="text" ng-model="msg"/>
<h1 ng-bind="msg" ng-class="{red:type}"></h1>

<ul ng-switch="msg">
    <li ng-switch-when="shuaige">shuaige</li>
    <li ng-switch-when="shuaige2">shuaige2</li>
</ul>

<ul ng-repeat="str in arrs" ng-switch="str">
    <li ng-switch-when="java">java技术</li>
    <li ng-switch-when="css">css技术</li>
    <li ng-switch-when="js">js技术</li>
</ul>
<button ng-click="click()">click</button>
<h1 ng-dblclick="dbclick1()">dbclick</h1>
<input type="text" ng-blur="blur()" />
<p ng-mouseout="mouseout()">这是一块区域</p>




3.2 自定义指令
    AngularJS允许根据实际业务需要自定义指令,通过angular全局对象下的directive方法实现。
        // E element 是一个元素  <tag></tag>
        // A attribute 是一个属性 <div tag></div>
        // C class 是一个类 <div class="tag"></div>
        // M mark replace 必须为true  <!-- directive:tag -->备注
    代码1:
        <div tag></div>
        var App = angular.module('App', []);
        App.directive('tag', function () {
            return {
                    restrict: 'EA',
                    template: '<p>hello shuaige</p>'
                }
        });
        界面会显示hello shuaige
    代码2:要用localhost访问,否则会显示报错
        <div tag></div>
        var App = angular.module('App', []);
        App.directive('tag', function () {
            return {
                    restrict: 'EA',
                    templateUrl: './lists.html'
                }
        });
        界面会显示lists.html中的内容
    代码3:要用localhost访问,否则会显示报错
        <div tag></div>
        var App = angular.module('App', []);
        App.directive('tag', function () {
            return {
                    restrict: 'EA',
                    templateUrl: './lists.html',
                    replace: true
                }
        });
        界面会显示lists.html中的内容,但是外层的标签会去掉。
3.3 具体使用
    $scope 是一个空对象{},此对象就是Model
        //普通字符串
        $scope.name = '李振杰';
            //view中取值 {{name}}
        //数组
        $scope.courses = ['MVC','指令','模块化']
            //view中取值 <li ng-repeat="(key, course) in courses">第{{key}}天:{{course}}</li>
                ng-repeat="(key, course) in courses"是就遍历courses数组
                    (key,course) : key是下标,course是游标代表数组当前位置的值。
                    取值key是{{key}}
                    取值course是{{course}}
        //数组
        $scope.stars = [{name: '刘德华', sex: '男', age: 62},{name: '王力宏', sex: '男', age: 40}];
            //view中取值 
                <tr ng-repeat="star in stars">
                    <td>{{star.name}}</td>
                    <td>{{star.sex}}</td>
                    <td>{{star.age}}</td>
                </tr>   
                ng-repeat="star in stars" 是遍历 stars 数组,数组当前位置的变量名为star
                {{star.name}},{{star.sex}},{{star.age}} 是获取当前变量中的name,sex,age属性的值。

        //数组
        $scope.names = [];
        //往数组中添加一条数据
        $scope.names.push('shuaige');
        //从数组中取出一条数据,并且从数组中移除
        $scope.names.splice(start,length);

第4章 数据绑定

4.0 说明
    AngularJS是以数据做为驱动的MVC框架,所有模型(Model)里的数据经由控制器(Controller)展示到视图(View)中。
    所谓数据绑定指的就是将模型(Model)中的数据与相应的视图(View)进行关联,分为单向绑定和双向绑定两种方式。
4.1 单向绑定
    单向数据绑定是指将模型(Model)数据,按着写好的视图(View)模板生成HTML标签,然后追加到DOM中显示.

4.2 双向绑定
    双向绑定则可以实现模型(Model)数据和视图(View)模板的双向传递
    <input type="text" ng-model="msg">

4.3 相关指令
    在AngularJS中通过“{{}}”和ng-bind指令来实现模型(Model)数据向视图模板(View)的绑定,
    模型数据通过一个内置服务$scope来提供,这个$scope是一个空对象,
    通过为这个对象添加属性或者方法便可以在相应的视图(View)模板里被访问。
        注:“{{}}”是ng-bind的简写形式,其区别在于通过“{{}}”绑定数据时会有“闪烁”现象,
            添加ng-cloak也可以解决“闪烁”现象,通过ng-bind-template可以绑定多个数据。

    ng-model 实现视图(View)模板向模型(Model)数据的绑定。
    ng-init 初始化模型(Model)也就是$scope
    
    AngularJS对事件也进行了扩展,无需显式的获取DOM元素便可以添加事件,易用性变的更强。
    通过在原有事件名称基础上添加ng-做为前缀,然后以属性的形式添加到相应的HTML标签上即可。
    如ng-click、ng-dblclick、ng-blur等。

    ng-repeat可以将数组或对象数据迭代到视图模板中,ng-switch、on、ng-switch-when可以对数据进行筛选。

第5章 作用域

5.0 说明
    通常AngularJS中应用(App)是由若干个视图(View)组合成而成的,而视图(View)又都是HTML元素,并且HTML元素是可以互相嵌套的,
    另一方面视图都隶属于某个控制器(Controller),进而控制器之间也必然会产生嵌套关系。
    每个控制器(Controller)又都对应一个模型(Model)也就是$scope对象,不同层级控制器(Controller)下的$scope便产生了作用域。
5.1 根作用域
    一个AngularJS的应用(App)在启动时会自动创建一个根作用域$rootScope,这个根作用域在整个应用范围(ng-app所在标签以内)都是可以被访问到的。
    <div ng-app="App" ng-init="name='shuaige';age=20"></div>
    ng-app 所在元素就是应用的根元素
    ng-init 为$rootScope添加数据
5.2 子作用域
    通过ng-controller指令可以创建一个子作用域,新建的作用域可以访问其父作用域的数据。
    $scope.name = '帅哥';

第6章 过滤器

6.0 说明
    在AngularJS中使用过滤器格式化展示数据,在“{{}}”中使用“|”来调用过滤器,使用“:”传递参数。
6.1 内置过滤器
    currency将数值格式化为货币格式
        {{price|currency:'¥'}}
        $scope.price = 11.11;
    date日期格式化,年(y)、月(M)、日(d)、星期(EEEE/EEE)、时(H/h)、分(m)、秒(s)、毫秒(.sss),也可以组合到一起使用。
        {{now|date:'yyyy/MM/dd hh:mm:ss'}}
        $scope.now = new Date;
    filter在给定数组中选择满足条件的一个子集,并返回一个新数组,其条件可以是一个字符串、对象、函数
        {{items|filter:'s'}}
        $scope.items = ['html', 'css', 'js']; 从中挑出包含s的数据显示在界面
        {{students|filter:{age: 16} }}
        $scope.students = [{name: '小红', age: 16},{name: '小明', age: 16},{name: '小米', age: 10}]; 从中挑出age等于16的数据
    json将Javascrip对象转成JSON字符串
        {{students|json}}
        $scope.students = [{name: '小红', age: 16},{name: '小明', age: 16},{name: '小米', age: 10}]; 会显示json字符串
    limitTo取出字符串或数组的前(正数)几位或后(负数)几位
        {{items|limitTo:-1}}
        $scope.items = ['html', 'css', 'js']; 页面只显示js
    lowercase将文本转换成小写格式
        {{str|lowercase}}
        $scope.str = 'hello Angular'; 小写展示数据
    uppercase将文本转换成大写格式
        {{str|uppercase|limitTo:3}}
        $scope.str = 'hello Angular'; 从开始取3位字符,大写展示数据
    number数字格式化,可控制小位位数
        {{num|number:2}}
        $scope.num = '10.2365'; 取小数点两位显示
    orderBy对数组进行排序,第2个参数可控制方向
        {{items|orderBy: '':true}}
        $scope.items = ['html', 'ass', 'js']; 排序,false按照自然顺序正向排序,true按照自然顺序逆向排序
        {{students|orderBy: 'age': false}}
        $scope.students = [{name: '小红', age: 16},{name: '小明', age: 9},{name: '小米', age: 10}];//按照age排序。false按照自然顺序正向排序,true按照自然顺序逆向排序
6.2 自定义过滤器
    除了使用AngularJS内建过滤器外,还可以根业务需要自定义过滤器,通过模块对象实例提供的filter方法自定义过滤器。
        //自定义过滤器
        App.filter('shuaige', function () {
            return function (input) {
                return 'hello ' + input;
            }
        });
        //使用过滤器
        {{info|shuaige}}
        $scope.info = 'shuaige';

第7章 依赖注入

7.0 说明
    AngularJS采用模块化的方式组织代码,将一些通用逻辑封装成一个对象或函数,实现最大程度的复用,这导致了使用者和被使用者之间存在依赖关系。 
    所谓依赖注入是指在运行时自动查找依赖关系,然后将查找到依赖传递给使用者的一种机制。
    常见的AngularJS内置服务有$http、$location、$timeout、$rootScope等
7.1 推断式注入
    没有明确声明依赖,AngularJS会将函数参数名称当成是依赖的名称。
    这种方式会带来一个问题,当代码经过压缩后函数的参数被压缩,这样便会造成依赖无法找到。
    AngularJS 内置一些具有特殊功能的“模块”
    开发者在开发的时候可以直接使用这些“模块”
    <div class="box" ng-controller="DemoController"></div>
    App.controller('DemoController', function ($scope, $http) {});
7.2 行内注入
    以数组形式明确声明依赖,数组元素都是包含依赖名称的字符串,数组最后一个元素是依赖注入的目标函数。
    推荐使用这种方式声明依赖
    <div ng-controller="DemoController"></div>
    App.controller('DemoController', ['$scope', '$http', function (abc, bcd) {}]);

第8章 服务

8.0 说明
    服务是一个对象或函数,对外提供特定的功能
8.1 内建服务
    1.$location是对原生Javascript中location对象属性和方法的封装
        App.controller('DemoController', ['$scope', '$location', function($scope, $location) {
            $scope.absUrl = $location.absUrl();//绝对地址
            $scope.url = $location.url();//锚点#后面的内容
            $scope.host = $location.host();//地址
            $scope.search = $location.search();
            $scope.hash = $location.hash();
            $scope.protocol = $location.protocol();//协议
            $scope.port = $location.port();//端口号
        }]);
    2.$timeout&$interval对原生Javascript中的setTimeout和setInterval进行了封装。
        App.controller('DemoController', ['$scope', '$timeout', '$interval',function ($scope, $timeout, $interval) {
            $timeout(function () {
                $scope.msg = '执行了';
            }, 3000);
            var timer = $interval(function () {
                $scope.now = new Date;
            }, 1000);
            $scope.stop = function () {
                $interval.cancel(timer);
            }
        }]);
    3.$filter在控制器中格式化数据
        App.controller('DemoController', ['$scope', '$filter', function ($scope, $filter) {
            // $filter是九种过滤器中任何一个
            $scope.price = 11.11;
            var currency = $filter('currency');
            $scope.price = currency($scope.price);
            $scope.str = 'hello angular';
            var uppercase = $filter('uppercase');
            $scope.str = uppercase($scope.str);
            $scope.str1 = $filter('limitTo')($scope.str, 2);
        }]);
    4.$log打印调试信息
        App.controller('DemoController', ['$log', function ($log) {
            $log.info('普通信息');
            $log.warn('警告信息');
            $log.error('错误信息');
            $log.log('打印信息');
            $log.debug('调试信息');
        }]);
    5.$http用于向服务端发起异步请求
        $http 本质是对XMLHttpRequest对象封装
        App.controller('DemoController', ['$scope', '$http', '$log', function ($scope, $http, $log) {
            $http({
                url: 'example.php',
                method: 'post',
                headers: {
                    'Content-Type': 'application/x-www-form-urlencoded'
                },
                params: { // get 参数
                    name: 'itcast',
                    sex: '男'
                },
                data: { // post 传参
                    age: 10
                }
            }).success(function (info) {
                $log.info(info);// info 就是返回的数据
            });
        }]);
        App.controller('StarController', ['$http', '$scope', '$log', function ($http, $scope, $log) {
            $scope.getData = function () {
                $http({
                    url: './stars.php',
                    method: 'get'
                }).success(function (info) {
                    $scope.stars = info;
                });
            }
        }]);
        $http跨域
        App.controller('DemoController', ['$http', '$scope', function ($http, $scope) {
            $http({
                url: 'jsonp.php?a=JSON_CALLBACK',
                method: 'jsonp' // 采用JSONP方式
            }).success(function (info) {
                console.log(info);
            });
        }]);
        请求一下百度天气http://api.map.baidu.com/telematics/v3/weather?callback=JSON_CALLBACK
        Weather.controller('WeatherController', ['$scope', '$http', function ($scope, $http) {
            $http({
                url: 'http://api.map.baidu.com/telematics/v3/weather',
                method: 'jsonp',
                params: {
                    location: '北京',
                    output: 'json',
                    ak: '0A5bc3c4fb543c8f9bc54b77bc155724',
                    callback: 'JSON_CALLBACK'
                }
            }).success(function (info) {
                $scope.weatherData = info.results[0].weather_data;
            });


        }]);
    6.同时还支持多种快捷方式如$http.get()、$http.post()、$http.jsonp
8.2 自定义服务
    通过上面例子得知,所谓服务是将一些通用性的功能逻辑进行封装方便使用,AngularJS允许将自定义服务。
    在介绍服务时曾提到服务本质就是一个对象或函数,所以自定义服务就是要返回一个对象或函数以供使用。
    1.factory方法
        //定义一个名叫showTime的服务
        App.factory('showTime', ['$filter', function ($filter) {
            var now = new Date();
            var date = $filter('date');
            return {
                now: date(now, 'y-M-d H:m:s')
            }
        }]);
        //使用服务
        App.controller('DemoController', ['$scope', 'showTime', function($scope, showTime) {
            $scope.now = showTime.now;
        }]);
        App.controller('DemoController', ['$scope', '$http', 'showTime', function ($scope, $http, showTime) {}]);
    2.service方法
        App.service('showTime', ['$filter', function($filter) {
            var now = new Date();
            var date = $filter('date');
            this.now = date(now, 'y-M-d H:mm:ss');
        }]);

        App.controller('DemoController', ['$scope', 'showTime', function($scope, showTime) {
            $scope.now = showTime.now;
        }]);
    3.value方法定义常量
        //本质上一个服务
        //从表现形式上是一个常量
        //常量就是不变的值与变对量相对应
        App.value('author', 'shuaige');
        App.controller('DemoController', ['$scope', 'author', function($scope, author) {
            $scope.author = author;
        }]);

第9章 模块加载

9.0 说明
    AngularJS模块可以在被加载和执行之前对其自身进行配置。我们可以在应用的加载阶段配置不同的逻辑。
    执行流程
        1.启动阶段
            开始->浏览器解析DOM树->遇到angular.js停止解析,开始执行脚本->angular监听到DOMContent Loaded事件->启动angular应用
        2.初始化阶段
            查找模块依赖->寻找ng-app指令->初始化必要组件($injector/$complie/$rootScope)->配置和运行->开始解析DOM树
        3.编译链接阶段
            $complie遍历DOM树,搜集指令->执行每个指令的complie函数->处理DOM转换,编译模板->调用链接函数,生成实时视图
        4.运行阶段
            等待事件触发,执行$digest循环->检测到变化,调用$watch函数->再次执行$digest循环,直到没有变化->结束
9.1 配置块
    通过config方法实现对模块的配置,AngularJS中的服务大部分都对应一个“provider”,用来执行与对应服务相同的功能或对其进行配置。
    比如$log、$http、$location都是内置服务,相对应的“provider”分别是$logProvider、$httpProvider、$locationPorvider。
    
    //配置一个服务
    App.config('$logProvider',function($logProvider){});
    //配置多个服务
    App.config(['$logProvider', '$filterProvider', function ($logProvider, $filterProvider) {
        // 禁用debug功能 $log.debug();
        $logProvider.debugEnabled(false);

        // 默认9个过滤器,通过配置可以新增一些过滤器
        $filterProvider.register('capitalize', function () {
            // 新增一个过滤器
            return function (input) {
                return input[0].toUpperCase() + input.slice(1);
            }
        });
    }]);
    
    
9.2 运行块
    服务也是模块形式存在的对且对外提供特定功能,前面学习中都是将服务做为依赖注入进去的,然后再进行调用,
    除了这种方式外我们也可以直接运行相应的服务模块,AngularJS提供了run方法来实现。
    不但如此,run方法还是最先执行的,利用这个特点我们可以将一些需要优先执行的功能通过run方法来运行,
    比如验证用户是否登录,未登录则不允许进行任何其它操作。
    注:此知识点意在了解AngularJS的加载机制。
    
    App.run(['$http',function ($http) {
        // 直接调用$http
        $http({
            url: 'example.php',
            method: 'get'
        });
    }]);

    App.run(['$http', '$rootScope', function ($http, $rootScope) {
        // 直接调用$http
        $http({
            url: 'example.php',
            method: 'get'
        });
        // 根作用域
        $rootScope.name = '帅哥';
    }]);

第10章 路由

10.0  说明
    一个应用是由若个视图组合而成的,根据不同的业务逻辑展示给用户不同的视图,路由则是实现这一功能的关键。
    angularjs默认支持restful接口
10.1  SPA   
    SPA(Single Page Application)指的是通单一页面展示所有功能,通过Ajax动态获取数据然后进行实时渲染,
    结合CSS3动画模仿原生App交互,然后再进行打包(使用工具把Web应用包一个壳,这个壳本质上是浏览器)变成一个“原生”应用。
    在PC端也有广泛的应用,通常情况下使用Ajax异步请求数据,然后实现内容局部刷新,局部刷新的本质是动态生成DOM,
    新生成的DOM元素并没有真实存在于文档中,所以当再次刷新页面时新添加的DOM元素会“丢失”,通过单页面应可以很好的解决这个问题。
10.2  路由
    在后端开发中通过URL地址可以实现页面(视图)的切换,但是AngularJS是一个纯前端MVC框架,
    在开发单页面应用时,所有功能都在同一页面完成,所以无需切换URL地址(即不允许产生跳转),
    但Web应用中又经常通过链接(a标签)来更新页面(视图),当点击链接时还要阻止其向服务器发起请求,
    通过锚点(页内跳转)可以实现这一点。
    
    1.实现单页面应用需要具备:
        a、只有一页面
        b、链接使用锚点
        通过上面的例子发现在单一页面中可以能过hashchange事件监听到锚点的变化,进而可以实现为不同的锚点准不同的视图,单页面应用就是基于这一原理实现的。
        AngularJS对这一实现原理进行了封装,将锚点的变化封装成路由(Route),这是与后端路由的根本区别。
        在1.2版前路由功能是包含在AngularJS核心代码当中,之后的版本将路由功能独立成一个模块,下载angular-route.js
        
    spa页面应用特点
        把多个项目链接放到同一个页面,不产生页面跳转
        把若干功能集成到一个页面
        动态生成数据,通过ajax异步获取
        为了增强用户体验
        可以提升性能
        仿制手机app的交互
        锚点是前端的路由
        hashchange监听锚点变化,绑定给window对象。
        
    路由js写法
        <!-- 导航菜单  js写法 -->
        <ul>
            <li class="active">
                <a href="#index">Index</a>
            </li>
            <li>
                <a href="#introduce">Introduce</a>
            </li>
            <li>
                <a href="#contact">Contact Us</a>
            </li>
        </ul>
        <!-- 内容 -->
        <div class="content">
            Index Page
        </div>
        // 监听锚点变化然后发送请求
        // hashchange事件可以监听锚点变化
        window.addEventListener('hashchange', function () {
            // 获取锚点
            var hash = location.hash;
            // 处理#
            hash = hash.slice(1);
            // 实例对象
            var xhr = new XMLHttpRequest;
            // 将锚点做为参数传递给服务端进处理
            xhr.open('get', '10-01.php?hash=' + hash);
            xhr.send(null);
            xhr.onreadystatechange = function () {
                if(xhr.readyState == 4 && xhr.status == 200) {
                    var result = xhr.responseText;
                    // 将返回结果添加到页面
                    document.querySelector('.content').innerHTML = result;
                }
            }
        });
        
    
    
        
10.2.1  使用
    1.引入angular-route.js
    2.实例化模块(App)时,当成依赖传进去(模块名称叫ngRoute)
    3.配置路由模块
    4.布局模板 - 通过ng-view指令布局模板,路由匹配的视图会被加载渲染到些区域
    
    路由angularjs写法
        <!-- 导航菜单 -->
        <ul>
            <li class="active">
                <a href="#/index">Index</a>
            </li>
            <li>
                <a href="#/introduce">Introduce</a>
            </li>
            <li>
                <a href="#/contact">Contact Us</a>
            </li>
            <li>
                <a href="#/list">List</a>
            </li>
        </ul>
        <!-- 内容 -->
        <div class="content">
            <!-- 占位符 -->
            <div ng-view></div>
        </div>
        <!-- AngularJS核心框架 -->
        <script src="./libs/angular.min.js"></script>
        <!-- 路由模块理解成插件 -->
        <script src="./libs/angular-route.js"></script>
        <script>
            // 依赖ngRoute模块
            var App = angular.module('App', ['ngRoute']);
            // 需要对路由模块进行配置,使其正常工作
            App.config(['$routeProvider', function ($routeProvider) {
                $routeProvider.when('/index', {
                    // template: '<h1>index Pages!</h1>',
                    templateUrl: './abc.html'
                })
                .when('/introduce', {
                    template: '<h1>introduce Pages!</h1>'
                })
                .when('/contact', {
                    // template: '<h1>contact US Pages!</h1>',
                    templateUrl: './contact.html',
                    controller: 'ContactController' // 定义控制器
                })
                .when('/list', {
                    templateUrl: './list.html', // 视图模板
                    controller: 'ListController' // 定义控制器 这个控制器就是控制list.html的
                })
                .otherwise({
                    redirectTo: '/index'
                });
            }]);
            // 列表控制器
            App.controller('ListController', ['$scope', '$http', function ($scope, $http) {
                // 模型数据
                $http({
                    url: '10-02.php',
                }).success(function (info) {
                    $scope.items = info;
                });
            }]);
            App.controller('ContactController', ['$scope', '$http', function ($scope, $http) {
                $http({
                    url: 'contact.php'
                }).success(function (info) {
                    $scope.content = info;
                });
            }]);
        </script>
    
10.2.1  路由参数
    1.提供两个方法匹配路由,分别是when和otherwise,when方法需要两个参数,otherwise方法做为when方法的补充只需要一个参数,其中when方法可以被多次调用。
    2.第1个参数是一个字符串,代表当前URL中的hash值
    3.第2个参数是一个对象,配置当前路由的参数,如视图、控制器等
        a、template 字符串形式的视图模板
        b、templateUrl 引入外部视图模板
        c、controller 视图模板所属的控制器
        d、redirectTo跳转到其它路由
    4.获取参数,在控制中注入$routeParams可以获取传递的参数
        a)url问号传参数
        b)占位符传参
    
    代码:
        <!-- AngularJS核心框架 -->
        <script src="./libs/angular.min.js"></script>
        <!-- 路由模块理解成插件 -->
        <script src="./libs/angular-route.js"></script>
        <script>
            // 依赖ngRoute模块
            var App = angular.module('App', ['ngRoute']);
            // 需要对路由模块进行配置,使其正常工作
            App.config(['$routeProvider', function ($routeProvider) {
                $routeProvider.when('/index/:id/:page/:p', {
                    templateUrl: 'abc.html',
                    controller: 'IndexController'
                })
                .otherwise({
                    redirectTo: '/index'
                });
            }]);
            // 提供了一个专门负责获取参数的一个服务$routeParams
            App.controller('IndexController', ['$scope', '$http', '$routeParams', function ($scope, $http, $routeParams) {
                $scope.content = '练习路由功能';
                console.log($routeParams);
            }]);
        </script>
        
        <!--
            后端会说明需要什么样的参数
            需要一个类型 ID:1,2,3,4
            1表示流行 2表示华语 3表示欧美 4表示日韩
        -->
        <nav>
            <a href="#/1"></a>
            <a href="#/2"></a>
            <a href="#/3"></a>
            <a href="#/4"></a>
        </nav>

        Music.config(['$routeProvider',function($routeProvider){
            $routeProvider.when("/:id",{
                templateUrl:'./views/list.html',
                controller:'ListController'
            });
        }]);
        Music.controller('ListController',['$scope','$http','$routeParams',function($scope,$http,$routeParams){
            var id = $routeParams.id;//获取地址上的参数
            $http({
                url:'./api/list.php',
                method:'get',
                params:{type:id}//将获取到的参数传给后端
            }).success(function(info){
                console.log(info);
            });
        }]);

第11章 其它

11.1    jQuery
    在没有引入jQuery的前提下AngularJS实现了简版的jQuery Lite,通过angular.element不能选择元素,但可以将一个DOM元素转成jQuery对象,
    如果引提前引入了jQuery则angular.element则完全等于jQuery。
    
    angular.element() 方法可以将一个原生DOM对象转成jquery对象
    但是angularJS 只是实现了jquery对象部分方法
    // 原生DOM对象
    var box = document.querySelector('.box');
    var btn = document.querySelector('button');
    // 转成jQuery对象
    box = angular.element(box);
    btn = angular.element(btn);
    btn.on('click', function () {
        box.css('color', 'red');
    });
    
    
11.2    bower
    基于NodeJS的一个静态资源管理工具,由twitter公司开发维,解决大型网站中静态资源的依赖问题。
    1、依赖NodeJS环境和git工具。
    2、npm install -g bower安装bower
    3、bower search 查找资源信息
    4、bower install  安装(下载)资源,通过#号可以指定版本号
    5、bower info 查看资源信息
    6、bower uninstall 卸载(删除)资源
    7、bower init初始化,用来记录资源信息及依赖。
    
    步骤:
        node -v 安装node.js
        npm -vv
        npm install -g brower / npm install -g brower --registry=https://registry.npm.taobao.org
        
        打开git工具/在项目目录下安装jQuery或者其它
        brower install jquery / brower install jquery#1.7.2
        brower info jquery 
        brower uninstall jquery 
        brower init / brower init -config.interactive

其它

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

推荐阅读更多精彩内容

  • 一、介绍 AngularJS是一款由Google公司开发维护的前端MVC框架,其克服了HTML在构建应用上的诸多不...
    福尔摩鸡阅读 752评论 0 2
  • 通过AngularJS仿豆瓣一刻的案例:https://github.com/zhongxiaolian/doub...
    中小恋阅读 1,753评论 1 21
  • 1. 介绍 AngularJS是一款由Google公司开发维护的前端MVC框架,其克服了HTML在构建应用上的诸多...
    崔皓翔阅读 835评论 0 5
  • AngularJS AngularJS概述 介绍 简称:ng Angular是一个MVC框架 其他前端框架: Vu...
    我爱开发阅读 2,335评论 0 8
  • 收花,是女孩最幸福的时刻,正因为幸福,女孩们百收不厌倦,看着美美的花儿,体会送花君那份心意,美好的心情可以持续好几...
    田舍君阅读 5,405评论 23 216