Live a simple life

沉默(zhu_xing@live.cn)
随笔 - 48, 文章 - 0, 评论 - 132, 引用 - 0
数据加载中……

【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)分析视图源码

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

posted on 2008-09-17 16:24 zhuxing 阅读(2424) 评论(3)  编辑  收藏 所属分类: Eclipse Plug-in & OSGIWTP(Web Tools Platform)

评论

# re: 【Eclipse插件开发】基于WTP开发自定义的JSP编辑器(六):IStructuredModel(DOM Document)分析视图   回复  更多评论   

HOHO,真棒!

什么时候讲到 EDITOR 啊~~
2008-09-18 10:08 | srdrm

# re: 【Eclipse插件开发】基于WTP开发自定义的JSP编辑器(六):IStructuredModel(DOM Document)分析视图   回复  更多评论   

大侠,代码还是前一节的啊,和前面的有重复,能不能换一下撒?谢谢啊!
2008-10-15 20:33 | 飞扬的麦子

# re: 【Eclipse插件开发】基于WTP开发自定义的JSP编辑器(六):IStructuredModel(DOM Document)分析视图   回复  更多评论   

代码搞混了^_^ 不好意思!
已经更新了
2008-10-16 11:20 | zhuxing

只有注册用户登录后才能发表评论。


网站导航: