扩展RefactoringWizard 框架
Eclipse中的RefactoringWizard框架扩展了Eclipse的Wizard框架,关于Wizard框架的介绍可以在Eclipse的帮助系统中找到,这里我们仅从OO设计和架构的角度探讨一下RefactoringWizard框架。
我们从Wizard相关的几个类开始:
1. WizardPage类
WizardPage是一个包含了多个界面元素(比如文本框Text,按钮Button)的一个界面组合部分。各个Page之间是独立的,是可以动态加载的。WizardPage类的职责有:
·组合SWT界面元素,构造出一个界面页。
·定义本身界面元素的操作行为。
在RefactoringWizard框架中预设了两个通用的属性页:PReviewWizardPage和ErrorWizardPage。PreviewWizardPage类是用来预览重构后的修改,对比代码或其他资源的变化。ErrorWizardPage类是用来处理条件检查及错误状态通知的。我们只需扩展RefactoringWizard框架就可以自动获取这两项强大的功能。
2. Wizard类
一个Wizard就是一个装载一系列WizardPage页的容器,Wizard类的职责有:
·装载一系列WizardPage,构造出一个复杂的界面。
·装载领域类来处理具体业务逻辑。(在RefactoringWizard框架中这个类就是Refactoring类)
维护WizardPage页之间以及页与领域类之间的数据传递和状态共享。(在这里要补充一点,其实在具体RefactoringWizard框架的实现中有专门的类来分担这部分职责。)
我们的界面行为可以千变万化(通过组合不同的WizardPage),而负责处理业务逻辑的领域类也可以独立的变化,你可以随意扩展Wizard的界面功能(-对扩展开放),而不用修改现有RefactoringWizard框架(-对修改封闭),这正是OO设计的最基本原则-OCP(Open-Close Principle)。
3. WizardDialog类
这个对话框类的主要职责是构造一个完整的GUI界面以及操作界面。它预设了一些按钮(Back,Next,Finish,Cancel)等界面元素,它负责装载Wizard类,操作时通过按钮Back、Next来在多个WizardPage之间切换。
下面我们给出RefactoringWizard框架的架构图:
图 5 Refactoring Wizard架构图
从图 5中我们可以看到,假如我们把每一个WizardPage页看作一项业务,那么Refactoring正是处理业务逻辑的控制中心,它封装了所有对业务逻辑的处理,当然它可以在将处理任务委任出去。但请注重,它并不负责实现业务流程,也就是说各业务(各个Page界面)之间的逻辑顺序关系不由它维护。
RefactoringWizard框架充分考虑到了应用的可扩展性,它在SWT的MVC(模型-视图-控制)元架构模式的基础上,添加了一些新的架构元素。MVC模式促使业务逻辑与界面分离,界面与控制行为分离,而RefactoringWizard框架增强了界面本身分离的特性,它将一个完整的界面分拆成多个页面,用户可以动态组合这些页面或添加新的页面来扩展界面行为。这种特性-界面的动态组合,低耦合,高内聚,封装良好的接口-让我们领略到了OO设计的精髓。
下面我们通过以下几个步骤来扩展RefactoringWizard框架:
·扩展RefactoringWizardPage
·扩展RefactoringWizard
·启动RefactoringWizard
第一步,扩展RefactoringWizardPage:首先我们新建一个类AnnotationRefactoringWizardPage,它需要继续UserInputWizardPage类(其父类是RefactoringWizardPage,而RefactoringWizardPage最终实现了IDialogPage接口)。接下来就是实现IDialogPage接口的createControl(…)方法,在这个方法里实现你的界面行为,比如我们例子中的TimeOut文本框,代码清单如下:
清单 14
/**
* create composite to add UI elements
*/
public void createControl(Composite parent) {
// define UI
Composite composite = new Composite(parent, SWT.NONE);
GridLayout lay = new GridLayout();
lay.numColumns = 2;
composite.setLayout(lay);
BTnCheck = new Button(composite, SWT.CHECK);
btnCheck.setText("Add timeout parameter");
GridData gdBtnCheck = new GridData();
gdBtnCheck.horizontalSpan = 2;
gdBtnCheck.horizontalAlignment = GridData.FILL;
btnCheck.setLayoutData(gdBtnCheck);
labName = new Label(composite, SWT.WRAP);
labName.setText("TimeOut:");
GridData gdLabName = new GridData();
gdLabName.horizontalAlignment = GridData.BEGINNING;
gdLabName.grabExcessHorizontalSpace = true;
labName.setLayoutData(gdLabName);
txtTimeOut = new Text(composite, SWT.SINGLE SWT.BORDER);
GridData gdTxtTimeOut = new GridData();
gdTxtTimeOut.horizontalAlignment = GridData.END;
gdLabName.grabExcessHorizontalSpace = true;
txtTimeOut.setLayoutData(gdTxtTimeOut);
txtTimeOut.setText("500");
// init status
labName.setEnabled(false);
txtTimeOut.setEnabled(false);
// add listener
defineListener();
// 将composite纳入框架的控制
setControl(composite);
Dialog.applyDialogFont(composite);
}
在这里我们要非凡注重的一点是在定义完我们的界面元素后,需要将自定义的Composite纳入框架的控制,就是这行代码:"setControl(composite);"
在我们处理完输入数据检查后进入下一页面之前,我们需要设置页面完成的状态,以及传递输入数据到领域类Refactoring。我们用以下代码设置好页面完成的状态后,下个页面ErrorWizardPage就会处理显示逻辑:
清单 15
private void notifyStatus(boolean valid, String message) {
//设置错误信息
setErrorMessage(message);
//设置页面完成状态
setPageComplete(valid);
}
传递输入数据通过以下代码来处理:
清单 16
private void setRefactoring(boolean selection, String text) {
AnnotationRefactoring refactoring = (AnnotationRefactoring) getRefactoring();
refactoring.setNeedTimeout(true);
if(selection) {
refactoring.setTimeout(Integer.valueOf(txtTimeOut.getText()).intValue());
}
}
其中getRefactoring()方法是继续自RefactoringWizardPage的方法,由于我们的RefactoringWizard类装载了RefactoringWizardPage和Refactoring类,这个方法是从RefactoringWizard类中获得的,这里面用到了Observer设计模式。至此,我们完成RefactoringWizardPage的扩展。
第二步,扩展RefactoringWizard:首先我们新建一个类AnnotationRefactoringWizard,它需要继续RefactoringWizard类,这个类中我们只需要加载定义好的AnnotationRefactoringWizardPage类和AnnotationRefactoring类,当然复杂的处理已经有RefactoringWizard框架处理好了。下面我们在构造函数中加载Refactoring类:
清单 17
public AnnotationRefactoringWizard(Refactoring refactoring) {
super(refactoring, WIZARD_BASED_USER_INTERFACE);
}
然后我们加载我们的AnnotationRefactoringWizardPage类,只需重载父类RefactoringWizard的addUserInputPages()方法就可以:
清单 18
protected void addUserInputPages() {
page = new AnnotationRefactoringWizardPage("refactor annotation");
addPage(page);
}
第三步,启动RefactoringWizard。扩展好RefactoringWizard之后,就需要在用户点击菜单项或是按钮时弹出这个对话框。RefactoringWizard最好使用RefactoringWizardOpenOperation类来打开(当然也可以用RefactoringWizardDialog)。RefactoringWizardOpenOperation首先进行重构的初始检查,通过后才打开RefactoringWinzard对话框,否则就会打开错误对话框。前面完成创建插件工程时我们提到,弹出RefactoringWizard对话框的代码应该放到响应菜单操作的类的run函数中。具体到本文工程中,就是把下面的代码放到AnnotationManageAction的run函数中。这些代码首先依次构造Refactoring和RefacoringWizard子类AnnotationRefactoring和AnnotationRefactoringWizard,并将AnnotationRefactoring的引用传递给AnnotationRefactoringWizard,然后用RefactoringWizardOpenOperation打开AnnotationRefactoringWizard,弹出向导对话框。
清单 19
public void run(IAction action) {
Shell shell = window.getShell();
AnnotationRefactoring refactor = new AnnotationRefactoring(select);
AnnotationRefactoringWizard wizard = new AnnotationRefactoringWizard(refactor);
RefactoringWizardOpenOperation op = new RefactoringWizardOpenOperation(wizard);
try {
op.run(shell, "Inserting @Override Annotation");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
小结
在Eclipse中有效的利用重构能够大大的减轻软件开发人员的工作负担,提高软件的健壮性。然而,目前重构仍然处在一个工具缺乏的时代。以Eclipse为例,只有JDT提供的重构工具最为完善,而针对其他语言例如C++、Python等的开发环境,都缺乏对应的重构功能。 通过本文提供的方法,我们能够有效的利用Eclipse中的重构框架创建新的重构,从而进一步提高已有开发环境的效率。