【Eclipse插件开发】基于WTP开发自定义的JSP编辑器(六):IStructuredModel(DOM Document)分析视图

【Eclipse插件开发】基于WTP开发自定义的JSP编辑器(六):IStructuredModel(DOM Document)分析视图

        前面的几节中,我们都已经完整的介绍过了WTP最核心的几个数据模型:语法Document(IStructuredDocument)、语义Document(IDOMDocument、ICSSDocument)和WTP模型(IStructuredModel)。IStructuredModel在某种程度上可以看作是语义Document和语法Document的门面,三者关系再罗唆一下:
        
        前面在讲完WTP 语法Document(IStructuredDocument)的时候,我们开发过一个Structured Document分析视图,我想通过那个视图可以加深对IStructuredDocument的理解。在本节中,我们在开发一个视图,来分析一下WTP的语义Document(我们只分析最常用的IDOMDocument),希望也有类似的作用。

        PS:这两个视图其实可以作为一个工具来用,对于想修改或者定制WTP源码(当然也包括基于WTP开发一些工具)的开发者可以做一个工具,当写代码分析IStructuredDocument(Text Region)和IDOMDocument(Indexed Region)遇到障碍的时候,这两个视图应该做为一个助手^_^。而且通过这两个视图内容显示的比较,应该会明白为什么IStructuredDocument是语法Document,为什么IDOMDocument(ICSSDocument)是语义Document。

          开发本IStructuredModel(DOM Document)分析视图很多地方和前面的Structured Document分析视图类似,有不明白的地方(涉及到技术实现的地方),可以参考一下前面的第四节。

        【需求】
          和前面的Structured Document分析视图需求比较类似,大致如下:
           1、提供一个Structured Model分析视图,以树状方式将当前编辑器中的IDOMDocument展示出来
           2、交互(编辑器 ---> Structured Model分析视图):
                激活WTP JSP编辑器(或者是我们前面自己定制的编辑器),即时更新Structured Model分析视图
                当用户光在编辑器中标移动时,自动选中Structured Model分析视图中对应的节点
                当编辑器中的内容改变时,即时更新Structured Model分析视图
                当前激活编辑器关闭时,清空Structured Model分析视图内容
           3、交互(Structured Model分析视图 ---> 编辑器)
                双击视图中树状控件中特定节点,对应内容在编辑器中被选中
            4、显示内容:
                 因为每个节点都是IDOMNode,则分别显示其实现类名称、位置信息和文本内容
                
          【效果预览】
            
            上面显示的效果是,双击视图中对应的IDOMNode,对应的文本内容在编辑器中被选中。

            【实现摘要(文章后门会附上对应的源码)】
             1、 创建插件工程wtp.stucturedmodel,创建视图。视图IViewPart对应实现类为StructuredModelView,这个和前面讲过的Structured Document分析视图类似,这边就不细讲了。
public   class  StructuredModelView  extends  ViewPart  implements  ISelectionListener{
    
private  TreeViewer viewer;
    
private  ITreeContentProvider contentProvider;
    
private  ILabelProvider labelProvider;
        
        
//
}
          
             2、利用workbench中的选择服务(seleciton service)。前面需求中说过,我们要监听光标在编辑器中的位置选择,所以使用此服务,所以我们的StructuredModelView要实现org.eclipse.ui.ISelectionListener接口。
                注册、销毁selection listener和前面开发Structured Document分析视图是一样的,在视图实现类init方法中注册,在dispose方法中销毁。        
1  /*  (non-Javadoc)
2       * @see org.eclipse.ui.part.ViewPart#init(org.eclipse.ui.IViewSite)
3        */
4       public   void  init(IViewSite site)  throws  PartInitException {
5           super .init(site);
6          
7           this .getSite().getPage().getWorkbenchWindow().getSelectionService().addPostSelectionListener( this );
8           this .getSite().getPage().getWorkbenchWindow().getPartService().addPartListener(partListener);
9      }

                我们看一下selection事件的处理代码:
 1  /*  (non-Javadoc)
 2       * @see org.eclipse.ui.ISelectionListener#selectionChanged(org.eclipse.ui.IWorkbenchPart, org.eclipse.jface.viewers.ISelection)
 3        */
 4       public   void  selectionChanged(IWorkbenchPart part, ISelection selection) {
 5           if  (part  instanceof  TextEditor) {
 6              IEditorInput editorInput  =  ((TextEditor)part).getEditorInput();
 7              IDocument document  =  ((TextEditor)part).getDocumentProvider().getDocument(editorInput);
 8              
 9               // 判断是否是IStructuredDocument
10               if  ( ! (document  instanceof  IStructuredDocument)) {
11                   this .viewer.setInput( new  Object[ 0 ]);
12                   return  ;
13              }
14              
15               // 对于editor使用的IStructuredModel,是用IModelManager来管理的
16              IModelManager modelManager  =  StructuredModelManager.getModelManager();
17              IStructuredModel structuredModel  =  modelManager.getModelForRead((IStructuredDocument)document);
18               if  (structuredModel  ==   null )
19                   return  ;
20              
21               // 根据判断是否需要更新tree vier输入做不同处理
22               if  ( this .needUpdateInput(structuredModel)) {
23                   // 减少old model的引用计数,并注销对应的model listener
24                   if  ( this .viewer.getInput()  instanceof  IStructuredModel) {
25                      IStructuredModel oldStructuredModel  =  ((IStructuredModel) this .viewer.getInput());
26                      
27                      oldStructuredModel.removeModelStateListener(modelStateListener);
28                      oldStructuredModel.releaseFromRead();
29                  }
30                  
31                   // 设置输入,并注册模型状态监听器
32                  structuredModel.addModelStateListener( this .modelStateListener);
36              }
37               else  {
38                   // 如果不需要此structuredModel作为输入,则将此structuredModel的引用计数复原
39                  structuredModel.releaseFromRead();
40              }
41              
42               // 根据编辑器中选择定位到model tree viewer中相应节点
43               if  (selection  instanceof  ITextSelection)
44               this .processTextSelection((ITextSelection)selection, structuredModel);
45              
46               this .sourcePart  =  part;
47          }
48      }
49      
50       /**
51       * 判断当前structuredModel和tree viewer中已有的structured model是否一致(判断是否==,而非equals)
52       * 
53       *  @param  structuredModel
54       *  @return
55        */
56       private   boolean  needUpdateInput(IStructuredModel structuredModel) {
57           if  ( this .viewer.getInput()  !=   null )
58               return   this .viewer.getInput()  !=  structuredModel;
59          
60           return   true ;
61      }
            看的出来,我们的selection处理逻辑如下:
            1、如果当前选择事件来自TextEditor类型的编辑器中(文本编辑器的共同超类),则去获取编辑器对应的IDocument,如果是IStrucuturedDocument则判断是否需要更新;如果不是,则不是我们的菜^_^
            2、利用WTP提供的模型管理器IModelManager(这个在下一节会详细讲,很重要^_^)获取和以上IStructuredDocument对应的IStructuredModel(通过IStructuredModel,可以获取到对应的语义Document--IDOMDocument,前面说过的^_^)。然后判断是否需要刷新模型tree viewer,判断的依据是看tree viewer中现有的input和本IStructuredModel是否一致。
                PS:这边有两点需要注意:一是IModelStateListener的注册;二是IModelManager的使用。

            3、处理text selection(参见org.eclipse.jface.text.ITexSelection),定位对应的dom node。大致过程为:首先判根据IStructuredModel.getIndexedRegion获取对应的节点(注意这边只能定位到对应的IDOMElement元素,并不能定到对应的IDOMAttr或者IDOMText);其次判断光标是否位于节点的attribute中,如果是,则定位到dom attr(具体可以参见代码)

         3、 处理视图中tree viewer双击,定位编辑器中对应内容
this .viewer.addDoubleClickListener( new  IDoubleClickListener() {
                
public   void  doubleClick(DoubleClickEvent event) {
                    TreeSelection selection 
=  (TreeSelection)event.getSelection();
                    
                    
// 树上的每个节点都是indexed region
                    IndexedRegion indexedRegion  =  (IndexedRegion)selection.getFirstElement();
                    
                    
// 处理编辑器选中
                     int  selectionOffset  =  indexedRegion.getStartOffset();
                    
int  length  =  indexedRegion.getEndOffset()  -  selectionOffset;
                    ((StructuredTextEditor)sourcePart).getTextViewer().setSelectedRange(selectionOffset, length);
                }
            });
            上面我们根据tree viewer中选中的indexed region对应的坐标,直接通过ITextViewer.setSelectedRange(int offset, int length)接口来进行文本选中。(PS:WTP提供的StructuredText本身就是一种ISourceViewer,ISourceViewer本身又是一种ITextViewer^_^)

            4、 利用IModelStateListener同步更新视图。我们在上一节在介绍IStructuredModel的时候,提到过WTP提供了一个IModelStateListener来允许用户监听IStructuredModel的状态变化,IStructuredModel本身又作为一个target,接受用户注册IModelStateListener实现。我们的IModelStateListener实现非常简单,只在目标IStructuredModel变化了之后,刷新视图中的tree viewer
            
private   class  ModelStateListener  implements  IModelStateListener {
     
/*  (non-Javadoc)
         * @see org.eclipse.wst.sse.core.internal.provisional.IModelStateListener#modelChanged(org.eclipse.wst.sse.core.internal.provisional.IStructuredModel)
         
*/
        
public   void  modelChanged(IStructuredModel model) {
            viewer.refresh();
        }

       
// 只覆写了该方法,其他方法代码省略
}

            5、 处理编辑器关闭行为,利用workbench的part service特性。当关联编辑器关闭时,削减目标IStructuredModel的引用计数,并注销之前注册的IModelStateListener,清空视图中的tree viewer。    
private   class  PartListener  implements  IPartListener {
        
public   void  partActivated(IWorkbenchPart part) {
            
//  TODO Auto-generated method stub
            
        }
        
        
public   void  partBroughtToTop(IWorkbenchPart part) {
            
//  TODO Auto-generated method stub
            
        }
        
        
/*  
         * 如果被关闭的workbench part是提供structured model信息的source part,则:
         * 1、削减该structured model的引用计数(因为已经不再引用)
         * 2、注销之前注册的IModelStateListener
         * 3、清空tree viewer
         * 
         * @see org.eclipse.ui.IPartListener#partClosed(org.eclipse.ui.IWorkbenchPart)
         
*/
        
public   void  partClosed(IWorkbenchPart part) {
            
// 削减引用计数,并注销对应的model listener
             if  (part  ==  StructuredModelView. this ) {
                
if  (viewer.getInput()  instanceof  IStructuredModel) {
                    IStructuredModel structuredModel 
=  ((IStructuredModel)viewer.getInput());
                    structuredModel.releaseFromRead();
                    
                    structuredModel.removeModelStateListener(modelStateListener);
                }
            }
            
            
// update model tree viewer
             if  (sourcePart  ==  part) {
                sourcePart 
=   null ;
                viewer.setInput(
null );
            }
        }
        
        
public   void  partDeactivated(IWorkbenchPart part) {
            
//  TODO Auto-generated method stub
            
        }
        
        
public   void  partOpened(IWorkbenchPart part) {
            
//  TODO Auto-generated method stub
        }
    }


            6、tree viewer对应的content provider实现。(其实和我们遍历一个普通的xml dom document很类似)    
public   class  ModelTreeContentProvider  implements  ITreeContentProvider {

    
/*  
     * IStructuredModel分为两种:IDOMModel和ICSSModel,对应的document实现分别为IDOMDocument和ICSSDocument。
     * 我们只分析IDOMModel(IDOMDocument)的情况,对于ICSSModel(ICSSDocument)的分析留给大家吧^_^
     * 
     * @see org.eclipse.jface.viewers.ITreeContentProvider#getChildren(java.lang.Object)
     
*/
    
public  Object[] getChildren(Object parentElement) {
        
if  (parentElement  ==   null )
            
return   new  Object[ 0 ];
        
        
// 如果是IDOMModel,则获取对应的IDOMDocument
         if  (parentElement  instanceof  IDOMModel) {
            IDOMDocument domDocument 
=  ((IDOMModel)parentElement).getDocument();
            
return   new  Object[]{domDocument};
        }
        
        
// 对于遵守xml dom规范的node,则按照xml node的结构来遍历
         if  (parentElement  instanceof  IDOMNode) {
            List children 
=   new  ArrayList();
            
            NamedNodeMap attributes 
=  ((IDOMNode)parentElement).getAttributes();
            
if  (attributes  !=   null ) {
                
for  ( int  i  =   0 ; i  <  attributes.getLength(); i ++ ) {
                    children.add(attributes.item(i));
                }
            }
            
            NodeList childNodes 
=  ((IDOMNode)parentElement).getChildNodes();
            
for  ( int  i  =   0 ; i  <  childNodes.getLength(); i ++ ) {
                children.add(childNodes.item(i));
            }
            
            
return  children.toArray();
        }
        
        
return   new  Object[ 0 ];
    }
     
// 其他方法省略
}

            

            以上基本上就是本视图的主要代码了,开发这个视图代码基本上也是300行左右。

            本插件工程需要依赖的插件列表为:
             org.eclipse.ui,
             org.eclipse.core.runtime,
             org.eclipse.core.resources,
              org.eclipse.wst.sse.core,
             org.eclipse.wst.xml.core,
             org.eclipse.wst.sse.ui,
             org.eclipse.jface.text,
             org.eclipse.ui.workbench.texteditor


            【源码下载】
             源码为实际工程以Export ---> Archive File方式导出的,下载链接: Structured Model(Dom Document)分析视图源码

本博客中的所有文章、随笔除了标题中含有引用或者转载字样的,其他均为原创。转载请注明出处,谢谢!

你可能感兴趣的:(【Eclipse插件开发】基于WTP开发自定义的JSP编辑器(六):IStructuredModel(DOM Document)分析视图)