| /******************************************************************************* |
| * Copyright (c) 2000, 2011 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 |
| * QNX Software System |
| * Anton Leherbauer (Wind River Systems) |
| *******************************************************************************/ |
| package org.eclipse.cdt.internal.ui.text; |
| |
| import org.eclipse.jface.text.BadLocationException; |
| import org.eclipse.jface.text.IDocument; |
| import org.eclipse.jface.text.IRegion; |
| import org.eclipse.jface.text.Region; |
| import org.eclipse.jface.text.TextUtilities; |
| import org.eclipse.jface.text.source.DefaultCharacterPairMatcher; |
| import org.eclipse.jface.text.source.ICharacterPairMatcher; |
| |
| import org.eclipse.cdt.core.dom.ILinkage; |
| import org.eclipse.cdt.core.model.ILanguage; |
| import org.eclipse.cdt.ui.text.ICPartitions; |
| |
| /** |
| * Helper class to match pairs of characters. |
| */ |
| public class CPairMatcher extends DefaultCharacterPairMatcher { |
| |
| private static final int ANGLE_BRACKETS_SEARCH_BOUND = 200; |
| |
| private boolean fMatchAngularBrackets= true; |
| private int fAnchor= -1; |
| |
| public CPairMatcher(char[] pairs) { |
| super(pairs, ICPartitions.C_PARTITIONING); |
| } |
| |
| /* @see ICharacterPairMatcher#match(IDocument, int) */ |
| @Override |
| public IRegion match(IDocument document, int offset) { |
| try { |
| return performMatch(document, offset); |
| } catch (BadLocationException ble) { |
| return null; |
| } |
| } |
| |
| /* |
| * @see org.eclipse.jface.text.source.DefaultCharacterPairMatcher#getAnchor() |
| */ |
| @Override |
| public int getAnchor() { |
| if (fAnchor < 0) { |
| return super.getAnchor(); |
| } |
| return fAnchor; |
| } |
| |
| /* |
| * Performs the actual work of matching for #match(IDocument, int). |
| */ |
| private IRegion performMatch(IDocument document, int offset) throws BadLocationException { |
| if (offset < 0 || document == null) return null; |
| final char prevChar= document.getChar(Math.max(offset - 1, 0)); |
| if ((prevChar == '<' || prevChar == '>') && !fMatchAngularBrackets) |
| return null; |
| final IRegion region; |
| if (prevChar == '<') { |
| region= findClosingAngleBracket(document, offset - 1); |
| fAnchor= ICharacterPairMatcher.LEFT; |
| } else if (prevChar == '>') { |
| region= findOpeningAngleBracket(document, offset - 1); |
| fAnchor= ICharacterPairMatcher.RIGHT; |
| } else { |
| region= super.match(document, offset); |
| fAnchor= -1; |
| } |
| if (region != null) { |
| if (prevChar == '>') { |
| final int peer= region.getOffset(); |
| if (isLessThanOperator(document, peer)) |
| return null; |
| } else if (prevChar == '<') { |
| final int peer= region.getOffset() + region.getLength() - 1; |
| if (isGreaterThanOperator(document, peer)) |
| return null; |
| } |
| } |
| return region; |
| } |
| |
| /** |
| * Returns the region enclosing the matching angle brackets. |
| * |
| * @param document a document |
| * @param offset an offset within the document pointing after the closing angle bracket |
| * @return the matching region or {@link NullPointerException} if no match could be found |
| * @throws BadLocationException |
| */ |
| private IRegion findOpeningAngleBracket(IDocument document, int offset) throws BadLocationException { |
| if (offset < 0) return null; |
| final String contentType= TextUtilities.getContentType(document, ICPartitions.C_PARTITIONING, offset, false); |
| CHeuristicScanner scanner= new CHeuristicScanner(document, ICPartitions.C_PARTITIONING, contentType); |
| if (isTemplateParameterCloseBracket(offset, document, scanner)) { |
| int pos= scanner.findOpeningPeer(offset - 1, Math.max(0, offset - ANGLE_BRACKETS_SEARCH_BOUND), '<', '>'); |
| if (pos != CHeuristicScanner.NOT_FOUND) { |
| return new Region(pos, offset - pos + 1); |
| } |
| } |
| return null; |
| } |
| |
| /** |
| * Returns the region enclosing the matching angle brackets. |
| * |
| * @param document a document |
| * @param offset an offset within the document pointing after the opening angle bracket |
| * @return the matching region or {@link NullPointerException} if no match could be found |
| * @throws BadLocationException |
| */ |
| private IRegion findClosingAngleBracket(IDocument document, int offset) throws BadLocationException { |
| if (offset < 0) return null; |
| final String contentType= TextUtilities.getContentType(document, ICPartitions.C_PARTITIONING, offset, false); |
| CHeuristicScanner scanner= new CHeuristicScanner(document, ICPartitions.C_PARTITIONING, contentType); |
| if (isTemplateParameterOpenBracket(offset, document, scanner)) { |
| int pos= scanner.findClosingPeer(offset + 1, Math.min(document.getLength(), offset + ANGLE_BRACKETS_SEARCH_BOUND), '<', '>'); |
| if (pos != CHeuristicScanner.NOT_FOUND) { |
| return new Region(offset, pos - offset + 1); |
| } |
| } |
| return null; |
| } |
| |
| /** |
| * Returns true if the character at the specified offset is a |
| * less-than sign, rather than an template parameter list open |
| * angle bracket. |
| * |
| * @param document a document |
| * @param offset an offset within the document |
| * @return true if the character at the specified offset is not |
| * a template parameter start bracket |
| * @throws BadLocationException |
| */ |
| private boolean isLessThanOperator(IDocument document, int offset) throws BadLocationException { |
| if (offset < 0) return false; |
| final String contentType= TextUtilities.getContentType(document, ICPartitions.C_PARTITIONING, offset, false); |
| CHeuristicScanner scanner= new CHeuristicScanner(document, ICPartitions.C_PARTITIONING, contentType); |
| return !isTemplateParameterOpenBracket(offset, document, scanner); |
| } |
| |
| /** |
| * Returns true if the character at the specified offset is a |
| * greater-than sign, rather than an template parameter list close |
| * angle bracket. |
| * |
| * @param document a document |
| * @param offset an offset within the document |
| * @return true if the character at the specified offset is not |
| * a template parameter end bracket |
| * @throws BadLocationException |
| */ |
| private boolean isGreaterThanOperator(IDocument document, int offset) throws BadLocationException { |
| if (offset < 0) return false; |
| final String contentType= TextUtilities.getContentType(document, ICPartitions.C_PARTITIONING, offset, false); |
| CHeuristicScanner scanner= new CHeuristicScanner(document, ICPartitions.C_PARTITIONING, contentType); |
| return !isTemplateParameterCloseBracket(offset, document, scanner); |
| } |
| |
| /** |
| * Checks if the angular bracket at <code>offset</code> is a template |
| * parameter bracket. |
| * |
| * @param offset the offset of the opening bracket |
| * @param document the document |
| * @param scanner a heuristic scanner on <code>document</code> |
| * @return <code>true</code> if the bracket is part of a template parameter, |
| * <code>false</code> otherwise |
| */ |
| private boolean isTemplateParameterOpenBracket(int offset, IDocument document, CHeuristicScanner scanner) { |
| int nextToken = scanner.nextToken(offset + 1, Math.min(document.getLength(), offset + ANGLE_BRACKETS_SEARCH_BOUND)); |
| if (nextToken == Symbols.TokenSHIFTLEFT || nextToken == Symbols.TokenLESSTHAN) |
| return false; |
| int prevToken= scanner.previousToken(offset - 1, Math.max(0, offset - ANGLE_BRACKETS_SEARCH_BOUND)); |
| if (prevToken == Symbols.TokenIDENT || prevToken == Symbols.TokenTEMPLATE) { |
| return true; |
| } |
| |
| return false; |
| } |
| |
| /** |
| * Checks if the angular bracket at <code>offset</code> is a template |
| * parameter bracket. |
| * |
| * @param offset the offset of the closing bracket |
| * @param document the document |
| * @param scanner a heuristic scanner on <code>document</code> |
| * @return <code>true</code> if the bracket is part of a template parameter, |
| * <code>false</code> otherwise |
| */ |
| private boolean isTemplateParameterCloseBracket(int offset, IDocument document, CHeuristicScanner scanner) { |
| if (offset >= document.getLength() - 1) |
| return true; |
| int thisToken= scanner.previousToken(offset, Math.max(0, offset - ANGLE_BRACKETS_SEARCH_BOUND)); |
| if (thisToken == Symbols.TokenSHIFTRIGHT) |
| return true; |
| if (thisToken != Symbols.TokenGREATERTHAN) |
| return false; |
| int prevToken= scanner.previousToken(scanner.getPosition(), Math.max(0, offset - ANGLE_BRACKETS_SEARCH_BOUND)); |
| if (prevToken == Symbols.TokenGREATERTHAN) |
| return true; |
| int nextToken= scanner.nextToken(offset + 1, Math.min(document.getLength(), offset + ANGLE_BRACKETS_SEARCH_BOUND)); |
| |
| switch (nextToken) { |
| case Symbols.TokenGREATERTHAN: |
| case Symbols.TokenCOMMA: |
| case Symbols.TokenSEMICOLON: |
| case Symbols.TokenCLASS: |
| case Symbols.TokenSTRUCT: |
| case Symbols.TokenUNION: |
| return true; |
| } |
| return false; |
| } |
| |
| /** |
| * Configure this bracket matcher for the given language. |
| * @param language |
| */ |
| public void configure(ILanguage language) { |
| fMatchAngularBrackets= language != null && language.getLinkageID() == ILinkage.CPP_LINKAGE_ID; |
| } |
| |
| } |