Part 1: 从页面点击"Add"过程找出事件处理函数:
在Liferay中,当我们从左边选择一个Portlet并且添加的时候,会触发一系列的动作,并且最终把这个Portlet显示在页面上,现在我们就对这个神秘的过程进行窥测。
在页面上,为了找到我们点击Add之后绑定的事件处理函数,我们先找到这段代码对应的jsp页面在/html/portlet/layout_configuration/view_category.jsp中:
- <div
- class="lfr-portlet-item lfr-archived-setup"
- id="<portlet:namespace />portletItem<%= portletItem.getPortletItemId() %>"
- instanceable="<%= portletInstanceable %>"
- plid="<%= plid %>"
- portletId="<%= portlet.getPortletId() %>"
- portletItemId="<%= portletItem.getPortletItemId() %>"
- title="<%= HtmlUtil.escape(portletItem.getName()) %>"
- >
- <p><%= HtmlUtil.escape(portletItem.getName()) %> <a href="javascript:;"><liferay-ui:message key="add" /></a></p>
- </div>
因为昨天我们研究过,这个文本任意变动都不会影响到添加Portlet事件的触发(参见http://supercharles888.blog.51cto.com/609344/908773)文章,所以我们确定点击事件和显示内容无关,而页面上除了这个文本以外任何地方点击都无效(不触发添加Portlet事件),由此看来,我们的事件最终是绑定到<a>元素的,因为这是它唯一和其他部分不同的地方。
最终,我们在/html/js/liferay/layout_configuration.js 中找到了事件绑定关联的地方,它在_loadContent方法中:
- _loadContent: function() {
- var instance = this;
- Liferay.fire('initLayout');
- instance.init();
- Util.addInputType();
- Liferay.on('closePortlet', instance._onPortletClose, instance);
- instance._portletItems = instance._dialogBody.all('div.lfr-portlet-item');
- var portlets = instance._portletItems;
- instance._dialogBody.delegate(
- 'mousedown',
- function(event) {
- var link = event.currentTarget;
- var portlet = link.ancestor('.lfr-portlet-item');
- instance._addPortlet(portlet);
- },
- 'a'
- );
从这里我们看出来,它先把左边对话框中所有的div.lfr-portlet-item都加进来作为portletItem,而我们的每个portletItem刚好位于类选择器lfr-portlet-item中,所以被选中。然后它用delegate方法吧为父元素绑定监听器监听子元素,在这里就是为lfr-portlet-item中绑定监听器来监听子元素a(锚点-port)【关于delegate我一直不太明白,还好同事js高手帮我解决了这个疑惑】,(小插曲,这里我不得不感慨Liferay框架设计者的精妙思路,因为页面上有多个portlet,每个portlet最后都有一个a元素,因为左边对话框元素是固定的,但是对话框中包含的portlet数量是不固定的,所以直接绑定事件有很大的困难,不如索性还是让对话框当监听器,让其委托其子元素中的a来触发事件,这样就算<a>再多,也不会影响事件的触发)所以现在,每个<a>当被鼠标按下去,就会触发事件处理函数_addPortlet(portlet),这就是我们关联到的事件处理函数:
- _addPortlet: function(portlet, options) {
- var instance = this;
- var portletMetaData = instance._getPortletMetaData(portlet);
- if (!portletMetaData.portletUsed) {
- var plid = portletMetaData.plid;
- var portletId = portletMetaData.portletId;
- var portletItemId = portletMetaData.portletItemId;
- var isInstanceable = portletMetaData.instanceable;
- if (!isInstanceable) {
- instance._disablePortletEntry(portletId);
- }
- var beforePortletLoaded = null;
- var placeHolder = A.Node.create('<div class="loading-animation" />');
- if (options) {
- var item = options.item;
- item.placeAfter(placeHolder);
- item.remove(true);
- beforePortletLoaded = options.beforePortletLoaded;
- }
- else {
- var layoutOptions = Layout.options;
- var firstColumn = Layout.getActiveDropNodes().item(0);
- if (firstColumn) {
- var dropColumn = firstColumn.one(layoutOptions.dropContainer);
- var referencePortlet = Layout.findReferencePortlet(dropColumn);
- if (referencePortlet) {
- referencePortlet.placeBefore(placeHolder);
- }
- else {
- if (dropColumn) {
- dropColumn.append(placeHolder);
- }
- }
- }
- }
- var portletOptions = {
- beforePortletLoaded: beforePortletLoaded,
- plid: plid,
- placeHolder: placeHolder,
- portletId: portletId,
- portletItemId: portletItemId
- };
- Liferay.Portlet.add(portletOptions);
- }
- }
从这段函数我们可以看出,它在04行通过_getPortletMetaData方法,先获得portlet的元数据,比如(instancable,plid,portletId,portletItemId等信息)。然后06行判断这个portlet是否未被使用,如果没有,则第07-10行从元数据的json对象中分类出instancable,plid等信息,然后12行做出是否instancable的判断来决定这个portlet是否可以被重复添加。然后第17行创建一个placeholder用于页面上加载新portlet的容器,然后进行一系列判断和检查,最终第47-53行,吧所有这个portlet有关的信息放入一个json对象叫portletOptions,然后第55行调用Liferay.Portlet.add方法来渲染这个portlet.
这个Liferay.Portlet.add方法定义在/html/js/liferay/portlet.js文件中:
- Liferay.provide(
- Portlet,
- 'add',
- function(options) {
- var instance = this;
- Liferay.fire('initLayout');
- var plid = options.plid || themeDisplay.getPlid();
- var portletId = options.portletId;
- var portletItemId = options.portletItemId;
- var doAsUserId = options.doAsUserId || themeDisplay.getDoAsUserIdEncoded();
- var placeHolder = options.placeHolder;
- if (!placeHolder) {
- placeHolder = A.Node.create('<div class="loading-animation" />');
- }
- else {
- placeHolder = A.one(placeHolder);
- }
- var positionOptions = options.positionOptions;
- var beforePortletLoaded = options.beforePortletLoaded;
- var onComplete = options.onComplete;
- var container = null;
- if (Liferay.Layout && Liferay.Layout.INITIALIZED) {
- container = Liferay.Layout.getActiveDropContainer();
- }
- if (!container) {
- return;
- }
- var portletPosition = 0;
- var currentColumnId = Util.getColumnId(container.attr('id'));
- if (options.placeHolder) {
- var column = placeHolder.get('parentNode');
- if (!column) {
- return;
- }
- placeHolder.addClass('portlet-boundary');
- portletPosition = column.all('.portlet-boundary').indexOf(placeHolder);
- currentColumnId = Util.getColumnId(column.attr('id'));
- }
- var url = themeDisplay.getPathMain() + '/portal/update_layout';
- var data = {
- cmd: 'add',
- dataType: 'json',
- doAsUserId: doAsUserId,
- p_l_id: plid,
- p_p_col_id: currentColumnId,
- p_p_col_pos: portletPosition,
- p_p_id: portletId,
- p_p_i_id: portletItemId,
- p_p_isolated: true,
- p_v_g_id: themeDisplay.getParentGroupId()
- };
- var firstPortlet = container.one('.portlet-boundary');
- var hasStaticPortlet = (firstPortlet && firstPortlet.isStatic);
- if (!options.placeHolder && !options.plid) {
- if (!hasStaticPortlet) {
- container.prepend(placeHolder);
- }
- else {
- firstPortlet.placeAfter(placeHolder);
- }
- }
- if (themeDisplay.isFreeformLayout()) {
- container.prepend(placeHolder);
- }
- data.currentURL = Liferay.currentURL;
- return instance.addHTML(
- {
- beforePortletLoaded: beforePortletLoaded,
- data: data,
- onComplete: onComplete,
- placeHolder: placeHolder,
- url: url
- }
- );
- },
- ['aui-base']
- );
从这里可以看出,它09-14行中从options中取出所有的信息,然后为渲染的内容添加一组样式类,最终发送到的目的地是var url = themeDisplay.getPathMain() + '/portal/update_layout';
因为我们在文章http://supercharles888.blog.51cto.com/609344/905695中已经给出了themeDisplay.getPathMain()代表的是'c',所以最终发送到的请求目的地是host:port/c/portal/update_layout,并且携带所有必要数据(56-67)行,然后吧在页面上嵌入一段url,吧从服务器端返回的内容进行填充,并且在第87-94行用数据并且调用Liferay.addHTML方法进行渲染。
对比浏览器调试器的信息,我们可以清楚的看到客户端发送到服务器端的json对象的请求内容刚好匹配56-67行:
Part 2: 分析这个携带json数据发送到/c/portal/update_layout的过程的幕后。
首先,因为我们Liferay服务器已经正确加载,而且Liferay的MainServlet是基于Struts中央控制器的,所以这个/c/portal/update_layout必须会进入到Struts框架:
对照struts-config.xml的url-mapping的设定:
- <action path="/portal/update_layout" type="com.liferay.portal.action.UpdateLayoutAction" />
我们可以看到,这个最终是UpdateLayoutAction类来处理这个请求:
这个UpdateLayoutAction是标准的Struts的Action并且它的直接父类是JSONAction:
它的execute方法最终会调用getJSON方法,先从json参数中获取cmd参数和获得portletId:
- String cmd = ParamUtil.getString(request, Constants.CMD);
- String portletId = ParamUtil.getString(request, "p_p_id");
因为我们的cmd参数的内容是"add",所以它会去执行以下代码:
- if (cmd.equals(Constants.ADD)) {
- String columnId = ParamUtil.getString(request, "p_p_col_id", null);
- int columnPos = ParamUtil.getInteger(request, "p_p_col_pos", -1);
- portletId = layoutTypePortlet.addPortletId(
- userId, portletId, columnId, columnPos);
- if (layoutTypePortlet.isCustomizable() &&
- layoutTypePortlet.isCustomizedView() &&
- !layoutTypePortlet.isColumnDisabled(columnId)) {
- updateLayout = false;
- }
- }
这段代码的内容就是02-03行获取p_p_col_id参数和p_p_col_pos参数。然后第05-06行用LayoutTypePortlet的addPorttletId方法吧这个portletId添加到Layout中(参见LayoutTypePortletImpl的addPortletId定义),现在这个portlet就在Layout中可用了。
执行完这段代码后,继续会去执行以下代码:
- if (cmd.equals(Constants.ADD) && (portletId != null)) {
- addPortlet(mapping, form, request, response, portletId);
- }
它会去调用UpdateLayoutAction的addPortlet方法:
- protected void addPortlet(
- ActionMapping mapping, ActionForm form, HttpServletRequest request,
- HttpServletResponse response, String portletId)
- throws Exception {
- // Run the render portlet action to add a portlet without refreshing.
- Action renderPortletAction = (Action)InstancePool.get(
- RenderPortletAction.class.getName());
- // Pass in the portlet id because the portlet id may be the instance id.
- // Namespace the request if necessary. See LEP-4644.
- long companyId = PortalUtil.getCompanyId(request);
- Portlet portlet = PortletLocalServiceUtil.getPortletById(
- companyId, portletId);
- DynamicServletRequest dynamicRequest = null;
- if (portlet.isPrivateRequestAttributes()) {
- String portletNamespace =
- PortalUtil.getPortletNamespace(portlet.getPortletId());
- dynamicRequest = new NamespaceServletRequest(
- request, portletNamespace, portletNamespace);
- }
- else {
- dynamicRequest = new DynamicServletRequest(request);
- }
- dynamicRequest.setParameter("p_p_id", portletId);
- String dataType = ParamUtil.getString(request, "dataType");
- if (dataType.equals("json")) {
- JSONObject jsonObject = JSONFactoryUtil.createJSONObject();
- StringServletResponse stringResponse = new StringServletResponse(
- response);
- renderPortletAction.execute(
- mapping, form, dynamicRequest, stringResponse);
- populatePortletJSONObject(
- request, stringResponse, portlet, jsonObject);
- response.setContentType(ContentTypes.TEXT_JAVASCRIPT);
- ServletResponseUtil.write(response, jsonObject.toString());
- }
- else {
- renderPortletAction.execute(
- mapping, form, dynamicRequest, response);
- }
- }
从这里可以看出,它实际是调用RenderPortletAction的execute方法来对这个portlet进行渲染。如何渲染呢?我们继续跟进。
- public ActionForward execute(
- ActionMapping mapping, ActionForm form, HttpServletRequest request,
- HttpServletResponse response)
- throws Exception {
- ServletContext servletContext = (ServletContext)request.getAttribute(
- WebKeys.CTX);
- String ajaxId = request.getParameter("ajax_id");
- long companyId = PortalUtil.getCompanyId(request);
- User user = PortalUtil.getUser(request);
- Layout layout = (Layout)request.getAttribute(WebKeys.LAYOUT);
- String portletId = ParamUtil.getString(request, "p_p_id");
- Portlet portlet = PortletLocalServiceUtil.getPortletById(
- companyId, portletId);
- String queryString = null;
- String columnId = ParamUtil.getString(request, "p_p_col_id");
- int columnPos = ParamUtil.getInteger(request, "p_p_col_pos");
- int columnCount = ParamUtil.getInteger(request, "p_p_col_count");
- boolean staticPortlet = ParamUtil.getBoolean(request, "p_p_static");
- boolean staticStartPortlet = ParamUtil.getBoolean(
- request, "p_p_static_start");
- if (staticPortlet) {
- portlet = (Portlet)portlet.clone();
- portlet.setStatic(true);
- portlet.setStaticStart(staticStartPortlet);
- }
- if (ajaxId != null) {
- response.setHeader("Ajax-ID", ajaxId);
- }
- WindowState windowState = WindowStateFactory.getWindowState(
- ParamUtil.getString(request, "p_p_state"));
- PortalUtil.updateWindowState(
- portletId, user, layout, windowState, request);
- PortalUtil.renderPortlet(
- servletContext, request, response, portlet, queryString, columnId,
- new Integer(columnPos), new Integer(columnCount), true);
- return null;
- }
从这里我们可以看出,从06-31行只是设置一些参数,然后从第38-42行获取并且更新窗口状态(最大,最小,中等),然后最后44行调用PortalUtil的renderPortlet方法来渲染Portlet:
我们继续跟进到PortalUtil的renderPortlet,经过一系列封装之后,它会调用PortalImpl的renderPortlet方法:
- public String renderPortlet(
- ServletContext servletContext, HttpServletRequest request,
- HttpServletResponse response, Portlet portlet, String queryString,
- String columnId, Integer columnPos, Integer columnCount,
- String path, boolean writeOutput)
- throws IOException, ServletException {
- queryString = GetterUtil.getString(queryString);
- columnId = GetterUtil.getString(columnId);
- if (columnPos == null) {
- columnPos = Integer.valueOf(0);
- }
- if (columnCount == null) {
- columnCount = Integer.valueOf(0);
- }
- request.setAttribute(WebKeys.RENDER_PORTLET, portlet);
- request.setAttribute(WebKeys.RENDER_PORTLET_QUERY_STRING, queryString);
- request.setAttribute(WebKeys.RENDER_PORTLET_COLUMN_ID, columnId);
- request.setAttribute(WebKeys.RENDER_PORTLET_COLUMN_POS, columnPos);
- request.setAttribute(WebKeys.RENDER_PORTLET_COLUMN_COUNT, columnCount);
- if (path == null) {
- path = "/html/portal/render_portlet.jsp";
- }
- RequestDispatcher requestDispatcher =
- servletContext.getRequestDispatcher(path);
- ..
可以发现,它最终是调用/html/portal/render_portlet.jsp去渲染页面。
- ...
- else {
- if (useDefaultTemplate) {
- renderRequestImpl.setAttribute(WebKeys.PORTLET_CONTENT, stringResponse.getString());
- >
- <tiles:insert template='<%= StrutsUtil.TEXT_HTML_DIR + "/common/themes/portlet.jsp" %>' flush="false">
- <tiles:put name="portlet_content" value="<%= StringPool.BLANK %>" />
- </tiles:insert>
- %
- }
- else {
- stringResponse.writeTo(pageContext.getOut());
- }
- }
- ...
而这页面,最终会去调用Tiles框架来进行渲染:比如为了解析07-09行的<tiles>标记,因为定义了它的tld:
- <%@ taglib uri="http://struts.apache.org/tags-tiles" prefix="tiles" %>
所以就可以去web.xml找它的标记库定义:
- <taglib>
- <taglib-uri>http://struts.apache.org/tags-tiles</taglib-uri>
- <taglib-location>/WEB-INF/tld/struts-tiles.tld</taglib-location>
- </taglib>
然后从去struts-tiles.tld去找到其标记库的定义,然后就可以找到解析类了(这里看出页面上<tiles:insert>会被InsertTag类所消费),后续不再一一展开:
- <!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.1//EN" "http://java.sun.com/j2ee/dtds/web-jsptaglibrary_1_1.dtd">
- <taglib>
- <tlibversion>1.2</tlibversion>
- <jspversion>1.1</jspversion>
- <shortname>tiles</shortname>
- <uri>http://struts.apache.org/tags-tiles</uri>
- <tag>
- <name>insert</name>
- <tagclass>org.apache.struts.taglib.tiles.InsertTag</tagclass>
- <bodycontent>JSP</bodycontent>
- <attribute>
- <name>template</name>
- <required>false</required>
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>component</name>
- <required>false</required>
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>page</name>
- <required>false</required>
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>definition</name>
- <required>false</required>
- <rtexprvalue>true</rtexprvalue>
- </attribute>
- <attribute>
- <name>attribute</name>
- <required>false</required>
- <rtexprvalue>false</rtexprvalue>
- </attribute>
- <attribute>
- <name>name</name>
- <required>false</required>
- <rtexprvalue>true</rtexprvalue>
- ..