/******************************************************************************* * Copyright (c) 2009 IBM Corporation and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * IBM Corporation - initial API and implementation * *******************************************************************************/ package org.eclipse.wst.sse.ui.internal.projection; import org.eclipse.jface.text.BadLocationException; import org.eclipse.jface.text.IDocument; import org.eclipse.jface.text.IRegion; import org.eclipse.jface.text.Position; import org.eclipse.jface.text.Region; import org.eclipse.jface.text.source.projection.IProjectionPosition; /** * Represents a folding position for an XML comment */ public abstract class AbstractStructuredCommentFoldingPosition extends Position implements IProjectionPosition { /** * Default constructor * * @param offset the offset of the folding position * @param length the length of the folidng position */ public AbstractStructuredCommentFoldingPosition(int offset, int length) { super(offset, length); } /** * @see org.eclipse.jface.text.source.projection.IProjectionPosition#computeCaptionOffset(org.eclipse.jface.text.IDocument) */ public int computeCaptionOffset(IDocument document) throws BadLocationException { return findFirstContent(document.get(offset, length)); } /** * @see org.eclipse.jface.text.source.projection.IProjectionPosition#computeProjectionRegions(org.eclipse.jface.text.IDocument) */ public IRegion[] computeProjectionRegions(IDocument document) throws BadLocationException { //get the content of the comment String content = document.get(offset, length); int contentStart = findFirstContent(content); //find the start line of the comment //find the end line of the comment //find the first line of text in the comment int startLineNum = document.getLineOfOffset(getStartOffset()); IRegion startLine = document.getLineInformation(startLineNum); int endLineNum = document.getLineOfOffset(getEndOffset()) +1; IRegion endLine = document.getLineInformation(endLineNum); int captionLineNum = document.getLineOfOffset(getStartOffset() + contentStart); int foldOffset; int foldEndOffset; synchronized (this) { foldOffset = startLine.getOffset(); if(foldOffset < offset) { offset = foldOffset; } foldEndOffset = endLine.getOffset(); if((foldEndOffset-offset) > length) { length = foldEndOffset-offset; } } //fold before the first line of text in the comment IRegion preRegion = null; IRegion postRegion = null; if(startLineNum < captionLineNum) { IRegion captionLine = document.getLineInformation(captionLineNum); preRegion = new Region(foldOffset, captionLine.getOffset()-foldOffset); } //fold after the first line of text in the comment if(captionLineNum < endLineNum) { int postRegionOffset = document.getLineOffset(captionLineNum+1); postRegion = new Region(postRegionOffset, foldEndOffset-postRegionOffset); } IRegion[] regions = null; if(preRegion != null && postRegion != null) { regions = new IRegion[] {preRegion, postRegion}; } else if(preRegion != null) { regions = new IRegion[] {preRegion}; } else if(postRegion != null) { regions = new IRegion[] {postRegion}; } return regions; } /** * Finds the offset of the first identifier part within <code>content</code>. * Returns 0 if none is found. * * @param content the content to search * @param prefixEnd the end of the prefix * @return the first index of a unicode identifier part, or zero if none can * be found */ private int findFirstContent(final CharSequence content) { int lenght= content.length(); for (int i= 0; i < lenght; i++) { if (Character.isUnicodeIdentifierPart(content.charAt(i))) return i; } return 0; } /** * @return the start offset of the folding position */ protected abstract int getStartOffset(); /** * @return the end offset of the folding position */ protected abstract int getEndOffset(); }