1、基于ui-router的页面跳转传参

(1) 用ui-router定义路由,比如有两个页面,一个页面(producers.html)放置了多个producers,点击其中一个目标,页面跳转到对应的producer页面,同时将producerId这个参数传过去。

.state('producers',{

  url: '/producers',

  templateUrl: 'views/producers.html',

  controller: 'ProducersCtrl'

})

.state('producers',{

  url: '/producer/:producerId',

  templateUrl: 'views/producer.html',

  controller: 'ProducerCtrl'

})

(2)在producer.html中,定义点击事件,比如ng-click="toProducer(producerId)", 在ProducersCtrl中,定义页面跳转函数(使用ui-router的$state.go接口):

.controller('ProducersCtrl', function ($scope, $state) {
$scope.toProducer = function (producerId) {
$state.go('producer', {producerId: producerId});
};
});

(3)在ProducerCtrl中,通过ui-router的$stateParams获取参数producerId,

.controller('ProducerCtrl', function($scope, $state, $stateParams){

  var producerId = $stateParams.producerId;

});

2、基于factory的页面跳转传参

举例:有N个页面,每个页面都需要用户填选信息,最终引导用户至尾页提交,同时后一个页面要显示前面所有页面填写的信息。这时用factory传参是比较合理的选择

 .factory('myFactory', function () {
//定义参数对象
var myObject = {}; /**
* 定义传递数据的setter函数
* @param {type} xxx
* @returns {*}
* @private
*/
var _setter = function (data) {
myObject = data;
}; /**
* 定义获取数据的getter函数
* @param {type} xxx
* @returns {*}
* @private
*/
var _getter = function () {
return myObject;
}; // Public APIs
// 在controller中通过调setter()和getter()方法可实现提交或获取参数的功能
return {
setter: _setter,
getter: _getter
};
});

3、基于factory和$rootScope.$broadcast()的传参

(1)举例:

在一个单页中定义了nested views,你希望让所有子作用域都监听到某个参数的变化,并且作出相应动作。比如一个地图应用,某个$state中定义元素input,输入地址后,地图要定位,同时另一个状态下的列表要显示出该位置周边商铺的信息,此时多个$scope都在监听地址变化。
PS: $rootScope.$broadcast()可以非常方便的设置全局事件,并让所有子作用域都监听到。
 
.factory('addressFactory', ['$rootScope', function ($rootScope) {
// 定义所要返回的地址对象
var address = {}; // 定义components数组,数组包括街道,城市,国家等
address.components = []; // 定义更新地址函数,通过$rootScope.$broadcast()设置全局事件'AddressUpdated'
// 所有子作用域都能监听到该事件
address.updateAddress = function (value) {
this.components = angular.copy(value);
$rootScope.$broadcast('AddressUpdated');
}; // 返回地址对象
return address;
}]);

(2)在获取地址的controller中:

// 动态获取地址,接口方法省略
var component = {
addressLongName: xxxx,
addressShortName: xx,
cityLongName: xxxx,
cityShortName: xx,
countryLongName: xxxx,
countryShortName: xx,
postCode: xxxxx
}; // 定义地址数组
$scope.components = []; $scope.$watch('components', function () {
// 将component对象推入$scope.components数组
components.push(component);
// 更新addressFactory中的components
addressFactory.updateAddress(components);
});

(3)在监听地址变化的controller中:

// 通过addressFactory中定义的全局事件'AddressUpdated'监听地址变化
$scope.$on('AddressUpdated', function () {
// 监听地址变化并获取相应数据
var street = address.components[0].addressLongName;
var city = address.components[0].cityLongName; // 通过获取的地址数据可以做相关操作,譬如获取该地址周边的商铺,下面代码为本人虚构
shopFactory.getShops(street, city).then(function (data) {
if(data.status === 200){
$scope.shops = data.shops;
}else{
$log.error('对不起,获取该位置周边商铺数据出错: ', data);
}
});
});

4. 基于localStorage或sessionStorage的页面跳转传参

注意事项:通过LS或SS传参,一定要监听变量,否则参数改变时,获取变量的一端不会更新。AngularJS有一些现成的WebStorage dependency可以使用,譬如gsklee/ngStorage · GitHubgrevory/angular-local-storage · GitHub。下面使用ngStorage来简述传参过程:
(1) 上传参数到localStorage - Controller A
// 定义并初始化localStorage中的counter属性
$scope.$storage = $localStorage.$default({
counter: 0
}); // 假设某个factory(此例暂且命名为counterFactory)中的updateCounter()方法
// 可以用于更新参数counter
counterFactory.updateCounter().then(function (data) {
// 将新的counter值上传到localStorage中
$scope.$storage.counter = data.counter;
});

(2)监听localStorage中的参数变化 - Controller B

$scope.counter = $localStorage.counter;
$scope.$watch('counter', function(newVal, oldVal) {
// 监听变化,并获取参数的最新值
$log.log('newVal: ', newVal);
});

5. 基于localStorage/sessionStorage和Factory的页面传参

由于传参出现的不同的需求,将不同方式组合起来可帮助你构建低耦合便于扩展和维护的代码。
举例:应用的Authentication(授权)。用户登录后,后端传回一个时限性的token,该用户下次访问应用,通过检测token和相关参数,可获取用户权限,因而无须再次登录即可进入相应页面(Automatically Login)。其次所有的APIs都需要在HTTP header里注入token才能与服务器传输数据。此时我们看到token扮演一个重要角色:(a)用于检测用户权限,(b)保证前后端数据传输安全性。以下实例中使用GitHub - gsklee/ngStorage: localStorage and sessionStorage done right for AngularJS.GitHub - Narzerus/angular-permission: Simple route authorization via roles/permissions
(1)定义一个名为auth.service.js的factory,用于处理和authentication相关的业务逻辑,比如login,logout,checkAuthentication,getAuthenticationParams等。此处略去其他业务,只专注Authentication的部分。
 (function() {
'use strict'; angular
.module('myApp')
.factory('authService', authService); /** @ngInject */
function authService($http, $log, $q, $localStorage, PermissionStore, ENV) {
var apiUserPermission = ENV.baseUrl + 'user/permission'; var authServices = {
login: login,
logout: logout,
getAuthenticationParams: getAuthenticationParams,
checkAuthentication: checkAuthentication
}; return authServices; //////////////// /**
* 定义处理错误函数,私有函数。
* @param {type} xxx
* @returns {*}
* @private
*/
function handleError(name, error) {
return $log.error('XHR Failed for ' + name + '.\n', angular.toJson(error, true));
} /**
* 定义login函数,公有函数。
* 若登录成功,把服务器返回的token存入localStorage。
* @param {type} xxx
* @returns {*}
* @public
*/
function login(loginData) {
var apiLoginUrl = ENV.baseUrl + 'user/login'; return $http({
method: 'POST',
url: apiLoginUrl,
params: {
username: loginData.username,
password: loginData.password
}
})
.then(loginComplete)
.catch(loginFailed); function loginComplete(response) {
if (response.status === 200 && _.includes(response.data.authorities, 'admin')) {
// 将token存入localStorage。
$localStorage.authtoken = response.headers().authtoken;
setAuthenticationParams(true);
} else {
$localStorage.authtoken = '';
setAuthenticationParams(false);
}
} function loginFailed(error) {
handleError('login()', error);
}
} /**
* 定义logout函数,公有函数。
* 清除localStorage和PermissionStore中的数据。
* @public
*/
function logout() {
$localStorage.$reset();
PermissionStore.clearStore();
} /**
* 定义传递数据的setter函数,私有函数。
* 用于设置isAuth参数。
* @param {type} xxx
* @returns {*}
* @private
*/
function setAuthenticationParams(param) {
$localStorage.isAuth = param;
} /**
* 定义获取数据的getter函数,公有函数。
* 用于获取isAuth和token参数。
* 通过setter和getter函数,可以避免使用第四种方法所提到的$watch变量。
* @param {type} xxx
* @returns {*}
* @public
*/
function getAuthenticationParams() {
var authParams = {
isAuth: $localStorage.isAuth,
authtoken: $localStorage.authtoken
};
return authParams;
} /*
* 第一步: 检测token是否有效.
* 若token有效,进入第二步。
*
* 第二步: 检测用户是否依旧属于admin权限.
*
* 只有满足上述两个条件,函数才会返回true,否则返回false。
* 请参看angular-permission文档了解其工作原理https://github.com/Narzerus/angular-permission/wiki/Managing-permissions
*/
function checkAuthentication() {
var deferred = $q.defer(); $http.get(apiUserPermission).success(function(response) {
if (_.includes(response.authorities, 'admin')) {
deferred.resolve(true);
} else {
deferred.reject(false);
}
}).error(function(error) {
handleError('checkAuthentication()', error);
deferred.reject(false);
}); return deferred.promise;
}
}
})();

(2)定义名为index.run.js的文件,用于在应用载入时自动运行权限检测代码。

 (function() {
'use strict'; angular
.module('myApp')
.run(checkPermission); /** @ngInject */ /**
* angular-permission version 3.0.x.
* https://github.com/Narzerus/angular-permission/wiki/Managing-permissions.
*
* 第一步: 运行authService.getAuthenticationParams()函数.
* 返回true:用户之前成功登陆过。因而localStorage中已储存isAuth和authtoken两个参数。
* 返回false:用户或许已logout,或是首次访问应用。因而强制用户至登录页输入用户名密码登录。
*
* 第二步: 运行authService.checkAuthentication()函数.
* 返回true:用户的token依旧有效,同时用户依然拥有admin权限。因而无需手动登录,页面将自动重定向到admin页面。
* 返回false:要么用户token已经过期,或用户不再属于admin权限。因而强制用户至登录页输入用户名密码登录。
*/
function checkPermission(PermissionStore, authService) {
PermissionStore
.definePermission('ADMIN', function() {
var authParams = authService.getAuthenticationParams();
if (authParams.isAuth) {
return authService.checkAuthentication();
} else {
return false;
}
});
}
})();

(3)定义名为authInterceptor.service.js的文件,用于在所有该应用请求的HTTP requests的header中注入token。关于AngularJS的Interceptor,请参看AngularJS

 (function() {
'use strict'; angular
.module('myApp')
.factory('authInterceptorService', authInterceptorService); /** @ngInject */
function authInterceptorService($q, $injector, $location) {
var authService = $injector.get('authService'); var authInterceptorServices = {
request: request,
responseError: responseError
}; return authInterceptorServices; //////////////// // 将token注入所有HTTP requests的headers。
function request(config) {
var authParams = authService.getAuthenticationParams();
config.headers = config.headers || {};
if (authParams.authtoken) config.headers.authtoken = authParams.authtoken; return config || $q.when(config);
} function responseError(rejection) {
if (rejection.status === 401) {
authService.logout();
$location.path('/login');
}
return $q.reject(rejection);
}
}
})();

转自知乎:https://www.zhihu.com/question/33565135

AngularJS中页面传参方法的更多相关文章

  1. angularJS 中的传参

    今天总结一下 angularJS 传参的 3种方式:(配合 ui-router) 现在有两个页面,page1.html 和 page2.html, 现由 page1.html 向 page2.html ...

  2. WebForm中 页面传参的总结

    页面与后台的数据传递是实现动态页面的前提---数据交互.无论是MVC还是WebFrom 都需要详细了解各种前后台的数据传输方式,熟悉每种方式的优缺点,这样才能提高网站的性能,技术上得到锻炼. 1.Fo ...

  3. FastAdmin 前端页面传参笔记

    FastAdmin 前端页面传参笔记 看到 QQ 群里的小伙伴询问如何传参,然后在社区里找到一笔记帖子 1 还要参考在线文档控制器部分2. 引用 Karson 的回复: 如果我们需要自己在控制器中透传 ...

  4. vue 通过 name 和 params 进行调整页面传参刷新参数丢失问题&vue路由可选参数

    vue  通过 name 和 params 进行调整页面传参刷新参数丢失问题 router.js: export default new Router({ routes: [ { path: '/', ...

  5. 学习chrome 插件 DHC ,http请求传参方法

    DHC的简介 DHC是一款可以帮助用户使用chrome插件模拟HTTP客户端发送测试数据到服务器的谷歌浏览器插件,在chrome中安装了DHC插件以后,就可在服务器端代码初步完成的时候,使用DHC进行 ...

  6. 关于页面传参,decodeURI和decodeURIComponent

    之前写过一个关于页面传参的,但是是前端相对于自己的页面做的跳转,也就是页面1,跳转到页面2,里面带的参数.这里可以参考我上一篇文章,包括里面参数中如果有数组和json格式的情况.但是需要注意的是,我前 ...

  7. Strut2页面传参跳转 --Struts2

    1.本案例借助struts2框架,完成页面传参.跳转功能 2.代码实现 index.jsp: <form action="helloStruts2.action" metho ...

  8. 在Java中动态传参调用Python脚本

    最近,又接触到一个奇葩的接口,基于老板不断催促赶时间的情况下,在重写java接口和复用已有的python脚本的两条路中选择了后者,但是其实后者并没有好很多,因为我是一个对python的认识仅限于其名称 ...

  9. ionic简单路由及页面传参

    1)页面跳转及传参方法 angular.module('app.routes', [])//routes路由模型 .config(function($stateProvider, $urlRouter ...

随机推荐

  1. Linux 命令 -- chmod

    chmod命令用来变更文件或目录的权限.在UNIX系统家族里,文件或目录权限的控制分别以读取.写入.执行3种一般权限来区分,另有3种特殊权限可供运用.用户可以使用chmod指令去变更文件与目录的权限, ...

  2. 十三、nginx 强制下载txt等文件

    当前的浏览器能够识别文件格式,如果浏览器本身能够解析就会默认打开,如果不能解析就会下载该文件. 那么使用nginx做资源服务器的时候,如何强制下载文件呢? location /back/upload/ ...

  3. Spring boot Mybatis 整合(注解版)

    之前写过一篇关于springboot 与 mybatis整合的博文,使用了一段时间spring-data-jpa,发现那种方式真的是太爽了,mybatis的xml的映射配置总觉得有点麻烦.接口定义和映 ...

  4. About custom Theme and Style

    For android system, of course you can custom your own style and theme, but you can't break compatibi ...

  5. 如何正确实现 IDisposable 接口

    MSDN建议按照下面的模式实现IDisposable接口: public class Foo: IDisposable { public void Dispose() { Dispose(true); ...

  6. 深入浅出ConcurrentHashMap1.8

    转载:https://www.jianshu.com/p/c0642afe03e0 好文 关于文章中的疑问:为什么要构造一个反序链表,放在nextTable的i+n的位置上呢,在<深入分析Con ...

  7. Maven+MyBatis 初试

    工作中一直使用的都是Hibernate,总是听见有人拿Mybatis和Hibernate做比较,今天尝试来看看. 一.用Maven建立web项目 此处参见 http://www.cnblogs.com ...

  8. 主动驱动事件执行--createEvent

    1. createEvent(eventType)参数:eventType 共5种类型:    Events :包括所有的事件.           HTMLEvents:包括 'abort', 'b ...

  9. nginx: [emerg] bind() to 0.0.0.0:80 failed (98: Address already in use)解决方案

    前提:已经配置好静态IP以防万一,先安装好iptables服务(不管你装没装,先执行,免得后面添乱)[root@localhost ~]# yum install iptables-services[ ...

  10. 微信jssdk 返回的 config invalid signature

    这几天一直在调试wxjssdk,按照api的需求,http://mp.weixin.qq.com/wiki/7/aaa137b55fb2e0456bf8dd9148dd613f.html ,先是根据a ...