diff options
author | John Ahlroos <john@vaadin.com> | 2012-09-20 10:07:32 +0300 |
---|---|---|
committer | John Ahlroos <john@vaadin.com> | 2012-09-20 10:07:32 +0300 |
commit | 5baf9bc92ddb66c93d0d2e16a95900ef49946c71 (patch) | |
tree | ea546085430a2385620361b1e846db0742e864d3 | |
parent | 96d1ba6b53c2b8380a6cb4b3b1040e291a1a8766 (diff) | |
parent | 0dda5a66cf4dc1f033f82c57565b0aa25506f124 (diff) | |
download | vaadin-framework-5baf9bc92ddb66c93d0d2e16a95900ef49946c71.tar.gz vaadin-framework-5baf9bc92ddb66c93d0d2e16a95900ef49946c71.zip |
Merge branch 'master' into html5-doctype
32 files changed, 1390 insertions, 664 deletions
diff --git a/all/build.xml b/all/build.xml index 4281a63b2f..2e9bf9a89d 100644 --- a/all/build.xml +++ b/all/build.xml @@ -10,9 +10,16 @@ <!-- global properties --> <property name="module.name" value="vaadin-all" /> <property name="result.dir" value="result" /> + <property name="javadoc.jar" location="${result.dir}/vaadin-all-${vaadin.version}-javadoc.jar" /> <property name="temp.dir" location="${result.dir}/temp" /> + <property name="temp.deps.dir" value="${temp.dir}/lib" /> + <property name="javadoc.temp.dir" location="${result.dir}/javadoc-temp" /> <property name="zip.file" location="${result.dir}/lib/${module.name}-${vaadin.version}.zip" /> + <path id="classpath.javadoc"> + <fileset dir="${temp.deps.dir}" includes="*.jar"> + </fileset> + </path> <target name="fetch.module.and.dependencies"> <fail unless="module" message="No 'module' parameter given" /> @@ -22,17 +29,53 @@ </copy> </target> - <target name="zip"> + <target name="unzip.to.javadoctemp"> + <property name="file" location="${temp.dir}/vaadin-${module}-${vaadin.version}.jar" /> + <unzip src="${file}" dest="${javadoc.temp.dir}" /> + </target> + + <target name="javadoc" depends="copy-jars"> + <!-- Unpack all source files to javadoc.temp.dir--> + <antcontrib:foreach list="${modules.to.publish.to.maven}" target="unzip.to.javadoctemp" param="module" /> + + <property name="javadoc.dir" location="${result.dir}/javadoc" /> + <property name="title" value="Vaadin" /> + <javadoc destdir="${javadoc.dir}" author="true" version="true" use="true" windowtitle="${title}" encoding="utf-8"> + <packageset dir="${javadoc.temp.dir}"> + <!-- TODO Javadoc throws ClassCastException if this is included (#9660)--> + <exclude name="com/google/gwt/uibinder/elementparsers" /> + </packageset> + <doctitle><h1>${title}</h1></doctitle> + <!-- <header><![CDATA[<script type="text/javascript" src=".html-style/style.js"></script>]]></header> --> + <bottom>${javadoc.bottom}</bottom> + <link offline="true" href="http://docs.oracle.com/javase/6/docs/api/" packagelistLoc="build/javadoc/j2se-1.6.0" /> + <link offline="true" href="http://java.sun.com/j2ee/1.4/docs/api/" packagelistLoc="build/javadoc/j2ee-1.4" /> + <classpath refid="classpath.javadoc" /> + </javadoc> + + <!-- Create a javadoc jar --> + <jar file="${javadoc.jar}" compress="true"> + <fileset dir="${javadoc.dir}" /> + <fileset refid="common.files.for.all.jars" /> + </jar> + + + </target> + + <target name="copy-jars"> <delete dir="${temp.dir}" /> <antcontrib:foreach list="${modules.to.publish.to.maven}" target="fetch.module.and.dependencies" param="module" /> <!-- All jars are now in temp.dir. Still need to separate vaadin and deps --> - <move todir="${temp.dir}/lib"> + <move todir="${temp.deps.dir}"> <fileset dir="${temp.dir}"> <exclude name="vaadin-*-${vaadin.version}.*" /> <exclude name="vaadin-*-${vaadin.version}-*.*" /> </fileset> </move> + + </target> + <target name="zip" depends="copy-jars"> <zip destfile="${zip.file}"> <fileset dir="${temp.dir}"> <!-- Avoid conflicts with servlet and portlet API. They are provided by the container --> @@ -48,8 +91,9 @@ </fileset> <fileset refid="common.files.for.all.jars" /> <fileset dir="${result.dir}/.."> - <include name="README.TXT"/> + <include name="README.TXT" /> </fileset> + <fileset file="${javadoc.jar}" /> </zip> </target> diff --git a/client/src/com/vaadin/client/ApplicationConnection.java b/client/src/com/vaadin/client/ApplicationConnection.java index 61d9792b82..ab28ad291b 100644 --- a/client/src/com/vaadin/client/ApplicationConnection.java +++ b/client/src/com/vaadin/client/ApplicationConnection.java @@ -555,7 +555,8 @@ public class ApplicationConnection { // TODO figure out how client and view size could be used better on // server. screen size can be accessed via Browser object, but other // values currently only via transaction listener. - String parameters = "repaintAll=1&" + nativeBootstrapParameters; + String parameters = ApplicationConstants.URL_PARAMETER_REPAINT_ALL + + "=1&" + nativeBootstrapParameters; return parameters; } diff --git a/client/src/com/vaadin/client/ComponentLocator.java b/client/src/com/vaadin/client/ComponentLocator.java index 1852b67260..3338147465 100644 --- a/client/src/com/vaadin/client/ComponentLocator.java +++ b/client/src/com/vaadin/client/ComponentLocator.java @@ -97,6 +97,8 @@ public class ComponentLocator { public String getPathForElement(Element targetElement) { String pid = null; + targetElement = getElement(targetElement); + Element e = targetElement; while (true) { @@ -176,6 +178,8 @@ public class ComponentLocator { return null; } + // The parent check is a work around for Firefox 15 which fails to + // compare elements properly (#9534) if (w.getElement() == targetElement) { /* * We are done if the target element is the root of the target @@ -206,6 +210,41 @@ public class ComponentLocator { } /** + * Returns the element passed to the method. Or in case of Firefox 15, + * returns the real element that is in the DOM instead of the element passed + * to the method (which is the same element but not ==). + * + * @param targetElement + * the element to return + * @return the element passed to the method + */ + private Element getElement(Element targetElement) { + if (targetElement == null) { + return null; + } + + if (!BrowserInfo.get().isFirefox()) { + return targetElement; + } + + if (BrowserInfo.get().getBrowserMajorVersion() != 15) { + return targetElement; + } + + // Firefox 15, you make me sad + if (targetElement.getNextSibling() != null) { + return (Element) targetElement.getNextSibling() + .getPreviousSibling(); + } + if (targetElement.getPreviousSibling() != null) { + return (Element) targetElement.getPreviousSibling() + .getNextSibling(); + } + // No siblings so this is the only child + return (Element) targetElement.getParentNode().getChild(0); + } + + /** * Finds the first widget in the hierarchy (moving upwards) that implements * SubPartAware. Returns the SubPartAware implementor or null if none is * found. @@ -316,27 +355,20 @@ public class ComponentLocator { Element e = element; String path = ""; while (true) { - Element parent = DOM.getParent(e); - if (parent == null) { - return null; - } - 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 null; + Element siblingIterator = e; + while (siblingIterator != null) { + childIndex++; + siblingIterator = siblingIterator.getPreviousSiblingElement() + .cast(); } path = PARENTCHILD_SEPARATOR + "domChild[" + childIndex + "]" + path; + Element parent = e.getParentElement().cast(); + // The parent check is a work around for Firefox 15 which fails to + // compare elements properly (#9534) if (parent == baseElement) { break; } diff --git a/server/src/com/vaadin/navigator/Navigator.java b/server/src/com/vaadin/navigator/Navigator.java index d3098903c5..257dfa208f 100644 --- a/server/src/com/vaadin/navigator/Navigator.java +++ b/server/src/com/vaadin/navigator/Navigator.java @@ -29,6 +29,7 @@ import com.vaadin.ui.Component; import com.vaadin.ui.ComponentContainer; import com.vaadin.ui.CssLayout; import com.vaadin.ui.CustomComponent; +import com.vaadin.ui.UI; /** * A navigator utility that allows switching of views in a part of an @@ -124,7 +125,7 @@ public class Navigator implements Serializable { @Override public void fragmentChanged(FragmentChangedEvent event) { - UriFragmentManager.this.navigator.navigateTo(getState()); + navigator.navigateTo(getState()); } } @@ -331,6 +332,7 @@ public class Navigator implements Serializable { } } + private final UI ui; private final NavigationStateManager stateManager; private final ViewDisplay display; private View currentView = null; @@ -354,30 +356,34 @@ public class Navigator implements Serializable { * the application should trigger navigation to the current fragment using * {@link #navigate()}. * + * @param ui + * The UI to which this Navigator is attached. * @param container - * ComponentContainer whose contents should be replaced with the - * active view on view change + * The ComponentContainer whose contents should be replaced with + * the active view on view change */ - public Navigator(ComponentContainer container) { - display = new ComponentContainerViewDisplay(container); - stateManager = new UriFragmentManager(Page.getCurrent(), this); + public Navigator(UI ui, ComponentContainer container) { + this(ui, new ComponentContainerViewDisplay(container)); } /** - * Creates a navigator that is tracking the active view using URI fragments. + * Creates a navigator that is tracking the active view using URI fragments + * of the Page containing the given UI. * <p> * After all {@link View}s and {@link ViewProvider}s have been registered, * the application should trigger navigation to the current fragment using * {@link #navigate()}. * - * @param page - * whose URI fragments are used + * @param ui + * The UI to which this Navigator is attached. * @param display - * where to display the views + * The ViewDisplay used to display the views. */ - public Navigator(Page page, ViewDisplay display) { + public Navigator(UI ui, ViewDisplay display) { + this.ui = ui; + this.ui.setNavigator(this); this.display = display; - stateManager = new UriFragmentManager(page, this); + stateManager = new UriFragmentManager(ui.getPage(), this); } /** @@ -391,14 +397,19 @@ public class Navigator implements Serializable { * the application should trigger navigation to the current fragment using * {@link #navigate()}. * + * @param ui + * The UI to which this Navigator is attached. * @param stateManager - * {@link NavigationStateManager} keeping track of the active - * view and enabling bookmarking and direct navigation + * The NavigationStateManager keeping track of the active view + * and enabling bookmarking and direct navigation * @param display - * {@ViewDisplay} used to display the views handled - * by this navigator + * The ViewDisplay used to display the views handled by this + * navigator */ - public Navigator(NavigationStateManager stateManager, ViewDisplay display) { + public Navigator(UI ui, NavigationStateManager stateManager, + ViewDisplay display) { + this.ui = ui; + this.ui.setNavigator(this); this.display = display; this.stateManager = stateManager; } @@ -547,6 +558,10 @@ public class Navigator implements Serializable { return display; } + public UI getUI() { + return ui; + } + /** * Fires an event after the current view has changed. * <p> diff --git a/server/src/com/vaadin/server/AbstractCommunicationManager.java b/server/src/com/vaadin/server/AbstractCommunicationManager.java index 5832b144ec..9ac33df7d2 100644 --- a/server/src/com/vaadin/server/AbstractCommunicationManager.java +++ b/server/src/com/vaadin/server/AbstractCommunicationManager.java @@ -84,6 +84,7 @@ import com.vaadin.ui.ConnectorTracker; import com.vaadin.ui.HasComponents; import com.vaadin.ui.UI; import com.vaadin.ui.Window; +import com.vaadin.util.CurrentInstance; /** * This is a common base class for the server-side implementations of the @@ -126,8 +127,6 @@ public abstract class AbstractCommunicationManager implements Serializable { } } - private static String GET_PARAM_REPAINT_ALL = "repaintAll"; - // flag used in the request to indicate that the security token should be // written to the response private static final String WRITE_SECURITY_TOKEN_FLAG = "writeSecurityToken"; @@ -549,7 +548,8 @@ public abstract class AbstractCommunicationManager implements Serializable { boolean repaintAll; final OutputStream out; - repaintAll = (request.getParameter(GET_PARAM_REPAINT_ALL) != null); + repaintAll = (request + .getParameter(ApplicationConstants.URL_PARAMETER_REPAINT_ALL) != null); // || (request.getSession().isNew()); FIXME What the h*ll is this?? out = response.getOutputStream(); @@ -2470,6 +2470,7 @@ public abstract class AbstractCommunicationManager implements Serializable { assert UI.getCurrent() == null; CombinedRequest combinedRequest = new CombinedRequest(request); + CurrentInstance.set(WrappedRequest.class, combinedRequest); response.setContentType("application/json; charset=UTF-8"); diff --git a/server/src/com/vaadin/server/AbstractVaadinService.java b/server/src/com/vaadin/server/AbstractVaadinService.java deleted file mode 100644 index 7150fdf0da..0000000000 --- a/server/src/com/vaadin/server/AbstractVaadinService.java +++ /dev/null @@ -1,99 +0,0 @@ -/* - * Copyright 2011 Vaadin Ltd. - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may not - * use this file except in compliance with the License. You may obtain a copy of - * the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the - * License for the specific language governing permissions and limitations under - * the License. - */ - -package com.vaadin.server; - -import java.lang.reflect.Constructor; -import java.util.Iterator; -import java.util.ServiceLoader; - -/** - * Abstract implementation of VaadinService that takes care of those parts that - * are common to both servlets and portlets. - * - * @author Vaadin Ltd - * @since 7.0.0 - */ -public abstract class AbstractVaadinService implements VaadinService { - - private AddonContext addonContext; - private final DeploymentConfiguration deploymentConfiguration; - - /** - * Creates a new vaadin service based on a deployment configuration - * - * @param deploymentConfiguration - * the deployment configuration for the service - */ - public AbstractVaadinService(DeploymentConfiguration deploymentConfiguration) { - this.deploymentConfiguration = deploymentConfiguration; - } - - @Override - public DeploymentConfiguration getDeploymentConfiguration() { - return deploymentConfiguration; - } - - @Override - public ClassLoader getClassLoader() { - final String classLoaderName = getDeploymentConfiguration() - .getApplicationOrSystemProperty("ClassLoader", null); - ClassLoader classLoader; - if (classLoaderName == null) { - classLoader = getClass().getClassLoader(); - } else { - try { - final Class<?> classLoaderClass = getClass().getClassLoader() - .loadClass(classLoaderName); - final Constructor<?> c = classLoaderClass - .getConstructor(new Class[] { ClassLoader.class }); - classLoader = (ClassLoader) c - .newInstance(new Object[] { getClass().getClassLoader() }); - } catch (final Exception e) { - throw new RuntimeException( - "Could not find specified class loader: " - + classLoaderName, e); - } - } - return classLoader; - } - - @Override - public Iterator<AddonContextListener> getAddonContextListeners() { - // Called once for init and then no more, so there's no point in caching - // the instance - ServiceLoader<AddonContextListener> contextListenerLoader = ServiceLoader - .load(AddonContextListener.class, getClassLoader()); - return contextListenerLoader.iterator(); - } - - @Override - public void setAddonContext(AddonContext addonContext) { - this.addonContext = addonContext; - } - - @Override - public AddonContext getAddonContext() { - return addonContext; - } - - @Override - public VaadinSession getVaadinSession(WrappedRequest request) { - return (VaadinSession) request.getAttribute(VaadinSession.class - .getName()); - } - -} diff --git a/server/src/com/vaadin/server/Constants.java b/server/src/com/vaadin/server/Constants.java index afb2d4dae1..f516b4b9a6 100644 --- a/server/src/com/vaadin/server/Constants.java +++ b/server/src/com/vaadin/server/Constants.java @@ -56,9 +56,6 @@ public interface Constants { + " Widgetset version: %s\n" + "================================================================="; - static final String URL_PARAMETER_RESTART_APPLICATION = "restartApplication"; - static final String URL_PARAMETER_CLOSE_APPLICATION = "closeApplication"; - static final String URL_PARAMETER_REPAINT_ALL = "repaintAll"; static final String URL_PARAMETER_THEME = "theme"; static final String SERVLET_PARAMETER_PRODUCTION_MODE = "productionMode"; diff --git a/server/src/com/vaadin/server/DeploymentConfiguration.java b/server/src/com/vaadin/server/DeploymentConfiguration.java index 0333091632..09405d6004 100644 --- a/server/src/com/vaadin/server/DeploymentConfiguration.java +++ b/server/src/com/vaadin/server/DeploymentConfiguration.java @@ -59,15 +59,15 @@ public interface DeploymentConfiguration extends Serializable { /** * Returns whether UIs that have no other activity than heartbeat requests - * should be closed after they have been idle the maximum inactivity time - * enforced by the session. + * should be removed from the session after they have been idle the maximum + * inactivity time enforced by the session. * * @see WrappedSession#getMaxInactiveInterval() * * @since 7.0.0 * * @return True if UIs receiving only heartbeat requests are eventually - * closed; false if heartbeat requests extend UI lifetime + * removed; false if heartbeat requests extend UI lifetime * indefinitely. */ public boolean isIdleUICleanupEnabled(); diff --git a/server/src/com/vaadin/server/GAEVaadinServlet.java b/server/src/com/vaadin/server/GAEVaadinServlet.java index 98b5f94c6a..42d03274bb 100644 --- a/server/src/com/vaadin/server/GAEVaadinServlet.java +++ b/server/src/com/vaadin/server/GAEVaadinServlet.java @@ -199,8 +199,8 @@ public class GAEVaadinServlet extends VaadinServlet { return; } - final HttpSession session = request - .getSession(requestCanCreateApplication(request, requestType)); + final HttpSession session = request.getSession(getVaadinService() + .requestCanCreateSession(request)); if (session == null) { handleServiceSessionExpired(request, response); cleanSession(request); @@ -292,7 +292,7 @@ public class GAEVaadinServlet extends VaadinServlet { } protected VaadinSession getApplicationContext(HttpServletRequest request, - MemcacheService memcache) { + MemcacheService memcache) throws ServletException { HttpSession session = request.getSession(); String id = AC_BASE + session.getId(); byte[] serializedAC = (byte[]) memcache.get(id); @@ -338,9 +338,14 @@ public class GAEVaadinServlet extends VaadinServlet { + " A new one will be created. ", e); } } - // will create new context if the above did not - return getApplicationContext(session); + // will create new context if the above did not + try { + return getVaadinService().findVaadinSession( + createWrappedRequest(request)); + } catch (Exception e) { + throw new ServletException(e); + } } private boolean isCleanupRequest(HttpServletRequest request) { diff --git a/server/src/com/vaadin/server/LegacyVaadinPortlet.java b/server/src/com/vaadin/server/LegacyVaadinPortlet.java index 067ef888b1..bdc03ff643 100644 --- a/server/src/com/vaadin/server/LegacyVaadinPortlet.java +++ b/server/src/com/vaadin/server/LegacyVaadinPortlet.java @@ -24,6 +24,28 @@ import com.vaadin.server.ServletPortletHelper.ApplicationClassException; public class LegacyVaadinPortlet extends VaadinPortlet { + @Override + public void init() throws PortletException { + super.init(); + + getVaadinService().addVaadinSessionInitializationListener( + new VaadinSessionInitializationListener() { + @Override + public void vaadinSessionInitialized( + VaadinSessionInitializeEvent event) + throws ServiceException { + try { + onVaadinSessionStarted(WrappedPortletRequest + .cast(event.getRequest()), + (VaadinPortletSession) event + .getVaadinSession()); + } catch (PortletException e) { + throw new ServiceException(e); + } + } + }); + } + protected Class<? extends LegacyApplication> getApplicationClass() throws ClassNotFoundException { try { @@ -44,8 +66,7 @@ public class LegacyVaadinPortlet extends VaadinPortlet { } } - @Override - protected void onVaadinSessionStarted(WrappedPortletRequest request, + private void onVaadinSessionStarted(WrappedPortletRequest request, VaadinPortletSession session) throws PortletException { if (shouldCreateApplication(request)) { // Must set current before running init() @@ -59,8 +80,6 @@ public class LegacyVaadinPortlet extends VaadinPortlet { legacyApplication.doInit(); session.addUIProvider(legacyApplication); } - - super.onVaadinSessionStarted(request, session); } protected boolean shouldCreateApplication(WrappedPortletRequest request) { diff --git a/server/src/com/vaadin/server/LegacyVaadinServlet.java b/server/src/com/vaadin/server/LegacyVaadinServlet.java index 4cdb66db44..8d55fddc39 100644 --- a/server/src/com/vaadin/server/LegacyVaadinServlet.java +++ b/server/src/com/vaadin/server/LegacyVaadinServlet.java @@ -16,6 +16,7 @@ package com.vaadin.server; +import javax.servlet.ServletConfig; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; @@ -24,6 +25,26 @@ import com.vaadin.server.ServletPortletHelper.ApplicationClassException; public class LegacyVaadinServlet extends VaadinServlet { + @Override + public void init(ServletConfig servletConfig) throws ServletException { + super.init(servletConfig); + + getVaadinService().addVaadinSessionInitializationListener( + new VaadinSessionInitializationListener() { + @Override + public void vaadinSessionInitialized( + VaadinSessionInitializeEvent event) + throws ServiceException { + try { + onVaadinSessionStarted(event.getRequest(), + event.getVaadinSession()); + } catch (ServletException e) { + throw new ServiceException(e); + } + } + }); + } + protected Class<? extends LegacyApplication> getApplicationClass() throws ClassNotFoundException { try { @@ -49,9 +70,10 @@ public class LegacyVaadinServlet extends VaadinServlet { return true; } - @Override - protected void onVaadinSessionStarted(WrappedHttpServletRequest request, - VaadinServletSession session) throws ServletException { + private void onVaadinSessionStarted(WrappedRequest wrappedRequest, + VaadinSession session) throws ServletException { + WrappedHttpServletRequest request = WrappedHttpServletRequest + .cast(wrappedRequest); if (shouldCreateApplication(request)) { // Must set current before running init() @@ -64,8 +86,6 @@ public class LegacyVaadinServlet extends VaadinServlet { legacyApplication.doInit(); session.addUIProvider(legacyApplication); } - - super.onVaadinSessionStarted(request, session); } } diff --git a/server/src/com/vaadin/server/ServiceException.java b/server/src/com/vaadin/server/ServiceException.java new file mode 100644 index 0000000000..538e8b30ea --- /dev/null +++ b/server/src/com/vaadin/server/ServiceException.java @@ -0,0 +1,29 @@ +/* + * Copyright 2011 Vaadin Ltd. + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may not + * use this file except in compliance with the License. You may obtain a copy of + * the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations under + * the License. + */ + +package com.vaadin.server; + +public class ServiceException extends Exception { + + public ServiceException(Exception e) { + super(e); + } + + public ServiceException(String message) { + super(message); + } + +} diff --git a/server/src/com/vaadin/server/VaadinPortlet.java b/server/src/com/vaadin/server/VaadinPortlet.java index 940a4925c8..dc1f93b2a8 100644 --- a/server/src/com/vaadin/server/VaadinPortlet.java +++ b/server/src/com/vaadin/server/VaadinPortlet.java @@ -28,7 +28,6 @@ import java.net.MalformedURLException; import java.net.URL; import java.security.GeneralSecurityException; import java.util.Enumeration; -import java.util.Locale; import java.util.Map; import java.util.Properties; import java.util.logging.Level; @@ -44,7 +43,6 @@ import javax.portlet.PortletContext; import javax.portlet.PortletException; import javax.portlet.PortletRequest; import javax.portlet.PortletResponse; -import javax.portlet.PortletSession; import javax.portlet.RenderRequest; import javax.portlet.RenderResponse; import javax.portlet.ResourceRequest; @@ -57,8 +55,6 @@ import com.liferay.portal.kernel.util.PortalClassInvoker; import com.liferay.portal.kernel.util.PropsUtil; import com.vaadin.DefaultDeploymentConfiguration; import com.vaadin.server.AbstractCommunicationManager.Callback; -import com.vaadin.server.ServletPortletHelper.ApplicationClassException; -import com.vaadin.server.VaadinSession.SessionStartEvent; import com.vaadin.ui.UI; import com.vaadin.util.CurrentInstance; @@ -78,7 +74,7 @@ public class VaadinPortlet extends GenericPortlet implements Constants { @Deprecated public static final String RESOURCE_URL_ID = "APP"; - public static class PortletService extends AbstractVaadinService { + public static class PortletService extends VaadinService { private final VaadinPortlet portlet; public PortletService(VaadinPortlet portlet, @@ -182,6 +178,63 @@ public class VaadinPortlet extends GenericPortlet implements Constants { return null; } + @Override + protected boolean requestCanCreateSession(WrappedRequest request) { + RequestType requestType = getRequestType(request); + if (requestType == RequestType.RENDER) { + // In most cases the first request is a render request that + // renders the HTML fragment. This should create an application + // instance. + return true; + } else if (requestType == RequestType.EVENT) { + // A portlet can also be sent an event even though it has not + // been rendered, e.g. portlet on one page sends an event to a + // portlet on another page and then moves the user to that page. + return true; + } + return false; + } + + /** + * Gets the request type for the request. + * + * @param request + * the request to get a request type for + * @return the request type + * + * @deprecated might be refactored or removed before 7.0.0 + */ + @Deprecated + protected RequestType getRequestType(WrappedRequest request) { + RequestType type = (RequestType) request + .getAttribute(RequestType.class.getName()); + if (type == null) { + type = getPortlet().getRequestType( + WrappedPortletRequest.cast(request)); + request.setAttribute(RequestType.class.getName(), type); + } + return type; + } + + @Override + protected AbstractCommunicationManager createCommunicationManager( + VaadinSession session) { + return new PortletCommunicationManager(session); + } + + public static WrappedPortletRequest getCurrentRequest() { + WrappedRequest currentRequest = VaadinService.getCurrentRequest(); + try { + return WrappedPortletRequest.cast(currentRequest); + } catch (ClassCastException e) { + return null; + } + } + + public static WrappedPortletResponse getCurrentResponse() { + return (WrappedPortletResponse) VaadinService.getCurrentResponse(); + } + } public static class WrappedHttpAndPortletRequest extends @@ -337,6 +390,8 @@ public class VaadinPortlet extends GenericPortlet implements Constants { @Override public void init(PortletConfig config) throws PortletException { + CurrentInstance.clearAll(); + setCurrent(this); super.init(config); Properties initParameters = new Properties(); @@ -357,9 +412,18 @@ public class VaadinPortlet extends GenericPortlet implements Constants { DeploymentConfiguration deploymentConfiguration = createDeploymentConfiguration(initParameters); vaadinService = createPortletService(deploymentConfiguration); + // Sets current service even though there are no request and response + vaadinService.setCurrentInstances(null, null); addonContext = new AddonContext(vaadinService); addonContext.init(); + + portletInitialized(); + CurrentInstance.clearAll(); + } + + protected void portletInitialized() { + } protected DeploymentConfiguration createDeploymentConfiguration( @@ -453,6 +517,9 @@ public class VaadinPortlet extends GenericPortlet implements Constants { RequestTimer requestTimer = new RequestTimer(); requestTimer.start(); + CurrentInstance.clearAll(); + setCurrent(this); + AbstractApplicationPortletWrapper portletWrapper = new AbstractApplicationPortletWrapper( this); @@ -461,8 +528,7 @@ public class VaadinPortlet extends GenericPortlet implements Constants { WrappedPortletResponse wrappedResponse = new WrappedPortletResponse( response, getVaadinService()); - CurrentInstance.set(WrappedRequest.class, wrappedRequest); - CurrentInstance.set(WrappedResponse.class, wrappedResponse); + getVaadinService().setCurrentInstances(wrappedRequest, wrappedResponse); RequestType requestType = getRequestType(wrappedRequest); @@ -492,8 +558,8 @@ public class VaadinPortlet extends GenericPortlet implements Constants { // TODO What about PARAM_UNLOADBURST & redirectToApplication?? /* Find out which application this request is related to */ - application = findApplicationInstance(wrappedRequest, - requestType); + application = getVaadinService().findVaadinSession( + wrappedRequest); if (application == null) { return; } @@ -621,7 +687,7 @@ public class VaadinPortlet extends GenericPortlet implements Constants { } finally { if (applicationRunning) { - application.closeInactiveUIs(); + application.cleanupInactiveUIs(); } CurrentInstance.clearAll(); @@ -767,36 +833,6 @@ public class VaadinPortlet extends GenericPortlet implements Constants { handleRequest(request, response); } - /** - * @param request - * @param requestType - * @return - * - * @deprecated might be refactored or removed before 7.0.0 - */ - @Deprecated - boolean requestCanCreateApplication(PortletRequest request, - RequestType requestType) { - if (requestType == RequestType.UIDL && isRepaintAll(request)) { - return true; - } else if (requestType == RequestType.RENDER) { - // In most cases the first request is a render request that renders - // the HTML fragment. This should create an application instance. - return true; - } else if (requestType == RequestType.EVENT) { - // A portlet can also be sent an event even though it has not been - // rendered, e.g. portlet on one page sends an event to a portlet on - // another page and then moves the user to that page. - return true; - } - return false; - } - - private boolean isRepaintAll(PortletRequest request) { - return (request.getParameter(URL_PARAMETER_REPAINT_ALL) != null) - && (request.getParameter(URL_PARAMETER_REPAINT_ALL).equals("1")); - } - private void endApplication(PortletRequest request, PortletResponse response, VaadinSession application) throws IOException { @@ -804,125 +840,6 @@ public class VaadinPortlet extends GenericPortlet implements Constants { // Do not send any redirects when running inside a portlet. } - private VaadinSession findApplicationInstance( - WrappedPortletRequest wrappedRequest, RequestType requestType) - throws PortletException, SessionExpiredException, - MalformedURLException { - PortletRequest request = wrappedRequest.getPortletRequest(); - - boolean requestCanCreateApplication = requestCanCreateApplication( - request, requestType); - - /* Find an existing application for this request. */ - VaadinSession application = getExistingApplication(request, - requestCanCreateApplication); - - if (application != null) { - /* - * There is an existing application. We can use this as long as the - * user not specifically requested to close or restart it. - */ - - final boolean restartApplication = (wrappedRequest - .getParameter(URL_PARAMETER_RESTART_APPLICATION) != null); - final boolean closeApplication = (wrappedRequest - .getParameter(URL_PARAMETER_CLOSE_APPLICATION) != null); - - if (restartApplication) { - closeApplication(application, request.getPortletSession(false)); - return createAndRegisterApplication(wrappedRequest); - } else if (closeApplication) { - closeApplication(application, request.getPortletSession(false)); - return null; - } else { - return application; - } - } - - // No existing application was found - - if (requestCanCreateApplication) { - return createAndRegisterApplication(wrappedRequest); - } else { - throw new SessionExpiredException(); - } - } - - private void closeApplication(VaadinSession application, - PortletSession session) { - if (application == null) { - return; - } - - application.close(); - application.removeFromSession(); - } - - private VaadinSession createAndRegisterApplication( - WrappedPortletRequest request) throws PortletException { - VaadinPortletSession newApplication = createApplication(); - - newApplication.storeInSession(new WrappedPortletSession(request - .getPortletRequest().getPortletSession())); - - Locale locale = request.getLocale(); - newApplication.setLocale(locale); - // No application URL when running inside a portlet - newApplication.start(new SessionStartEvent(null, getVaadinService() - .getDeploymentConfiguration(), new PortletCommunicationManager( - newApplication))); - onVaadinSessionStarted(request, newApplication); - - return newApplication; - } - - protected void onVaadinSessionStarted(WrappedPortletRequest request, - VaadinPortletSession session) throws PortletException { - addonContext.fireApplicationStarted(session); - try { - ServletPortletHelper.checkUiProviders(session); - } catch (ApplicationClassException e) { - throw new PortletException(e); - } - } - - private VaadinPortletSession createApplication() throws PortletException { - VaadinPortletSession application = new VaadinPortletSession(); - - try { - ServletPortletHelper.initDefaultUIProvider(application, - getVaadinService()); - } catch (ApplicationClassException e) { - throw new PortletException(e); - } - - return application; - } - - private VaadinSession getExistingApplication(PortletRequest request, - boolean allowSessionCreation) throws MalformedURLException, - SessionExpiredException { - - final PortletSession session = request - .getPortletSession(allowSessionCreation); - - if (session == null) { - throw new SessionExpiredException(); - } - - VaadinSession application = VaadinSession - .getForSession(new WrappedPortletSession(session)); - if (application == null) { - return null; - } - if (!application.isRunning()) { - application.removeFromSession(); - return null; - } - - return application; - } - private void handleServiceException(WrappedPortletRequest request, WrappedPortletResponse response, VaadinSession application, Throwable e) throws IOException, PortletException { @@ -1031,4 +948,43 @@ public class VaadinPortlet extends GenericPortlet implements Constants { return Logger.getLogger(VaadinPortlet.class.getName()); } + /** + * Gets the currently used Vaadin portlet. The current portlet is + * automatically defined when initializing the portlet and when processing + * requests to the server and in threads started at a point when the current + * portlet is defined (see {@link InheritableThreadLocal}). In other cases, + * (e.g. from background threads started in some other way), the current + * portlet is not automatically defined. + * + * @return the current vaadin portlet instance if available, otherwise + * <code>null</code> + * + * @see #setCurrent(VaadinPortlet) + * + * @since 7.0 + */ + public static VaadinPortlet getCurrent() { + return CurrentInstance.get(VaadinPortlet.class); + } + + /** + * Sets the current Vaadin portlet. This method is used by the framework to + * set the current portlet whenever a new request is processed and it is + * cleared when the request has been processed. + * <p> + * The application developer can also use this method to define the current + * portlet outside the normal request handling, e.g. when initiating custom + * background threads. + * </p> + * + * @param portlet + * the Vaadin portlet to register as the current portlet + * + * @see #getCurrent() + * @see InheritableThreadLocal + */ + public static void setCurrent(VaadinPortlet portlet) { + CurrentInstance.setInheritable(VaadinPortlet.class, portlet); + } + } diff --git a/server/src/com/vaadin/server/VaadinService.java b/server/src/com/vaadin/server/VaadinService.java index 827902def2..80590df4a0 100644 --- a/server/src/com/vaadin/server/VaadinService.java +++ b/server/src/com/vaadin/server/VaadinService.java @@ -18,10 +18,24 @@ package com.vaadin.server; import java.io.File; import java.io.Serializable; +import java.lang.reflect.Constructor; +import java.lang.reflect.Method; +import java.net.MalformedURLException; +import java.net.URL; import java.util.Iterator; +import java.util.Locale; +import java.util.ServiceLoader; import javax.portlet.PortletContext; import javax.servlet.ServletContext; +import javax.servlet.ServletException; + +import com.vaadin.LegacyApplication; +import com.vaadin.event.EventRouter; +import com.vaadin.server.ServletPortletHelper.ApplicationClassException; +import com.vaadin.server.VaadinSession.SessionStartEvent; +import com.vaadin.util.CurrentInstance; +import com.vaadin.util.ReflectTools; /** * Provide deployment specific settings that are required outside terminal @@ -31,7 +45,39 @@ import javax.servlet.ServletContext; * * @since 7.0 */ -public interface VaadinService extends Serializable { +public abstract class VaadinService implements Serializable { + + private static final Method SESSION_INIT_METHOD = ReflectTools.findMethod( + VaadinSessionInitializationListener.class, + "vaadinSessionInitialized", VaadinSessionInitializeEvent.class); + + /** + * @deprecated Only supported for {@link LegacyApplication}. + */ + @Deprecated + public static final String URL_PARAMETER_RESTART_APPLICATION = "restartApplication"; + + /** + * @deprecated Only supported for {@link LegacyApplication}. + */ + @Deprecated + public static final String URL_PARAMETER_CLOSE_APPLICATION = "closeApplication"; + + private AddonContext addonContext; + private final DeploymentConfiguration deploymentConfiguration; + + private final EventRouter eventRouter = new EventRouter(); + + /** + * Creates a new vaadin service based on a deployment configuration + * + * @param deploymentConfiguration + * the deployment configuration for the service + */ + public VaadinService(DeploymentConfiguration deploymentConfiguration) { + this.deploymentConfiguration = deploymentConfiguration; + } + /** * Return the URL from where static files, e.g. the widgetset and the theme, * are served. In a standard configuration the VAADIN folder inside the @@ -46,7 +92,7 @@ public interface VaadinService extends Serializable { * @return The location of static resources (should contain the VAADIN * directory). Never ends with a slash (/). */ - public String getStaticFileLocation(WrappedRequest request); + public abstract String getStaticFileLocation(WrappedRequest request); /** * Gets the widgetset that is configured for this deployment, e.g. from a @@ -56,7 +102,7 @@ public interface VaadinService extends Serializable { * the request for which a widgetset is required * @return the name of the widgetset */ - public String getConfiguredWidgetset(WrappedRequest request); + public abstract String getConfiguredWidgetset(WrappedRequest request); /** * Gets the theme that is configured for this deployment, e.g. from a portal @@ -66,7 +112,7 @@ public interface VaadinService extends Serializable { * the request for which a theme is required * @return the name of the theme */ - public String getConfiguredTheme(WrappedRequest request); + public abstract String getConfiguredTheme(WrappedRequest request); /** * Checks whether the Vaadin application will be rendered on its own in the @@ -79,7 +125,7 @@ public interface VaadinService extends Serializable { * the request for which the application is loaded * @return a boolean indicating whether the application should be standalone */ - public boolean isStandalone(WrappedRequest request); + public abstract boolean isStandalone(WrappedRequest request); /** * Get the class loader to use for loading classes loaded by name, e.g. @@ -88,7 +134,28 @@ public interface VaadinService extends Serializable { * * @return the class loader to use, or <code>null</code> */ - public ClassLoader getClassLoader(); + public ClassLoader getClassLoader() { + final String classLoaderName = getDeploymentConfiguration() + .getApplicationOrSystemProperty("ClassLoader", null); + ClassLoader classLoader; + if (classLoaderName == null) { + classLoader = getClass().getClassLoader(); + } else { + try { + final Class<?> classLoaderClass = getClass().getClassLoader() + .loadClass(classLoaderName); + final Constructor<?> c = classLoaderClass + .getConstructor(new Class[] { ClassLoader.class }); + classLoader = (ClassLoader) c + .newInstance(new Object[] { getClass().getClassLoader() }); + } catch (final Exception e) { + throw new RuntimeException( + "Could not find specified class loader: " + + classLoaderName, e); + } + } + return classLoader; + } /** * Returns the MIME type of the specified file, or null if the MIME type is @@ -103,27 +170,39 @@ public interface VaadinService extends Serializable { * @see ServletContext#getMimeType(String) * @see PortletContext#getMimeType(String) */ - public String getMimeType(String resourceName); + public abstract String getMimeType(String resourceName); /** * Gets the deployment configuration. * * @return the deployment configuration */ - public DeploymentConfiguration getDeploymentConfiguration(); + public DeploymentConfiguration getDeploymentConfiguration() { + return deploymentConfiguration; + } - public Iterator<AddonContextListener> getAddonContextListeners(); + public Iterator<AddonContextListener> getAddonContextListeners() { + // Called once for init and then no more, so there's no point in caching + // the instance + ServiceLoader<AddonContextListener> contextListenerLoader = ServiceLoader + .load(AddonContextListener.class, getClassLoader()); + return contextListenerLoader.iterator(); + } - public AddonContext getAddonContext(); + public AddonContext getAddonContext() { + return addonContext; + } - public void setAddonContext(AddonContext vaadinContext); + public void setAddonContext(AddonContext addonContext) { + this.addonContext = addonContext; + } /** * Gets the system messages object * * @return the system messages object */ - public SystemMessages getSystemMessages(); + public abstract SystemMessages getSystemMessages(); /** * Returns application context base directory. @@ -137,10 +216,45 @@ public interface VaadinService extends Serializable { * @return The application base directory or null if the application has no * base directory. */ - public File getBaseDirectory(); + public abstract File getBaseDirectory(); + + /** + * Adds a listener that gets notified when a new Vaadin session is + * initialized for this service. + * <p> + * Because of the way different service instances share the same session, + * the listener is not necessarily notified immediately when the session is + * created but only when the first request for that session is handled by + * this service. + * + * @see #removeVaadinSessionInitializationListener(VaadinSessionInitializationListener) + * @see VaadinSessionInitializationListener + * + * @param listener + * the vaadin session initialization listener + */ + public void addVaadinSessionInitializationListener( + VaadinSessionInitializationListener listener) { + eventRouter.addListener(VaadinSessionInitializeEvent.class, listener, + SESSION_INIT_METHOD); + } /** - * Gets the Vaadin session associated with this request. + * Removes a Vaadin session initialization listener from this service. + * + * @see #addVaadinSessionInitializationListener(VaadinSessionInitializationListener) + * + * @param listener + * the Vaadin session initialization listener to remove. + */ + public void removeVaadinSessionInitializationListener( + VaadinSessionInitializationListener listener) { + eventRouter.removeListener(VaadinSessionInitializeEvent.class, + listener, SESSION_INIT_METHOD); + } + + /** + * Attempts to find a Vaadin session associated with this request. * * @param request * the request to get a vaadin session for. @@ -151,5 +265,265 @@ public interface VaadinService extends Serializable { * session is found and this is a request for which a new session * shouldn't be created. */ - public VaadinSession getVaadinSession(WrappedRequest request); + public VaadinSession findVaadinSession(WrappedRequest request) + throws ServiceException, SessionExpiredException { + + boolean requestCanCreateApplication = requestCanCreateSession(request); + + /* Find an existing application for this request. */ + VaadinSession session = getExistingSession(request, + requestCanCreateApplication); + + if (session != null) { + /* + * There is an existing application. We can use this as long as the + * user not specifically requested to close or restart it. + */ + + final boolean restartApplication = (request + .getParameter(URL_PARAMETER_RESTART_APPLICATION) != null); + final boolean closeApplication = (request + .getParameter(URL_PARAMETER_CLOSE_APPLICATION) != null); + + if (restartApplication) { + closeApplication(session, request.getWrappedSession(false)); + return createAndRegisterApplication(request); + } else if (closeApplication) { + closeApplication(session, request.getWrappedSession(false)); + return null; + } else { + return session; + } + } + + // No existing application was found + + if (requestCanCreateApplication) { + /* + * If the request is such that it should create a new application if + * one as not found, we do that. + */ + return createAndRegisterApplication(request); + } else { + /* + * The application was not found and a new one should not be + * created. Assume the session has expired. + */ + throw new SessionExpiredException(); + } + + } + + private VaadinSession createAndRegisterApplication(WrappedRequest request) + throws ServiceException { + VaadinSession session = createVaadinSession(request); + + session.storeInSession(request.getWrappedSession()); + + URL applicationUrl; + try { + applicationUrl = getApplicationUrl(request); + } catch (MalformedURLException e) { + throw new ServiceException(e); + } + + // Initial locale comes from the request + Locale locale = request.getLocale(); + session.setLocale(locale); + session.start(new SessionStartEvent(applicationUrl, + getDeploymentConfiguration(), + createCommunicationManager(session))); + + onVaadinSessionStarted(request, session); + + return session; + } + + /** + * Get the base URL that should be used for sending requests back to this + * service. + * <p> + * This is only used to support legacy cases. + * + * @param request + * @return + * @throws MalformedURLException + * + * @deprecated Only used to support {@link LegacyApplication}. + */ + @Deprecated + protected URL getApplicationUrl(WrappedRequest request) + throws MalformedURLException { + return null; + } + + /** + * Create a communication manager to use for the given Vaadin session. + * + * @param session + * the vaadin session for which a new communication manager is + * needed + * @return a new communication manager + */ + protected abstract AbstractCommunicationManager createCommunicationManager( + VaadinSession session); + + /** + * Creates a new vaadin session. + * + * @param request + * @return + * @throws ServletException + * @throws MalformedURLException + */ + private VaadinServletSession createVaadinSession(WrappedRequest request) + throws ServiceException { + VaadinServletSession session = new VaadinServletSession(); + + try { + ServletPortletHelper.initDefaultUIProvider(session, this); + } catch (ApplicationClassException e) { + throw new ServiceException(e); + } + + return session; + } + + private void onVaadinSessionStarted(WrappedRequest request, + VaadinSession session) throws ServiceException { + addonContext.fireApplicationStarted(session); + eventRouter.fireEvent(new VaadinSessionInitializeEvent(this, session, + request)); + + try { + ServletPortletHelper.checkUiProviders(session); + } catch (ApplicationClassException e) { + throw new ServiceException(e); + } + } + + private void closeApplication(VaadinSession application, + WrappedSession session) { + if (application == null) { + return; + } + + application.close(); + if (session != null) { + application.removeFromSession(); + } + } + + protected VaadinSession getExistingSession(WrappedRequest request, + boolean allowSessionCreation) throws SessionExpiredException { + + // Ensures that the session is still valid + final WrappedSession session = request + .getWrappedSession(allowSessionCreation); + if (session == null) { + throw new SessionExpiredException(); + } + + VaadinSession sessionApplication = VaadinSession.getForSession(session); + + if (sessionApplication == null) { + return null; + } + + if (!sessionApplication.isRunning()) { + sessionApplication.removeFromSession(); + return null; + } + + return sessionApplication; + } + + /** + * Checks whether it's valid to create a new Vaadin session as a result of + * the given request. + * + * @param request + * the request + * @return <code>true</code> if it's valid to create a new Vaadin session + * for the request; else <code>false</code> + */ + protected abstract boolean requestCanCreateSession(WrappedRequest request); + + /** + * Gets the currently used Vaadin service. The current service is + * automatically defined when processing requests related to the service and + * in threads started at a point when the current service is defined (see + * {@link InheritableThreadLocal}). In other cases, (e.g. from background + * threads started in some other way), the current service is not + * automatically defined. + * + * @return the current Vaadin service instance if available, otherwise + * <code>null</code> + * + * @see #setCurrentInstances(WrappedRequest, WrappedResponse) + */ + public static VaadinService getCurrent() { + return CurrentInstance.get(VaadinService.class); + } + + /** + * Sets the this Vaadin service as the current service and also sets the + * current wrapped request and wrapped response. This method is used by the + * framework to set the current instances when a request related to the + * service is processed and they are cleared when the request has been + * processed. + * <p> + * The application developer can also use this method to define the current + * instances outside the normal request handling, e.g. when initiating + * custom background threads. + * </p> + * + * @param request + * the wrapped request to set as the current request, or + * <code>null</code> if no request should be set. + * @param response + * the wrapped response to set as the current response, or + * <code>null</code> if no response should be set. + * + * @see #getCurrent() + * @see #getCurrentRequest() + * @see #getCurrentResponse() + */ + public void setCurrentInstances(WrappedRequest request, + WrappedResponse response) { + CurrentInstance.setInheritable(VaadinService.class, this); + CurrentInstance.set(WrappedRequest.class, request); + CurrentInstance.set(WrappedResponse.class, response); + } + + /** + * Gets the currently processed wrapped request. The current request is + * automatically defined when the request is started. The current request + * can not be used in e.g. background threads because of the way server + * implementations reuse request instances. + * + * @return the current wrapped request instance if available, otherwise + * <code>null</code> + * + * @see #setCurrentInstances(WrappedRequest, WrappedResponse) + */ + public static WrappedRequest getCurrentRequest() { + return CurrentInstance.get(WrappedRequest.class); + } + + /** + * Gets the currently processed wrapped request. The current request is + * automatically defined when the request is started. The current request + * can not be used in e.g. background threads because of the way server + * implementations reuse request instances. + * + * @return the current wrapped request instance if available, otherwise + * <code>null</code> + * + * @see #setCurrentInstances(WrappedRequest, WrappedResponse) + */ + public static WrappedResponse getCurrentResponse() { + return CurrentInstance.get(WrappedResponse.class); + } + } diff --git a/server/src/com/vaadin/server/VaadinServlet.java b/server/src/com/vaadin/server/VaadinServlet.java index fd95852c6a..c42d029dc4 100644 --- a/server/src/com/vaadin/server/VaadinServlet.java +++ b/server/src/com/vaadin/server/VaadinServlet.java @@ -31,7 +31,6 @@ import java.util.Arrays; import java.util.Collection; import java.util.Enumeration; import java.util.HashSet; -import java.util.Locale; import java.util.Properties; import java.util.logging.Level; import java.util.logging.Logger; @@ -47,8 +46,6 @@ import javax.servlet.http.HttpSession; import com.vaadin.DefaultDeploymentConfiguration; import com.vaadin.sass.ScssStylesheet; import com.vaadin.server.AbstractCommunicationManager.Callback; -import com.vaadin.server.ServletPortletHelper.ApplicationClassException; -import com.vaadin.server.VaadinSession.SessionStartEvent; import com.vaadin.shared.ApplicationConstants; import com.vaadin.ui.UI; import com.vaadin.util.CurrentInstance; @@ -56,7 +53,7 @@ import com.vaadin.util.CurrentInstance; @SuppressWarnings("serial") public class VaadinServlet extends HttpServlet implements Constants { - public static class ServletService extends AbstractVaadinService { + public static class ServletService extends VaadinService { private final VaadinServlet servlet; public ServletService(VaadinServlet servlet, @@ -148,6 +145,72 @@ public class VaadinServlet extends HttpServlet implements Constants { } return new File(realPath); } + + @Override + protected boolean requestCanCreateSession(WrappedRequest request) { + RequestType requestType = getRequestType(request); + if (requestType == RequestType.BROWSER_DETAILS) { + // This is the first request if you are embedding by writing the + // embedding code yourself + return true; + } else if (requestType == RequestType.OTHER) { + /* + * I.e URIs that are not application resources or static (theme) + * files. + */ + return true; + } + + return false; + } + + /** + * Gets the request type for the request. + * + * @param request + * the request to get a request type for + * @return the request type + * + * @deprecated might be refactored or removed before 7.0.0 + */ + @Deprecated + protected RequestType getRequestType(WrappedRequest request) { + RequestType type = (RequestType) request + .getAttribute(RequestType.class.getName()); + if (type == null) { + type = getServlet().getRequestType( + WrappedHttpServletRequest.cast(request)); + request.setAttribute(RequestType.class.getName(), type); + } + return type; + } + + @Override + protected URL getApplicationUrl(WrappedRequest request) + throws MalformedURLException { + return getServlet().getApplicationUrl( + WrappedHttpServletRequest.cast(request)); + } + + @Override + protected AbstractCommunicationManager createCommunicationManager( + VaadinSession session) { + return new CommunicationManager(session); + } + + public static WrappedHttpServletRequest getCurrentRequest() { + WrappedRequest currentRequest = VaadinService.getCurrentRequest(); + try { + return WrappedHttpServletRequest.cast(currentRequest); + } catch (ClassCastException e) { + return null; + } + } + + public static WrappedHttpServletResponse getCurrentResponse() { + return (WrappedHttpServletResponse) VaadinService + .getCurrentResponse(); + } } private static class AbstractApplicationServletWrapper implements Callback { @@ -192,6 +255,8 @@ public class VaadinServlet extends HttpServlet implements Constants { @Override public void init(javax.servlet.ServletConfig servletConfig) throws javax.servlet.ServletException { + CurrentInstance.clearAll(); + setCurrent(this); super.init(servletConfig); Properties initParameters = new Properties(); @@ -213,9 +278,58 @@ public class VaadinServlet extends HttpServlet implements Constants { DeploymentConfiguration deploymentConfiguration = createDeploymentConfiguration(initParameters); servletService = createServletService(deploymentConfiguration); + // Sets current service even though there are no request and response + servletService.setCurrentInstances(null, null); addonContext = new AddonContext(servletService); addonContext.init(); + + servletInitialized(); + + CurrentInstance.clearAll(); + } + + protected void servletInitialized() { + // Empty by default + } + + /** + * Gets the currently used Vaadin servlet. The current servlet is + * automatically defined when initializing the servlet and when processing + * requests to the server and in threads started at a point when the current + * servlet is defined (see {@link InheritableThreadLocal}). In other cases, + * (e.g. from background threads started in some other way), the current + * servlet is not automatically defined. + * + * @return the current Vaadin servlet instance if available, otherwise + * <code>null</code> + * + * @see #setCurrent(VaadinServlet) + * + * @since 7.0 + */ + public static VaadinServlet getCurrent() { + return CurrentInstance.get(VaadinServlet.class); + } + + /** + * Sets the current Vaadin servlet. This method is used by the framework to + * set the current servlet whenever a new request is processed and it is + * cleared when the request has been processed. + * <p> + * The application developer can also use this method to define the current + * servlet outside the normal request handling, e.g. when initiating custom + * background threads. + * </p> + * + * @param servlet + * the Vaadin servlet to register as the current servlet + * + * @see #getCurrent() + * @see InheritableThreadLocal + */ + public static void setCurrent(VaadinServlet servlet) { + CurrentInstance.setInheritable(VaadinServlet.class, servlet); } protected DeploymentConfiguration createDeploymentConfiguration( @@ -255,6 +369,8 @@ public class VaadinServlet extends HttpServlet implements Constants { @Override protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { + CurrentInstance.clearAll(); + setCurrent(this); service(createWrappedRequest(request), createWrappedResponse(response)); } @@ -264,8 +380,7 @@ public class VaadinServlet extends HttpServlet implements Constants { RequestTimer requestTimer = new RequestTimer(); requestTimer.start(); - CurrentInstance.set(WrappedResponse.class, response); - CurrentInstance.set(WrappedRequest.class, request); + getVaadinService().setCurrentInstances(request, response); AbstractApplicationServletWrapper servletWrapper = new AbstractApplicationServletWrapper( this); @@ -297,13 +412,13 @@ public class VaadinServlet extends HttpServlet implements Constants { && request.getParameterMap().containsKey( ApplicationConstants.PARAM_UNLOADBURST) && request.getContentLength() < 1 - && getExistingApplication(request, false) == null) { + && getVaadinService().getExistingSession(request, false) == null) { redirectToApplication(request, response); return; } // Find out which application this request is related to - application = findApplicationInstance(request, requestType); + application = getVaadinService().findVaadinSession(request); if (application == null) { return; } @@ -376,7 +491,7 @@ public class VaadinServlet extends HttpServlet implements Constants { } finally { if (applicationRunning) { - application.closeInactiveUIs(); + application.cleanupInactiveUIs(); } CurrentInstance.clearAll(); @@ -433,7 +548,7 @@ public class VaadinServlet extends HttpServlet implements Constants { private boolean ensureCookiesEnabled(RequestType requestType, WrappedHttpServletRequest request, WrappedHttpServletResponse response) throws IOException { - if (requestType == RequestType.UIDL && !isRepaintAll(request)) { + if (requestType == RequestType.UIDL) { // In all other but the first UIDL request a cookie should be // returned by the browser. // This can be removed if cookieless mode (#3228) is supported @@ -560,140 +675,6 @@ public class VaadinServlet extends HttpServlet implements Constants { } /** - * Returns the application instance to be used for the request. If an - * existing instance is not found a new one is created or null is returned - * to indicate that the application is not available. - * - * @param request - * @param requestType - * @return - * @throws MalformedURLException - * @throws IllegalAccessException - * @throws InstantiationException - * @throws ServletException - * @throws SessionExpiredException - */ - private VaadinSession findApplicationInstance( - WrappedHttpServletRequest request, RequestType requestType) - throws MalformedURLException, ServletException, - SessionExpiredException { - - boolean requestCanCreateApplication = requestCanCreateApplication( - request, requestType); - - /* Find an existing application for this request. */ - VaadinSession application = getExistingApplication(request, - requestCanCreateApplication); - - if (application != null) { - /* - * There is an existing application. We can use this as long as the - * user not specifically requested to close or restart it. - */ - - final boolean restartApplication = (request - .getParameter(URL_PARAMETER_RESTART_APPLICATION) != null); - final boolean closeApplication = (request - .getParameter(URL_PARAMETER_CLOSE_APPLICATION) != null); - - if (restartApplication) { - closeApplication(application, request.getSession(false)); - return createAndRegisterApplication(request); - } else if (closeApplication) { - closeApplication(application, request.getSession(false)); - return null; - } else { - return application; - } - } - - // No existing application was found - - if (requestCanCreateApplication) { - /* - * If the request is such that it should create a new application if - * one as not found, we do that. - */ - return createAndRegisterApplication(request); - } else { - /* - * The application was not found and a new one should not be - * created. Assume the session has expired. - */ - throw new SessionExpiredException(); - } - - } - - private VaadinSession createAndRegisterApplication( - WrappedHttpServletRequest request) throws ServletException, - MalformedURLException { - VaadinServletSession session = createVaadinSession(request); - - session.storeInSession(new WrappedHttpSession(request.getSession())); - - final URL applicationUrl = getApplicationUrl(request); - - // Initial locale comes from the request - Locale locale = request.getLocale(); - session.setLocale(locale); - session.start(new SessionStartEvent(applicationUrl, getVaadinService() - .getDeploymentConfiguration(), - createCommunicationManager(session))); - - onVaadinSessionStarted(request, session); - - return session; - } - - protected void onVaadinSessionStarted(WrappedHttpServletRequest request, - VaadinServletSession session) throws ServletException { - addonContext.fireApplicationStarted(session); - - try { - ServletPortletHelper.checkUiProviders(session); - } catch (ApplicationClassException e) { - throw new ServletException(e); - } - } - - /** - * Check if the request should create an application if an existing - * application is not found. - * - * @param request - * @param requestType - * @return true if an application should be created, false otherwise - * - * @deprecated might be refactored or removed before 7.0.0 - */ - @Deprecated - boolean requestCanCreateApplication(HttpServletRequest request, - RequestType requestType) { - if (requestType == RequestType.UIDL && isRepaintAll(request)) { - /* - * UIDL request contains valid repaintAll=1 event, the user probably - * wants to initiate a new application through a custom index.html - * without using the bootstrap page. - */ - return true; - } else if (requestType == RequestType.BROWSER_DETAILS) { - // This is the first request if you are embedding by writing the - // embedding code yourself - return true; - } else if (requestType == RequestType.OTHER) { - /* - * I.e URIs that are not application resources or static (theme) - * files. - */ - return true; - - } - - return false; - } - - /** * Gets resource path using different implementations. Required to * supporting different servlet container implementations (application * servers). @@ -725,28 +706,6 @@ public class VaadinServlet extends HttpServlet implements Constants { return resultPath; } - /** - * Creates a new vaadin session. - * - * @param request - * @return - * @throws ServletException - * @throws MalformedURLException - */ - private VaadinServletSession createVaadinSession(HttpServletRequest request) - throws ServletException { - VaadinServletSession session = new VaadinServletSession(); - - try { - ServletPortletHelper.initDefaultUIProvider(session, - getVaadinService()); - } catch (ApplicationClassException e) { - throw new ServletException(e); - } - - return session; - } - private void handleServiceException(WrappedHttpServletRequest request, WrappedHttpServletResponse response, VaadinSession application, Throwable e) throws IOException, ServletException { @@ -1377,51 +1336,6 @@ public class VaadinServlet extends HttpServlet implements Constants { } /** - * Gets the existing application for given request. Looks for application - * instance for given request based on the requested URL. - * - * @param request - * the HTTP request. - * @param allowSessionCreation - * true if a session should be created if no session exists, - * false if no session should be created - * @return Application instance, or null if the URL does not map to valid - * application. - * @throws MalformedURLException - * if the application is denied access to the persistent data - * store represented by the given URL. - * @throws IllegalAccessException - * @throws InstantiationException - * @throws SessionExpiredException - * - * @deprecated might be refactored or removed before 7.0.0 - */ - @Deprecated - protected VaadinSession getExistingApplication(HttpServletRequest request, - boolean allowSessionCreation) throws MalformedURLException, - SessionExpiredException { - - // Ensures that the session is still valid - final HttpSession session = request.getSession(allowSessionCreation); - if (session == null) { - throw new SessionExpiredException(); - } - - VaadinSession sessionApplication = getApplicationContext(session); - - if (sessionApplication == null) { - return null; - } - - if (!sessionApplication.isRunning()) { - sessionApplication.removeFromSession(); - return null; - } - - return sessionApplication; - } - - /** * Ends the application. * * @param request @@ -1490,35 +1404,6 @@ public class VaadinServlet extends HttpServlet implements Constants { return resourcePath + theme + "/" + resource.getResourceId(); } - private boolean isRepaintAll(HttpServletRequest request) { - return (request.getParameter(URL_PARAMETER_REPAINT_ALL) != null) - && (request.getParameter(URL_PARAMETER_REPAINT_ALL).equals("1")); - } - - private void closeApplication(VaadinSession application, HttpSession session) { - if (application == null) { - return; - } - - application.close(); - if (session != null) { - application.removeFromSession(); - } - } - - /** - * @param session - * @return - * - * @deprecated might be refactored or removed before 7.0.0 - */ - @Deprecated - protected VaadinSession getApplicationContext(final HttpSession session) { - VaadinSession sessionApplication = VaadinSession - .getForSession(new WrappedHttpSession(session)); - return sessionApplication; - } - public class RequestError implements Terminal.ErrorEvent, Serializable { private final Throwable throwable; @@ -1535,29 +1420,6 @@ public class VaadinServlet extends HttpServlet implements Constants { } /** - * Override this method if you need to use a specialized communicaiton - * mananger implementation. - * - * @deprecated Instead of overriding this method, override - * {@link VaadinServletSession} implementation via - * {@link VaadinServlet#getApplicationContext(HttpSession)} - * method and in that customized implementation return your - * CommunicationManager in - * {@link VaadinServletSession#getApplicationManager(VaadinSession, VaadinServlet)} - * method. - * - * @param application - * @return - * - * @deprecated might be refactored or removed before 7.0.0 - */ - @Deprecated - public CommunicationManager createCommunicationManager( - VaadinSession application) { - return new CommunicationManager(application); - } - - /** * Escapes characters to html entities. An exception is made for some * "safe characters" to keep the text somewhat readable. * diff --git a/server/src/com/vaadin/server/VaadinServletSession.java b/server/src/com/vaadin/server/VaadinServletSession.java index b2e3b9f54c..b2dc84d857 100644 --- a/server/src/com/vaadin/server/VaadinServletSession.java +++ b/server/src/com/vaadin/server/VaadinServletSession.java @@ -23,7 +23,7 @@ import javax.servlet.http.HttpSession; import javax.servlet.http.HttpSessionBindingEvent; import javax.servlet.http.HttpSessionBindingListener; -import com.vaadin.util.CurrentInstance; +import com.vaadin.server.VaadinServlet.ServletService; /** * Web application context for Vaadin applications. @@ -58,6 +58,12 @@ public class VaadinServletSession extends VaadinSession { * to avoid session fixation attacks. */ public void reinitializeSession() { + WrappedHttpServletRequest currentRequest = ServletService + .getCurrentRequest(); + if (currentRequest == null) { + throw new IllegalStateException( + "Can not reinitialize session outside normal request handling."); + } HttpSession oldSession = getHttpSession(); @@ -77,8 +83,7 @@ public class VaadinServletSession extends VaadinSession { reinitializingSession = false; // Create a new session - HttpSession newSession = WrappedHttpServletRequest.cast( - CurrentInstance.get(WrappedRequest.class)).getSession(); + HttpSession newSession = currentRequest.getSession(); // Restores all attributes (security key, reference to this context // instance) diff --git a/server/src/com/vaadin/server/VaadinSession.java b/server/src/com/vaadin/server/VaadinSession.java index 35a53fc6db..f1d4ab343a 100644 --- a/server/src/com/vaadin/server/VaadinSession.java +++ b/server/src/com/vaadin/server/VaadinSession.java @@ -316,7 +316,7 @@ public class VaadinSession implements HttpSessionBindingListener, Serializable { public void close() { applicationIsRunning = false; for (UI ui : getUIs()) { - ui.fireCloseEvent(); + ui.fireCleanupEvent(); } } @@ -1166,12 +1166,12 @@ public class VaadinSession implements HttpSessionBindingListener, Serializable { /** * Removes all those UIs from the session for which {@link #isUIAlive} - * returns false. Close events are fired for the removed UIs. + * returns false. Cleanup events are fired for the removed UIs. * <p> * Called by the framework at the end of every request. * - * @see UI.CloseEvent - * @see UI.CloseListener + * @see UI.CleanupEvent + * @see UI.CleanupListener * @see #isUIAlive(UI) * * @since 7.0.0 @@ -1179,13 +1179,13 @@ public class VaadinSession implements HttpSessionBindingListener, Serializable { * @deprecated might be refactored or removed before 7.0.0 */ @Deprecated - public void closeInactiveUIs() { + public void cleanupInactiveUIs() { for (Iterator<UI> i = uIs.values().iterator(); i.hasNext();) { UI ui = i.next(); if (!isUIAlive(ui)) { i.remove(); retainOnRefreshUIs.values().remove(ui.getUIId()); - ui.fireCloseEvent(); + ui.fireCleanupEvent(); getLogger().fine( "Closed UI #" + ui.getUIId() + " due to inactivity"); } @@ -1200,7 +1200,7 @@ public class VaadinSession implements HttpSessionBindingListener, Serializable { * timeout never occurs. * * @see #getUidlRequestTimeout() - * @see #closeInactiveUIs() + * @see #cleanupInactiveUIs() * @see DeploymentConfiguration#getHeartbeatInterval() * * @since 7.0.0 @@ -1225,7 +1225,7 @@ public class VaadinSession implements HttpSessionBindingListener, Serializable { * * @see DeploymentConfiguration#isIdleUICleanupEnabled() * @see #getHeartbeatTimeout() - * @see #closeInactiveUIs() + * @see #cleanupInactiveUIs() * * @since 7.0.0 * diff --git a/server/src/com/vaadin/server/VaadinSessionInitializationListener.java b/server/src/com/vaadin/server/VaadinSessionInitializationListener.java new file mode 100644 index 0000000000..11b14cc9fc --- /dev/null +++ b/server/src/com/vaadin/server/VaadinSessionInitializationListener.java @@ -0,0 +1,49 @@ +/* + * Copyright 2011 Vaadin Ltd. + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may not + * use this file except in compliance with the License. You may obtain a copy of + * the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations under + * the License. + */ + +package com.vaadin.server; + +/** + * Event listener that can be registered to a {@link VaadinService} to get an + * event when a new Vaadin session is initialized for that service. + * <p> + * Because of the way different service instances share the same session, the + * listener is not necessarily notified immediately when the session is created + * but only when the first request for that session is handled by a specific + * service. + * + * @see VaadinService#addVaadinSessionInitializationListener(VaadinSessionInitializationListener) + * + * @author Vaadin Ltd + * @since 7.0.0 + */ +public interface VaadinSessionInitializationListener { + /** + * Invoked when a new Vaadin session is initialized for that service. + * <p> + * Because of the way different service instances share the same session, + * the listener is not necessarily notified immediately when the session is + * created but only when the first request for that session is handled by a + * specific service. + * + * @param event + * the initialization event + * @throws ServiceException + * a problem occurs when processing the event + */ + public void vaadinSessionInitialized(VaadinSessionInitializeEvent event) + throws ServiceException; +} diff --git a/server/src/com/vaadin/server/VaadinSessionInitializeEvent.java b/server/src/com/vaadin/server/VaadinSessionInitializeEvent.java new file mode 100644 index 0000000000..cc4a0990d6 --- /dev/null +++ b/server/src/com/vaadin/server/VaadinSessionInitializeEvent.java @@ -0,0 +1,89 @@ +/* + * Copyright 2011 Vaadin Ltd. + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may not + * use this file except in compliance with the License. You may obtain a copy of + * the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations under + * the License. + */ + +package com.vaadin.server; + +import java.util.EventObject; + +/** + * Event gets fired when a new Vaadin session is initialized for a Vaadin + * service. + * <p> + * Because of the way different service instances share the same session, the + * event is not necessarily fired immediately when the session is created but + * only when the first request for that session is handled by a specific + * service. + * + * @see VaadinSessionInitializationListener#vaadinSessionInitialized(VaadinSessionInitializeEvent) + * + * @author Vaadin Ltd + * @since 7.0.0 + */ +public class VaadinSessionInitializeEvent extends EventObject { + + private final VaadinSession session; + private final WrappedRequest request; + + /** + * Creates a new event. + * + * @param service + * the Vaadin service from which the event originates + * @param session + * the Vaadin session that has been initialized + * @param request + * the request that triggered the initialization + */ + public VaadinSessionInitializeEvent(VaadinService service, + VaadinSession session, WrappedRequest request) { + super(service); + this.session = session; + this.request = request; + } + + @Override + public VaadinService getSource() { + return (VaadinService) super.getSource(); + } + + /** + * Gets the Vaadin service from which this event originates + * + * @return the Vaadin service instance + */ + public VaadinService getVaadinService() { + return getSource(); + } + + /** + * Gets the Vaadin session that has been initialized. + * + * @return the Vaadin session + */ + public VaadinSession getVaadinSession() { + return session; + } + + /** + * Gets the request that triggered the initialization. + * + * @return the request + */ + public WrappedRequest getRequest() { + return request; + } + +} diff --git a/server/src/com/vaadin/ui/Label.java b/server/src/com/vaadin/ui/Label.java index 8b5cd87648..9434e92186 100644 --- a/server/src/com/vaadin/ui/Label.java +++ b/server/src/com/vaadin/ui/Label.java @@ -170,6 +170,16 @@ public class Label extends AbstractComponent implements Property<String>, // Use internal value if we are running without a data source return getState().text; } + return getDataSourceValue(); + } + + /** + * Returns the current value of the data source. Assumes there is a data + * source. + * + * @return + */ + private String getDataSourceValue() { return ConverterUtil.convertFromModel(getPropertyDataSource() .getValue(), String.class, getConverter(), getLocale()); } @@ -250,14 +260,20 @@ public class Label extends AbstractComponent implements Property<String>, ((Property.ValueChangeNotifier) dataSource).removeListener(this); } - if (!ConverterUtil.canConverterHandle(getConverter(), String.class, - newDataSource.getType())) { + if (newDataSource != null + && !ConverterUtil.canConverterHandle(getConverter(), + String.class, newDataSource.getType())) { // Try to find a converter Converter<String, ?> c = ConverterUtil.getConverter(String.class, newDataSource.getType(), getSession()); setConverter(c); } dataSource = newDataSource; + if (dataSource != null) { + // Update the value from the data source. If data source was set to + // null, retain the old value + getState().text = getDataSourceValue(); + } // Listens the new data source if possible if (dataSource != null diff --git a/server/src/com/vaadin/ui/UI.java b/server/src/com/vaadin/ui/UI.java index 3ad0cc57fa..b15a83cdce 100644 --- a/server/src/com/vaadin/ui/UI.java +++ b/server/src/com/vaadin/ui/UI.java @@ -33,6 +33,7 @@ import com.vaadin.event.Action.Handler; import com.vaadin.event.ActionManager; import com.vaadin.event.MouseEvents.ClickEvent; import com.vaadin.event.MouseEvents.ClickListener; +import com.vaadin.navigator.Navigator; import com.vaadin.server.LegacyComponent; import com.vaadin.server.Page; import com.vaadin.server.Page.BrowserWindowResizeEvent; @@ -411,11 +412,11 @@ public abstract class UI extends AbstractComponentContainer implements /** * Event fired when a UI is removed from the application. */ - public static class CloseEvent extends Event { + public static class CleanupEvent extends Event { - private static final String CLOSE_EVENT_IDENTIFIER = "uiClose"; + private static final String CLEANUP_EVENT_IDENTIFIER = "uiCleanup"; - public CloseEvent(UI source) { + public CleanupEvent(UI source) { super(source); } @@ -425,23 +426,23 @@ public abstract class UI extends AbstractComponentContainer implements } /** - * Interface for listening {@link UI.CloseEvent UI close events}. + * Interface for listening {@link UI.CleanupEvent UI cleanup events}. * */ - public interface CloseListener extends EventListener { + public interface CleanupListener extends EventListener { - public static final Method closeMethod = ReflectTools.findMethod( - CloseListener.class, "click", CloseEvent.class); + public static final Method cleanupMethod = ReflectTools.findMethod( + CleanupListener.class, "cleanup", CleanupEvent.class); /** - * Called when a CloseListener is notified of a CloseEvent. + * Called when a CleanupListener is notified of a CleanupEvent. * {@link UI#getCurrent()} returns <code>event.getUI()</code> within * this method. * * @param event * The close event that was fired. */ - public void close(CloseEvent event); + public void cleanup(CleanupEvent event); } /** @@ -654,10 +655,10 @@ public abstract class UI extends AbstractComponentContainer implements /** * For internal use only. */ - public void fireCloseEvent() { + public void fireCleanupEvent() { UI current = UI.getCurrent(); UI.setCurrent(this); - fireEvent(new CloseEvent(this)); + fireEvent(new CleanupEvent(this)); UI.setCurrent(current); } @@ -845,6 +846,8 @@ public abstract class UI extends AbstractComponentContainer implements private String theme; + private Navigator navigator; + /** * This method is used by Component.Focusable objects to request focus to * themselves. Focus renders must be handled at window level (instead of @@ -1126,9 +1129,9 @@ public abstract class UI extends AbstractComponentContainer implements * @param listener * The CloseListener that should be added. */ - public void addCloseListener(CloseListener listener) { - addListener(CloseEvent.CLOSE_EVENT_IDENTIFIER, CloseEvent.class, - listener, CloseListener.closeMethod); + public void addCleanupListener(CleanupListener listener) { + addListener(CleanupEvent.CLEANUP_EVENT_IDENTIFIER, CleanupEvent.class, + listener, CleanupListener.cleanupMethod); } /** @@ -1154,14 +1157,14 @@ public abstract class UI extends AbstractComponentContainer implements /** * Removes a close listener from the UI, if previously added with - * {@link #addCloseListener(CloseListener)}. + * {@link #addCleanupListener(CleanupListener)}. * * @param listener * The CloseListener that should be removed */ - public void removeCloseListener(CloseListener listener) { - removeListener(CloseEvent.CLOSE_EVENT_IDENTIFIER, CloseEvent.class, - listener); + public void removeCleanupListener(CleanupListener listener) { + removeListener(CleanupEvent.CLEANUP_EVENT_IDENTIFIER, + CleanupEvent.class, listener); } /** @@ -1188,6 +1191,25 @@ public abstract class UI extends AbstractComponentContainer implements } /** + * Returns the navigator attached to this UI or null if there is no + * navigator. + * + * @return + */ + public Navigator getNavigator() { + return navigator; + } + + /** + * For internal use only. + * + * @param navigator + */ + public void setNavigator(Navigator navigator) { + this.navigator = navigator; + } + + /** * Setting the caption of a UI is not supported. To set the title of the * HTML page, use Page.setTitle * @@ -1362,7 +1384,7 @@ public abstract class UI extends AbstractComponentContainer implements * heartbeat for this UI. * * @see #heartbeat() - * @see VaadinSession#closeInactiveUIs() + * @see VaadinSession#cleanupInactiveUIs() * * @return The time the last heartbeat request occurred. */ diff --git a/server/tests/src/com/vaadin/tests/server/navigator/NavigatorTest.java b/server/tests/src/com/vaadin/tests/server/navigator/NavigatorTest.java index a78c76cb70..f97c2fdeb5 100644 --- a/server/tests/src/com/vaadin/tests/server/navigator/NavigatorTest.java +++ b/server/tests/src/com/vaadin/tests/server/navigator/NavigatorTest.java @@ -73,7 +73,7 @@ public class NavigatorTest extends TestCase { public static class TestNavigator extends Navigator { public TestNavigator() { - super(new NullFragmentManager(), new TestDisplay()); + super(null, new NullFragmentManager(), new TestDisplay()); } public View getView(String viewAndParameters) { @@ -221,7 +221,7 @@ public class NavigatorTest extends TestCase { control.replay(); // create and test navigator - Navigator navigator = new Navigator(manager, display); + Navigator navigator = new Navigator(null, manager, display); navigator.addProvider(provider); navigator.navigateTo("test1"); @@ -264,7 +264,7 @@ public class NavigatorTest extends TestCase { control.replay(); // create and test navigator - Navigator navigator = new Navigator(manager, display); + Navigator navigator = new Navigator(null, manager, display); navigator.addProvider(provider); navigator.navigateTo("test2"); @@ -283,7 +283,7 @@ public class NavigatorTest extends TestCase { ViewChangeTestListener listener = new ViewChangeTestListener(); // create navigator to test - Navigator navigator = new Navigator(manager, display); + Navigator navigator = new Navigator(null, manager, display); // prepare mocks: what to expect EasyMock.expect(provider.getViewName("test1")).andReturn("test1"); @@ -333,7 +333,7 @@ public class NavigatorTest extends TestCase { ViewChangeTestListener listener1 = new ViewChangeTestListener(); ViewChangeTestListener listener2 = new ViewChangeTestListener(); - Navigator navigator = new Navigator(manager, display); + Navigator navigator = new Navigator(null, manager, display); // prepare mocks: what to expect // first listener blocks first view change @@ -467,7 +467,7 @@ public class NavigatorTest extends TestCase { } public void testAddViewWithNullName() throws Exception { - Navigator navigator = new Navigator(new NullFragmentManager(), + Navigator navigator = new Navigator(null, new NullFragmentManager(), new NullDisplay()); try { @@ -483,7 +483,7 @@ public class NavigatorTest extends TestCase { } public void testAddViewWithNullInstance() throws Exception { - Navigator navigator = new Navigator(new NullFragmentManager(), + Navigator navigator = new Navigator(null, new NullFragmentManager(), new NullDisplay()); try { @@ -494,7 +494,7 @@ public class NavigatorTest extends TestCase { } public void testAddViewWithNullClass() throws Exception { - Navigator navigator = new Navigator(new NullFragmentManager(), + Navigator navigator = new Navigator(null, new NullFragmentManager(), new NullDisplay()); try { diff --git a/server/tests/src/com/vaadin/ui/LabelDataSource.java b/server/tests/src/com/vaadin/ui/LabelDataSource.java new file mode 100644 index 0000000000..3abe12535b --- /dev/null +++ b/server/tests/src/com/vaadin/ui/LabelDataSource.java @@ -0,0 +1,128 @@ +/* + * Copyright 2011 Vaadin Ltd. + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may not + * use this file except in compliance with the License. You may obtain a copy of + * the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations under + * the License. + */ +package com.vaadin.ui; + +import java.util.Locale; + +import org.junit.Assert; +import org.junit.Before; +import org.junit.Test; + +import com.vaadin.data.util.ObjectProperty; +import com.vaadin.server.VaadinSession; +import com.vaadin.server.WrappedRequest; + +public class LabelDataSource { + + Label label; + private static final String STRING_DS_VALUE = "String DatA source"; + private static final int INTEGER_DS_VALUE = 1587; + private static final String INTEGER_STRING_VALUE_FI = "1 587"; + private static final String INTEGER_STRING_VALUE_EN_US = "1,587"; + private static final Object INTEGER_STRING_VALUE_DE = "1.587"; + ObjectProperty<String> stringDataSource; + private ObjectProperty<Integer> integerDataSource; + VaadinSession vaadinSession; + + @Before + public void setup() { + vaadinSession = new VaadinSession(); + VaadinSession.setCurrent(vaadinSession); + + label = new Label(); + stringDataSource = new ObjectProperty<String>(STRING_DS_VALUE); + integerDataSource = new ObjectProperty<Integer>(INTEGER_DS_VALUE); + } + + @Test + public void stringDataSource() { + label.setPropertyDataSource(stringDataSource); + Assert.assertEquals(STRING_DS_VALUE, label.getState().text); + Assert.assertEquals(STRING_DS_VALUE, label.getValue()); + Assert.assertEquals(stringDataSource, label.getPropertyDataSource()); + label.setPropertyDataSource(null); + Assert.assertEquals(STRING_DS_VALUE, label.getState().text); + Assert.assertEquals(STRING_DS_VALUE, label.getValue()); + Assert.assertEquals(null, label.getPropertyDataSource()); + label.setValue("foo"); + Assert.assertEquals("foo", label.getState().text); + Assert.assertEquals("foo", label.getValue()); + Assert.assertNull(label.getPropertyDataSource()); + + } + + @Test + public void integerDataSourceFi() { + label.setLocale(new Locale("fi", "FI")); + label.setPropertyDataSource(integerDataSource); + Assert.assertEquals(INTEGER_STRING_VALUE_FI, label.getState().text); + Assert.assertEquals(INTEGER_STRING_VALUE_FI, label.getValue()); + Assert.assertEquals(integerDataSource, label.getPropertyDataSource()); + } + + @Test + public void integerDataSourceEn() { + label.setLocale(new Locale("en", "US")); + label.setPropertyDataSource(integerDataSource); + Assert.assertEquals(INTEGER_STRING_VALUE_EN_US, label.getState().text); + Assert.assertEquals(INTEGER_STRING_VALUE_EN_US, label.getValue()); + Assert.assertEquals(integerDataSource, label.getPropertyDataSource()); + } + + @Test + public void changeLocaleAfterDataSource() { + label.setLocale(new Locale("en", "US")); + label.setPropertyDataSource(integerDataSource); + label.setLocale(new Locale("fi", "FI")); + Assert.assertEquals(INTEGER_STRING_VALUE_FI, label.getState().text); + Assert.assertEquals(INTEGER_STRING_VALUE_FI, label.getValue()); + Assert.assertEquals(integerDataSource, label.getPropertyDataSource()); + } + + @Test + public void setRemoveDataSource() { + label.setValue("before"); + label.setPropertyDataSource(stringDataSource); + Assert.assertEquals(STRING_DS_VALUE, label.getValue()); + label.setPropertyDataSource(null); + Assert.assertEquals(STRING_DS_VALUE, label.getValue()); + label.setValue("after"); + Assert.assertEquals("after", label.getValue()); + } + + public class MockUI extends UI { + + public MockUI() { + setSession(vaadinSession); + } + + @Override + protected void init(WrappedRequest request) { + } + + } + + @Test + public void attachToSessionWithDifferentLocale() { + label.setValue("before"); + // label.setLocale(Locale.GERMANY); + label.setPropertyDataSource(integerDataSource); + UI ui = new MockUI(); + ui.setLocale(Locale.GERMANY); + ui.addComponent(label); + Assert.assertEquals(INTEGER_STRING_VALUE_DE, label.getState().text); + } +} diff --git a/shared/src/com/vaadin/shared/ApplicationConstants.java b/shared/src/com/vaadin/shared/ApplicationConstants.java index 0bacd2d256..2c0c0c4af0 100644 --- a/shared/src/com/vaadin/shared/ApplicationConstants.java +++ b/shared/src/com/vaadin/shared/ApplicationConstants.java @@ -45,4 +45,11 @@ public class ApplicationConstants { @Deprecated public static final String DRAG_AND_DROP_CONNECTOR_ID = "DD"; + + /** + * URL parameter used in UIDL requests to indicate that the full server-side + * state should be returned to the client, i.e. without any incremental + * changes. + */ + public static final String URL_PARAMETER_REPAINT_ALL = "repaintAll"; } diff --git a/uitest/src/com/vaadin/launcher/ApplicationRunnerServlet.java b/uitest/src/com/vaadin/launcher/ApplicationRunnerServlet.java index 032957bffc..f31b17333a 100644 --- a/uitest/src/com/vaadin/launcher/ApplicationRunnerServlet.java +++ b/uitest/src/com/vaadin/launcher/ApplicationRunnerServlet.java @@ -33,8 +33,11 @@ import com.vaadin.LegacyApplication; import com.vaadin.server.AbstractUIProvider; import com.vaadin.server.DeploymentConfiguration; import com.vaadin.server.LegacyVaadinServlet; +import com.vaadin.server.ServiceException; import com.vaadin.server.UIProvider; -import com.vaadin.server.VaadinServletSession; +import com.vaadin.server.VaadinSession; +import com.vaadin.server.VaadinSessionInitializationListener; +import com.vaadin.server.VaadinSessionInitializeEvent; import com.vaadin.server.WrappedHttpServletRequest; import com.vaadin.server.WrappedRequest; import com.vaadin.tests.components.TestBase; @@ -69,6 +72,21 @@ public class ApplicationRunnerServlet extends LegacyVaadinServlet { } } + @Override + protected void servletInitialized() { + super.servletInitialized(); + getVaadinService().addVaadinSessionInitializationListener( + new VaadinSessionInitializationListener() { + @Override + public void vaadinSessionInitialized( + VaadinSessionInitializeEvent event) + throws ServiceException { + onVaadinSessionStarted(event.getRequest(), + event.getVaadinSession()); + } + }); + } + private void addDirectories(File parent, LinkedHashSet<String> packages, String parentPackage) { packages.add(parentPackage); @@ -120,9 +138,8 @@ public class ApplicationRunnerServlet extends LegacyVaadinServlet { } } - @Override - protected void onVaadinSessionStarted(WrappedHttpServletRequest request, - VaadinServletSession session) throws ServletException { + protected void onVaadinSessionStarted(WrappedRequest request, + VaadinSession session) throws ServiceException { try { final Class<?> classToRun = getClassToRun(); if (UI.class.isAssignableFrom(classToRun)) { @@ -138,21 +155,20 @@ public class ApplicationRunnerServlet extends LegacyVaadinServlet { } else if (UIProvider.class.isAssignableFrom(classToRun)) { session.addUIProvider((UIProvider) classToRun.newInstance()); } else { - throw new ServletException(classToRun.getCanonicalName() + throw new ServiceException(classToRun.getCanonicalName() + " is neither an Application nor a UI"); } } catch (final IllegalAccessException e) { - throw new ServletException(e); + throw new ServiceException(e); } catch (final InstantiationException e) { - throw new ServletException(e); + throw new ServiceException(e); } catch (final ClassNotFoundException e) { - throw new ServletException( + throw new ServiceException( new InstantiationException( "Failed to load application class: " - + getApplicationRunnerApplicationClassName(request))); + + getApplicationRunnerApplicationClassName(WrappedHttpServletRequest + .cast(request)))); } - - super.onVaadinSessionStarted(request, session); } private String getApplicationRunnerApplicationClassName( diff --git a/uitest/src/com/vaadin/tests/components/label/LabelPropertySourceValue.html b/uitest/src/com/vaadin/tests/components/label/LabelPropertySourceValue.html new file mode 100644 index 0000000000..e62ef4bba1 --- /dev/null +++ b/uitest/src/com/vaadin/tests/components/label/LabelPropertySourceValue.html @@ -0,0 +1,86 @@ +<?xml version="1.0" encoding="UTF-8"?> +<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> +<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en"> +<head profile="http://selenium-ide.openqa.org/profiles/test-case"> +<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /> +<link rel="selenium.base" href="http://localhost:8888/" /> +<title>New Test</title> +</head> +<body> +<table cellpadding="1" cellspacing="1" border="1"> +<thead> +<tr><td rowspan="1" colspan="3">New Test</td></tr> +</thead><tbody> +<tr> + <td>open</td> + <td>/run/LabelPropertySourceValue?restartApplication</td> + <td></td> +</tr> +<tr> + <td>assertText</td> + <td>//div[@id='runLabelPropertySourceValue-596713704']/div/div[2]/div[2]/div/div/div</td> + <td>Hello Vaadin user</td> +</tr> +<!--set data source--> +<tr> + <td>click</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VButton[0]/domChild[0]</td> + <td></td> +</tr> +<tr> + <td>assertText</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[0]/VLabel[0]</td> + <td>This text should appear on the label after clicking the button.</td> +</tr> +<!--remove data source--> +<tr> + <td>click</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[2]/VButton[0]/domChild[0]/domChild[0]</td> + <td></td> +</tr> +<tr> + <td>assertText</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[0]/VLabel[0]</td> + <td>This text should appear on the label after clicking the button.</td> +</tr> +<!--set value to foo--> +<tr> + <td>click</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[3]/VButton[0]/domChild[0]/domChild[0]</td> + <td></td> +</tr> +<tr> + <td>assertText</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[0]/VLabel[0]</td> + <td>foo</td> +</tr> +<!--set data source--> +<tr> + <td>click</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VButton[0]/domChild[0]</td> + <td></td> +</tr> +<tr> + <td>assertText</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[0]/VLabel[0]</td> + <td>This text should appear on the label after clicking the button.</td> +</tr> +<!--set value to foo (should fail)--> +<tr> + <td>click</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[3]/VButton[0]/domChild[0]/domChild[0]</td> + <td></td> +</tr> +<tr> + <td>assertText</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[0]/VLabel[0]</td> + <td>This text should appear on the label after clicking the button.</td> +</tr> +<tr> + <td>assertCSSClass</td> + <td>vaadin=runLabelPropertySourceValue::/VVerticalLayout[0]/VOrderedLayout$Slot[1]/VVerticalLayout[0]/VOrderedLayout$Slot[3]/VButton[0]/domChild[0]/domChild[0]</td> + <td>v-errorindicator</td> +</tr> +</tbody></table> +</body> +</html> diff --git a/uitest/src/com/vaadin/tests/components/label/LabelPropertySourceValue.java b/uitest/src/com/vaadin/tests/components/label/LabelPropertySourceValue.java new file mode 100644 index 0000000000..c3d0d9cccd --- /dev/null +++ b/uitest/src/com/vaadin/tests/components/label/LabelPropertySourceValue.java @@ -0,0 +1,57 @@ +package com.vaadin.tests.components.label;
+
+import com.vaadin.data.util.ObjectProperty;
+import com.vaadin.server.WrappedRequest;
+import com.vaadin.tests.components.AbstractTestUI;
+import com.vaadin.ui.Button;
+import com.vaadin.ui.Button.ClickEvent;
+import com.vaadin.ui.Button.ClickListener;
+import com.vaadin.ui.Label;
+
+public class LabelPropertySourceValue extends AbstractTestUI {
+ private Label label;
+
+ @Override
+ public void setup(WrappedRequest request) {
+ label = new Label("Hello Vaadin user");
+ addComponent(label);
+ Button button = new Button("Give label a new property data source...");
+ button.addClickListener(new ClickListener() {
+ public void buttonClick(ClickEvent event) {
+ ObjectProperty<String> p = new ObjectProperty<String>(
+ "This text should appear on the label after clicking the button.");
+
+ label.setPropertyDataSource(p);
+ }
+ });
+ addComponent(button);
+ button = new Button("Remove data source", new ClickListener() {
+
+ @Override
+ public void buttonClick(ClickEvent event) {
+ label.setPropertyDataSource(null);
+ }
+ });
+ addComponent(button);
+
+ button = new Button("Set label value to 'foo'", new ClickListener() {
+
+ @Override
+ public void buttonClick(ClickEvent event) {
+ label.setValue("foo");
+ }
+ });
+ addComponent(button);
+ }
+
+ @Override
+ protected String getTestDescription() {
+ return "The value should change by clicking the button";
+ }
+
+ @Override
+ protected Integer getTicketNumber() {
+ return 9618;
+ }
+
+}
\ No newline at end of file diff --git a/uitest/src/com/vaadin/tests/components/window/AttachShouldBeCalledForSubWindows.java b/uitest/src/com/vaadin/tests/components/window/AttachShouldBeCalledForSubWindows.java index 6f21346b7d..4d0dea4f91 100644 --- a/uitest/src/com/vaadin/tests/components/window/AttachShouldBeCalledForSubWindows.java +++ b/uitest/src/com/vaadin/tests/components/window/AttachShouldBeCalledForSubWindows.java @@ -1,6 +1,7 @@ package com.vaadin.tests.components.window; import com.vaadin.event.ShortcutAction.KeyCode; +import com.vaadin.server.VaadinService; import com.vaadin.server.WrappedRequest; import com.vaadin.tests.components.AbstractTestCase; import com.vaadin.tests.util.Log; @@ -11,7 +12,6 @@ import com.vaadin.ui.Component; import com.vaadin.ui.Label; import com.vaadin.ui.UI; import com.vaadin.ui.Window; -import com.vaadin.util.CurrentInstance; public class AttachShouldBeCalledForSubWindows extends AbstractTestCase { private static final long serialVersionUID = 1L; @@ -23,7 +23,7 @@ public class AttachShouldBeCalledForSubWindows extends AbstractTestCase { @Override public void init() { - WrappedRequest request = CurrentInstance.get(WrappedRequest.class); + WrappedRequest request = VaadinService.getCurrentRequest(); if (request.getParameter("attachMainFirst") != null) { addSubWindowBeforeMainWindow = false; } else { diff --git a/uitest/src/com/vaadin/tests/navigator/NavigatorTest.java b/uitest/src/com/vaadin/tests/navigator/NavigatorTest.java index ef9130a2b1..4986bd21e6 100644 --- a/uitest/src/com/vaadin/tests/navigator/NavigatorTest.java +++ b/uitest/src/com/vaadin/tests/navigator/NavigatorTest.java @@ -104,7 +104,7 @@ public class NavigatorTest extends UI { @Override protected void init(WrappedRequest req) { try { - navi = new Navigator(naviLayout); + navi = new Navigator(this, naviLayout); navi.addView("", new DefaultView()); @@ -113,7 +113,8 @@ public class NavigatorTest extends UI { navi.addView("forbidden", new ForbiddenView()); navi.addViewChangeListener(new NaviListener()); - // navi.navigate(); + + navi.navigate(); addComponent(new NaviButton("list")); addComponent(new NaviButton("edit")); diff --git a/uitest/src/com/vaadin/tests/resources/NonExistingFileResource.java b/uitest/src/com/vaadin/tests/resources/NonExistingFileResource.java index c102291b50..f55c1f1f1d 100644 --- a/uitest/src/com/vaadin/tests/resources/NonExistingFileResource.java +++ b/uitest/src/com/vaadin/tests/resources/NonExistingFileResource.java @@ -3,11 +3,10 @@ package com.vaadin.tests.resources; import java.io.File; import com.vaadin.server.FileResource; -import com.vaadin.server.WrappedRequest; +import com.vaadin.server.VaadinService; import com.vaadin.tests.components.TestBase; import com.vaadin.ui.Button; import com.vaadin.ui.Button.ClickEvent; -import com.vaadin.util.CurrentInstance; public class NonExistingFileResource extends TestBase { @@ -26,10 +25,8 @@ public class NonExistingFileResource extends TestBase { @Override public void buttonClick(ClickEvent event) { - FileResource res = new FileResource(new File(CurrentInstance - .get(WrappedRequest.class).getVaadinService() - .getBaseDirectory() - + "/" + filename)); + FileResource res = new FileResource(new File(VaadinService + .getCurrent().getBaseDirectory() + "/" + filename)); getMainWindow().open(res); } diff --git a/uitest/src/com/vaadin/tests/tickets/Ticket1975.java b/uitest/src/com/vaadin/tests/tickets/Ticket1975.java index a688fd9908..f41585ff5d 100644 --- a/uitest/src/com/vaadin/tests/tickets/Ticket1975.java +++ b/uitest/src/com/vaadin/tests/tickets/Ticket1975.java @@ -5,14 +5,13 @@ import java.io.File; import java.io.FileInputStream; import com.vaadin.LegacyApplication; -import com.vaadin.server.WrappedRequest; +import com.vaadin.server.VaadinService; import com.vaadin.ui.Button; import com.vaadin.ui.Button.ClickEvent; import com.vaadin.ui.Button.ClickListener; import com.vaadin.ui.CustomLayout; import com.vaadin.ui.GridLayout; import com.vaadin.ui.UI.LegacyWindow; -import com.vaadin.util.CurrentInstance; public class Ticket1975 extends LegacyApplication { @@ -47,8 +46,8 @@ public class Ticket1975 extends LegacyApplication { } })); - File baseDir = CurrentInstance.get(WrappedRequest.class) - .getVaadinService().getBaseDirectory().getAbsoluteFile(); + File baseDir = VaadinService.getCurrent().getBaseDirectory() + .getAbsoluteFile(); File f = new File(baseDir + "/VAADIN/themes/" + getTheme() + "/layouts/Ticket1975.html"); diff --git a/uitest/src/com/vaadin/tests/util/SampleDirectory.java b/uitest/src/com/vaadin/tests/util/SampleDirectory.java index 140f778551..5c45e0d64e 100644 --- a/uitest/src/com/vaadin/tests/util/SampleDirectory.java +++ b/uitest/src/com/vaadin/tests/util/SampleDirectory.java @@ -19,13 +19,12 @@ package com.vaadin.tests.util; import java.io.File; import com.vaadin.server.SystemError; +import com.vaadin.server.VaadinService; import com.vaadin.server.VaadinSession; -import com.vaadin.server.WrappedRequest; import com.vaadin.shared.ui.label.ContentMode; import com.vaadin.ui.Label; import com.vaadin.ui.Panel; import com.vaadin.ui.UI; -import com.vaadin.util.CurrentInstance; /** * Provides sample directory based on application directory. If this fails then @@ -48,8 +47,7 @@ public class SampleDirectory { + "context base directory failed, " + "possible security constraint with Application " + "Server or Servlet Container.<br />"; - File file = CurrentInstance.get(WrappedRequest.class) - .getVaadinService().getBaseDirectory(); + File file = VaadinService.getCurrent().getBaseDirectory(); if ((file == null) || (!file.canRead()) || (file.getAbsolutePath() == null)) { // cannot access example directory, possible security issue with |