科技行者

行者学院 转型私董会 科技行者专题报道 网红大战科技行者

知识库

知识库 安全导航

至顶网软件频道扩展Eclipse的Java开发工具(二)

扩展Eclipse的Java开发工具(二)

  • 扫一扫
    分享文章到微信

  • 扫一扫
    关注官方公众号
    至顶头条

扩展Eclipse的Java开发工具(二)

作者:gmpost 来源:赛迪网技术社区 2007年12月1日

关键字: Eclipse 扩展

  • 评论
  • 分享微博
  • 分享邮件
在用户界面中,如何显示扩展,显示在何处? @kyb y  
  这在很大程度上是一个温和的提示,因为我们已得到了答案。我们希望对一个或多个选中的方法显示上下文菜单选项,这些菜单选项允许我们只用一个操作就可以更改方法的可视性。我们更喜欢在可以显示方法的任何地方都能使用这些菜单选项,如在 Hierarchy 视图和 Package Explorer 中。这把我们带到下一个问题。 y:zQ-T`N  
   j\*|p&EB  
  通常如何扩展用户界面? TJu(Fh  
  通过示例来学习会更有趣,这方面 Plug-in Project 向导可以提供帮助,它提供了一些样本代码,我们可以修改这些代码来满足我们的需要。我们将回答该向导中的几个问题,它将自动启动用于插件开发的专门透视图,称为 Plug-in Development Environment(PDE),以准备测试。该向导包含了可以帮助我们入手的许多示例。事实上,我们的老朋友“Hello World”也在那里。为了沿袭这个传统,我们将生成这个“Hello World”,查看结果以验证是否正确安装了该环境,随后修改它以帮助我们回答当前的问题,并把我们带到下一个问题:对用户界面的扩展如何知道类似于“选择”这样的基本事件?这很重要,因为我们希望将我们新近引入的菜单选项应用到当前选中的方法上。 .qQc" i"  
   rO`!GN  
  请注意,这些指示信息假定您正从全新的 Eclipse 安装开始。如果修改了该环境或更改了首选项,那么可能不会完全象如下所述那样工作。您可以考虑从全新的工作空间启动 Eclipse:打开命令提示符窗口,更改到 \eclipse 目录,然后使用 -data 参数启动 Eclipse,如清单 1 所示。 :BN2Q$z#  
   ek=A\aO  
  清单 1. 启动全新的 Eclipse 实例 7K{<NETM  
   <}\g=zbX  
  cd c:\eclipse2.1\eclipse b2y rXH  
  eclipse.exe -data workspaceDevWorks hzKL4iT  
   Xd=1FN I  
  从使用 New Plug-in Project 向导创建一个插件项目开始。选择 File > New > Project。在 New Project 对话框中,在向导列表中选择 Plug-in Development and Plug-in Project,然后选择 Next。将项目命名为 com.ibm.lab.helloworld。该向导将根据这个名称创建插件标识,所以它在系统中必须是唯一的(按惯例,项目名和插件标识相同)。使用显示在“Project contents”下面的推荐缺省工作空间位置就可以了;选择 Next。 s&hMJ0  
   M|j"m8;6  
  在下一页上,选择 Next 以接受缺省插件项目结构。该插件代码生成器页推荐了许多样本,向导可以帮助您进一步对该项目进行参数化。选择“Hello, World”选项,然后选择 Next。下一页(显示在图 4 中)推荐了插件名和插件类名。这些名称基于插件项目 com.ibm.lab.helloworld 的最后一个单词。这个示例不需要任何插件类便利方法,所以取消对三个代码生成选项的选择(如图 4 所示),然后选择 Next(不是 Finish;您还有一页要完成)。 gyFI73OY  
   Z,HS0|f+3  
  |;4'zf-  
.I90Wh#  
   R2,>Qpm  
图 4. Simple Plug-in Content ,3vuF 2GHv  
/g.erjZ<  
  您可以在下一页(显示在图 5 中)中指定参数,这些参数对于“Hello, Worlds”示例是唯一的,比如,将要显示的消息。 `&x z-k;  
   KBZ.~ij  
  F y=%9}X  
{ ;UMV]Gc  
   ] O R>*4z  
图 5. Sample Action Set % u8 a`C  
d:>PM  
  要简化所产生的代码,将该操作的目标包名从 com.ibm.lab.helloworld.actions 更改成 com.ibm.lab.helloworld,即与该项目的名称相同。尽管在实际的插件中,您可以选择用不同的包对相关的类进行分组,但在本例中,只有两个类,所以不必这样做。这样也遵循了“主”包名和项目名相同这个惯例。现在选择 Finish。 n36"7i2  
   ! cbV5D\  
  您应该看到一个信息消息:“Plug-ins required to compile Java classes in this plug-in are currently disabled. The wizard will enable them to avoid compile errors.”。选择 OK 继续。如果这是个全新的工作空间,那么您还将看到另一个信息消息:“This kind of project is associated with the Plug-in Development Perspective. Do you want to switch to this perspective now?”。 选择 Yes 以根据这个消息的建议进行切换。 9<rz`vUz  
s\ITR#(`k  
要验证所有的东西是否都设置正确,让我们测试新插件。选择 Run > Run As > Run-Time Workbench。这将启动 Eclipse 的第二个实例,它将包含您的插件。这个新实例将创建一个新的名为 runtime-workspace 的工作空间目录,所以不必担心;对这个实例所作的任何测试都不会影响开发设置。您应该看到类似图 6 的样子,其中有一个新的下拉菜单,其标签为 Sample Menu,它有单一的选项 Sample Action。选择它将显示下面的信息消息。如果您不是从全新的工作空间启动,那么可以选择 Window > Reset Perspective 以查看新生成的下拉菜单;从现有工作空间启动时不会显示这个菜单,因为工作台“记得”上次 Eclipse 运行时哪些操作集是活动的(您还可以从 Window > Customize Perspective...下拉菜单选项上添加/删除操作集)。 pY'K8x]=  
   n5=!L=  
 cqu{ \k  
T6;T<d  
Jahp'S]  
   RJqEHF  
ZLb]q=["o  
图 6. Hello, Eclipse world 8nQC |.s  
+ejw<4Cn  
  让我们快速浏览一下插件清单文件 plugin.xml。双击它,以在 Plug-in Manifest 编辑器中打开它。这个编辑器提供了几个类似于向导的页和一个“原始”源代码页。通过选择 Source 选项卡转到源代码页。您将看到与以下清单 2 显示的代码相似的内容;我们感兴趣的是用粗体显示的那几部分。 $S,79R~t"  
   Pb;QmmG8*  
  清单 2. 所生成的“Hello, World” plugin.xml QE)o &s  
   lOiCb@0DG}  
      point="org.eclipse.ui.actionSets">> !x5BL(-%  
        label="Sample Action Set" VV7;J&aA  
     visible="true" *(+[:AGl  
     id="com.ibm.lab.helloworld.actionSet"> :Gdn_Z'9<  
          label="Sample &Menu" MwY5,/~R  
      id="sampleMenu"> 7$Sh_:W  
           name="sampleGroup"> ?S. :XP#+s  
      p3?jlBb%  
     qJczT1a  
~k6!OGU<  
#fq5 N6  
          label="&Sample Action" ;@M/;]  
      icon="icons/sample.gif" '_(v1:b[s  
      class="com.ibm.lab.helloworld.SampleAction" 6V@ u:/N]  
      tooltip="Hello, Eclipse world" FJ1fZ9  
      menubarPath="sampleMenu/sampleGroup" iz`?EL+7  
      toolbarPath="sampleGroup" ;:lFD6MH  
      id="com.ibm.lab.helloworld.SampleAction"> h<6.ZN +>  
     k`eK+-\ j  
    dT6Vh=  
   6,M}@bX*An  
   B:qT^J  
  并不需要过于深入地研究这个代码。我们“旅行”的第 II 部分的目的只是让您熟悉一些基本机制,借此我们可以介绍 JDT 的扩展。这里,您会看到这样一种技术的一个样本:它将菜单和菜单选项作为操作集添加到工作台。它以一个用 标记声明的扩展开始。工作台用户界面插件定义了这个扩展点 org.eclipse.ui.actionSets,以及几个类似的扩展点,通过这几个扩展点可以向各种用户界面元素提供其它插件。 "R.,;3R^]  
   iKr\yUv9  
  我们还未回答如何将菜单选项添加到 Java 方法的上下文菜单中。一个简单示例可以给我们一些提示。首先打开显示“Hello, World”消息的类 SampleAction,请注意其 run 方法。它不是特别有趣;不过我们还看到了另一个方法 selectionChanged。啊哈!下一个问题的答案有了。 ^S3LadD  
   ` DsOm!s  
  对用户界面的扩展如何知道类似于“选择”这样的基本事件? Z] @99]"n  
   yEb[h(  
  工作台“选择”更改时会告知所提供的操作(象我们提供的菜单下拉选项)。这在 selectionChanged 方法前面的 Javadoc 注释中得到了确认。让我们修改这个方法以告知有关“选择”的更多信息。首先,如果您还没有关闭工作台的运行时实例,那么现在就关闭。然后对 selectionChanged 方法添加清单 3 中的代码。 phhLU"E  
   ,.TK l4n  
  清单 3. selectionChanged 方法,首次修改 PP|o3tqd  
   _< 5h=]T  
  public void selectionChanged(IAction action, ISelection selection) { *5EgZ  
   System.out.println("==========> selectionChanged"); b7[L 3uF  
   System.out.println(selection); gp5,VzI  
  } SWLg:q-1M  
   F FOmR!u  
  有了这个调试代码,我们将看到选择了什么,并了解到有关什么使 Eclipse 工作的更多信息。保存该方法,然后重新启动运行时工作台。 !4_.f\:~  
Hg + uO  
重要:Eclipse 有一个延迟装入的策略,以在用户执行需要插件代码的操作时才装入插件。所以您必须先选择 Sample Action 菜单选项,以在调用 selectionChanged 方法之前装入您的插件。 BA6(a=rmC  
   g)r)gR  
  现在选择其它东西,如编辑器中的文本、Navigator 中的文件,当然还有 Outline 视图中的成员(回忆一下:您必须创建一个 Java 项目和示例 Java 类来做到这一点,因为运行时实例使用不同的工作空间)。清单 4 显示了您将在 Eclipse 的开发实例的控制台中看到的某个示例输出。 'nSi<5}5  
   '';7 >ivEv  
  清单 4. selectionChanged 输出,首次修改 v1/,DpbU  
   *-nav[  
  ==========> selectionChanged .V1C_bb  
  [package com.ibm.lab.soln.jdt.excerpt [in [Working copy] ChangeIMemberFlagAction.java UpRV=}|  
    [in com.ibm.lab.soln.jdt.excerpt [in src [in com.ibm.lab.soln.jdt.excerpt]]]]] *;Qq)YP  
  ==========> selectionChanged aX-G'A`W  
   EHv#Q]enm  
  ==========> selectionChanged _VSC$i~  
  org.eclipse.jface.text.TextSelection@9fca283 @n\4 kqe  
  ==========> selectionChanged :-Vcq4(Cz  
   .{qr[^ #  
  ==========> selectionChanged *@0&~1  
  [package com.ibm.lab.soln.jdt.excerpt [in [Working copy] ChangeIMemberFlagAction.java 8(PU)  
    [in com.ibm.lab.soln.jdt.excerpt [in src [in com.ibm.lab.soln.jdt.excerpt]]]]] Fq(HS=N  
  ==========> selectionChanged T AX[go(T  
  [IMember[] members [in ChangeIMemberFlagAction [in [Working copy] ChangeIMemberFlagAction.java iPbv *ge  
    [in com.ibm.lab.soln.jdt.excerpt [in src [in com.ibm.lab.soln.jdt.excerpt]]]]]] F<[8"tf  
  ==========> selectionChanged uM{,I'  
   `"=_ fqm!  
  ==========> selectionChanged 6:0=A" jUF  
  [ChangeIMemberFlagAction.java [in com.ibm.lab.soln.jdt.excerpt `Zh}|BNFb  
     [in src [in com.ibm.lab.soln.jdt.excerpt]]] 3L<@_ k%  
   package com.ibm.lab.soln.jdt.excerpt V,([y,V695  
   import org.eclipse.jdt.core.Flags U ~:Y  
   import org.eclipse.jdt.core.IBuffer "n~g2A&]  
   ...lines omitted... 7$Dfl\Qk  
    void selectionChanged(IAction, ISelection)] N)c-85_  
  ==========> selectionChanged &_ O)(4i  
  [boolean isChecked(IAction, IMember) [in ToggleIMemberFinalAction ;/edE*o  
    [in ToggleIMemberFinalAction.java [in com.ibm.lab.soln.jdt.excerpt  5 D `$X  
    [in src [in com.ibm.lab.soln.jdt.excerpt]]]]]] 9%>>]To  
   X`=fU84  
  很明显,这个“选择”不象 String 的实例那么基本,但没有明示涉及了什么类,因为这些类很明显覆盖了它们的缺省 toString 方法。如果不用多做一点研究就可以明白这些类向我们显示的信息,那我们就轻松了,但目前我们还没有达到这种程度。回到 selectionChanged 方法,浏览 selection 参数的接口 ISelection 的层次结构。其层次结构表明它的通用子类型接口并不多,只有 IStructuredSelection(用于列表)和 ITextSelection。通过输出所选的类,我们可以使 selectionChanged 方法稍微更智能一点。如清单 5 所示修改 selectionChanged 方法。 *D_KrIy@v  
   -v+0+N.Db  
  清单 5. selectionChanged 方法,第二次修改 Sw' $ZMz  
   Lc:fo`Z  
  public void selectionChanged(IAction action, ISelection selection) { @j,1dH,.  
   System.out.println("==========> selectionChanged"); i=agE  
   if (selection != null) { JqQg~1@  
    if (selection instanceof IStructuredSelection) { 4mOW1$P1  
     IStructuredSelection ss = (IStructuredSelection) selection; !MmMsJr#  
     if (ss.isEmpty()) ('}tDc3  
      System.out.println(""); r1mjaO  $  
     else Wy{ AC!  
      System.out.println("First selected element is " + ss.getFirstElement().getClass()); Hw U1i  
    } else if (selection instanceof ITextSelection) { w #,IcSVVh  
     ITextSelection ts = (ITextSelection) selection; wR-\zt||i  
*GkfG> :J  
 System.out.println("Selected text is <" + ts.getText() + ">"); z7QE  
    } pF ` @PUr  
   } else { nAeQ=Nr  
    System.out.println(""); Ar^k}8|C  
   } |V.em@`M]  
  } L#( ;0,_  
   Te J{/}  
  同样,请记住关闭运行时实例,然后重新启动。现在当您选择用户界面的各种元素时,它们是不是提供了更多信息,如清单 6 所示。 z3. 2d /  
   O#O 1MTZ  
  清单 6. selectionChanged 输出,第二次修改 %i5<CJ/ez#  
   y wplNgz!f  
  ????selected some methods in the Outline view uIQZm#Zf  
  ==========> selectionChanged BB:~IXBg  
  First selected element is class org.eclipse.jdt.internal.core.SourceMethod Up {dp'%  
  ==========> selectionChanged IEAm4-R  
  First selected element is class org.eclipse.jdt.internal.core.SourceMethod F%OSD{0p  
  ==========> selectionChanged K<Z?[_"  
   .w*wIu^d  
   , }TmX0,<s  
    activated the Java editor _0xK~.@\  
  ==========> selectionChanged 4^ (ZK&=*  
  Selected text is RwTl9?^i}  
  ==========> selectionChanged g6 ?+  
   r~rr9V  
   d Hv'FJ  
    selected same methods and classes, package in the Package Explorer OqM%7K  
  ==========> selectionChanged [EH1f/k.W  
  First selected element is class org.eclipse.jdt.internal.core.SourceMethod Gi(`mUpeWL  
  ==========> selectionChanged K@ZK^I~_  
  First selected element is class org.eclipse.jdt.internal.core.SourceType "TI]lL X  
  ==========> selectionChanged  BLmYsJ  
  First selected element is class org.eclipse.jdt.internal.core.PackageFragment NXSAu:3s  
   poHLNk*.  
    activated the Navigator view, selected some files, folders, and projects MC$ (fHGc  
  ==========> selectionChanged R=YtH5=x  
  First selected element is class org.eclipse.core.internal.resources.File ;:;d[jH  
  ==========> selectionChanged B: ISC1:q<  
   aIB"_&;,  
  ==========> selectionChanged w96`E&{4  
  First selected element is class org.eclipse.core.internal.resources.File \hcXCXwh(`  
  ==========> selectionChanged m/F4]d)G  
  First selected element is class org.eclipse.core.internal.resources.Project ]G&cnB-3X  
  ==========> selectionChanged B7.=|l  
  First selected element is class org.eclipse.core.internal.resources.Folder 0rV =1L)T  
  ==========> selectionChanged < s!Ta#f  
   /9Ie`]|c  
   e -i~[$}  
    reactivated the Package Explorer, 1K.# "  
    selected some classes and methods in JARs of reference libraries V+TEdr   
  ==========> selectionChanged y 8"9^>  
  First selected element is class org.eclipse.jdt.internal.core.JarPackageFragment ^#'iMKZ  
  ==========> selectionChanged 6Fy[uM0a  
  First selected element is class org.eclipse.jdt.internal.core.ClassFile p.`-pz#  
  ==========> selectionChanged Y= zc`^QP  
  First selected element is class org.eclipse.jdt.internal.core.BinaryMethod Lyx1Js=  
   MFzKA:1  
  特别地,我们确认在用户界面中看到的东西与 JDT 模型类一一对应。我们之所以了解所显示的是作为选择的模型,而不是类似于字符串和图像的较低级的基本类型,要归功于另一个 Eclipse 框架,称为 JFace。这个框架在象字符串这样的基本类型(接近操作系统的窗口小部件希望使用这些基本类型)和更高级的模型对象(您的代码更愿意使用这些对象)之间进行映射。本文只是略微提及这个主题,因为我们设定的目标是扩展 JDT。参考资料一节推荐了有关 JFace 的其它参考资料,它们将拓展您的理解。本文将只讨论理解 JDT 扩展的基础所必需的知识。 {P)#ef  
   ziUv&O  
  回到输出,特殊的选项结果引起了我们的注意:它们对应于用户界面中 Java 成员的选择。清单 7 中重复了它们。 FA]MC  
   piG{7c@H%  
  清单 7. selectionChanged 输出,再次研究 dC*D Ql`  
   1gm4>[  
  ==========> selectionChanged Fxq!}Y  
  First selected element is class org.eclipse.jdt.internal.core.SourceMethod Fj:CmpA'  
_w=- jO^  
 ==========> selectionChanged /Ko-.J Ez  
  First selected element is class org.eclipse.jdt.internal.core.BinaryMethod rZW[ v  
   ?IkL/u  
  这些类的包名中间的 internal 使人有点担心。但是,正如您通常会发现的,Eclipse 会有一个公共(public)接口,它对应于(内部)实现类,就如这里的例子。快速类查找揭示出:这些类都实现了看来很有希望成为这个问题答案的一组公共接口,也就是 ISourceReference、IJavaElement,尤其还有 IMember 查看本文来源
    • 评论
    • 分享微博
    • 分享邮件
    邮件订阅

    如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。

    重磅专题
    往期文章
    最新文章