diff options
author | Artur Signell <artur.signell@itmill.com> | 2009-04-03 09:14:07 +0000 |
---|---|---|
committer | Artur Signell <artur.signell@itmill.com> | 2009-04-03 09:14:07 +0000 |
commit | c0e35f4ea8fcc7c61ae2b2d4882373217da1b3af (patch) | |
tree | 760de7fa3f063d9070eb7ee77a36e677387fe324 /src/com/itmill/toolkit/terminal/gwt | |
parent | 1216fd25d6ee545db3b205369d7624d3201fba5f (diff) | |
download | vaadin-framework-c0e35f4ea8fcc7c61ae2b2d4882373217da1b3af.tar.gz vaadin-framework-c0e35f4ea8fcc7c61ae2b2d4882373217da1b3af.zip |
Merged Testing Tools branch
svn changeset:7298/svn branch:6.0
Diffstat (limited to 'src/com/itmill/toolkit/terminal/gwt')
6 files changed, 342 insertions, 124 deletions
diff --git a/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConfiguration.java b/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConfiguration.java index 69006a81e2..0560385799 100644 --- a/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConfiguration.java +++ b/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConfiguration.java @@ -21,6 +21,7 @@ public class ApplicationConfiguration { private String communicationErrorCaption; private String communicationErrorMessage; private String communicationErrorUrl; + private boolean useDebugIdInDom = true; private static ArrayList<ApplicationConnection> unstartedApplications = new ArrayList<ApplicationConnection>(); private static ArrayList<ApplicationConnection> runningApplications = new ArrayList<ApplicationConnection>(); @@ -85,6 +86,9 @@ public class ApplicationConfiguration { if(jsobj.windowName) { this.@com.itmill.toolkit.terminal.gwt.client.ApplicationConfiguration::windowName = jsobj.windowName; } + if('useDebugIdInDom' in jsobj && typeof(jsobj.useDebugIdInDom) == "boolean") { + this.@com.itmill.toolkit.terminal.gwt.client.ApplicationConfiguration::useDebugIdInDom = jsobj.useDebugIdInDom; + } if(jsobj.versionInfo) { this.@com.itmill.toolkit.terminal.gwt.client.ApplicationConfiguration::versionInfo = jsobj.versionInfo; } @@ -190,4 +194,8 @@ public class ApplicationConfiguration { /*-{ return this.@com.itmill.toolkit.terminal.gwt.client.ApplicationConfiguration::versionInfo.applicationVersion; }-*/; + + public boolean useDebugIdInDOM() { + return useDebugIdInDom; + } } diff --git a/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConnection.java b/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConnection.java index df0f9f5519..f3541b5bb3 100755 --- a/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConnection.java +++ b/src/com/itmill/toolkit/terminal/gwt/client/ApplicationConnection.java @@ -31,7 +31,6 @@ import com.google.gwt.user.client.Element; import com.google.gwt.user.client.Event; import com.google.gwt.user.client.Timer; import com.google.gwt.user.client.Window; -import com.google.gwt.user.client.WindowCloseListener; import com.google.gwt.user.client.impl.HTTPRequestImpl; import com.google.gwt.user.client.ui.FocusWidget; import com.google.gwt.user.client.ui.HasWidgets; @@ -72,8 +71,6 @@ public class ApplicationConnection { private static Console console; - private static boolean testingMode; - private final Vector<String> pendingVariables = new Vector<String>(); private final ComponentDetailMap idToPaintableDetail = ComponentDetailMap @@ -92,21 +89,6 @@ public class ApplicationConnection { private boolean applicationRunning = false; - /** - * True if each Paintable objects id is injected to DOM. Used for Testing - * Tools. - */ - private boolean usePaintableIdsInDOM = false; - - /** - * Contains reference for client wrapper given to Testing Tools. - * - * Used in JSNI functions - * - */ - @SuppressWarnings("unused") - private final JavaScriptObject ttClientWrapper = null; - private int activeRequests = 0; /** Parameters for this application connection loaded from the web-page */ @@ -137,26 +119,19 @@ public class ApplicationConnection { this.widgetSet = widgetSet; configuration = cnf; windowName = configuration.getInitialWindowName(); - if (isDebugMode()) { console = new IDebugConsole(this, cnf, !isQuietDebugMode()); } else { console = new NullConsole(); } - if (checkTestingMode()) { - usePaintableIdsInDOM = true; - initializeTestingTools(); - Window.addWindowCloseListener(new WindowCloseListener() { - public void onWindowClosed() { - uninitializeTestingTools(); - } + ComponentLocator componentLocator = new ComponentLocator(this); - public String onWindowClosing() { - return null; - } - }); - } + String appRootPanelName = cnf.getRootPanelId(); + // remove the end (window name) of autogenarated rootpanel id + appRootPanelName = appRootPanelName.replaceFirst("-\\d+$", ""); + + initializeTestingToolsHooks(componentLocator, appRootPanelName); initializeClientHooks(); @@ -176,33 +151,8 @@ public class ApplicationConnection { makeUidlRequest("", true, false, false); } - /** - * Method to check if application is in testing mode. Can be used after - * application init. - * - * @return true if in testing mode - */ - public static boolean isTestingMode() { - return testingMode; - } - - /** - * Check is application is run in testing mode. - * - * @return true if in testing mode - */ - private native static boolean checkTestingMode() - /*-{ - try { - @com.itmill.toolkit.terminal.gwt.client.ApplicationConnection::testingMode = $wnd.top.itmill && $wnd.top.itmill.registerToTT ? true : false; - return @com.itmill.toolkit.terminal.gwt.client.ApplicationConnection::testingMode; - } catch(e) { - // if run in iframe SOP may cause exception, return false then - return false; - } - }-*/; - - private native void initializeTestingTools() + private native void initializeTestingToolsHooks( + ComponentLocator componentLocator, String TTAppId) /*-{ var ap = this; var client = {}; @@ -215,8 +165,19 @@ public class ApplicationConnection { return vi; } } - $wnd.top.itmill.registerToTT(client); - this.@com.itmill.toolkit.terminal.gwt.client.ApplicationConnection::ttClientWrapper = client; + + client.getElementByPath = function(id) { + return componentLocator.@com.itmill.toolkit.terminal.gwt.client.ComponentLocator::getElementByPath(Ljava/lang/String;)(id); + } + client.getPathForElement = function(element) { + return componentLocator.@com.itmill.toolkit.terminal.gwt.client.ComponentLocator::getPathForElement(Lcom/google/gwt/user/client/Element;)(element); + } + + if(!$wnd.itmill.clients) { + $wnd.itmill.clients = {}; + } + + $wnd.itmill.clients[TTAppId] = client; }-*/; /** @@ -227,11 +188,6 @@ public class ApplicationConnection { return configuration.getVersionInfoJSObject(); } - private native void uninitializeTestingTools() - /*-{ - $wnd.top.itmill.unregisterFromTT(this.@com.itmill.toolkit.terminal.gwt.client.ApplicationConnection::ttClientWrapper); - }-*/; - /** * Publishes a JavaScript API for mash-up applications. * <ul> @@ -804,15 +760,19 @@ public class ApplicationConnection { el.tkPid = pid; }-*/; - private String getPid(Paintable paintable) { + public String getPid(Paintable paintable) { return getPid(((Widget) paintable).getElement()); } - private native String getPid(Element el) + public native String getPid(Element el) /*-{ return el.tkPid; }-*/; + public Element getElementByPid(String pid) { + return ((Widget) getPaintable(pid)).getElement(); + } + public void unregisterPaintable(Paintable p) { if (p == null) { ApplicationConnection.getConsole().error( @@ -1165,8 +1125,9 @@ public class ApplicationConnection { } } - if (usePaintableIdsInDOM) { - DOM.setElementProperty(component.getElement(), "id", uidl.getId()); + if (configuration.useDebugIdInDOM() && uidl.getId().startsWith("PID_S")) { + DOM.setElementProperty(component.getElement(), "id", uidl.getId() + .substring(5)); } /* @@ -1499,10 +1460,8 @@ public class ApplicationConnection { public IContextMenu getContextMenu() { if (contextMenu == null) { contextMenu = new IContextMenu(); - if (usePaintableIdsInDOM) { - DOM.setElementProperty(contextMenu.getElement(), "id", - "PID_TOOLKIT_CM"); - } + DOM.setElementProperty(contextMenu.getElement(), "id", + "PID_TOOLKIT_CM"); } return contextMenu; } @@ -1660,4 +1619,9 @@ public class ApplicationConnection { public void analyzeLayouts() { makeUidlRequest("", true, false, true); } + + public IView getView() { + return view; + } + } diff --git a/src/com/itmill/toolkit/terminal/gwt/client/ComponentLocator.java b/src/com/itmill/toolkit/terminal/gwt/client/ComponentLocator.java new file mode 100644 index 0000000000..b0078868dc --- /dev/null +++ b/src/com/itmill/toolkit/terminal/gwt/client/ComponentLocator.java @@ -0,0 +1,278 @@ +package com.itmill.toolkit.terminal.gwt.client; + +import java.util.Iterator; + +import com.google.gwt.user.client.DOM; +import com.google.gwt.user.client.Element; +import com.google.gwt.user.client.ui.HasWidgets; +import com.google.gwt.user.client.ui.Widget; +import com.itmill.toolkit.terminal.gwt.client.ui.IView; +import com.itmill.toolkit.terminal.gwt.client.ui.SubPartAware; + +/** + * Helper class for TestingTools that builds appropriate locators for browser + * bot. + */ +public class ComponentLocator { + + private static final String PARENTCHILD_SEPARATOR = "/"; + private static final String SUBPART_SEPARATOR = "#"; + + private ApplicationConnection client; + + public ComponentLocator(ApplicationConnection client) { + this.client = client; + } + + public String getPathForElement(Element targetElement) { + String pid = null; + + Element e = targetElement; + + while (true) { + pid = client.getPid(e); + if (pid != null) { + break; + } + + e = DOM.getParent(e); + if (e == null) { + break; + } + } + + if (e == null || pid == null) { + return null; + } + + Widget w = (Widget) client.getPaintable(pid); + if (w == null) { + return null; + } + // ApplicationConnection.getConsole().log( + // "First parent widget: " + Util.getSimpleName(w)); + + String path = getPathForWidget(w); + // ApplicationConnection.getConsole().log( + // "getPathFromWidget returned " + path); + if (w.getElement() == targetElement) { + // ApplicationConnection.getConsole().log( + // "Path for " + Util.getSimpleName(w) + ": " + path); + + return path; + } else if (w instanceof SubPartAware) { + return path + SUBPART_SEPARATOR + + ((SubPartAware) w).getSubPartName(targetElement); + } else { + path = path + getDOMPathForElement(targetElement, w.getElement()); + // ApplicationConnection.getConsole().log( + // "Path with dom addition for " + Util.getSimpleName(w) + // + ": " + path); + + return path; + } + } + + private Element getElementByDOMPath(Element baseElement, String path) { + String parts[] = path.split(PARENTCHILD_SEPARATOR); + Element element = baseElement; + + for (String part : parts) { + if (part.startsWith("domChild[")) { + String childIndexString = part.substring("domChild[".length(), + part.length() - 1); + try { + int childIndex = Integer.parseInt(childIndexString); + element = DOM.getChild(element, childIndex); + } catch (Exception e) { + // ApplicationConnection.getConsole().error( + // "Failed to parse integer in " + childIndexString); + return null; + } + } + } + + return element; + } + + private String getDOMPathForElement(Element element, Element baseElement) { + Element e = element; + String path = ""; + while (true) { + Element parent = DOM.getParent(e); + if (parent == null) { + return "ERROR, baseElement is not a parent to element"; + } + + int childIndex = -1; + + int childCount = DOM.getChildCount(parent); + for (int i = 0; i < childCount; i++) { + if (e == DOM.getChild(parent, i)) { + childIndex = i; + break; + } + } + if (childIndex == -1) { + return "ERROR, baseElement is not a parent to element."; + } + + path = PARENTCHILD_SEPARATOR + "domChild[" + childIndex + "]" + + path; + + if (parent == baseElement) { + break; + } + + e = parent; + } + + return path; + } + + public Element getElementByPath(String path) { + // ApplicationConnection.getConsole() + // .log("getElementByPath(" + path + ")"); + + // Path is of type "PID/componentPart" + String parts[] = path.split(SUBPART_SEPARATOR, 2); + String widgetPath = parts[0]; + Widget w = getWidgetFromPath(widgetPath); + if (w == null) { + return null; + } + + if (parts.length == 1) { + int pos = widgetPath.indexOf("domChild"); + if (pos == -1) { + return w.getElement(); + } + + // Contains dom reference to a sub element of the widget + String subPath = widgetPath.substring(pos); + return getElementByDOMPath(w.getElement(), subPath); + } else if (parts.length == 2) { + if (w instanceof SubPartAware) { + // ApplicationConnection.getConsole().log( + // "subPartAware: " + parts[1]); + return ((SubPartAware) w).getSubPartElement(parts[1]); + } else { + // ApplicationConnection.getConsole().error( + // "getElementByPath failed because " + // + Util.getSimpleName(w) + // + " is not SubPartAware"); + return null; + } + } + + return null; + } + + private String getPathForWidget(Widget w) { + String pid = client.getPid(w.getElement()); + if (isStaticPid(pid)) { + return pid; + } + + if (w instanceof IView) { + return ""; + } + + Widget parent = w.getParent(); + + String basePath = getPathForWidget(parent); + + String simpleName = Util.getSimpleName(w); + + Iterator i = ((HasWidgets) parent).iterator(); + int pos = 0; + while (i.hasNext()) { + Object child = i.next(); + if (child == w) { + return basePath + PARENTCHILD_SEPARATOR + simpleName + "[" + + pos + "]"; + } + String simpleName2 = Util.getSimpleName(child); + if (simpleName.equals(simpleName2)) { + pos++; + } + } + + return "NOTFOUND"; + } + + private Widget getWidgetFromPath(String path) { + Widget w = null; + String parts[] = path.split(PARENTCHILD_SEPARATOR); + + // ApplicationConnection.getConsole().log( + // "getWidgetFromPath(" + path + ")"); + + for (String part : parts) { + // ApplicationConnection.getConsole().log("Part: " + part); + // ApplicationConnection.getConsole().log( + // "Widget: " + Util.getSimpleName(w)); + if (part.equals("")) { + w = client.getView(); + } else if (w == null) { + w = (Widget) client.getPaintable(part); + } else if (part.startsWith("domChild[")) { + break; + } else if (w instanceof HasWidgets) { + HasWidgets parent = (HasWidgets) w; + + String simpleName = Util.getSimpleName(parent); + + Iterator i = parent.iterator(); + + boolean ok = false; + String[] split = part.split("\\["); + int pos = Integer.parseInt(split[1].substring(0, split[1] + .length() - 1)); + // ApplicationConnection.getConsole().log( + // "Looking for child " + pos); + while (i.hasNext()) { + // ApplicationConnection.getConsole().log("- child found"); + + Widget child = (Widget) i.next(); + String simpleName2 = Util.getSimpleName(child); + + if (split[0].equals(simpleName2)) { + if (pos == 0) { + w = child; + ok = true; + break; + } + pos--; + } + } + + if (!ok) { + // Did not find the child + // ApplicationConnection.getConsole().error( + // "getWidgetFromPath(" + path + ") - did not find '" + // + part + "' for " + // + Util.getSimpleName(parent)); + + return null; + } + } else { + // ApplicationConnection.getConsole().error( + // "getWidgetFromPath(" + path + ") - failed for '" + part + // + "'"); + return null; + } + } + + return w; + } + + private boolean isStaticPid(String pid) { + if (pid == null) { + return false; + } + + return pid.startsWith("PID_S"); + } + +} diff --git a/src/com/itmill/toolkit/terminal/gwt/client/ui/IFilterSelect.java b/src/com/itmill/toolkit/terminal/gwt/client/ui/IFilterSelect.java index e76374cfc2..3b0fdd7927 100644 --- a/src/com/itmill/toolkit/terminal/gwt/client/ui/IFilterSelect.java +++ b/src/com/itmill/toolkit/terminal/gwt/client/ui/IFilterSelect.java @@ -131,11 +131,9 @@ public class IFilterSelect extends Composite implements Paintable, Field, Collection<FilterSelectSuggestion> currentSuggestions, int currentPage, int totalSuggestions) { - if (ApplicationConnection.isTestingMode()) { - // Add TT anchor point - DOM.setElementProperty(getElement(), "id", paintableId - + "_OPTIONLIST"); - } + // Add TT anchor point + DOM.setElementProperty(getElement(), "id", + "TOOLKIT_COMBOBOX_OPTIONLIST"); menu.setSuggestions(currentSuggestions); final int x = IFilterSelect.this.getAbsoluteLeft(); diff --git a/src/com/itmill/toolkit/terminal/gwt/client/ui/SubPartAware.java b/src/com/itmill/toolkit/terminal/gwt/client/ui/SubPartAware.java new file mode 100644 index 0000000000..76ba7524be --- /dev/null +++ b/src/com/itmill/toolkit/terminal/gwt/client/ui/SubPartAware.java @@ -0,0 +1,11 @@ +package com.itmill.toolkit.terminal.gwt.client.ui;
+
+import com.google.gwt.user.client.Element;
+
+public interface SubPartAware {
+
+ Element getSubPartElement(String subPart);
+
+ String getSubPartName(Element subElement);
+
+}
\ No newline at end of file diff --git a/src/com/itmill/toolkit/terminal/gwt/server/ApplicationServlet.java b/src/com/itmill/toolkit/terminal/gwt/server/ApplicationServlet.java index bada4af66f..b9c8d58ff5 100644 --- a/src/com/itmill/toolkit/terminal/gwt/server/ApplicationServlet.java +++ b/src/com/itmill/toolkit/terminal/gwt/server/ApplicationServlet.java @@ -10,7 +10,6 @@ import java.io.InputStream; import java.io.OutputStream; import java.io.OutputStreamWriter; import java.io.PrintWriter; -import java.io.Writer; import java.lang.reflect.Constructor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; @@ -165,10 +164,6 @@ public class ApplicationServlet extends HttpServlet { private ClassLoader classLoader; - private boolean testingToolsActive = false; - - private String testingToolsServerUri = null; - /** * Called by the servlet container to indicate to a servlet that the servlet * is being placed into service. @@ -234,14 +229,6 @@ public class ApplicationServlet extends HttpServlet { System.err.println(NOT_PRODUCTION_MODE_INFO); } - // Gets Testing Tools parameters if feature is activated - if (getApplicationOrSystemProperty("testingToolsActive", "false") - .equals("true")) { - testingToolsActive = true; - testingToolsServerUri = getApplicationOrSystemProperty( - "testingToolsServerUri", null); - } - // Gets custom class loader final String classLoaderName = getApplicationOrSystemProperty( "ClassLoader", null); @@ -844,9 +831,6 @@ public class ApplicationServlet extends HttpServlet { themeUri = staticFilePath + "/" + THEME_DIRECTORY_PATH + themeName; } - boolean testingApplication = testingToolsActive - && request.getParameter("TT") != null; - if (!fragment) { // Window renders are not cacheable response.setHeader("Cache-Control", "no-cache"); @@ -879,12 +863,6 @@ public class ApplicationServlet extends HttpServlet { page.write("<title>" + title + "</title>"); - if (testingApplication) { - // TT script needs to be in head as it needs to be the first - // to hook capturing event listeners - writeTestingToolsScripts(page, request); - } - page .write("\n</head>\n<body scroll=\"auto\" class=\"i-generated-body\">\n"); } @@ -896,7 +874,11 @@ public class ApplicationServlet extends HttpServlet { appId = appId.replaceAll("[^a-zA-Z0-9]", ""); // Add hashCode to the end, so that it is still (sort of) predictable, // but indicates that it should not be used in CSS and such: - appId = appId + appId.hashCode(); + int hashCode = appId.hashCode(); + if (hashCode < 0) { + hashCode = -hashCode; + } + appId = appId + "-" + hashCode; // Get system messages Application.SystemMessages systemMessages = null; @@ -1119,29 +1101,6 @@ public class ApplicationServlet extends HttpServlet { return false; } - private void writeTestingToolsScripts(Writer page, - HttpServletRequest request) throws IOException { - // Testing Tools script and CSS files are served from Testing Tools - // Server - String ext = getTestingToolsUri(request); - ext = ext.substring(0, ext.lastIndexOf('/')); - page.write("<script src=\"" + ext + "/ext/TT.js" - + "\" type=\"text/javascript\"></script>\n"); - page.write("<link rel=\"stylesheet\" href=\"" + ext + "/ext/TT.css" - + "\" type=\"text/css\" />\n"); - - } - - private String getTestingToolsUri(HttpServletRequest request) { - if (testingToolsServerUri == null) { - // Default behavior is that Testing Tools Server application exists - // on same host as current application does in port 8099. - testingToolsServerUri = "http" + "://" + request.getServerName() - + ":8099" + "/TestingToolsServer"; - } - return testingToolsServerUri; - } - /** * Handles the requested URI. An application can add handlers to do special * processing, when a certain URI is requested. The handlers are invoked |