注意在目前了解到的所有的js操作中:
//只有下面的方法,查找有 点+类名 ,比如【.edit-btn】 这个算标签内的自定义属性。
//其他地方,查找类名,都不需要添加 . 号 【原因就是 . 代表查找同类名的所有元素 】
$(document).on("click",".edit-btn",function(){ });
一、思路概述:
0、新建了一个 全局变量 当前页码信息,当保存提交时,显示的 还是当前页码。
1、get deps方法中 ajax 的异步和同步问题 获取部门请求,会影响更新操作前 获取员工信息,并赋值给 select 的dep 部门。两个都是 异步操作,容易出现提前赋值 select ,后在出现 select 部门选项
2、//2、发送ajax请求保存更新的员工数据 添加 _method
$.ajax({
url:"${APP_PATH}/emp/"+$(this).attr("edit-id"),
type:"post",
data:$("#empUpdateModal form").serialize()+"&_method=put",
success:function(result){
这样做会来到,web.xml 中的过滤器
<!-- 使用Rest风格的URI,将页面普通的post请求转为指定的delete或者put请求 -->
<filter>
<filter-name>HiddenHttpMethodFilter</filter-name>
<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>HiddenHttpMethodFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
3、 @RequestMapping(value="/emp/{id}”,method=RequestMethod.PUT)
改成
@RequestMapping(value="/emp/{empId}",method=RequestMethod.PUT)
4、如果改成 ajax 的 put 方法,则不需要在data中添加 "&_method=put" ,但是 tomcat 无法封装 请求体,导致 报错。
* 原因:
* Tomcat:
* 1、将请求体中的数据,封装一个map。
* 2、request.getParameter("empName")就会从这个map中取值。
* 3、SpringMVC封装POJO对象的时候。
* 会把POJO中每个属性的值,request.getParamter("email");
* AJAX发送PUT请求引发的血案:
* PUT请求,请求体中的数据,request.getParameter("empName")拿不到
* Tomcat一看是PUT不会封装请求体中的数据为map,只有POST形式的请求才封装请求体为map
* 查看下面的 tomcat 源码 解析 ,发现只有post
* org.apache.catalina.connector.Request(类名)—parseParameters() (方法名)( 行号 3111);
*
* protected String parseBodyMethods = "POST";
* if( !getConnector().isParseBodyMethod(getMethod()) ) {
success = true;
return;
}
*
*
* 解决方案;
* 我们要能支持直接发送PUT之类的请求还要封装请求体中的数据
* 1、配置上HttpPutFormContentFilter;
* 2、他的作用;将请求体中的数据解析包装成一个map。
* 3、request被重新包装,request.getParameter()被重写,就会从自己封装的map中取数据
* 员工更新方法
二、添加模态框:
1、直接复制并修改之前的员工添加模态框:
<!-- 员工修改的模态框 -->
<div class="modal fade" id="empUpdateModal" tabindex="-1" role="dialog" aria-labelledby="exampleModalCenterTitle" aria-hidden="true">
<div class="modal-dialog modal-dialog-centered" role="document">
<div class="modal-content">
<div class="modal-header">
<h5 class="modal-title" id="exampleModalLongTitle">员工修改</h5>
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
</div>
<div class="modal-body">
<form>
<div class="form-group row">
<label class="col-sm-3 col-form-label">empName</label>
<div class="col-sm-9">
<input type="text" readonly class="form-control-plaintext" id="empName_update_input_disable" name="empName" value="none">
</div>
</div>
<div class="form-group row">
<label class="col-sm-3 col-form-label">email</label>
<div class="col-sm-9">
<input type="text" class="form-control" id="email_update_input" name="email" placeholder="[email protected]">
<div class="invalid-feedback"> 邮箱输入错误。</div>
</div>
</div>
<div class="form-group row">
<label class="col-sm-3 col-form-label">gender</label>
<div class="col-sm-9">
<div class="form-check form-check-inline">
<input class="form-check-input" type="radio" name="gender" id="gender1_update_input" checked value="M">
<label class="form-check-label" >男</label>
</div>
<div class="form-check form-check-inline">
<input class="form-check-input" type="radio" name="gender" id="gender2_update_input" value="F">
<label class="form-check-label" >女</label>
</div>
</div>
</div>
<div class="form-group row">
<label class="col-sm-3 col-form-label">departName</label>
<div class="col-sm-6">
<!-- 部门提交部门id即可 -->
<select class="custom-select my-1 mr-sm-2" name="dId" id="dept_update_select">
</select>
</div>
</div>
</form>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-secondary" data-dismiss="modal">关闭</button>
<button type="button" class="btn btn-primary" id="emp_update_done_btn">更新</button>
</div>
</div>
</div>
</div>
2、添加点击出现模态框事件
注意绑定的时间,因为在生成button后,才能完成绑定。所以绑定的方法不是普通方法。[ edit-btn 这个伪类]
/***
1、我们是按钮创建之前就绑定了click,所以绑定不上。
$(".edit-btn").click(function(){
alert("edit");
});
****/
//1)、可以在创建按钮的时候绑定。
//2)、绑定点击 live() 方法,这个方法就算是后来添加的元素,也能绑定方法
/***
$(".edit-btn").live(function(){
alert("edit");
});
***/
//但是新版jquery没有live方法,使用on方法进行替代
/*** 下面的写法 不对
$(".edit-btn").on("click",function(){
alert("edit");
});
***/
// edit-btn 这个是在一开始展示表格信息的时候,就已经添加了。
//注意 这里是 .edit-btn 算标签内的自定义属性,只是这里使用时 有个点号
$(document).on("click",".edit-btn",function(){
//0、清空表单样式和内容
reset_form("#empUpdateModal form");
//1、查出部门信息,并显示部门列表
getDepts("#empUpdateModal select");
//2、查询员工信息,并显示员工信息 // 坑爹 一开始写成了 attr(edit-id)
getEmp($(this).attr("edit-id"));
//3、把员工的id传递给模态框的更新按钮,为了发送ajax时传递id
$("#emp_update_done_btn").attr("edit-id",$(this).attr("edit-id"));
//alert("edit");
$("#empUpdateModal").modal({
backdrop:"static"
});
});
3、将添加雇员信息用到的 getDepts()
方法,进行了抽取
async:false
, 这里用到了同步请求方法,如果是异步方法,可能会出现第四步设置设置部门信息时,可能部门信息还没有获取到,就已经开始设置哪个部门是员工现在默认的了。
//1、查出部门信息,并显示部门列表
function getDepts(ele){
$.ajax({
url:"${APP_PATH}/depts",
type:"GET",
async:false,
success:function(result){
console.log(result);
//显示部门信息,在下拉列表中
// $("#dept_add_select") 换一种找法
//$("#empAddModal select")
//清除之前留下的 option 标签
$(ele).empty();
$.each(result.extend.depts,function(){
var optionEle = $("<option></option>").attr("value",this.deptId).append(this.deptName);
optionEle.appendTo(ele);
});
}
});
}
4、显示员工信息方法
(1)前端请求代码:
function getEmp(id){
$.ajax({
url:"${APP_PATH}/emp/"+id,
type:"get",
success:function(result){
console.log(result);
var empData = result.extend.emp;
$("#empName_update_input_disable").val(empData.empName);
$("#email_update_input").val(empData.email);
$("#empUpdateModal input[name=gender]").val([empData.gender]);
$("#empUpdateModal select").val([empData.dId]);
}
})
}
(2)后端处理代码:
EmployeeController.java
/**
* 根据id查询员工
* @param id
* @return
*/
@RequestMapping(value="/emp/{id}",method=RequestMethod.GET)
@ResponseBody
public Msg getEmp(@PathVariable("id")Integer id){
Employee employee = employeeService.getEmp(id);
return Msg.success().add("emp", employee);
}
EmployeeService.java
/**
* 按照员工Id 查询员工
* @param id
* @return
*/
public Employee getEmp(Integer id) {
// TODO Auto-generated method stub
Employee employee = employeeMapper.selectByPrimaryKey(id);
return employee;
}
三、保存员工修改信息
1、添加保存按钮点击事件
之前对前端框架不熟悉,忘记添加这句话了$("#empUpdateModal").modal("hide");
导致点击提交事件后,一直没有关闭模态框,我以为会自动关闭模态框,原因找了很久,后来才发现。
//点击更新,更新员工信息
$("#emp_update_done_btn").click(function(){
console.log("提交更新方法---邮箱校验");
//验证邮箱是否合法
if(!validate_form_ele("#email_update_input",g_email_reg,g_email_valid,g_email_invalid_format))
return false;
//2、发送ajax请求保存更新的员工数据
$.ajax({
url:"${APP_PATH}/emp/"+$(this).attr("edit-id"),
type:"post",
data:$("#empUpdateModal form").serialize()+"&_method=put",
success:function(result){
//alert(result);
//console.log(result);
if(result.code=100){
//1、员工修改成功,需要关闭模态框。
$("#empUpdateModal").modal("hide");
//2、来到最后一页,显示刚才的数据;发送ajax请求,显示最后一页数据即可(用总记录数请求,保证是请求足够大,mybatis 分页插件 已经在mybatis-config.xml中配置了数据合法性校验,只会返回最后一页数据)
to_page(g_currentPage);
}else{
//后端校验 显示失败信息
if(undefined!=result.extend.errorFileds.email){
//显示邮箱错误信息
show_validate_msg("#email_update_input","error",result.extend.errorFileds.email);
}
}
}//success 方法结束
});
});
特别注意:
1、to_page(g_currentPage);
g_currentPage 是一个全局变量,我在显示分页信息时,将其赋了值。
2、保存时,ajax 发送的是post请求,然后用参数 data:$("#empUpdateModal form").serialize()+"&_method=put",
利用 web.xml的
<!-- 4、使用Rest风格的URI,将页面普通的post请求转为指定的delete或者put请求 -->
<filter>
<filter-name>HiddenHttpMethodFilter</filter-name>
<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>HiddenHttpMethodFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
2、后端处理
EmployeeController.java
@ResponseBody
@RequestMapping(value="/emp/{empId}",method=RequestMethod.PUT)
public Msg saveEmp(Employee employee){
System.out.println("将要更新的员工数据:"+employee);
employeeService.updateEmp(employee);
return Msg.success();
}
特别注意:value="/emp/{empId}"
,不是 value="/emp/{id}"
因为empId要赋值给 employee 中的empId属性,前端的post 请求中,不包括empId。
前端put请求举例如下:
empName=adafaf&email=123123%4013.com&gender=F&dId=1&_method=put
EmployeeService.java
public void saveEmp(Employee employee) {
// 下面这个方法 是全部插入 ,包括 连自增的id 都是
// employeeMapper.insert(employee);
// 下面这个方法是 有选择的插入,自增Id 被忽略 不会插入。
employeeMapper.insertSelective(employee);
}
四、保存员工修改信息改良方法
1、前端直接用ajax的put请求:
$.ajax({
url:"${APP_PATH}/emp/"+$(this).attr("edit-id"),
type:"PUT",
data:$("#empUpdateModal form").serialize(),
success:function(result){
//alert(result.msg);
//1、关闭对话框
$("#empUpdateModal").modal("hide");
//2、回到本页面
to_page(currentPage);
}
});
2、后端解决方法
/**
* 如果直接发送ajax=PUT形式的请求
* 封装的数据
* Employee
* [empId=1014, empName=null, gender=null, email=null, dId=null]
*
* 问题:
* 请求体中有数据;
* 但是Employee对象封装不上;
* update tbl_emp where emp_id = 1014;
*
* 原因:
* Tomcat:
* 1、将请求体中的数据,封装一个map。
* 2、request.getParameter("empName")就会从这个map中取值。
* 3、SpringMVC封装POJO对象的时候。
* 会把POJO中每个属性的值,request.getParamter("email");
* AJAX发送PUT请求引发的血案:
* PUT请求,请求体中的数据,request.getParameter("empName")拿不到
* Tomcat一看是PUT不会封装请求体中的数据为map,只有POST形式的请求才封装请求体为map
* Tomcat 源码,查看 put 的处理逻辑
* org.apache.catalina.connector.Request【类名】--parseParameters()【方法名】 (行号:3111);
*
* protected String parseBodyMethods = "POST";
* if( !getConnector().isParseBodyMethod(getMethod()) ) {
success = true;
return;
}
*
*
* 解决方案;
* 我们要能支持直接发送PUT之类的请求还要封装请求体中的数据
* 1、配置上HttpPutFormContentFilter;
* 2、他的作用;将请求体中的数据解析包装成一个map。
* 3、request被重新包装,request.getParameter()被重写,就会从自己封装的map中取数据
* 员工更新方法
* @param employee
* @return
*/
@ResponseBody
@RequestMapping(value="/emp/{empId}",method=RequestMethod.PUT)
public Msg saveEmp(Employee employee,HttpServletRequest request){
System.out.println("请求体中的值:"+request.getParameter("gender"));
System.out.println("将要更新的员工数据:"+employee);
employeeService.updateEmp(employee);
return Msg.success() ;
}
注意一下:需要在 web.xml中添加
<!-- 5、直接支持put请求的拦截器,即在put请求时将请求体数据封装成map,并让springmvc能够通过request.getParameter(属性名) 来获取数据 -->
<filter>
<filter-name>HttpPutFormContentFilter</filter-name>
<filter-class>org.springframework.web.filter.HttpPutFormContentFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>HttpPutFormContentFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>