模型維護的是元素集合,需要自己開發,當模型內容改變時應向外界發出一個"屬性改變"的事件。
通過內容提供器對模型解析,所有內容提供者都實現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();
}