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
*/

publicinterfaceContent...{

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

publicvoidaddChild(Pagechild);

publicvoidremoveChild(Pagechild);
}
Page.java

/***//**
*@model
*/

publicinterfacePage...{

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


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

publicvoidaddChild(Controlchild);

publicvoidremoveChild(Controlchild);

}
Control.java

/***//**
*@model
*/

publicinterfaceControl...{

}
Container.java

/***//**
*@model
*/

publicinterfaceContainerextendsControl...{

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

publicvoidaddChild(Columnchild);

publicvoidremoveChild(Columnchild);
}
Column.java

/***//**
*@model
*/

publicinterfaceColumn...{

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


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

publicvoidaddChild(Nodenode,intindex);

publicvoidaddChild(Nodenode);

publicvoidremoveChild(Nodenode);

}
Node.java

/***//**
*@model
*/

publicinterfaceNode...{

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

}
Label.java

/***//**
*@model
*/

publicinterfaceLabelextendsControl...{

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

}
MyButton.java

/***//**
*@model
*/

publicinterfaceMyButtonextendsControl...{

/***//**
*@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

protectedvoidhookIntoModel(EObjectmodel)...{

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


protectedvoidunhookFromModel(EObjectmodel)...{

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

publicvoidactivate()...{
hookIntoModel((EObject)getModel());
super.activate();

}

@Override

publicvoiddeactivate()...{
unhookFromModel((EObject)getModel());
super.deactivate();
}
然后每个EditPart实现notifyChanged方法:
举个例子,比如PageEditPart:
PageEditPart.java

publicvoidnotifyChanged(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=newCreationToolEntry("drawpage",

"createpage",newCreationFactory()...{


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


publicObjectgetObjectType()...{
returnnull;
}
},descriptor,descriptor);
drawer.add(creationPage);
基本上这样就OK了,其实不麻烦.
代码下载
参考资源:
EMF
Using GEF with EMF
GEF学习教程-Unplugged版
相关文章: