EMF是eclipse下面的一个相当优秀的自动生成代码的框架.IBM很多大型软件都是基于EMF做出来的,这足已说明它的成熟和优秀.
特别地,如果是要做eclipse插件,EMF带来的帮助更大,除了根据模型生成model代码外,它还能够生成和eclipes UI相关的viewer和相应的viewprovider(控制层).
EMF其实不难,网上有很多关于EMF入门的文章,我这里也就不多废话了.
今天的主题是把前面的例子用EMF+GEF实现.
回顾一下前面的model,他们的UML图如下:
EMF需要我们先定义好model,然后它会根据model自动生成代码.可以有好几种方法,UML图(Rose),Java Codes, XML文件.我自己偏向于用Java代码来定义,因为自己没钱买rose,而XML我很容易出错(就没成功过).
根据上面的UML图,每个类的Java代码如下
Content.java
/***/
/**
*@model
*/

public
interface
Content
...
{

/***//**
*@modelcontainment="true"type="Page"
*/
publicListgetChildren();

publicvoidaddChild(Pagechild);

publicvoidremoveChild(Pagechild);
}
Page.java
/***/
/**
*@model
*/

public
interface
Page
...
{

/***//**
*@modelcontainment="true"type="Control"
*/
publicListgetChildren();


/***//**
*@model
*/
publicRectanglegetConstraint();

publicvoidaddChild(Controlchild);

publicvoidremoveChild(Controlchild);

}
Control.java
/***/
/**
*@model
*/

public
interface
Control
...
{

}
Container.java
/***/
/**
*@model
*/

public
interface
Container
extends
Control
...
{

/***//**
*@modelcontainment="true"type="Column"
*/
publicListgetChildren();

publicvoidaddChild(Columnchild);

publicvoidremoveChild(Columnchild);
}
Column.java
/***/
/**
*@model
*/

public
interface
Column
...
{

/***//**
*@modelcontainment="true"type="Node"
*/
publicListgetChildren();


/***//**
*@modeldefault="column"
*/
publicStringgetName();

publicvoidaddChild(Nodenode,intindex);

publicvoidaddChild(Nodenode);

publicvoidremoveChild(Nodenode);

}
Node.java
/***/
/**
*@model
*/

public
interface
Node
...
{

/***//**
*@modeldefault="node"
*/
publicStringgetName();

}
Label.java
/***/
/**
*@model
*/

public
interface
Label
extends
Control
...
{

/***//**
*@modeldefault="label"
*/
publicStringgetName();

}
MyButton.java
/***/
/**
*@model
*/

public
interface
MyButton
extends
Control
...
{

/***//**
*@modeldefault="button"
*/
publicStringgetName();

}
选择model文件夹,右键新建EMF Model,选择从Java代码得到model,文件名写geftest.genmodel,完成.然后就会自动生成geftest.ecore和geftest.genmodel两个文件,ecore文件保存的是emf core需要的信息, 而genmodel是一些比如生成的包名等和core无关的其它信息.
右键点ecore文件,然后选择:Initialize ecore_diagram diagram file,生成diagram_file文件,打开后得到下图:
可以看出来我们的model定义的是正确的.
然后修改一些地方就可以了.
EMF生成的代码有一个完整的事件机制,每个对象都可以有许多adapter,对象属性被改后会通知这些adapter.
所以我把以前的AbstractModel删掉了.然后把每个editpart做为它的model的adapter就可以了.这需要我们的所有editpart都实现Adapter接口.
因为我们的editpart都继承AbstractEditPartWithListener,所以我把里面原来的加入到listener list的代码都删掉了,加了如下代码:
EditPartWithListener.java
protected
void
hookIntoModel(EObjectmodel)
...
{

if(model!=null)...{
model.eAdapters().add(this);
}
}


protected
void
unhookFromModel(EObjectmodel)
...
{

if(model!=null)...{
model.eAdapters().remove(this);
}
}
@Override

public
void
activate()
...
{
hookIntoModel((EObject)getModel());
super.activate();

}

@Override

public
void
deactivate()
...
{
unhookFromModel((EObject)getModel());
super.deactivate();
}
然后每个EditPart实现notifyChanged方法:
举个例子,比如PageEditPart:
PageEditPart.java
public
void
notifyChanged(Notificationnotification)
...
{
inttype=notification.getEventType();
intfeatureId=notification.getFeatureID(ModelPackage.class);

switch(type)...{
caseNotification.ADD:
caseNotification.REMOVE:

switch(featureId)...{
caseModelPackage.PAGE__CHILDREN:
refreshChildren();
break;
}
caseNotification.SET:

switch(featureId)...{
caseModelPackage.PAGE__CONSTRAINT:
refreshVisuals();
break;
}
}
}
最后修改Editor的ToolEntry.因为我们不能用SimpleFactory来简单的用反射生成每个model对象.
而应该用ModelFactory.eINSTANCE.createXXX方法.
Editor.java
CreationToolEntrycreationPage
=
new
CreationToolEntry(
"
drawpage
"
,

"
createpage
"
,
new
CreationFactory()
...
{


publicObjectgetNewObject()...{
returnModelFactory.eINSTANCE.createPage();
}


publicObjectgetObjectType()...{
returnnull;
}
}
,descriptor,descriptor);
drawer.add(creationPage);
基本上这样就OK了,其实不麻烦.
代码下载
参考资源:
EMF
Using GEF with EMF
GEF学习教程-Unplugged版
特别地,如果是要做eclipse插件,EMF带来的帮助更大,除了根据模型生成model代码外,它还能够生成和eclipes UI相关的viewer和相应的viewprovider(控制层).
EMF其实不难,网上有很多关于EMF入门的文章,我这里也就不多废话了.
今天的主题是把前面的例子用EMF+GEF实现.
回顾一下前面的model,他们的UML图如下:
EMF需要我们先定义好model,然后它会根据model自动生成代码.可以有好几种方法,UML图(Rose),Java Codes, XML文件.我自己偏向于用Java代码来定义,因为自己没钱买rose,而XML我很容易出错(就没成功过).
根据上面的UML图,每个类的Java代码如下
Content.java
/***/
/**
*@model
*/

public
interface
Content
...
{
/***//**
*@modelcontainment="true"type="Page"
*/
publicListgetChildren();
publicvoidaddChild(Pagechild);
publicvoidremoveChild(Pagechild);
}
Page.java
/***/
/**
*@model
*/

public
interface
Page
...
{
/***//**
*@modelcontainment="true"type="Control"
*/
publicListgetChildren();

/***//**
*@model
*/
publicRectanglegetConstraint();
publicvoidaddChild(Controlchild);
publicvoidremoveChild(Controlchild);
}
Control.java
/***/
/**
*@model
*/

public
interface
Control
...
{
}
Container.java
/***/
/**
*@model
*/

public
interface
Container
extends
Control
...
{
/***//**
*@modelcontainment="true"type="Column"
*/
publicListgetChildren();
publicvoidaddChild(Columnchild);
publicvoidremoveChild(Columnchild);
}
Column.java
/***/
/**
*@model
*/

public
interface
Column
...
{
/***//**
*@modelcontainment="true"type="Node"
*/
publicListgetChildren();

/***//**
*@modeldefault="column"
*/
publicStringgetName();
publicvoidaddChild(Nodenode,intindex);
publicvoidaddChild(Nodenode);
publicvoidremoveChild(Nodenode);
}
Node.java
/***/
/**
*@model
*/

public
interface
Node
...
{
/***//**
*@modeldefault="node"
*/
publicStringgetName();
}
Label.java
/***/
/**
*@model
*/

public
interface
Label
extends
Control
...
{
/***//**
*@modeldefault="label"
*/
publicStringgetName();
}
MyButton.java
/***/
/**
*@model
*/

public
interface
MyButton
extends
Control
...
{
/***//**
*@modeldefault="button"
*/
publicStringgetName();
}
选择model文件夹,右键新建EMF Model,选择从Java代码得到model,文件名写geftest.genmodel,完成.然后就会自动生成geftest.ecore和geftest.genmodel两个文件,ecore文件保存的是emf core需要的信息, 而genmodel是一些比如生成的包名等和core无关的其它信息.
右键点ecore文件,然后选择:Initialize ecore_diagram diagram file,生成diagram_file文件,打开后得到下图:
可以看出来我们的model定义的是正确的.
然后修改一些地方就可以了.
EMF生成的代码有一个完整的事件机制,每个对象都可以有许多adapter,对象属性被改后会通知这些adapter.
所以我把以前的AbstractModel删掉了.然后把每个editpart做为它的model的adapter就可以了.这需要我们的所有editpart都实现Adapter接口.
因为我们的editpart都继承AbstractEditPartWithListener,所以我把里面原来的加入到listener list的代码都删掉了,加了如下代码:
EditPartWithListener.java
protected
void
hookIntoModel(EObjectmodel)
...
{
if(model!=null)...{
model.eAdapters().add(this);
}
}


protected
void
unhookFromModel(EObjectmodel)
...
{
if(model!=null)...{
model.eAdapters().remove(this);
}
}
@Override
public
void
activate()
...
{
hookIntoModel((EObject)getModel());
super.activate();
}

@Override
public
void
deactivate()
...
{
unhookFromModel((EObject)getModel());
super.deactivate();
}
举个例子,比如PageEditPart:
PageEditPart.java
public
void
notifyChanged(Notificationnotification)
...
{
inttype=notification.getEventType();
intfeatureId=notification.getFeatureID(ModelPackage.class);
switch(type)...{
caseNotification.ADD:
caseNotification.REMOVE:
switch(featureId)...{
caseModelPackage.PAGE__CHILDREN:
refreshChildren();
break;
}
caseNotification.SET:
switch(featureId)...{
caseModelPackage.PAGE__CONSTRAINT:
refreshVisuals();
break;
}
}
}
最后修改Editor的ToolEntry.因为我们不能用SimpleFactory来简单的用反射生成每个model对象.
而应该用ModelFactory.eINSTANCE.createXXX方法.
Editor.java
CreationToolEntrycreationPage
=
new
CreationToolEntry(
"
drawpage
"
,
"
createpage
"
,
new
CreationFactory()
...
{

publicObjectgetNewObject()...{
returnModelFactory.eINSTANCE.createPage();
}

publicObjectgetObjectType()...{
returnnull;
}
}
,descriptor,descriptor);
drawer.add(creationPage);
基本上这样就OK了,其实不麻烦.
代码下载
参考资源:
EMF
Using GEF with EMF
GEF学习教程-Unplugged版
本文介绍如何使用EMF和GEF实现一个具体案例。通过定义模型并利用EMF自动生成代码,结合GEF进行图形界面编辑。文章详细展示了如何通过Java代码定义模型,并将模型与编辑器组件相结合。
266

被折叠的 条评论
为什么被折叠?



