更新:我即将为此问题添加赏金。我在这里将整个项目的源代码添加到GitHub:
https://github.com/doctrang/gwt-activities-places-mvp-example
我决定将SimpleModule
重命名为WebModule
;所以,在我下面的所有代码片段中,GWT模块名为SimpleModule
或simplemodule
,在我的最新代码中,您将分别看到名为WebModule
和webmodule
的模块 - 但他们是一样的!
注意:我知道我可能没有在这里完美设置,并且可能有一些完全没有使用的死代码(SimpleModule.css
等),但是这是我的第一个GWT应用程序,我只想让这个东西运行起来!
我正在尝试使用我的第一个GWT应用程序(2.5.1),并尝试使用推荐的位置和放大器来显示一个简单的UI。活动框架(用于历史管理)以及利用基本的MVP架构。
为简单起见,我已将所有Java代码放在SimpleModule.java
(入口点)中。一旦我得到这个概念验证工作,我就会将SimpleModule.java
分解为更多的类。
我的目标是在用户访问我的主页(me.example.com)时加载GWT应用。因此,我创建了SimpleModule implements EntryPoint
,然后将主机页面从SimpleModule.html
重命名为index.html
(以便用户访问me.example.com或me.example.com/ index.html,他们将下拉SimpleModule
)。
我的WAR的目录结构:
war/
hosts/
simplemodule/
SimpleModule.css
WEB-INF/
classes/
lib/
deploy/
web.xml
simplemodule/
css/
font/
gwt/
img/
js/
prettify/
clear.cache.gif
hosted.html
simplemodule.nocache.js
img/
mylogo.jpg
index.html
index.html
:
<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8">
<script type="text/javascript" language="javascript" src="simplemodule/simplemodule.nocache.js"></script>
</head>
<body>
<iframe src="javascript:''" id="__gwt_historyFrame" tabIndex='-1' style="position:absolute;width:0;height:0;border:0"></iframe>
<noscript>
<div style="width: 22em; position: absolute; left: 50%; margin-left: -11em; color: red; background-color: white; border: 1px solid red; padding: 4px; font-family: sans-serif">
Your web browser must have JavaScript enabled in order for this application to display correctly.
</div>
</noscript>
</body>
</html>
web.xml
:
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5" xmlns="http://java.sun.com/xml/ns/javaee">
<servlet>
<servlet-name>greetServlet</servlet-name>
<servlet-class>com.myapp.server.GreetingServiceImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>greetServlet</servlet-name>
<url-pattern>/simplemodule/greet</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>remoteLogging</servlet-name>
<servlet-class>com.google.gwt.logging.server.RemoteLoggingServiceImpl</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>remoteLogging</servlet-name>
<url-pattern>/simplemodule/remote_logging</url-pattern>
</servlet-mapping>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
</welcome-file-list>
</web-app>
SimpleModule.java
:
public class SimpleModule implements EntryPoint {
private EventBus eventBus = new SimpleEventBus();
private PlaceController placeController = new PlaceController(eventBus);
private PlaceHistoryMapper placeHistoryMapper;
private PlaceHistoryHandler placeHistoryHandler;
private Place defaultPlace;
private ActivityMapper activityMapper;
private ActivityManager activityManager;
private LoginDisplay loginDisplay = new LoginDisplay();
@Override
public void onModuleLoad() {
bootstrap();
RootPanel.get().add(loginDisplay);
activityManager.setDisplay(loginDisplay);
placeHistoryHandler.register(placeController, eventBus, defaultPlace);
placeHistoryHandler.handleCurrentHistory();
}
private void bootstrap() {
placeHistoryMapper = new PlaceHistoryMapper() {
@Override
public String getToken(Place arg0) {
return "home";
}
@Override
public Place getPlace(String arg0) {
return defaultPlace;
}
};
placeHistoryHandler = new PlaceHistoryHandler(placeHistoryMapper);
defaultPlace = new Place() {};
activityMapper = new ActivityMapper() {
@Override
public Activity getActivity(Place arg0) {
return new LoginActivity();
}
};
activityManager = new ActivityManager(activityMapper, eventBus);
}
public class LoginDisplay extends SimplePanel {
private LoginDisplayUiBinder uiBinder = GWT
.create(LoginDisplayUiBinder.class);
public LoginDisplay() {
super();
uiBinder.createAndBindUi(this);
}
}
public interface LoginDisplayUiBinder extends UiBinder<Widget, LoginDisplay> {
// ???
}
public class LoginActivity extends AbstractActivity {
@Override
public void start(AcceptsOneWidget panel,
com.google.gwt.event.shared.EventBus eventBus) {
panel.setWidget(loginDisplay);
}
}
}
LoginDisplay.ui.xml
:
<!DOCTYPE ui:UiBinder SYSTEM "http://dl.google.com/gwt/DTD/xhtml.ent">
<ui:UiBinder xmlns:ui="urn:ui:com.google.gwt.uibinder"
xmlns:g="urn:import:com.google.gwt.user.client.ui"
xmlns:b="urn:import:com.github.gwtbootstrap.client.ui">
<g:HTMLPanel>
<img src="img/mylogo.jpg" />
<hr/>
<form action="doStuff" method="post" class="form-horizontal"
id="someForm" accept-charset="utf-8">
<div class="control-group">
<label for="username" class="control-label">
Username:
</label>
<div class="controls">
<input name="username" type="text" value="" id="username"/>
</div>
</div>
<div class="control-group">
<label for="password" class="control-label">
Password:
</label>
<div class="controls">
<input name="password" type="password" value="" id="password"/>
</div>
</div>
<div class="control-group">
<div class="controls">
<input type="button" class="btn-danger" value="Login"/>
</div>
</div>
</form>
</g:HTMLPanel>
</ui:UiBinder>
更新:和SimpleModule.gwt.xml
:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE module PUBLIC "-//Google Inc.//DTD Google Web Toolkit 2.5.1//EN"
"http://google-web-toolkit.googlecode.com/svn/tags/2.5.1/distro-source/core/src/gwt-module.dtd">
<module rename-to='simplemodule'>
<inherits name='com.google.gwt.user.User'/>
<!-- Configure logging. -->
<inherits name="com.google.gwt.logging.Logging"/>
<set-property name="gwt.logging.logLevel" value="FINEST"/>
<set-property name="gwt.logging.enabled" value="TRUE"/>
<set-property name="gwt.logging.consoleHandler" value="ENABLED"/>
<set-property name="gwt.logging.developmentModeHandler" value="DISABLED" />
<set-property name="gwt.logging.popupHandler" value="DISABLED" />
<set-property name="gwt.logging.systemHandler" value="DISABLED" />
<set-property name="gwt.logging.firebugHandler" value="DISABLED" />
<set-property name="gwt.logging.simpleRemoteHandler" value="DISABLED" />
<!-- GWT-Bootstrap. -->
<inherits name ="com.github.gwtbootstrap.Bootstrap"/>
<inherits name='com.google.gwt.user.theme.clean.Clean'/>
<entry-point class='com.myapp.client.SimpleModule'/>
<source path='client'/>
<source path='shared'/>
</module>
服务器启动后的网页源:
<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8">
<script type="text/javascript" language="javascript" src="simplemodule/simplemodule.nocache.js"></script>
</head>
<body height="100%" width="100%">
<iframe src="javascript:''" id="__gwt_historyFrame" tabIndex='-1' style="position:absolute;width:0;height:0;border:0"></iframe>
<noscript>
<div style="width: 22em; position: absolute; left: 50%; margin-left: -11em; color: red; background-color: white; border: 1px solid red; padding: 4px; font-family: sans-serif">
Your web browser must have JavaScript enabled in order for this application to display correctly.
</div>
</noscript>
</body>
</html>
当我在开发模式下运行它(在com.google.gwt.dev.DevMode
上调用Java的Ant目标)时,我启动了DevMode工具而没有错误。当我选择“启动应用程序”时,我会得到一个空白网页。当我打开Firebug并因错误而刺激时,我看不到任何。
我的设置有什么问题??? 为什么我看不到带有登录字段的简单图像徽标?提前谢谢!
修改:我的最新更新。我按照@ Raphael的建议(继承Place和Activity模块,并将@UITemplate
注释添加到LoginDisplayUiBinder
),我得到标签“Hello,GWT!”打印到我的浏览器!然后,我将onModuleLoader()
方法修改为:
@Override
public void onModuleLoad() {
bootstrap();
// RootPanel.get().add(new Label("Hello, GWT!"));
RootPanel.get().add(loginDisplay);
activityManager.setDisplay(loginDisplay);
// Connect the PlaceController to the EventBus, and set the
// defaultPlace as our first place in history.
placeHistoryHandler.register(placeController, eventBus, defaultPlace);
placeHistoryHandler.handleCurrentHistory();
}
我现在得到以下例外:
onModuleLoad() threw an exception
Exception while loading module com.dummylandapp.client.WebModule. See Development Mode for details.
java.lang.reflect.InvocationTargetException
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:616)
at com.google.gwt.dev.shell.ModuleSpace.onLoad(ModuleSpace.java:406)
at com.google.gwt.dev.shell.OophmSessionHandler.loadModule(OophmSessionHandler.java:200)
at com.google.gwt.dev.shell.BrowserChannelServer.processConnection(BrowserChannelServer.java:526)
at com.google.gwt.dev.shell.BrowserChannelServer.run(BrowserChannelServer.java:364)
at java.lang.Thread.run(Thread.java:679) Caused by: com.google.web.bindery.event.shared.UmbrellaException: Exception caught: Exception caught: (HierarchyRequestError) @com.google.gwt.dom.client.Node::appendChild(Lcom/google/gwt/dom/client/Node;)([JavaScript object(15)]): Node cannot be inserted
at the specified point in the hierarchy
at com.google.web.bindery.event.shared.SimpleEventBus.doFire(SimpleEventBus.java:203)
at com.google.web.bindery.event.shared.SimpleEventBus.fireEvent(SimpleEventBus.java:88)
at com.google.gwt.place.shared.PlaceController.goTo(PlaceController.java:156)
at com.google.gwt.place.shared.PlaceHistoryHandler.handleHistoryToken(PlaceHistoryHandler.java:192)
at com.google.gwt.place.shared.PlaceHistoryHandler.handleCurrentHistory(PlaceHistoryHandler.java:118)
at com.dummylandapp.client.WebModule.onModuleLoad(WebModule.java:66) ... 9 more Caused by: com.google.gwt.event.shared.UmbrellaException: Exception caught: (HierarchyRequestError) @com.google.gwt.dom.client.Node::appendChild(Lcom/google/gwt/dom/client/Node;)([JavaScript object(15)]): Node cannot be inserted
at the specified point in the hierarchy
at com.google.gwt.activity.shared.ActivityManager.onPlaceChange(ActivityManager.java:168)
at com.google.gwt.place.shared.PlaceChangeEvent.dispatch(PlaceChangeEvent.java:70)
at com.google.gwt.place.shared.PlaceChangeEvent.dispatch(PlaceChangeEvent.java:1)
at com.google.gwt.event.shared.GwtEvent.dispatch(GwtEvent.java:1)
at com.google.web.bindery.event.shared.EventBus.dispatchEvent(EventBus.java:40)
at com.google.web.bindery.event.shared.SimpleEventBus.doFire(SimpleEventBus.java:193) ... 14 more
答案 0 :(得分:3)
问题出在这里:
public class LoginDisplay extends SimplePanel {
private LoginDisplayUiBinder uiBinder = GWT
.create(LoginDisplayUiBinder.class);
public LoginDisplay() {
super();
uiBinder.createAndBindUi(this);
}
}
您创建了窗口小部件,但从未将其添加到面板中:
uiBinder.createAndBindUi(this);
尝试:
this.add(uiBinder.createAndBindUi(this));
请注意,您也可以将uiBinder接口放在LoginDisplay
类中,因为这里只需要它(使代码更清晰):
public class LoginDisplay extends SimplePanel {
public interface LoginDisplayUiBinder extends UiBinder<Widget, LoginDisplay> {}
public LoginDisplay() {
super();
LoginDisplayUiBinder uiBinder=GWT.create(LoginDisplayUiBinder.class);
this.add(uiBinder.createAndBindUi(this));
}
}
答案 1 :(得分:2)
您在项目中使用放置和活动,但您的模块未继承它们。
将以下行添加到WebModule.xml
:
<inherits name='com.google.gwt.place.Place'/>
<inherits name='com.google.gwt.activity.Activity'/>
此外,如果视图模板与其封闭类的名称不同,则必须将UiTemplate
注释添加到UiBinder接口。
在您的情况下,GWT在尝试创建WebModule.ui.xml
时会寻找LoginDisplayUiBinder
。
@UiTemplate("LoginDisplay.ui.xml")
public interface LoginDisplayUiBinder extends UiBinder<Widget, LoginDisplay> {}
<强>更新强>
您的新例外是由此引起的:
RootPanel.get().add(loginDisplay);
activityManager.setDisplay(loginDisplay);
您首次添加登录表单并将其设置为ActivityManager
的主显示。显示是在其start方法中为您的活动提供的panel
参数:
void start(AcceptsOneWidget panel, com.google.gwt.event.shared.EventBus eventBus);
在您的情况下,当您触发网页更改事件时(使用placeHistoryHandler.handleCurrentHistory();
),LoginActivity
会尝试将LoginDisplay
添加到自身,从而导致Exception
:
panel.setWidget(loginDisplay);
您可以执行的一项简单修复是创建其他SimplePanel
并将其设置为主显示:
final SimplePanel mainPanel = new SimplePanel();
RootPanel.get().add(mainPanel);
activityManager.setDisplay(mainPanel);
答案 2 :(得分:0)
你有yourapp.gwt.xml文件吗?如果是,请发布。如果没有,那就是原因。更多关于GWT documentation的信息。因为看起来您可能缺少<entry-point class='com.myapp.MyEntryPointClass' />
因为这会毫无问题地加载您的应用程序,所以启动您的html文件,但是不会执行任何代码,而是保留未修改的html文件。
你如何构建这个应用程序?
答案 3 :(得分:0)
我总是使用相同的项目名称,相同的模块名称和相同的Html页面运行我的应用程序。
像这样:
项目:SimpleModule
XML:SimpleModule.gwt.xml
包含此:
最后
SimpleModule.html
您无需将其重命名为index.html。
当gwt编译时,假设你在tomcat中运行它,它不需要index.html,它将SimpleModule.html识别为索引。
希望有所帮助。
你应该把你的日志错误更清楚,我建议你去做一个新的项目。看看它是如何工作的,并分析你的遗失。
也不要忘记de web.xml。
答案 4 :(得分:0)
1)您是使用Eclipse还是其他IDE?当我尝试使用不同的IDE(因此没有插件)和ANT文件时,我无法使GWT正常工作。
2)你把整个GreetingService servlet放在那里,你确定这不会给你带来麻烦吗?
答案 5 :(得分:0)
活动&amp;地方是一个高尚的概念,由于其固有的复杂性,永远不会起飞。请改为尝试gwtp
(GWT Platform)。你永远不会回去。您可以在几分钟内完成正在运行的MVP示例,并且GWTP具有A&amp; P所具有的所有历史管理/书签,并且具有更容易理解和编码的API。
相信我,GWTP是所有MVP应用的方式。