GEF实践总结(二)常见基础性问题

一. 怎么通过model创建对应的EditPart?
EditPartFactory :通过model创建对应EditPart的工厂类。
就和jface.TableView一样,能知道通过model得到TableItem。和jface.TreeViewer一样,能通过model知道要生成对应的TreeItem一样的功效。

@Override
public EditPart createEditPart(EditPart context, Object model) {
EditPart part = null;
if (model instanceof DiagramModel) {
part = new DiagramEditPart();
} else if (model instanceof RectModel) {
part = new RectEditPart();
}
part.setModel(model);
return part;
}

 


二. EditPart怎么创建对应的Figure?
在EditPart中有一个方法必须实现,就是createFigure方法。

@Override
protected IFigure createFigure() {
RectFigure figure = new RectFigure();
figure.setModel((BaseModel) this.getModel());
return figure;
}

 


这样就把EditPart和对应的Figure关联起来了,知道如何通过EditPart创建出一个对应的Figure。

三. Model改变,如何对外通知?
Model中有PropertyChangeSupport,外部可以往model中注册PropertyChangeListener,也就是对模型属性改变的监听器,如果model的属性发生了改变,那么就会对所有的listener通知这个属性变化。通知内容包括改变的属性名称、老的值、新的值。

 

protected void firePropertyChange(String propertyName, Object oldValue, Object newValue) {
changeSupport.firePropertyChange(propertyName, oldValue, newValue);
}
public void setX(int x) {
int oldX = this.x;
this.x = x;
this.firePropertyChange(“X”, oldX, x);
}

 

比如:上面的代码,当属性x发生变化的时候就对外通知”X”发生了变化,值从oldX变成了x。

四. EditPart是何时将自己注册到model的呢?
EditPart都是实现了PropertyChangeListener接口,表示:自身就是一个属性变化的监听器,可以对被监听者(比如model)的属性改变进行监听。见下面的代码:

 

/**
* @see org.eclipse.gef.editparts.AbstractGraphicalEditPart#activate()
*/
public void activate() {
if (!isActive()) {
super.activate();
((BaseModel) getModel()).addPropertyChangeListener(this);
}
}

/**
* @see org.eclipse.gef.editparts.AbstractGraphicalEditPart#deactivate()
*/
public void deactivate() {
if (isActive()) {
super.deactivate();
((BaseModel) getModel()).removePropertyChangeListener(this);
}
}

 

在EditPart被激活的时候(activate),就已经将自己作为监听器放入了model中了,在EditPart被销毁的时候(deactivate)的时候去掉的model的监听。

五. EditPart得知属性改变后做了什么事情?

/**
* @see java.beans.PropertyChangeListener#propertyChange(java.beans.PropertyChangeEvent)
*/
@Override
public void propertyChange(PropertyChangeEvent evt) {
String property = evt.getPropertyName();

if (IPropertyConst.CHILDREN.equals(property)) {
refreshChildren();
} else {
refreshVisuals();
}
}

 

上面的这段代码,说明了,当被监听者(model)的属性改变的时候,EditPart得知属性改变后做了什么事情。如果是添加或删除了孩子模型(Children),那么就是刷新所有小孩,重新绘制他们。其他的属性改变的时候,就把自己刷新一下(当然refreshVisuals中还有更多的逻辑)。

六. Model如何告诉Figure具体绘图的位置?

/**
* @see org.eclipse.gef.editparts.AbstractEditPart#refreshVisuals()
*/
protected void refreshVisuals() {
BaseModel model = (BaseModel) getModel();
Rectangle bounds = new Rectangle(model.getX(), model.getY(), model.getW(), model.getY());
((GraphicalEditPart) getParent()).setLayoutConstraint(this, getFigure(), bounds);
}
 

上面的代码在EditPart中,当属性改变时候,就告诉父EditPart,重新调整当前的位置,当然,新的位置信息来自于model。
注意:上面这段代码用于XYLayout的情况下,如果是GridLayout,需要传递GridData对象。其他的Layout也会有其他不同的参数传递和处理。


七. Model如果是父子结构的,EditPart如何知道呢?

/**
* @see org.eclipse.gef.editparts.AbstractEditPart#getModelChildren()
*/
protected List<BaseModel> getModelChildren() {
return ((BaseContainerModel) this.getModel()).getChilds();
}

 

就在于上面的代码,如果model有小孩,就将所有的model的小孩返回,那么EditPart就能知道当前的model是否是父子架构的了。

八. 总结
通过上面的内容已经能够明白:model->EditPart->Figure 的属性变化的传递和如何响应。这其实是GEF中模型变化影响UI界面的一条途径。
还有另一条途径就是通过界面操作,产生Request,通过EditPart的EditPolicy生成Command,Command改变model,从而影响界面。这第二条路以后会提到。

 

你可能感兴趣的:(eclipse,UI)