GEF框架学习(2)
写在前面
本系列文章主要参考了GEF入门Demo (github.com),上一次讲完了HelloWorld,可以看到GEF框架把JAVA编码繁琐的特性完全体现了出来。这次文章主要讲了如何设置一个布局,通过这个布局来统一管理,并且可以对图形进行操作。
创建图形集模型
我们可以类比上次说到GEF框架开发中的一个常规流程。首先是Model,然后是EditPart,同时我们需要修改EditPartFactory来讲EditPart和Model联系起来,最后修改Editor,如下图:
创建Model
首先先来创建Model,如下代码:
public class ContentsModel{
}
我们先什么都不写,创建完所有类后再来写。
创建对应的控制器ContentsEditPart
直接看代码:
public class ContentsEditPart extends AbstractGraphicalEditPart{
@Override
protected IFigure createFigure(){
Layer figure = new Layer();
return figure;
}
@Override
protected void createEditPolicies() {
}
}
修改Factory连接Model和EditPart
在getPartForElement方法中多加个If就好了:
if(modelElement instanceof HelloModel) {
return new HelloEditPart();
}else if(modelElement instanceof ContentsModel) {
return new ContentsEditPart();
}
修改Editor
这次修改Editor中的初始化方法变为初始化ContentsModel:
ContentsModel contentsModel = new ContentsModel();
viewer.setContents(contentsModel);
运行一下,会发现啥都没有。毕竟我们甚至没有在ContentsModel类中写代码。
为图形集添加子集
为了添加子集,我们肯定需要一个父容器。因此,我们要修改ContentsModel和他对应的ContentsEditPart。
修改ContentsModel
添加一个列表存储孩子,然后赋予他添加子孩子和获得子孩子的方法:
public class ContentsModel{
private List<Object> children = new ArrayList<>();// 子模型列表
public void addChild(Object child) {
children.add(child);
}
public List<Object> getChildren() {
return children;
}
}
修改ContentsEditPart
重写该类中的getModelChidren方法,返回其所有的孩子:
@Override
protected List getModelChildren() {
return ((ContentsModel) getModel()).getChildren();
}
美化一下HelloWorld图形
为了让效果更好一点,这里我们先对HelloWorld模型进行一个美化操作,更好看一些。修改HelloModelEditPart的createFigure方法,添加一些配置样式的代码:
@Override
protected IFigure createFigure() {
HelloModel model = (HelloModel) getModel();
Label label = new Label();
label.setText(model.getText());
label.setBorder(new CompoundBorder(new LineBorder(),new MarginBorder(3)));
// 设置背景色
label.setBackgroundColor(ColorConstants.orange);
// 设置背景色不透明
label.setOpaque(true);
return label;
}
配置布局管理器
我们刚才设置了图层类Layer,但没有正确设置他们的布局管理器。这里我们直接使用XYLayout,这是一种可以允许图形自由移动的布局:
@Override
protected IFigure createFigure() {
Layer figure = new Layer();
figure.setLayoutManager(new XYLayout());
return figure;
}
对图形类设置约束
在布局中每个子控件肯定有自己的位置。由于是每个模型有自己对应的位置,因此我们设置到子控件中去:
public class HelloModel extends AbstractModel{
private String text = "Hello World";
private Rectangle constraint;
public Rectangle getConstraint() {
return constraint;
}
public void setConstraint(Rectangle constraint) {
this.constraint = constraint;
}
public String getText() {
return text;
}
public void setText(String text) {
this.text = text;
}
}
然后我们要设置这些约束,我们要在其对应的EditPart中重载refreshVisuals()方法,这个方法是GEF提供的:
@Override
protected void refreshVisuals() {
// 设置定位
Rectangle constraint = ((HelloModel) getModel()).getConstraint();
((GraphicalEditPart)getParent()).setLayoutConstraint(this, getFigure(), constraint);
}
在Editor中显示出父子关系
设置好之后,修改Editor让他显示出刚才咱们定义的模式吧:
@Override
protected void initializeGraphicalViewer() {
// 设置最上层模型
ContentsModel contentsModel = new ContentsModel();
// 分别设置三个孩子的定位
HelloModel child1 = new HelloModel();
child1.setConstraint(new Rectangle(0,0,-1,-1));
HelloModel child2 = new HelloModel();
child2.setConstraint(new Rectangle(60,60,-1,-1));
HelloModel child3 = new HelloModel();
child3.setConstraint(new Rectangle(10,80,80,50));
contentsModel.addChild(child1);
contentsModel.addChild(child2);
contentsModel.addChild(child3);
viewer.setContents(contentsModel);
}
需要注意的是,这里的-1,-1即是设定他的宽度和高度为适应字的长度和宽度。效果如图:
对图形进行操作
为了对图形进行操作,GEF采用了命令模式。这里先放一张我理解的图:
这里重点介绍Policy和Command。Policy是Command的汇总,用户的每个操作都会发起Command,然后我们在Policy中监听到Command修改即可对Model进行操作。而Model的修改则会引起EditPart的变更,并且通知Editor修改。总体就是命令模式的应用。接下来慢慢用代码实现:
创建EditingPolicy
这里我们创建一个CustomXYLayoutEditPolicy,这个Policy中包含了很多Command,后面再细说:
public class CustomXYLayoutEditPolicy extends XYLayoutEditPolicy{
@Override
protected Command getCreateCommand(CreateRequest request) {
return null;
}
}
安装EditingPolicy
我们在对应的ContentsEditPart中注册咱们的Policy:
@Override
protected void createEditPolicies() {
// 安装editing policy
installEditPolicy(EditPolicy.LAYOUT_ROLE, new CustomXYLayoutEditPolicy());
}
创建Command
这里我们创建一个ChangeConstraintCommand即可:
public class ChangeConstraintCommand extends Command{
private HelloModel helloModel;
private Rectangle constraint;
@Override
public void execute() {
// 执行命令
helloModel.setConstraint(constraint);
}
public void setConstraint(Rectangle constraint) {
this.constraint = constraint;
}
public void setModel(Object helloModel) {
this.helloModel = (HelloModel) helloModel;
}
}
这里execute方法为核心方法,通过这个方法来执行命令。
修改EditingPolicy
我们刚才创建了命令,但没有跟Policy连接起来,我们把这俩连接起来:
@Override
protected Command createChangeConstraintCommand(ChangeBoundsRequest request, EditPart child, Object constraint) {
// 创建Command
ChangeConstraintCommand command = new ChangeConstraintCommand();
// 设置model和constraint
command.setModel(child.getModel());
command.setConstraint((Rectangle) constraint);
return command;
}
对模型添加监听器
这里我们已经修改好了,但我们的EditingPart还无法监听Model的改变。因此我们要挨个进行更改。首先对模型添加监听器。由于我们之后所有的模型都需要监听器, 这里我们抽象一个基类出来:
public class AbstractModel{
private PropertyChangeSupport listeners = new PropertyChangeSupport(this);
public void addPropertyChangeListener(PropertyChangeListener listener) {
listeners.addPropertyChangeListener(listener);
}
public void firePropertyChange(String propName,Object oldValue,Object newValue) {
listeners.firePropertyChange(propName,oldValue,newValue);
}
public void removePropertyChangeListener(PropertyChangeListener listener) {
listeners.removePropertyChangeListener(listener);
}
}
接下来,我们修改HelloModel类,让他继承这个基类:
public class HelloModel extends AbstractModel{
// 标识更改类别
public static final String P_CONSTRAINT = "_constraint";
private String text = "Hello World";
private Rectangle constraint;
public Rectangle getConstraint() {
return constraint;
}
public void setConstraint(Rectangle constraint) {
this.constraint = constraint;
firePropertyChange(P_CONSTRAINT, null, constraint);
}
public String getText() {
return text;
}
public void setText(String text) {
this.text = text;
}
}
在EditPart注册监听器
由于我们所有的EditPart都需要监听器,因此我们再抽象一个基类出来:
public abstract class EditPartWithListener extends AbstractGraphicalEditPart implements PropertyChangeListener{
@Override
public void activate() {
super.activate();
// 注册监听器
((AbstractModel)getModel()).addPropertyChangeListener(this);
}
@Override
public void deactivate() {
super.deactivate();
// 删除监听器
((AbstractModel)getModel()).removePropertyChangeListener(this);
}
}
然后我们修改HelloEditPart继承自这个基类,并且可以在模型改变时刷新视图:
@Override
public void propertyChange(PropertyChangeEvent evt) {
if(evt.getPropertyName().equals(HelloModel.P_CONSTRAINT)) {
// 刷新界面
refreshVisuals();
}
}
再运行一下,我们可以随便拖拽图形了:
UNDO和REDO操作
GEF中所有的Command都放在EditDomain中的CommandStack中。undo时弹出,redo时放进去。因此我们重载undo方法即可实现undo:
private Rectangle oldConstraint;// 记录之前的约束
public void setModel(Object helloModel) {
this.helloModel = (HelloModel) helloModel;
oldConstraint = this.helloModel.getConstraint();
}
@Override
public void undo() {
helloModel.setConstraint(oldConstraint);
}
这里还添加了一个变量来存储旧的变量。那我们为什么不重载redo方法呢?因为redo方法底层其实调用了execute方法,因此没有重载的必要。
添加UNDO和REDO按钮
有了UNDO和REDO操作,但没有按钮也是无法使用的。因此我们要添加按钮到工具栏上。这里我们创建一个DiagramActionBarContributor类:
public class DiagramActionBarContributor extends ActionBarContributor{
@Override
protected void buildActions() {
addRetargetAction(new UndoRetargetAction());
addRetargetAction(new RedoRetargetAction());
}
@Override
protected void declareGlobalActionKeys() {
}
@Override
public void contributeToToolBar(IToolBarManager toolBarManager) {
toolBarManager.add(getAction(ActionFactory.UNDO.getId()));
toolBarManager.add(getAction(ActionFactory.REDO.getId()));
}
}
这里用的都是GEF框架提供的。因为这些操作实在太常见了,我们都不需要自己写。最后直接修改plugin.xml绑定咱们自己的类:
接下来就可以实现了:
总结
这次主要讲了GEF中的很多东西吧。可以看出来用java开发客户端还是比较反人类的。也不知道目前主流的方案是什么。下次随缘更新吧~