使用Spring boot,Thymeleaf,AngularJS从零开始构建新的Web应用程序–第3部分

在之前的博客中,我们使用Thymeleaf,Bower和Bootstrap构建了登录页面,并将其部署到了Heroku。
在此博客中,我们将介绍用于前端的AngularJS和在后端的Spring Boot Web服务的一些功能。

我们将从“登录/注销”部分开始。 让我们开始使用Bower在我们的应用程序中安装AngularJS。

AngularJS简介

$ bower install angular --save

这将在bower_components文件夹内添加角度依赖性。 为了将角度注入到我们HTML页面中,我们需要做两件事
1)在我们的html文件中添加angular.min.js的路径

<script src="./bower_components/angular/angular.min.js" type="text/javascript"></script>

2.将主体标记为ng-app =“ myModule”,这意味着我们需要定义一个JavaScript模块,Angular将其识别为名为“ myModule”的应用程序。 Angular模块就像是应用程序不同部分的容器-控制器,服务,过滤器,指令等。

<body ng-app="myModule">

现在,我们需要在JS文件中定义此模块“ myModule”,我们可以将该文件称为具有成分的app.js:

angular.module('myModule', []).controller('home', function($http) {var self = this;$http.get('/resource/').then(function(response) {self.message = response.data;})
});

在这里,我们看到我们在“ myModule”中定义了一个名为“ home”的控制器,该控制器将用于通过数据绑定将值传递到HTML,此处的模型为“消息”。

数据将通过HTTP GET通过对我们的Spring后端进行REST调用来检索。
因此,让我们看看如何创建Spring Rest Controller:

弹簧架控制器

我们已经添加了依赖项"spring-boot-starter-web"因此我们现在不再需要任何spring boot依赖项来构建Rest Controller。

我们将创建一个LoginController

@RestController
public class LoginController {@RequestMapping("/resource")public Map<String,Object> home() {Map<String,Object> model = new HashMap<String,Object>();model.put("id", UUID.randomUUID().toString());model.put("content", "Hello World");return model;}

现在,如果将这段代码放在单独的div中的index.html中,则可以看到该div现在正在从后端动态加载数据。

要查看HTML中的数据,我们需要这样做:

<div ng-controller="home as home"><p>The ID is {{home.message.id}}</p><p>The content is {{home.message.content}}</p>
</div>

到目前为止,我们仅能在静态页面中使用angularJS并从后端Rest Controller检索一些信息并使用AngularJS进行渲染。

建筑物登录/注销

现在我们在索引页面中有了angularJS模块,让我们使用Spring Security实现登录/注销功能。

要构建登录/注销功能,我们需要制作3个HTML页面。
1)index.html –这将是登录页面,我们将在其中添加导航以登录和注销。 (我们已经有)
2)home.html –这是用户登录后将重定向到的页面。 3)login.html –这将是带有用户名和密码字段的页面,用于输入登录凭据。

对于这些页面之间的路由,我们将使用ng-route 。 ngRoute模块为angular应用程序提供路由和深层链接服务和指令。 要使用bower安装ngRoute,我们将执行以下命令:

bower install angular-route@X.Y.Z

然后将其包含在index.html中:

<script src="path/to/angular.js"></script>
<script src="path/to/angular-route.js"></script>

步骤1:在index.html中添加导航

为了添加导航,我们需要在主体下面添加此代码块。

<div ng-controller="navigation as nav" class="container"><ul class="nav nav-pills" role="tablist"><li class="active"><a href="#/">home</a></li><li><a href="#/login">login</a></li><li ng-show="authenticated"><a href="" ng-click="nav.logout()">logout</a></li></ul></div><div ng-view class="container"></div>

“ ng-view”部分将显示“ login.html”部分页面。

更改app.js文件以定义路由:

angular.module('myModule', [ 'ngRoute' ]).config(function($routeProvider, $httpProvider) {$routeProvider.when('/', {templateUrl : 'home.html',controller : 'home',controllerAs: 'controller'}).when('/login', {templateUrl : 'login.html',controller : 'navigation',controllerAs: 'controller'}).otherwise('/');}).controller('home', function($http) {var self = this;$http.get('/resource/').then(function(response) {self.message = response.data;})}).controller('navigation',function($rootScope, $http, $location) {var self = thisvar authenticate = function(credentials, callback){var headers = credentials ? {authentication : "Basic "+ btoa(credentials.username + ":"+credentials.password)} : {};$http.get('user', {headers : headers}).then(function(response) {if (response.data.name) {$rootScope.authenticated = true;} else {$rootScope.authenticated = false;}callback && callback();}, function() {$rootScope.authenticated = false;callback && callback();});}authenticate();self.credentials = {};self.login = function() {authenticate(self.credentials, function() {if ($rootScope.authenticated) {$location.path("/");self.error = false;} else {$location.path("/login");self.error = true;}});};self.logout = function() {$http.post('logout', {}).finally(function() {$rootScope.authenticated = false;$location.path("/");});}});

在这里,我们定义了2个控制器和1个配置,以及几个将由控制器使用的功能。 “ Config”用于使用ng-route定义路径和路由。 控制器“导航”用于按定义调用函数登录,注销和认证。 控制器“主页”用于在主页上发送问候语。

定义了authenticated变量以提供对页面上已认证用户的访问。

第2步:添加login.html部分页面

在这里,我们将制作一个局部页面login.html,该页面将呈现在ng-view标记的div中。

Login.html

<div class="alert alert-danger" ng-show="controller.error">There was a problem logging in. Please try again.
</div>
<form role="form" ng-submit="controller.login()"><div class="form-group"><label for="username">Username:</label> <input type="text"class="form-control" id="username" name="username" ng-model="controller.credentials.username"/></div><div class="form-group"><label for="password">Password:</label> <input type="password"class="form-control" id="password" name="password" ng-model="controller.credentials.password"/></div><button type="submit" class="btn btn-primary">Submit</button>
</form>

然后,我们需要RestControllers来使用Spring Security进行身份验证。 我们将使用spring安全的默认身份验证用户。 我们将使用以下方法制作一个UserController:

@RequestMapping("/user")public Principal user(Principal user) {return user;}

为了使Spring Security正常工作,我们需要在Application.java中添加它

@Configuration@Order(SecurityProperties.ACCESS_OVERRIDE_ORDER)protected static class SecurityConfiguration extends WebSecurityConfigurerAdapter {@Overrideprotected void configure(HttpSecurity http) throws Exception {http.httpBasic().and().authorizeRequests().antMatchers("/index.html", "/home.html", "/login.html", "/").permitAll().anyRequest().authenticated();}}

第3步:添加带有欢迎消息的home.html页面。

<h1>Message</h1>
<div ng-show="authenticated"><p>The ID is {{controller.message.id}}</p><p>The content is {{controller.message.content}}</p>
</div>
<div  ng-show="!authenticated"><p>Login to see your message</p>
</div>

“已认证”变量用于提供对页面的访问。

到目前为止,我们已经成功地使用Angular JS通过Spring Security登录和注销了。 但是在实际应用中,我们需要根据用户数据库对实际用户进行身份验证。 在下一个博客中,我们将使用从用户DB进行身份验证的用户来构建应用程序。

PS:示例从Spring博客引用: https : //spring.io/guides/tutorials/spring-security-and-angular-js/

翻译自: https://www.javacodegeeks.com/2016/05/build-new-web-application-scratch-using-spring-boot-thymeleaf-angularjs-part-3.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/335813.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

android的oomkiller_Android Low memory killer

Android Low memorykillerby 永远的伊苏Android中&#xff0c;进程的生命周期都是由系统控制的&#xff0c;即使用户关掉了程序&#xff0c;进程依然是存在于内存之中。这样设计的目的是为了下次能快速启动。当然&#xff0c;随着系统运行时间的增长&#xff0c;内存会越来越少…

C++ 11 深度学习(十二)函数新特性、内联函数、const详解

函数后置返回类型 //后置返回类型 auto fun(int, int)->int; 内联函数 在函数定义前增加关键字 inline ,使得该函数变成内联函数 (1) 适用于函数体很小&#xff0c;调用很频繁的函数类型&#xff0c;可以引入内联函数 (2) inline影响编译器&#xff0c;在编译阶段对inc…

apache pdfbox_Apache PDFBox 2

apache pdfboxApache PDFBox 2已于今年早些时候发布 &#xff0c; Apache PDFBox 2.0.1和Apache PDFBox 2.0.2已发布。 Apache PDFBox是开源的&#xff08; Apache许可证版本2 &#xff09;并且基于Java&#xff08;因此易于使用&#xff0c;包括Java &#xff0c; Groovy &…

cad坐标归零lisp_CAD图怎么归零

展开全部原理&#xff0c;就是把图元Z轴线移动的负无穷远&#xff0c;然e68a84e8a2ad62616964757a686964616f31333335336530后移动到正无穷&#xff0c;除了块就可以Z轴线归零了。用autoLISP来解决。;;;;;;Z坐标归零;;(defun c:z0 ( / &kw &k1 #os1)(setvar "cmde…

dynamodb java_使用Java查询DynamoDB项

dynamodb java在上一篇文章中&#xff0c;我们继续在DynamoDB数据库上插入数据。 在本教程中&#xff0c;我们将对DynamoDB表发出一些基本查询。 主要规则是每个查询都必须使用哈希键。 查询的最简单形式是仅使用哈希键。 我们将在此表上查询Users表。 结果只有一个&#xff…

冰点还原离线激活_冰点还原密钥,小编告诉你如何激活冰点还原

冰点还原软件&#xff0c;它具有轻松安装、动态保护、实时瞬间恢复&#xff0c;操作简单的特点。安装了冰点还原的系统&#xff0c;无论进行了安装文件&#xff0c;还是删除文件、更改系统设置等操作&#xff0c;计算机重新启动后&#xff0c;一切将恢复成初始状态。不过这些都…

【WebRTC---源码篇】(四)WebRTC线程模型

常见的线程模型 1.为了解决频繁线程创建与销毁,在此模型中使用的线程池。在线程池创建的时候就将一些线程创建起来,以提高效率。通过控制线程数量来解决线程频繁切换。 2.一般线程与线程存在前后关系的,线程执行完毕之后生成一个新的任务(task1 , task2,task3---)插入到任…

java cuba_CUBA平台–用于快速应用程序开发的开源Java框架

java cuba传统上&#xff0c;自计算时代开始以来&#xff0c;企业软件开发自然面临着一个挑战&#xff0c;当时自然而然地&#xff0c;企业软件开发本应专注于解决实际的业务问题&#xff0c;但与此同时&#xff0c;开发人员必须在技术上花费大量时间和精力。解决方案的一面&am…

web project、web service project和java project的区别

java project就是普通的用java写的程序&#xff0c;直接就能运行的&#xff0c;web project,web service project不能直接运行&#xff0c;必须在容器里面运行&#xff0c;这个容器就是web server&#xff0c;例如tomcat,jboss。web project就是web工程&#xff0c;简单的讲做网…

WebRTC Qos优化杂记

WebRTC视频JitterBuffer详解 详细内容链接 JitterBuffer延迟换流畅 在丢包+延迟+抖动的弱网环境下,推流端通过NACK+FEC等方式做了一定的容错,但是可能并不能百分之百解决所有问题,而且通过SFU分发到拉流端之后,拉流端网络也可能有问题,这个时候需要JitterBuffer来做媒体…

keras添加正则化全连接_TensorFlow keras卷积神经网络 添加L2正则化

model keras.models.Sequential([#卷积层1keras.layers.Conv2D(32,kernel_size5,strides1,padding"same",data_format"channels_last",activationtf.nn.relu,kernel_regularizerkeras.regularizers.l2(0.01)),#池化层1keras.layers.MaxPool2D(pool_size2,…

java中什么时候应用异常_生产Java应用程序中的十大异常类型-基于1B事件

java中什么时候应用异常Pareto记录原理&#xff1a;97&#xff05;的记录错误语句是由3&#xff05;的唯一错误引起的 在最新的数据整理帖子之后&#xff0c;我们收到了很多反馈和问题&#xff0c;我们发现97&#xff05;的记录错误是由10个唯一错误引起的 。 根据大众的需求&…

C++ 11 深度学习(十四)C++类

(一)综述&#xff1a;类是我们自己定义的数据类型 设计时要考虑的角度&#xff1a; 站在设计和实现的角度来考虑&#xff1b;站在使用者的角度考虑&#xff1b;父类&#xff0c;子类之间的考虑&#xff1b; &#xff08;二&#xff09;explicit 首先, C中的explicit关键字只…

手动编译 lombok_Lombok,一种编译时Java注释预处理器,可最大程度地减少代码大小...

手动编译 lombok在本文中&#xff0c;我们将看到如何在常规Java代码中使用lombok来最大程度地减少代码长度和冗余。 什么是Lombok&#xff1f; Lombok&#xff0c;一个编译时注释预处理器&#xff0c;有助于在编译时注入一些代码。 在详细介绍之前&#xff0c;我要求您应该从…

MyEclipse for Windows快捷键

文章目录编辑查询/替换导航调试重构其他自定义快捷键技巧编辑 快捷键功能说明Ctrl1快速修复&#xff08;最经典的快捷键,就不用多说了&#xff0c;可以解决很多问题&#xff0c;比如import类、try catch包围等&#xff09;CtrlShiftF格式化代码。团队有统一的代码格式&#xf…

微信小程序view动态长度_微信小程序实现动态获取元素宽高的方法分析

本文实例讲述了微信小程序实现动态获取元素宽高的方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;我以前一直以为微信小程序不能动态获取view元素的宽高。但是自从看到&#xff1a; wx.createSelectorQuery() 这个api接口&#xff0c;以前的某些问题就能得到解决了…

数据结构【双指针算法】

双指针一般应用于维护两个队列&#xff0c;或者同一队列。 常见写法如下 for (int i 0, j 0; i < n; i){while (j < i && check(i, j)) j;//每道题的具体逻辑} 双指针的核心思想是把如下O(n^2)的算法优化位O(n) for (int i 0; i < n; i){for (int j 0; …

antlr 语言 库_关于ANTLR的通用库的需求:使用反射来构建元模型

antlr 语言 库我是一名语言工程师&#xff1a;我使用多种工具来定义和处理语言。 在其他工具中&#xff0c;我使用ANTLR&#xff1a;它简单&#xff0c;灵活&#xff0c;可以围绕它进行构建。 但是我发现自己围绕ANTLR为不同的项目重建了类似的工具。 我看到两个问题&#xff…

Windows 如何通过命令启动和关闭 Tomcat

首先你的系统要安装tomcat&#xff0c;至于如何安装tomcat这里就不讲解了&#xff0c;接着你打开windows的DOS命令终端&#xff08;winR打开运行窗口&#xff0c;输入cmd即可&#xff09;&#xff0c;然后通过DOS命令切换到tomcat安装目录下的bin目录&#xff0c;最后执行start…