SSM小项目-(9)-修改雇员模块

注意在目前了解到的所有的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">&times;</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>

 

 

 

 

 

0 0 votes
Article Rating
Subscribe
Notify of
guest
0 Comments
Inline Feedbacks
View all comments