JFace Viewer

組成:1)模型和元素,模型爲元素的集合;2)內容提供者與標籤提供者;3)控件 ;4)查看器(負責協調前面三個部分)。
模型維護的是元素集合,需要自己開發,當模型內容改變時應向外界發出一個"屬性改變"的事件。

通過內容提供器對模型解析,所有內容提供者都實現IContentProvider接口,對於具體的子類,又有子接口,如列表和表格查看器實現 IStructuredContentProvider,樹查看器實現ITreeContentProvider。內容提供器的另一個工作是監聽模型的變化,並操作查看器將變化反映在界面上。

LabelProvider負責每個元素如何在界面顯示,需要實現IBaseLabelProvider,表格查看器要實現 ITableLabelProvider。其中的isLabelProperty(object element ,String property)指定property屬性是否會用於顯示,在更新元素時,查看器會調用這個方法,若元素的變化不影響顯示,則沒有必要更新

要實現對模型的監聽,需要將模型模擬成一個事件源,將內容提供方模擬成一個監聽器。協調關係如圖:

[img]http://dl.iteye.com/upload/attachment/157229/5d703693-2940-3ec2-a983-9cf486e95638.jpg[/img]


下面給個例子,代碼來自《eclipse學習筆記》:

User:爲元素,是自定義的Java類

public class ListModel {

public static final String ADD_ELEMENT = "addElement";
public static final String REMOVE_ELEMENT = "removeElement";
private PropertyChangeSupport delegate;
private Vector content;
public ListModel() {
content = new Vector();
delegate = new PropertyChangeSupport(this);
}

//模擬事件源

public void addPropertyChangeListener(PropertyChangeListener listener) {
delegate.addPropertyChangeListener(listener);
}

public void firePropertyChange(PropertyChangeEvent evt) {
delegate.firePropertyChange(evt);
}

public void removePropertyChangeListener(PropertyChangeListener listener) {
delegate.removePropertyChangeListener(listener);
}

//觸發屬性改變事件

public void add(Object element) {
if (content.add((User) element))
firePropertyChange(new PropertyChangeEvent(this, ADD_ELEMENT, null,
element));
}

//觸發屬性改變事件
public void remove(Object element) {
if (content.remove(element))
firePropertyChange(new PropertyChangeEvent(this, REMOVE_ELEMENT,
null, element));
}

public Object[] elements() {
return content.toArray();
}
}


public class ListContentProvider implements IStructuredContentProvider,PropertyChangeListener{
private ListViewer viewer;
private ListModel model;

//返回元素集合
@Override
public Object[] getElements(Object inputElement) {
return model.elements();
}

@Override
public void dispose() {
}

//設置輸入模型,註冊監聽器開始對新模型監聽

@Override
public void inputChanged(Viewer viewer, Object oldInput, Object newInput) {
this.viewer = (ListViewer) viewer;
if (oldInput instanceof ListModel)
((ListModel) oldInput).removePropertyChangeListener(this);
if (newInput instanceof ListModel) {
this.model = (ListModel) newInput;
((ListModel) newInput).addPropertyChangeListener(this);
}
}

//響應屬性改變事件

@Override
public void propertyChange(PropertyChangeEvent evt) {
if (ListModel.ADD_ELEMENT.equals(evt.getPropertyName()))
viewer.add(evt.getNewValue());
if (ListModel.REMOVE_ELEMENT.equals(evt.getPropertyName()))
viewer.remove(evt.getNewValue());
}
}


public class ListLabelProvider implements ILabelProvider{
@Override
public Image getImage(Object element) {
return null;
}

@Override
public String getText(Object element) {
if(element instanceof User)
return ((User)element).getName();
return element.toString();
}
}

//使用
public static void main(String[] args) {
Display display = Display.getDefault();
Shell shell = new Shell(display);
shell.setSize(400, 400);
shell.setLayout(new FillLayout());

ListViewer viewer = new ListViewer(shell, SWT.BORDER);

viewer.setContentProvider(new ListContentProvider());
viewer.setLabelProvider(new ListLabelProvider());
ListModel input = new ListModel();
viewer.setInput(input);

shell.open();
shell.layout();
//只需要操作模型就能改變視圖,實現界面與模型完全分離
input.add(new User("1","張三"));
input.add(new User("2","李四"));
input.add(new User("3","王五"));

while (!shell.isDisposed())
if (!display.readAndDispatch())
display.sleep();
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章