GEF框架学习(2)

写在前面

本系列文章主要参考了GEF入门Demo (github.com),上一次讲完了HelloWorld,可以看到GEF框架把JAVA编码繁琐的特性完全体现了出来。这次文章主要讲了如何设置一个布局,通过这个布局来统一管理,并且可以对图形进行操作。

创建图形集模型

我们可以类比上次说到GEF框架开发中的一个常规流程。首先是Model,然后是EditPart,同时我们需要修改EditPartFactory来讲EditPart和Model联系起来,最后修改Editor,如下图:

image-20220730194310343

创建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即是设定他的宽度和高度为适应字的长度和宽度。效果如图:

image-20220730200913985

对图形进行操作

为了对图形进行操作,GEF采用了命令模式。这里先放一张我理解的图:

image-20220730201455302

这里重点介绍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();
        }
		
	}

再运行一下,我们可以随便拖拽图形了:

image-20220730203124315

image-20220730203129473

image-20220730203137664

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绑定咱们自己的类:

image-20220730203604599

接下来就可以实现了:

image-20220730203702282

总结

这次主要讲了GEF中的很多东西吧。可以看出来用java开发客户端还是比较反人类的。也不知道目前主流的方案是什么。下次随缘更新吧~

posted @ 2022-07-30 20:39  武神酱丶  阅读(320)  评论(5编辑  收藏  举报