您现在的位置是:首页 > 文章详情

Windchill的JSP页面跳转到Ext页面的分析

日期:2018-09-19点击:373

今天花了一天时间,看了一块代码,低效但是不得不看懂。
具体内容是这样的:
Windchill中按钮菜单栏嵌入了一个菜单项,点击它会触发弹出一个新的浏览器页面,相当于弹框,让你输入内容,然后点击保存按钮保存输入信息。
问题就在于菜单项本身是一个JSP页面,点击以后弹出窗体本身会是一个Ext的页面。
上图描述一下:
_
上图这样的一个菜单项
_
这样配置的一个菜单项
点击菜单项,也就是这个url以后会跳转到这个url地址代表的jsp页面
JSP的代码如下:

<%@page import="com.bplead.drawingApproval.util.QsWebUtil"%> <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <% //重新定向action地址 //QsWebUtil.forceNavigateNetmarketsURL(request, response); QsWebUtil.BbURL(request, response); //---------------// //获取对象oid String oid = request.getParameter("oid"); //获取action type(编制、查看) String type = request.getParameter("type"); if("view".equals(type)){ type = "查看"; }else if("edit".equals(type)){ type = "编制"; } %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>图纸发布审批流程 </title> <!-- ExtJS4.2 css--> <link rel="stylesheet" type="text/css" href="/Windchill/netmarkets/javascript/bplead/extjs-4.2/resources/css/ext-all.css"></link> <!-- Util CSS --> <link rel="stylesheet" type="text/css" href="app/css/util.css"></link> <!-- ExtJS4.2 --> <script src="/Windchill/netmarkets/javascript/bplead/extjs-4.2/bootstrap.js"></script> <script src="/Windchill/netmarkets/javascript/bplead/extjs-4.2/locale/ext-lang-zh_CN.js"></script> <!-- Custom App --> <script src="app.js"></script> <!-- JSP Params --> <script type="text/javascript"> var oid = '<%=oid%>'; var type = '<%=type%>'; </script> </head> <body>

上述代码中的BbURL方法的代码如下:

public static boolean BbURL(HttpServletRequest request, HttpServletResponse response) throws Exception{ String key = SB_MARKS;//一个无关紧要的普通字符串 boolean hasRedirect = request.getSession().getAttribute(key) != null; if (!hasRedirect) { request.getSession().setAttribute(key, new Object()); response.sendRedirect(request.getRequestURI() + "?" + request.getQueryString()); } else { request.getSession().removeAttribute(key); } return !hasRedirect; }

上述代码中的BbURL()方法究竟起什么作用呢?
经过测试搞清楚了:
上述代码中的if和else代码块,在正常的访问中,会分别进入1次,也就是上述BbURL()方法一共被访问2次。
第一次,JSP页面首次加载,目的是JSP页面中的java代码获取type和oid这两个java变量的值,然后将java变量的值赋值给javascript的对应两个变量
第一次进入JSP页面的核心语句就是下面这一段

<script type="text/javascript"> var oid = '<%=oid%>'; var type = '<%=type%>'; </script>

第二次进入JSP页面的原因就是BbURL()方法,因为这个方法的java代码也就是If代码块中调用了

response.sendRedirect(request.getRequestURI() + "?" + request.getQueryString());

这句代码实际上是一句废话,目的是让同样的JSP页面被第二次访问。
在第二次访问JSP页面的时候,BbURL()方法进入else代码块,实际上并没有触发任何有实际意义的业务代码。
但是第二次访问JSP页面的时候,由于此时拥有了第一次访问时获取到的两个javascript变量:type和oid,因此第二次访问就可以顺利将这两个参数传递给ext的js代码,生成网页。
上述逻辑,记录一下。
上述内容总体来说,是无知的阐述,下面结合下一篇博文,详细阐述一下Windchill系统中的jsp页面跳转以及windchill与spring的关系
简单的说,Windchill的web.xml文件中配置了spring和springmvc。

Spring的配置文件较为简单,仅仅注册了1个ptc封装的bean,我估计用途就是,为日后java二次开发的时候,依靠依赖注入将这个bean导入到java代码中,用来后续通过这个bean访问windchill的数据库之类的。
SpringMVC的配置文件比较复杂,这个配置文件导入了133个其他的xml配置文件,非常壮观。
_
说了这么多,重点在于,springmvc的配置文件在web.xml中是怎么配置的呢?换句话说,web.xml文件中,有哪些和web访问有关的配置呢?这是我们考虑页面跳转的关键:
首先web.xml文件中配置了如下内容,也就是说:
所有访问路径中包含/ptc1/*的url的请求都会被springmvc单独处理
当然,具体怎么处理的通过这133个配置文件也许能找到头绪

 <servlet> <description>MVC Dispatcher Servlet</description> <servlet-name>MVCDispatcher</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextClass</param-name> <param-value>com.ptc.mvc.components.support.ComponentXmlWebApplicationContext</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>MVCDispatcher</servlet-name> <url-pattern>/servlet/WizardServlet/*</url-pattern> <url-pattern>/servlet/ActionsMenu/*</url-pattern> <url-pattern>/servlet/RecentList/*</url-pattern> <url-pattern>/servlet/Navigation/*</url-pattern> <url-pattern>/servlet/SuggestServlet/*</url-pattern> <url-pattern>/servlet/TypeBasedIncludeServlet/*</url-pattern> <url-pattern>/servlet/UIValidationAJAXServlet/*</url-pattern> <url-pattern>/ptc1/*</url-pattern> <url-pattern>/app/*</url-pattern> <url-pattern>/gwt/*</url-pattern> </servlet-mapping>

我们尝试在Windchill服务器的springmvc目录中检索133个配置文件中,哪些对ptc1这个字符串进行了配置
在上述目录中执行命令grep -nr "component-scan" *
下图所示的内容,就是这133个文件中都扫描了那些地方,这些地方就是controller的所在,也就是说,如果url中出现了ptc1,那么下面这些位置的controller中的java代码回针对性的对ptc1进行匹配,然后进行相应。但是这种分析是徒劳的,因为下图所示的所有包以及其中的java文件都是不公开的,而且也实在太多了。
_
以上分析说明,妄图去浩如烟海的java类中找到对ptc1这个字符串匹配的后续的service dao的处理代码,是徒劳的。
而且即使分析出ptc1字符串的处理代码,也是舍本求末,因为老大,你自己的二次开发代码压根没有主动向带有ptc1的url地址发送请求,你首先要搞清楚你的页面为什么会指向带有ptc1字样的url

所以我们继续分析web.xml文件,找到了如下代码:

 <filter> <description>Filter for converting Netmarkets URL's with 7.0 or 8.0 JSP's names to URL's having 10.0 JSP's names.</description> <filter-name>NetmarketsRedirectFilter</filter-name> <filter-class>com.ptc.core.components.filter.NetmarketsRedirect</filter-class> </filter> <filter-mapping> <filter-name>NetmarketsRedirectFilter</filter-name> <url-pattern>/netmarkets/jsp/*</url-pattern> </filter-mapping>

上述代码说明,web.xml中为Windchill配置了filter
只要主动访问windchill下/netmarkets/jsp/这个目录下的http request都会被com.ptc.core.components.filter.NetmarketsRedirect这个filter过滤一下
那我们最初关注的url哈记得是什么样子吗?就是Windchill中通过配置文件二次开发的按钮,我们看看
_
可以很清楚的看到,我们自己配置的url是:

url="/netmarkets/jsp/bplead/drawingApproval/drawingApproval.jsp?type=edit"

也就是我们主动访问的/netmarkets/jsp/目录下的地址
那么依据web.xml文件中filter的配置,这个request一定会被com.ptc.core.components.filter.NetmarketsRedirect这个filter过滤一下
下面我们看看这个filter的内容
_
_
_
如上图所示,web.xml文件中将NetmarketsRedirect类注册为url特征为/netmarkets/jsp/*的filter。ptc自行实现了Filter名为JCAFilter,而NetmarketsRedirect类是JCAFilter的子类。
当/netmarkets/jsp/*类型的url被web.xml文件中注册的filter过滤后,依据filter的规则,会调用NetmarketsRedirect类的dofilter()方法,由于NetmarketsRedirect类是JCAFilter的子类,因此就会调用它的dofilter()方法,如下图:

public abstract class JCAFilter implements Filter{ public void doFilter(ServletRequest paramServletRequest, ServletResponse paramServletResponse, FilterChain paramFilterChain) throws IOException, ServletException { paramServletRequest.setCharacterEncoding("UTF-8"); paramServletResponse.setContentType("text/html; charset=UTF-8"); if ((!(paramServletRequest instanceof HttpServletRequest)) || (!(paramServletResponse instanceof HttpServletResponse))) { log.debug("Request/Response was not HttpServlet"); paramFilterChain.doFilter(paramServletRequest, paramServletResponse); return; } if (log.isDebugEnabled()) { log.debug("Requested " + ((HttpServletRequest)paramServletRequest).getRequestURL()); } HttpServletRequest localHttpServletRequest = (HttpServletRequest)paramServletRequest; // String str = getURLForRedirect(localHttpServletRequest); }//doFilter()方法结束 //父类中只有一个抽象方法,NetmarketsRedirect类继承后再实现 protected abstract String getURLForRedirect(HttpServletRequest paramHttpServletRequest); public void destroy() {} public void init(FilterConfig paramFilterConfig) throws ServletException {} }//JCAFilter类结束

上述代码是JCAFilter类的部分代码,可以看到doFilter()方法的核心就是调用getURLForRedirect()方法。
那么NetmarketsRedirect类中getURLForRedirect()方法的内容是怎样的呢?

public class NetmarketsRedirect extends JCAFilter { @SuppressWarnings(value={"BC_IMPOSSIBLE_CAST"}, justification="String[] portletValues = (String[])qs.get(NmContextBean.PORTLET) is possible cast becuase, the map adds string array as value") protected String getURLForRedirect(String paramString1, String paramString2, String paramString3) { String str1 = null; if ((paramString1.contains("ptc1")) || (paramString1.contains("app/#"))) { log.debug("No redirect, the url is already an X-20 style"); return null; } if (!paramString1.endsWith(".jsp")) { log.debug("No redirect, this filter doesn't handle cases that don't end with \".jsp\""); return null; } if (paramString1.endsWith("/work/list.jsp")) { log.debug("Home -> Assignments"); str1 = getNewURL("work/listAssignments", paramString2); } else if (paramString1.endsWith("/netmarkets/overview.jsp")) { log.debug("Home Overview"); str1 = getNewURL("homepage", null); } else if (paramString1.endsWith("/events/list_mine.jsp")) { log.debug("Home -> Meetings"); str1 = getNewURL("meeting/list_mine", paramString2); } else if (paramString1.endsWith("/subscription/list.jsp")) { log.debug("Home -> Subscriptions"); str1 = getNewURL("subscription/listSubscriptions", paramString2); } else if ((paramString1.endsWith("/work/listProduct.jsp")) || (paramString1.endsWith("/work/listProduct2.jsp"))) { log.debug("Product -> Assignments"); str1 = getNewURL("work/listProductAssignments", paramString2); } else if (paramString1.endsWith("listFiles.jsp")) { log.debug("Folder Browser listFiles.jsp"); HashMap localHashMap = NmURLFactoryBean.parseQueryString(paramString2); String[] arrayOfString = (String[])localHashMap.get("portlet"); String str2 = arrayOfString == null ? null : arrayOfString[0]; if (!"component".equals(str2)) { str1 = getNewURL(NmURLFactoryBean.relativizeUrl(paramString1, false), paramString2); } } else if ((paramString1.endsWith("/folder/view.jsp")) || (paramString1.endsWith("/project/view.jsp"))) { log.debug("Folder Info Pages"); str1 = null; if (paramString3.indexOf("PDMLinkProduct") > 0) { log.debug("Product -> Folder Info Page"); str1 = getNewURL("product/listFiles", paramString2); } else if (paramString3.indexOf("Project2") > 0) { log.debug("Project -> Folder Info Page"); str1 = getNewURL("project/listFiles", paramString2); } else if (paramString3.indexOf("WTLibrary") > 0) { log.debug("Library -> Folder Info Page"); str1 = getNewURL("library/listFiles", paramString2); } else if (paramString3.indexOf("SubFolder") > 0) { log.debug("Folder Info Page"); str1 = getInfoPageURL(paramString2); } else { log.debug("This case is not covered. Do not redirect."); return null; } } else if (paramString1.endsWith("/project/list3.jsp")) { log.debug("Project -> Project List (7.0 only)"); str1 = getNewURL("project/list", null); } else if (paramString1.endsWith("/projectResource/list.jsp")) { log.debug("Project -> Resources (7.0 only)"); str1 = getNewURL("project/listProjectResource", paramString2); } else if (paramString1.endsWith("/work/listProject.jsp")) { log.debug("Project -> Assignments"); str1 = getNewURL("work/listProjectAssignments", paramString2); } else if (paramString1.endsWith("/report/projectReport.jsp")) { log.debug("Project -> Reports (7.0 only)"); str1 = getNewURL("project/reports", paramString2); } else if (paramString1.endsWith("/wtcore/jsp/com/ptc/windchill/search/ExecuteSearch.jsp")) { log.debug("All -> Search"); str1 = getNewURL("search/executeSearch?", null); } else if (paramString1.endsWith("/change/list3.jsp")) { log.debug("Change -> Problem Reports"); str1 = getNewURL("change/list", null); } else if (paramString1.endsWith("/change/listChangeRequests2.jsp")) { log.debug("Change -> Change Requests"); str1 = getNewURL("change/listChangeRequests", null); } else if (paramString1.endsWith("/change/listChangeNotices2.jsp")) { log.debug("Change -> Change Notices"); str1 = getNewURL("change/listChangeNotices", null); } else if ((paramString1.endsWith("/work/listLibrary.jsp")) || (paramString1.endsWith("/work/listLibrary2.jsp"))) { log.debug("Library -> Assignments"); str1 = getNewURL("work/listLibraryAssignments", paramString2); } else if (paramString1.endsWith("/library/list2.jsp")) { log.debug("Library -> Libraries List"); str1 = getNewURL("library/list", paramString2); } else if (paramString1.endsWith("/site/listOrgs.jsp")) { log.debug("Site -> Organizations"); str1 = getNewURL("org/listOrgs?tab=org", null); } else if (paramString1.endsWith("/org/view.jsp")) { log.debug("Org info page"); str1 = getNewURL("org/details", paramString2); } else if ((paramString1.endsWith("/view.jsp")) && ( (paramString1.endsWith("/part/view.jsp")) || (paramString1.endsWith("/document/view.jsp")) || (paramString1.endsWith("/bookmark/view.jsp")) || (paramString1.endsWith("/milestone/view.jsp")) || (paramString1.endsWith("/projectActivity/view.jsp")) || (paramString1.endsWith("/summaryActivity/view.jsp")) || (paramString1.endsWith("/projectProxy/view.jsp")) || (paramString1.endsWith("/meeting/view.jsp")) || (paramString1.endsWith("/actionitem/view.jsp")) || (paramString1.endsWith("/epmdocument/view.jsp")))) { log.debug("Project Link info pages"); str1 = getInfoPageURL(paramString2); } else { log.debug("This URL doesn't contain an oid, so we don't need to do anything."); return null; } return str1; }

上述代码其实调用了很多ptc自己封装的内容,至于具体这个filter到底做了什么,对url做了什么处理,我们编写测试代码查看一下。
我们修改web.xml配置1个单独的filter然后配置1个自定义的servlet
然后我们用浏览器访问servlet和filter对应的url,这样,我们在filter的代码中再现NetmarketsRedirect类的语句以及它引用的语句,这样我们就能搞清楚,Windchill中所谓重定向,在搞什么鬼了。
我们在自己的eclipse上的windchill项目上测试servlet和filter
_
_
_
_
在eclipse中创建servlet后,eclipse会帮助你在web.xml文件自动添加相关代码:
_
这并非我们想要的,所以修改一下:
_
servlet中的核心方法的代码如下:

 /** * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response) */ protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub response.getWriter().append("Served at: ").append(request.getContextPath()); }

我们先测试一下效果:
_
说明servlet正常运行
下面创建一个filter

原文链接:https://yq.aliyun.com/articles/642610
关注公众号

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。

持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。

转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。

文章评论

共有0条评论来说两句吧...

文章二维码

扫描即可查看该文章

点击排行

推荐阅读

最新文章