实现项目特性插件
本节将从一个具体的实例出发,在Eclipse 项目中添加用户自己的项目特性,扩展Eclipse的选项,让用户的插件具有一个产品的雏形。
项目扩展点
用户已经了解了如何在Eclipse中添加视图、编辑器、透视图菜单和工具栏按钮。本节将要介绍如何加入自己的项目相关特性,其中包括向导、项目特性、构建器和属性页。项目特性扩展点如图1所示。
图1 项目特性扩展点
项目特性扩展点及实现类
“nature”为用户添加项目的特有特性,Eclipse中一个项目可以同时具有多个nature特性。例如用户要在Java 项目的基础上扩展自己的项目,加入了自己项目的特有特性。这样JDT就能认识用户新建的项目,用户也可以判断此Java项目是否是自己的扩展。一般来说,用户期望用到JDT的强大功能,就要使自己的项目从Java项目扩展过来。
在插件中添加nature扩展点,步骤如下。
1. 在插件清单文件的Extensions页中添加“org.eclipse.core.resources.natures”扩展点,设置ID属性为“helloworldnature”。
2. 在“org.eclipse.core.resources.natures”节点下添加runtime子节点。
3. 在runtime节点下添加run节点,设置实现类class为“com.free.project. HelloWorldNature”。
扩展点在plugin.xml文件中的描述如下。
point="org.eclipse.core.resources.natures">
nature扩展点的实现类要实现“IProjectNature”接口,并要实现“IProjectNature”的如下几个方法。
setProject(IProject project):在nature类中保存当前项目。
getProject():从nature中得到当前项目。
configure():当nature被添加到项目中时,通过configure方法配置当前项目。
deconfigure():当nature从项目删除时,通过deconfigure方法删除相应配置。
nature对应的实现类为“HelloWorldNature”,在此实现类中主要功能是通过configure方法配置当前项目的构建器,代码如例程1所示。
例程1 HelloWorldNature.java
public class HelloWorldNature implements IProjectNature {
private IProject _project;
//配置Nature
public void configure() throws CoreException {
System.out.println("configure");
IProjectDescription projectDesc = _project.getDescription();
ICommand[] buildSpec = projectDesc.getBuildSpec();
boolean hasBuilder = false;
//遍历项目的构建器
for (int i = 0; i < buildSpec.length; ++i) {
if (buildSpec[i]
.getBuilderName()
.equals("com.free.proj.helloproject.helloworldbuilder")) {
hasBuilder = true;
System.out.println("true");
break;
}
}
if (hasBuilder == false) {
System.out.println("false");
ICommand newCommand = projectDesc.newCommand();
newCommand.setBuilderName("com.free.proj.helloproject.helloworldbuilder");
ICommand[] buildSpecs = new ICommand[buildSpec.length + 1];
System.arraycopy(buildSpec, 0, buildSpecs, 1, buildSpec.length);
buildSpecs[0] = newCommand;
projectDesc.setBuildSpec(buildSpecs);
_project.setDescription(projectDesc, null);
}
}
public void deconfigure() throws CoreException {
}
//获得Nature所关联的项目
public IProject getProject() {
System.out.println("getProject");
return _project;
}
//设置Nature所关联的项目
public void setProject(IProject project) {
System.out.println("setProject");
_project = project;
}
}
提示:此外构建器的ID号为“com.free.proj.helloproject.helloworldbuilder”,ID号为插件ID号加构建器ID号,在Eclipse中和产品及项目相关的ID都采用这种方式。
构建器扩展点及实现类
Builder为项目的构建器,通过构建器能实现项目的增量构建和完全构建。在插件中添加Builder扩展点,步骤如下。
1. 在插件清单文件的Extensions页中添加“org.eclipse.core.resources.builders”扩展点,设置ID属性为“helloworldbuilder”,name属性为“hello world, bulder”。
2. 在“org.eclipse.core.resources.builders”节点下添加builder子节点。
3. 在builder节点下添加run节点,设置实现类class为“com.free.project. HelloWorldBuilder”。
扩展点在plugin.xml文件中的描述如下。
point="org.eclipse.core.resources.builders">
builder 扩展点的实现类继承于“IncrementalProjectBuilder”类,并实现了两个资源监听器。当项目中资源发生变化的时候调用构建,本例中只打印出相应的数据,并没有实现构建的动作,当用户需要构建时可以替换相应代码,例如编译文件等。代码如例程2所示。
例程2 HelloWorldBuilder.java
public class HelloWorldBuilder extends IncrementalProjectBuilder {
//实现资源监听器
private class HelloWorldVisitor implements IResourceVisitor {
public boolean visit(IResource resource) throws CoreException {
System.out.println("**** HelloWorldVisitor.visit() ****");
switch (resource.getType()) {
case IResource.PROJECT:
System.out.println("Project added: " + resource.getName());
break;
case IResource.FOLDER:
System.out.println("Folder added: " + resource.getName());
break;
case IResource.FILE:
System.out.println("File added: " + resource.getName());
break;
}
return true;
}
}
//实现资源监听器
private class HelloWorldDeltaVisitor implements IResourceDeltaVisitor {
public boolean visit(IResourceDelta delta) throws CoreException {
System.out.println("**** HelloWorldDeltaVisitor.visit() ****");
String type = null;
switch (delta.getResource().getType()) {
case IResource.ROOT:
type = "ROOT";
break;
case IResource.PROJECT:
type = "Project";
break;
case IResource.FOLDER:
type = "Folder";
break;
case IResource.FILE:
type = "File";
break;
}
switch (delta.getKind()) {
case IResourceDelta.ADDED:
System.out.println(type + " added: "+ delta.getResource().getName());
break;
case IResourceDelta.CHANGED:
System.out.println(type + " changed: "
+ delta.getResource().getName());
break;
case IResourceDelta.REMOVED:
System.out.println(type + " removed: "
+ delta.getResource().getName());
break;
}
return true;
}
}
public HelloWorldBuilder() {
System.out.println("HelloWorldBuilder.constructor()");
}
//开始构建
protected IProject[] build(int kind, Map args, IProgressMonitor monitor)
throws CoreException {
System.out.println("HelloWorldBuilder.build()");
//当构建类型不同时,添加不同的资源监听器
if (kind == IncrementalProjectBuilder.FULL_BUILD) {
System.out.println("FULL_BUILD");
getProject().accept(new HelloWorldVisitor());
} else {
IResourceDelta delta = getDelta(getProject());
if (delta == null) {
System.out.println("AUTO_BUILD");
getProject().accept(new HelloWorldVisitor());
} else {
System.out.println("INCREMENTAL_BUILD");
delta.accept(new HelloWorldDeltaVisitor());
}
}
return null;
}
}
在用户建立的项目中,如果资源发生改变,builder会调用“HelloWorldDeltaVisitor”类进行增量构建处理,当清除项目时会调用 “HelloWorldVisitor”类进行完全构建处理。插件运行后,当用户选择自定义项目时,用户可以打开项目属性窗口看到“hello world, builder”构建器,如图2所示。
图2 项目属性窗口
工程向导扩展点及实现类
用户如果要在项目中添加自定义的资源,采用Eclipse中的向导是最好的方式。项目也是资源的一种,一般来说如果对新建的资源有输入条件的要求,都采用向导的方式。例如新建Java项目,要输入项目名称和构建路径等。在插件中添加newWizard扩展点,步骤如下。
1. 在插件清单文件的Extensions页中添加“org.eclipse.ui.newWizards”扩展点。
2. 在“org.eclipse.ui.newWizards”节点下添加category子节点,设置ID属性为“com.free.project.wizard.category”,name属性为“HelloWorld Project Category”。
3. 在“org.eclipse.ui.newWizards”节点下添加wizard节点,设置实现类class为“com.free.project.HelloWorldWizard”,ID属性为“com.free.project.wizard”。
扩展点在plugin.xml文件中的描述如下:
id="com.free.project.wizard.category"/>
category="com.free.project.wizard.category"
name="HelloWorld Project Wizard"
id="com.free.project.wizard"/>
newWizards 扩展点的实现类继承于“org.eclipse.jface.wizard.Wizard”类,并实现了 “org.eclipse.ui.INewWizard”接口。在实现类中,通过addPages方法添加向导页到当前向导,通过 performFinish方法完成向导的结束。实现类“HelloWorldWizard”的代码如例程3所示。
例程3 HelloWorldWizard.java
public class HelloWorldWizard extends Wizard implements INewWizard {
private WizardNewProjectCreationPage _page;
public HelloWorldWizard() {
super();
}
//添加向导页
public void addPages() {
_page = new WizardNewProjectCreationPage("page1");
_page.setDescription("Page 1 of Helloworld New Wizard");
_page.setTitle("HelloWorld Wizard");
addPage(_page);
}
//完成向导
public boolean performFinish() {
boolean result = false;
System.out.println("project page.name = " + _page.getProjectName());
System.out.println("project page.path = " + _page.getLocationPath());
IProject project = _page.getProjectHandle();
try {
project.create(null);
result = true;
} catch (CoreException e) {
e.printStackTrace();
}
IProjectDescription projectDesc = null;
try {
project.open(null);
projectDesc = project.getDescription();
} catch (CoreException e1) {
e1.printStackTrace();
}
String[] natureIds = projectDesc.getNatureIds();
String[] newNatureIds = new String[natureIds.length + 1];
System.arraycopy(natureIds, 0, newNatureIds, 0, natureIds.length);
newNatureIds[natureIds.length] = "com.free.proj.helloproject.helloworldnature";
projectDesc.setNatureIds(newNatureIds);
try {
project.setDescription(projectDesc, null);
} catch (CoreException e2) {
e2.printStackTrace();
}
return result;
}
public void init(IWorkbench workbench, IStructuredSelection selection) {
}
}
本例程中没有实现用户定义的向导页,而是用Eclipse下的“WizardNewProjectCreationPage”页来作为向导页。用户也可以定义向导页,如果有兴趣可以参考多页签编辑器中的例子源代码,多页签编辑器插件已经实现了向导。本例中可以通过新建菜单新建用户定义的项目,向导对话框如图 3所示。
图3 向导对话框
首选项页扩展点及实现类
在用户定义的项目中,一般都会有相应的参数配制。例如说Tomcat要配置相应的Web服务器,JDT可以配置代码的风格、JDK和JRE等。在插件中可以添加preferencePages扩展点,实现项目参数的配置,步骤如下。
1. 在插件清单文件的Extensions页中添加“org.eclipse.ui.preferencePages”扩展点。
2. 在“org.eclipse.ui.preferencePages”节点下添加page子节点,设置节点的ID属性为 “com.free.project.page”,name属性为“HelloWorld”,class属性为 “com.free.project.Workbench- PreferencePage”。
扩展点在plugin.xml文件中的描述如下。
name="HelloWorld"
id="com.free.project.page"/>
preferencePages 扩展点的实现类继承于“org.eclipse.jface.preference.PreferencePage”类,并实现了 “org.eclipse.ui.IWorkbenchPreferencePage”接口。在实现类中,通过createContents方法构建自己的属性页,通过performOk方法完成向导的结束。实现类“WorkbenchPreferencePage”的代码如例程4所示。
例程4 WorkbenchPreferencePage.java
public class WorkbenchPreferencePage extends PreferencePage implements
IWorkbenchPreferencePage {
private Text _greeting;
public WorkbenchPreferencePage() {
super();
}
public WorkbenchPreferencePage(String title) {
super(title);
}
public WorkbenchPreferencePage(String title, ImageDescriptor image) {
super(title, image);
}
protected Control createContents(Composite parent) {
Label label = new Label(parent, SWT.CENTER);
label.setText("Greeting");
_greeting = new Text(parent, SWT.SINGLE | SWT.BORDER);
return parent;
}
protected IPreferenceStore doGetPreferenceStore() {
System.out.println("WorkbenchPreferencePage.doGetPreferenceStore()");
return HelloWorldPlugin.getDefault().getPreferenceStore();
}
public void init(IWorkbench workbench) {
}
protected void performDefaults() {
System.out.println("WorkbenchPreferencePage.performDefaults()");
IPreferenceStore prefStore = getPreferenceStore();
prefStore.setValue(HelloWorldPlugin.GREETING, "world");
HelloWorldPlugin.getDefault().savePluginPreferences();
_greeting.setText(prefStore.getString(HelloWorldPlugin.GREETING));
}
public boolean performOk() {
boolean result = false;
System.out.println("WorkbenchPreferencePage.performOk()");
IPreferenceStore prefStore = getPreferenceStore();
prefStore.setValue(HelloWorldPlugin.GREETING, _greeting.getText());
HelloWorldPlugin.getDefault().savePluginPreferences();
result = true;
return result;
}
}
本例程只添加了一个Label控件和一个Text控件。若用户有兴趣可以添加自己完成的属性页,选择“window”→“preferences”菜单,打开首选项对话框,程序运行效果如图4所示。
图4 首选项窗口
提示:在Eclipse中,首选项页中定义的信息是一个全局的信息,这些信息为Workspace中所有的工程所共享,而属性页(property page)中的信息为某一个资源所独有,例如工程的属性。