当Abacus Research AG,一家商业软件开发公司,决定将开发平台从 Borland 的 Delphi 转向java Swing时,它发现自己站在了软件开发的十字路口。
我们大量的应用程序开发团队拥有多年的Delphi GUI开发经验,却没有任何的Java/Swing 经验,并且我们需要在极短的时间内将大量的Delphi应用程序转换为Swing。在研究了当时可用的GUI工具后,我们觉得我们应该通过赞助一个GPL开源Java GUI编译器来达到自己的目标,而且这个编译器还可以帮助其他的应用程序开发者面对相同的挑战。
AbaGUIBuilder ,Abacus Java GUI编译器被设计用来实现轻松地从VB或Delphi开发转向Java Swing。这一策略帮助Abacus Research实现了其Java版应用程序的交付,缩短了开发时间,使其开发人员把精力投向商业应用,而不是Swing框架的错综复杂。
如图1所示,AbaGUIBuilder有能够在许多其他开发环境如VB和Delphi找到的要素。位于视窗左上角是活动对象树,其下是Swing组件选择区。位于视窗的中上部是开发面板,或者说是“画布”,在这里放入Swing组件将其实例化。在右上部是属性编辑器。下部是事件面板,包括两个制表符页,分别是事件代码编辑器,为每个单独的可视对象添加事件代码,和消息面板,显示AbaGUIBuilder的状态消息。
图1. Abacus GUI编译器.
为什么做我们自己的GUI工具?
为什么我们选择写自己的工具?开始,我们决定我们的工具应该包括以下五个要求:
1. 轻易使用
2.不需要布局治理器的使用经验
3. 不需要Swing的使用经验
4. 不需要输出Swing代码
5. 实现商业逻辑与UI的分离
我们在评估当时可用的GUI开发工具时发现好的IDE都是手写Swing代码,这就不符合我们的第一个和最要害要求。
我们定义的轻易使用是指任一开发者不管他/她的Java Swing经验,都具有在几分钟而不是几小时内开发一个可运行的GUI表单,因此就要是开发者的精力集中在商业逻辑上而不是框架的细节。一开始我们的开发者就需要所见即所得的环境,以便能够真实地感知当应用程序发布的时候的情况。这一功能的实现节省了开发者的大量时间。但当时我们评估的工具没有一个能够实现这个功能;设计的屏幕与输出的屏幕并不一样还有糟糕的调度算法,对我们来说简直就是巨大的倒退。AbaGUIBuilder通过预览模式来实现这一功能,你可以在开发时看到它运行起来的样子。
为什么Abacus Research要求UI和商业逻辑的分离?因为企业要处理不断变换的政府规则,例如税收和工资计算;我们需要在一个独立的包装模块里维护规则和公式时保持UI不变的能力,使得当公式改变时,应用程序JAR(UI)并不需要变化。因此只有改变的公式需要测试,使得发布的进程简单化。
为了满足这个需求,我们设计的GUI编译器将应用程序和事件汇编到一个应用程序JAR,JAR隐藏了所有的GUI Swing 代码并且通过一个renderer类(AdaRenderer)在运行时来执行应用程序。
不需要Swing经验
为了确保开发工具能够做到真正的所见即所得,我们决定采用类似Delphi和VB中用XY坐标来定位的布局治理方式。采用这种方式,Swing JFrame就是一个画布,开发者可以将Swing组件放入JFrame的XY布局中。有了XY布局治理器,我们的开发者不需要理解Swing中并不熟悉的,复杂的布局治理器,这样我们的应用程序开发者就可以把精力集中到应用程序UI和它的商业逻辑中。
事实上,AbaGUIBuilder包含了大部分的Swing可视组件,从面板到制表符页,还有对菜单﹑菜单项,带有Java数据库连接的数据库感知组件,JFreeChart组件的支持,和导入第三方可视类库的能力。所有这些组件都可以从组件选择区拖入一个空框架来开发你的GUI应用程序。
示例项目: 跟踪开发商的联络信息
最佳的展示AbaGUIBuilder RAD(快速应用程序开发)的方式就是开发一个带有菜单栏的多文本制表符面板的示例程序。首先,你必须通过在组件选择区选择JFrame,将其拖入应用程序画布中来实现将JFrame对象放入空的表单面板中。然后,将所有的可视组件托在JFrame上。注重:当你开始一个新的项目时,必须先放入JFrame,使其成为你的对象画布,见图2。
图2. 应用程序JFrame 作为画布. 点击缩略图查看完整图.
其次,增加一个制表符格和两个制表符页。在组件选择区的容器区内选择JTabbedPane将其拖入JFrame1。当JTabbedPane放置好后,右键,在弹出菜单中选择 Add JATabPage在面板上增加两个制表符页。如图3。
图3. 增机制表符页: 右键制表符页选择Add JATabPage.
然后,在属性栏中选择TaBTitle为每个制表符页设置标题。
图4.可以放入Swing组件的空制表符页.
这时,你已经有了两个空的制表符页,你可以从组件选择区选择任意的Swing组件放入其中。在我们的例子中,开发者制表符页中包含两个屏面,上面有一系列的JLabel和JTextField对象。你能够很快的开发出一个类似图5的应用程序。作为一个可视化应用程序开发者,你能够体会到使用Abacus GUI编译器能给你开发复杂的GUI应用程序所带来的方便。
图5. 短时开发的应用程序.
编译器带的透视图模式是一个很有用的特征,因为它能够预览运行时的应用程序。通过选择工具栏的Run选项或者按F9键,即可进入透视图模式
图6. 所见即所得: 开发时检查视觉效果.
这时保存你的工作,按下Ctrl-S或者选择Save选项,给项目起名叫devteam,将其保存到示例目录中。在你保存完项目后,打开AbaGUIBuilder的示例目录,打开文件devteam.PRoj。proj文件是一个xml文件,其中定义了应用程序中的所有类,对象,对象的性质,甚至是代码。这是一个对你的可视项目的表达平台。
增加事件句柄
为每个对象增加事件句柄是一个直截了当的过程。你所要做的就是点击对象,从事件列表中选择你希望中断产生的事件,并为事件填写合适的Java代码。再一次证实,AbaGUIBuilder的设计符合Delphi和VB的模式。我们的目标是对应用程序开发者隐藏UI的实现,例如事件监听器等。举个例子,如图7,为了实现按下Exit(退出)按钮时,弹出一个确认对话框,首先选择Exit按钮的actionPerformed事件,然后填入如下Java代码
图7. 为可视化组件增加事件句柄.
切记,事件句柄在IDE透视图模式下并不能使用,只有在运行时才被激活,所以你必须编译,运行应用程序才能检查结果。
可视化地增加一个菜单和菜单项是另一个强大的并节省时间的特征。你所要做的就是从组件选择区选择JMenuBar并将其放在框架的任意地方。JMenuBar一般放在顶部且XY坐标无关。然后右键点击JMenuBar,弹出一个弹出框,如图8。为菜单增加JMenu,JMenuItem(s),和事件句柄。
图8. 可视化增加菜单和菜单项.
然后,复制Exit按钮的actionPerformed事件代码到Exit菜单项中,使得不管选择Exit按钮还是Exit菜单项都回弹出一样的确认对话框。最后,点击工具栏中的Save(With Compile)。现在你完成了你的第一个AbaGUIBuilder应用程序。
代码到哪里去了?
我们经常听到这样的问题:代码去哪了?AbaGUIBuilder并不是不创造Java代码,只不过是编译器内部生成的Java代码不被看到和用到而已。GUI编译器通过运行独立的包装程序(renderer)将内部代码编译到应用程序的jar文件中。事实上,你可以在\bin\output目录下检测AbaGUIBuilder产生的Java代码。但是切记,这些代码并使被外部使用的;它只不过作为一个辅助工具,当编译器生成的代码万一有问题时使用。
生成项目文件(.proj)后,GUI编译器生成两个文件,你的应用程序jar文件和decl文件。随后当你写包装器文件时,decl文件会有帮助。它包含了所有的对象定义和一个通用入口函数getReferences(),并且它将UI表单的所有可视元素写为私有变量。我们可以使用decl文件程序段中的这些私有变量实现商业逻辑和UI的分离。
开发过程中,我们推荐你使用例子文件夹中的runproz脚本来运行你的应用程序jar文件。这个脚本文件设置了classpath,填交了所有需要的JAR,使用示例Java包装器来运行应用程序JAR。如图9,当你运行devteam应用程序时,命令如下:runproz \abaguibuilder-1.7\samples\devteam.jar
图9. 部署应用程序.
理解runproz的机制很重要,虽然很简单,让我们注重下行:"%JAVA_HOME%\bin\java.exe" exec %1
exec是示例包装器AbaRenderer提供的默认加载机制,参数%1指的是AbaGUIBuilder应用程序JAR的名字。当JAR的路径和名字传给exec.java时,它将加载和运行指定的应用程序JAR。这就是应用程序JAR运行的机制。
编写自己的包装器
AbaRenderer包装器是一个用AbaRenderer 对象来加载应用程序jar的Java程序。包装器是一个仅仅需要几行代码的简单程序。
public class exec
{
public static AbaRenderer m_AbaRenderer ;
// Assignments for this user interface
public static void main(String[] args)
{
String docname = new String(args[0]);
try
{
System.out.println("Loading.." + docname);
m_AbaRenderer = new AbaRenderer(docname, true , null);
boolean bTestLoad = m_AbaRenderer.load();
if(bTestLoad)
m_AbaRenderer.renderInterface(); }
catch(Exception e)
{e.printStackTrace();}
}
}
当部署应用程序时,你可以选择使用runproz和示例包装器来发布它,也可以使用你自己写的包装器和classpath。当你需要为你的包装器增加其它的一些特征时我们推荐你编写自己的包装器,例如:使用第三方程序库的监听器,手工初始化对象,为组合框增加代码,或者最后将UI从业务逻辑中分离出来等。
将业务逻辑从UI中分离出来
为了将应用程序的UI和业务逻辑分离开,你必须写自己的包装器。一个分离的简单例子就是在开发工具和项目以外,通过包装器手工地初始化应用程序的可视组件。当你再次调用时,GUI编译器生成一个声明文件。这个文件有一个访问方法可以初始化一系列私有变量,这些私有变量是表单上可视对象的引用。getReferences()方法答应开发者访问和操作表单上的任一可视组件。
下面的exec2程序在第一个制表符位置初始化了一个组合框,示范了一个简单的应用程序UI和业务逻辑的分离:
public class exec2
{
public static AbaRenderer m_AbaRenderer ;
// Declarations of variables for this user interface.
…….
private JComboBox JStComboBox;
// Assignments for this user interface
public void getReferences()
{
…..
//Loads the visual object JComboBox1 to private data JComboBox1
JStComboBox= (JComboBox)m_AbaRenderer.getObject("JComboBox1");
…..
}
// Assignments for this user interface
public static void main(String[] args)
{
String docname = new String(args[0]);
try
{
System.out.println("Loading.." + docname);
m_AbaRenderer = new AbaRenderer(docname, true , null);
boolean bTestLoad = m_AbaRenderer.load();
if(bTestLoad)
{
m_AbaRenderer.renderInterface();
getReferences();
// Sample access to objects
JStComboBox.addItem("FL");
JStComboBox.addItem("CA");
JStComboBox.addItem("WA");
JStComboBox.addItem("MD");
JStComboBox.addItem("PA");
}
}
catch(Exception e)
{
e.printStackTrace();
}
}
}