视图之间实现事件监听

两个视图中的组件之间的互动,在开发插件的时候是经常碰到的问题.点击视图1列表的某项时,视图2的文本框显示相应的字符.

第一种主动式:

主动式就是在视图1的代码块中获取对视图2的对象的引用.然后将视图1中的对象主动的传给视图2.

修改View1.java和View2.java

Eclipse通过plugin.xml来加载插件和插件中的扩展点(如视图扩展点),所以可以在View1.java中由id标识来取得视图2对象.

View1.java

 public class View1 extends ViewPart {
private List list; // 将列表写成类的实例变量,以扩大它的可访问范围
//注意这个List并不是java.util包下的.而是org.eclipse.swt.widgets.List;包下的.
public void createPartControl(Composite parent) {
IWorkbenchHelpSystem help = PlatformUI.getWorkbench().getHelpSystem();
help.setHelp(parent, "cn.com.kxh.myplugin.buttonHelpId");
Composite topComp = new Composite(parent, SWT.NONE);
topComp.setLayout(new FillLayout());
list = new List(topComp, SWT.BORDER);
list.add("中国");
list.add("美国");
list.add("法国");
// 列表选择事件监听
list.addSelectionListener(new SelectionAdapter() {
public void widgetSelected(SelectionEvent e) {
// 由IWorkbenchPage获得view2对象
IWorkbenchPage wbp = getViewSite().getPage();
//在插件中IWorkbenchPage对象比较重要,这里再给出一种获得此对象的通用的方法.
// Activator.getDefault().getWorkbench().getActiveWorkbenchWindow().getActivePage();
IViewPart view2 = wbp.findView("cn.com.kxh.myplugin.View2");
//这个地方的参数是"视图2"在plugin.xml中的id标识.由此可见plugin.xml文件在插件中的地位是极其重要的.
// 将当前选择的列表项显示在文本框中
Text text = ((View2) view2).getText();
text.setText(list.getSelection()[0]);
}
});
}
@Override
public void setFocus() {}
}

View2.java (将View2.java的文本框对象改成类的实例变量,并编写相应的Setter和Getter方法)

 public class View2 extends ViewPart {
private Text text;
public void createPartControl(Composite parent) {
Composite topComp = new Composite(parent,SWT.NONE);
topComp.setLayout(new FillLayout());
text = new Text(topComp,SWT.BORDER);
text.setText("我是Text框");
}
public void setFocus() {} public Text getText() {
return text;
}
public void setText(Text text) {
this.text = text;
}
}

SamplePerspective.java(和我的上一篇博客上没有做任何修改)

 public class SamplePerspective implements IPerspectiveFactory {
// 参数IPageLayout是用于透视图的布局管理器
public void createInitialLayout(IPageLayout layout) {
// 得到本透视图的编辑空间标识
String editorArea = layout.getEditorArea();
// 在透视图左部创建一个空间,并将“视图1”放入其中。
// "left"是此空间的标识;IPageLayout.LEFT指出此空间在透视图布局中的位置靠左;
// 0.2f 指此空间占用透视图20%的宽度;editorArea 指使用透视图的编辑空间
IFolderLayout left = layout.createFolder("left", IPageLayout.LEFT, 0.2f, editorArea);
left.addView("cn.com.kxh.myplugin.View1"); // 参数为plugin.xml中“视图1”的id标识
// 将“视图2”加入到透视图的底部
IFolderLayout bottom = layout.createFolder("bottom", IPageLayout.BOTTOM, 0.8f, editorArea);
bottom.addView(View2.class.getName());// 由于我们把视图的id取成和类全名一样,所以也可以用这种写法
// 将以前定义的actionset(主菜单、工具栏按钮)加入到本透视图。这要在plugin.xml文
// 件的action设置中将visible="false"才看得出效果,这时打开其他透视图,action设置的
// 主菜单、工具栏按钮将不会出现在界面上,只有打开本透视图才会出现。
layout.addActionSet("myplugin.actionSet");// 参数为actionSet在plugin.xml中的id标识
}
}

总结:

(1)在插件中IWorkbenchPage对象比较重要,这里再给出一种获得此对象的通用方法,不过他是获得当前活动的IWorkbenchPage对象.

Activator.getDefault().getWorkbench().getActiveWorkbenchWindow().getActivepage();

(2)IWorkbenchPage.findView("cn.com.kxh.myplugin.View2")中的参数为"视图2"在plugin.xml中设置的id标识.

由此可见.plugin.xml文件在插件中的地位是及其重要的.IWorkbenchPage处理findView方法之外.还用findEditor方法来得到编辑器对象.像"cn.com.kxh.myplugin.View2"这种标识符在系统开发中会经常用到,最好建立一个类来集中放置这些字符串常量.然后系统中用的时候只用其常量名即可,否则把标识符的字串分散在代码中,以后改起来会非常麻烦.常量类的示例代码如下:

public final class StringConstants{
public final static String VIEW1="cn.com.kxh.myplugin.View1";
public final static String VIEW2=View2.class.getName();
}

要用的时候时候直接类名点调用就可以了.

附上上面代码的运行结果.

第二种监听式:

Eclipse环境的3个视图:"包资源管理器,大纲,属性"当双击包资源管理器中的结点时,大纲和属性视图也跟着改变.当然用前面讲的主动式来实现这个效果,不超过一个视图,用主动式就比较麻烦了,可以随着包资源管理器结点而需要改变的可能不止是大纲,属性视图,这时主动式就力所不及了.对于这种情况,则可以使用监听式.

  1.基本实例

  例如:View1,View2和View3视图,其中View2,View3需要监听View1中表格的选择事件.可以这样实现.

  (1)在View1类的createPartControl方法中加上如下一句:

  getSite().setSelectionProvider(tableViewer);//假设视图中有一个表格对象tableViewer

  setSelectionProvider方法的参数类型是ISelectionProvider(provider翻译为提供者),

  而TableViewer类正好实现了这一个接口(TreeViewer也一样).加此一句之后,如果再选择表格行时,

  底层事件机制将会通知所有监听者.

  (2)接着需要在View2,View3中各添加一个监听器到底层,一般也是写在createPartControl方法中.代码如下所示:

 getSite().getPage().addSelectionListener(new ISelectionListener(){
public void selectionChanged(IWorkbenchPart part,ISelection selection){
String partId = part.getSite().getId();
if(partId.equals("cn.com.kxh.myplugin.View1")){
System.out.println(part.getTitle());//part就是View1对象
System.out.println(selection);//selection就是被选择的表格行所代表的记录对象
}
}
});

这样,就在View2,View3中截获了View1的选择事件.由于底层的选择提供者可能不仅仅是View1,所以才需要再View2,View3的监听代码中根据View1对plugin.xml中的id标识做一下判断.当然,也可以将这个判断交由底层来负责.如下所示:

 getSite().getPage().addSelectionListener("cn.com.kxh.myplugin.View1",new ISelectionListener(){
public void selectionChanged(IWorkbenchPart part,ISelection selection){
System.out.println(part.getTitle());//part就是View1对象
System.out.println(selection);//selection就是被选择的表格行所代表的记录对象
}
});

如果View1中有两个表格怎么办?像下面的这样是行不通的.

getSite().setSelectionProvider(tableViewer1);

getSite().setSelectionProvider(tableViewer2);

既然是一个视图中只能设置一个选择提供者,那么可以换一种思路:创建一个自定义的选择提供者,然后由这个选择提供者收集tableViewer1,tableViewer2的选择事件集中传到底层.

自定义选择提供者就需要实现ISelectionProvider接口,查了一下该接口的层次结构发现有一个SelectionProviderAdapter适配器类,可惜它不是public类,无法继承它.那么久将SelectionProviderAdapter的代码复制到如下的MySelectionProvider类中.并略做修改.

 class MySelectionProvider implements ISelectionProvider {
List listeners = new ArrayList();
ISelection theSelection = StructuredSelection.EMPTY; public void addSelectionChangedListener(ISelectionChangedListener listener) {
listeners.add(listener);
} public ISelection getSelection() {
return theSelection;
} public void removeSelectionChangedListener(
ISelectionChangedListener listener) {
listeners.remove(listener);
} public void setSelection(ISelection selection) {
theSelection = selection;
final SelectionChangedEvent e = new SelectionChangedEvent(this, selection);
Object[] listenersArray = listeners.toArray(); for (int i = 0; i < listenersArray.length; i++) {
final ISelectionChangedListener l = (ISelectionChangedListener) listenersArray[i];
Platform.run(new SafeRunnable() {
public void run() {
l.selectionChanged(e);
}
});
}
}
}

现在有了选择提供器,但是View1中两个表格选择事件的功能还没有实现,可以发现TableViewer有一个addSelectionChangedListener方法.它能够监听表格的选择事件,但它接受的参数类型是ISelectionChangedListener.可以再单独创建一个ISelectionChangedListener接口的实现类,也可以让MySelectionProvider实现此接口.从而让MySelectionProvider即是底层的选择提供者,又是表格的选择事件的监听者.这里采用后一方案,让MySelectionProvider再实现ISelectionChangedListener接口.如下所示:

class MySelectionProvider implements ISelectionProvider,ISelectionChangedListener{

  ...原代码不改变.省略

  public void selectionChanged(SelectionChangedEvent event){

    setSelection(event.getSelection());

  }

}

View2,View3中的代码不必修改,只需要把View1类中的相应代码修改如下:

MyselectionProvider selectionProvider = new MySelectionProvider();

tableViewer1.addSelectionChangedListener(selectionProvider);

tableViewer2.addSelectionChangedListener(selectionProvider);

getSite().setSelectioinProvider(selectionProvider);

这里只提到了视图,实际上任何WorkbenchPart的子类都可以使用这种机制,包括编辑器,另外,由于可以创建自定义选择提供者,所以可以不仅局限于监听TreeViewer或者TableViewer,也可以监听Combo,Text等组件的非选择事件,只需要将要传送的信息包装成一个ISelection对象传给MySelectionProvider.setSelection方法既可.下面的代码就可以使得View1中的文本框组件的每次击键字符传播给各个视图的监听器.

final MySelectionProvider selectionProvider = new MySelectionProvider();

text.addKeyListener(new KeyListener(){

  public void keyPressed(KeyEvent e){

    String s = String.valueOf(e.character);

    ISelection selection = new StructuredSelection(s);

    selectionProvider.setSelection(selection);

  }

  public void keyReleased(KeyEvent e){}

});

注意:滥用底层事件广播机制可能会对性能有影响,但这需要用户在实际开发中做出测试和评估,以确定方法是否真的对性能造成了影响,而不是想当然.

关于Eclipse开发插件(三)的更多相关文章

  1. 程序猿必备的8款web前端开发插件三

    1.HTML5 Canvas 3D波浪翻滚动画 之前我们分享过好几款基于HTML5 Canvas的波浪和水波纹动画,比如这款HTML5 3D波浪起伏动画特效和这款超酷无比的HTML5 WebGL水面水 ...

  2. Hadoop伪分布配置与基于Eclipse开发环境搭建

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  3. 在windows下用eclipse + pydev插件来配置python的开发环境

    在windows下用eclipse + pydev插件来配置python的开发环境 一.安装 python 可以到网上下个Windows版的python,官网为:https://www.python. ...

  4. Spring(一):eclipse上安装spring开发插件&下载Spring开发包

    eclipse上安装spring开发插件 1)下载安装插件包:https://spring.io/tools/sts/all 由于我的eclipse版本是mars 4.5.2,因此我这里下载的插件包是 ...

  5. 第三篇:配置Hadoop的Eclipse开发环境

    前言 在先前的文章中,已经介绍了如何在Ubuntu Kylin操作系统下搭建Hadoop运行环境,而现在将在之前工作的基础上搭建Eclipse开发环境. 配置 开发环境:Eclipse 4.2,其他同 ...

  6. Python开发环境(1):Eclipse+PyDev插件

    电脑:小米笔记本电脑Pro 15.6寸(i5-8250U),操作系统:Windows 10,JDK版本:1.8.0_152(环境变量已配置) Step 1.下载Eclipse 根据我的CPU型号,选择 ...

  7. Eclipse开发环境配置,打磨Eclipse,安装插件(适用3.4,3.5,3.6,3.7)

    转载自:http://elf8848.iteye.com/blog/354035 打磨Eclipse -- 磨刀不误砍柴工 -------------------------------------- ...

  8. eclipse SE增加Web开发插件

    最近接触了些java项目,之前安装了eclipse SE版本.没有Web开发插件,调试不了Web代码.点击“Window”--“Preference” 左边菜单栏是找不到“Server”项来配置服务器 ...

  9. windows下用eclipse+goclipse插件+gdb搭建go语言开发调试环境

    windows下用eclipse+goclipse插件+gdb搭建go语言开发调试环境   http://rongmayisheng.com/post/windows%E4%B8%8B%E7%94%A ...

随机推荐

  1. mac远程连接 win8

    首先 打开组策略 .windows8 下 按win+R 可以唤出运行,输入[gpedit.msc]按下 回车键 选择[计算机配置]>[管理模板]>[Windows组件]>[远程桌面服 ...

  2. devexpress 中Grid 的使用:为零不显示

    如果要让为0的列不显示: this.gridColumn_FAmount.DisplayFormat.FormatType = DevExpress.Utils.FormatType.Numeric; ...

  3. 【DOM】学习笔记

    三. 一份文档就是一颗节点树 节点类型:元素节点——属性节点.文本节点 getElementById()返回一个对象,对应一个元素节点 getElementByTagName()返回一个对象数组,分别 ...

  4. RC522天线匹配参数【worldsing笔记】

    图为Device读卡器的参数值 EMC电路对读写距离影响不大:                   L3 和L4 固定为2.2uH:                  C11和C12也是固定值,如果P ...

  5. 高性能javascript

    高性能javascript开发 标签(空格分隔): javascript DOM访问与修改 访问DOM元素是具有代价的,修改元素代价较为昂贵,会导致浏览器重新计算页面的几何变化. 尽量减少DOM访问, ...

  6. C++学习笔记(七):函数

    函数通用格式: typeName functionName(parameterList) { //statements return value;//value is type cast to typ ...

  7. Azure 云服务中的实例端点

    Azure云服务(cloud Servive)中有三种端点类型(endpoint type):输入端点(input);内部端点(internal);实例端点(InstanceInput) 1.输入端点 ...

  8. matlab inpolygon 判断点在多边形内

    如何判断一个点在多边形内部? xv= [0 3 3 0 0]; %x坐标 yv= [0 0 3 3 0];%y坐标 x=1.5; y=1.5; in=inpolygon(x,y,xv,yv) plot ...

  9. oracle数据库表空间及归档

    --表空间(TableSpace)是Oracle的开创性理念.表空间使得数据库管理更加灵活,而且极大地提高了数据库性能. --作用 :1.避免磁盘空间突然耗竭的风险 2.规划数据更灵活 3.提高数据库 ...

  10. Python 对新浪微博的博文元素 (Word, Screen Name)的频率分析

    CODE: #!/usr/bin/python # -*- coding: utf-8 -*- ''' Created on 2014-7-9 @author: guaguastd @name: we ...