/******************************************************************************* * Copyright (c) 2000, 2005 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 com.aptana.ide.core.ui.contentassist; import java.util.ArrayList; import org.eclipse.jface.contentassist.IContentAssistSubjectControl; import org.eclipse.jface.contentassist.ISubjectControlContentAssistProcessor; import org.eclipse.jface.contentassist.SubjectControlContentAssistant; import org.eclipse.jface.contentassist.SubjectControlContextInformationValidator; import org.eclipse.jface.text.DefaultInformationControl; import org.eclipse.jface.text.IDocument; import org.eclipse.jface.text.IInformationControl; import org.eclipse.jface.text.IInformationControlCreator; import org.eclipse.jface.text.ITextViewer; import org.eclipse.jface.text.contentassist.CompletionProposal; import org.eclipse.jface.text.contentassist.ICompletionProposal; import org.eclipse.jface.text.contentassist.IContentAssistProcessor; import org.eclipse.jface.text.contentassist.IContentAssistant; import org.eclipse.jface.text.contentassist.IContextInformation; import org.eclipse.jface.text.contentassist.IContextInformationValidator; import org.eclipse.swt.widgets.Shell; import com.aptana.ide.epl.Activator; /** * Content assist processor for regular expressions. * * @since 3.0 */ public final class LoggingRegExContentAssistProcessor implements IContentAssistProcessor, ISubjectControlContentAssistProcessor { /** * Proposal computer. */ private static class ProposalComputer { /** * The whole regular expression. */ private final String fExpression; /** * The document offset. */ private final int fDocumentOffset; /** * The high-priority proposals. */ private final ArrayList fPriorityProposals; /** * The low-priority proposals. */ private final ArrayList fProposals; /** * <code>true</code> iff <code>fExpression</code> ends with an open escape. */ private final boolean fIsEscape; /** * Creates a new Proposal Computer. * * @param contentAssistSubjectControl * the subject control * @param documentOffset * the offset */ public ProposalComputer(IContentAssistSubjectControl contentAssistSubjectControl, int documentOffset) { this.fExpression = contentAssistSubjectControl.getDocument().get(); this.fDocumentOffset = documentOffset; this.fPriorityProposals = new ArrayList(); this.fProposals = new ArrayList(); boolean isEscape = false; esc: for (int i = documentOffset - 1; i >= 0; i--) { if (this.fExpression.charAt(i) == '\\') { isEscape = !isEscape; } else { break esc; } } this.fIsEscape = isEscape; } /** * Computes applicable proposals for the find field. * * @return the proposals */ public ICompletionProposal[] computeFindProposals() { // // characters // this.addBsProposal("\\\\", RegExMessages.displayString_bs_bs, RegExMessages.additionalInfo_bs_bs); //$NON-NLS-1$ // this.addBracketProposal("\\0", 2, RegExMessages.displayString_bs_0, RegExMessages.additionalInfo_bs_0); //$NON-NLS-1$ // this.addBracketProposal("\\x", 2, RegExMessages.displayString_bs_x, RegExMessages.additionalInfo_bs_x); //$NON-NLS-1$ // this.addBracketProposal("\\u", 2, RegExMessages.displayString_bs_u, RegExMessages.additionalInfo_bs_u); //$NON-NLS-1$ this.addBsProposal("\\t", RegExMessages.displayString_bs_t, RegExMessages.additionalInfo_bs_t); //$NON-NLS-1$ this.addBsProposal("\\v", RegExMessages.displayString_bs_v, RegExMessages.additionalInfo_bs_v); //$NON-NLS-1$ this.addBsProposal("\\n", RegExMessages.displayString_bs_n, RegExMessages.additionalInfo_bs_n); //$NON-NLS-1$ this.addBsProposal("\\r", RegExMessages.displayString_bs_r, RegExMessages.additionalInfo_bs_r); //$NON-NLS-1$ this.addBsProposal("\\f", RegExMessages.displayString_bs_f, RegExMessages.additionalInfo_bs_f); //$NON-NLS-1$ // this.addBsProposal("\\a", RegExMessages.displayString_bs_a, RegExMessages.additionalInfo_bs_a); //$NON-NLS-1$ // this.addBsProposal("\\e", RegExMessages.displayString_bs_e, RegExMessages.additionalInfo_bs_e); //$NON-NLS-1$ // this.addBsProposal("\\c", RegExMessages.displayString_bs_c, RegExMessages.additionalInfo_bs_c); //$NON-NLS-1$ // if (!this.fIsEscape) { this.addBracketProposal(".", 1, RegExMessages.displayString_dot, RegExMessages.additionalInfo_dot); //$NON-NLS-1$ } this.addBsProposal("\\d", RegExMessages.displayString_bs_d, RegExMessages.additionalInfo_bs_d); //$NON-NLS-1$ this.addBsProposal("\\D", RegExMessages.displayString_bs_D, RegExMessages.additionalInfo_bs_D); //$NON-NLS-1$ this.addBsProposal("\\s", RegExMessages.displayString_bs_s, RegExMessages.additionalInfo_bs_s); //$NON-NLS-1$ this.addBsProposal("\\S", RegExMessages.displayString_bs_S, RegExMessages.additionalInfo_bs_S); //$NON-NLS-1$ this.addBsProposal("\\w", RegExMessages.displayString_bs_w, RegExMessages.additionalInfo_bs_w); //$NON-NLS-1$ this.addBsProposal("\\W", RegExMessages.displayString_bs_W, RegExMessages.additionalInfo_bs_W); //$NON-NLS-1$ // // // backreference // this.addBsProposal("\\", RegExMessages.displayString_bs_i, RegExMessages.additionalInfo_bs_i); //$NON-NLS-1$ // // // quoting // this.addBsProposal("\\", RegExMessages.displayString_bs, RegExMessages.additionalInfo_bs); //$NON-NLS-1$ // this.addBsProposal("\\Q", RegExMessages.displayString_bs_Q, RegExMessages.additionalInfo_bs_Q); //$NON-NLS-1$ // this.addBsProposal("\\E", RegExMessages.displayString_bs_E, RegExMessages.additionalInfo_bs_E); //$NON-NLS-1$ // // character sets if (!this.fIsEscape) { this.addBracketProposal("[]", 1, RegExMessages.displayString_set, RegExMessages.additionalInfo_set); //$NON-NLS-1$ this.addBracketProposal( "[^]", 2, RegExMessages.displayString_setExcl, RegExMessages.additionalInfo_setExcl); //$NON-NLS-1$ this.addBracketProposal( "[-]", 1, RegExMessages.displayString_setRange, RegExMessages.additionalInfo_setRange); //$NON-NLS-1$ // this.addProposal("&&", RegExMessages.displayString_setInter, RegExMessages.additionalInfo_setInter); //$NON-NLS-1$ } // if (!this.fIsEscape && (this.fDocumentOffset > 0) // && (this.fExpression.charAt(this.fDocumentOffset - 1) == '\\')) // { // this.addProposal("\\p{}", 3, RegExMessages.displayString_posix, RegExMessages.additionalInfo_posix); //$NON-NLS-1$ // this.addProposal( // "\\P{}", 3, RegExMessages.displayString_posixNot, RegExMessages.additionalInfo_posixNot); //$NON-NLS-1$ // } // else // { // this.addBracketProposal( // "\\p{}", 3, RegExMessages.displayString_posix, RegExMessages.additionalInfo_posix); //$NON-NLS-1$ // this.addBracketProposal( // "\\P{}", 3, RegExMessages.displayString_posixNot, RegExMessages.additionalInfo_posixNot); //$NON-NLS-1$ // } // // // boundary matchers if (this.fDocumentOffset == 0) { this.addPriorityProposal("^", RegExMessages.displayString_start, RegExMessages.additionalInfo_start); //$NON-NLS-1$ } else if ((this.fDocumentOffset == 1) && (this.fExpression.charAt(0) == '^')) { this.addBracketProposal("^", 1, RegExMessages.displayString_start, RegExMessages.additionalInfo_start); //$NON-NLS-1$ } if (this.fDocumentOffset == this.fExpression.length()) { this.addProposal("$", RegExMessages.displayString_end, RegExMessages.additionalInfo_end); //$NON-NLS-1$ } this.addBsProposal("\\b", RegExMessages.displayString_bs_b, RegExMessages.additionalInfo_bs_b); //$NON-NLS-1$ this.addBsProposal("\\B", RegExMessages.displayString_bs_B, RegExMessages.additionalInfo_bs_B); //$NON-NLS-1$ this.addBsProposal("\\A", RegExMessages.displayString_bs_A, RegExMessages.additionalInfo_bs_A); //$NON-NLS-1$ // this.addBsProposal("\\G", RegExMessages.displayString_bs_G, RegExMessages.additionalInfo_bs_G); //$NON-NLS-1$ this.addBsProposal("\\Z", RegExMessages.displayString_bs_Z, RegExMessages.additionalInfo_bs_Z); //$NON-NLS-1$ // this.addBsProposal("\\z", RegExMessages.displayString_bs_z, RegExMessages.additionalInfo_bs_z); //$NON-NLS-1$ if (!this.fIsEscape) { // capturing groups this.addBracketProposal("()", 1, RegExMessages.displayString_group, RegExMessages.additionalInfo_group); //$NON-NLS-1$ // flags // this.addBracketProposal("(?)", 2, RegExMessages.displayString_flag, RegExMessages.additionalInfo_flag); //$NON-NLS-1$ // this.addBracketProposal( // "(?:)", 3, RegExMessages.displayString_flagExpr, RegExMessages.additionalInfo_flagExpr); //$NON-NLS-1$ // noncapturing group this.addBracketProposal( "(?:)", 3, RegExMessages.displayString_nonCap, RegExMessages.additionalInfo_nonCap); //$NON-NLS-1$ // this.addBracketProposal( // "(?>)", 3, RegExMessages.displayString_atomicCap, RegExMessages.additionalInfo_atomicCap); //$NON-NLS-1$ // lookaraound this.addBracketProposal( "(?=)", 3, RegExMessages.displayString_posLookahead, RegExMessages.additionalInfo_posLookahead); //$NON-NLS-1$ this.addBracketProposal( "(?!)", 3, RegExMessages.displayString_negLookahead, RegExMessages.additionalInfo_negLookahead); //$NON-NLS-1$ // this // .addBracketProposal( // "(?<=)", 4, RegExMessages.displayString_posLookbehind, RegExMessages.additionalInfo_posLookbehind); //$NON-NLS-1$ // this // .addBracketProposal( // "(?<!)", 4, RegExMessages.displayString_negLookbehind, RegExMessages.additionalInfo_negLookbehind); //$NON-NLS-1$ // // // greedy quantifiers this.addBracketProposal("?", 1, RegExMessages.displayString_quest, RegExMessages.additionalInfo_quest); //$NON-NLS-1$ this.addBracketProposal("*", 1, RegExMessages.displayString_star, RegExMessages.additionalInfo_star); //$NON-NLS-1$ this.addBracketProposal("+", 1, RegExMessages.displayString_plus, RegExMessages.additionalInfo_plus); //$NON-NLS-1$ this.addBracketProposal("{}", 1, RegExMessages.displayString_exact, RegExMessages.additionalInfo_exact); //$NON-NLS-1$ this .addBracketProposal( "{,}", 1, RegExMessages.displayString_least, RegExMessages.additionalInfo_least); //$NON-NLS-1$ this .addBracketProposal( "{,}", 1, RegExMessages.displayString_count, RegExMessages.additionalInfo_count); //$NON-NLS-1$ this .addBracketProposal( "{.}", 2, RegExMessages.displayString_zeromore, RegExMessages.additionalInfo_zeromore); //$NON-NLS-1$ // // lazy quantifiers // this.addBracketProposal( // "??", 1, RegExMessages.displayString_questLazy, RegExMessages.additionalInfo_questLazy); //$NON-NLS-1$ // this.addBracketProposal( // "*?", 1, RegExMessages.displayString_starLazy, RegExMessages.additionalInfo_starLazy); //$NON-NLS-1$ // this.addBracketProposal( // "+?", 1, RegExMessages.displayString_plusLazy, RegExMessages.additionalInfo_plusLazy); //$NON-NLS-1$ // this.addBracketProposal( // "{}?", 1, RegExMessages.displayString_exactLazy, RegExMessages.additionalInfo_exactLazy); //$NON-NLS-1$ // this.addBracketProposal( // "{,}?", 1, RegExMessages.displayString_leastLazy, RegExMessages.additionalInfo_leastLazy); //$NON-NLS-1$ // this.addBracketProposal( // "{,}?", 1, RegExMessages.displayString_countLazy, RegExMessages.additionalInfo_countLazy); //$NON-NLS-1$ // // // possessive quantifiers // this.addBracketProposal( // "?+", 1, RegExMessages.displayString_questPoss, RegExMessages.additionalInfo_questPoss); //$NON-NLS-1$ // this.addBracketProposal( // "*+", 1, RegExMessages.displayString_starPoss, RegExMessages.additionalInfo_starPoss); //$NON-NLS-1$ // this.addBracketProposal( // "++", 1, RegExMessages.displayString_plusPoss, RegExMessages.additionalInfo_plusPoss); //$NON-NLS-1$ // this.addBracketProposal( // "{}+", 1, RegExMessages.displayString_exactPoss, RegExMessages.additionalInfo_exactPoss); //$NON-NLS-1$ // this.addBracketProposal( // "{,}+", 1, RegExMessages.displayString_leastPoss, RegExMessages.additionalInfo_leastPoss); //$NON-NLS-1$ // this.addBracketProposal( // "{,}+", 1, RegExMessages.displayString_countPoss, RegExMessages.additionalInfo_countPoss); //$NON-NLS-1$ // alternative this.addBracketProposal("|", 1, RegExMessages.displayString_alt, RegExMessages.additionalInfo_alt); //$NON-NLS-1$ } this.fPriorityProposals.addAll(this.fProposals); return (ICompletionProposal[]) this.fPriorityProposals.toArray(new ICompletionProposal[this.fProposals .size()]); } /** * Computes applicable proposals for the replace field. * * @return the proposals */ public ICompletionProposal[] computeReplaceProposals() { if ((this.fDocumentOffset > 0) && ('$' == this.fExpression.charAt(this.fDocumentOffset - 1))) { this.addProposal("", RegExMessages.displayString_dollar, RegExMessages.additionalInfo_dollar); //$NON-NLS-1$ } else { this.addProposal("$", RegExMessages.displayString_dollar, RegExMessages.additionalInfo_dollar); //$NON-NLS-1$ this.addBsProposal( "\\", RegExMessages.displayString_replace_bs, RegExMessages.additionalInfo_replace_bs); //$NON-NLS-1$ this.addProposal("\t", RegExMessages.displayString_tab, RegExMessages.additionalInfo_tab); //$NON-NLS-1$ } return (ICompletionProposal[]) this.fProposals.toArray(new ICompletionProposal[this.fProposals.size()]); } /** * Adds a proposal. * * @param proposal * the string to be inserted * @param displayString * the proposal's label * @param additionalInfo * the additional information */ private void addProposal(String proposal, String displayString, String additionalInfo) { this.fProposals.add(new CompletionProposal(proposal, this.fDocumentOffset, 0, proposal.length(), null, displayString, null, additionalInfo)); } /** * Adds a proposal. * * @param proposal * the string to be inserted * @param cursorPosition * the cursor position after insertion, relative to the start of the proposal * @param displayString * the proposal's label * @param additionalInfo * the additional information */ private void addProposal(String proposal, int cursorPosition, String displayString, String additionalInfo) { this.fProposals.add(new CompletionProposal(proposal, this.fDocumentOffset, 0, cursorPosition, null, displayString, null, additionalInfo)); } /** * Adds a proposal to the priority proposals list. * * @param proposal * the string to be inserted * @param displayString * the proposal's label * @param additionalInfo * the additional information */ private void addPriorityProposal(String proposal, String displayString, String additionalInfo) { this.fPriorityProposals.add(new CompletionProposal(proposal, this.fDocumentOffset, 0, proposal.length(), null, displayString, null, additionalInfo)); } /** * Adds a proposal. Ensures that existing pre- and postfixes are not duplicated. * * @param proposal * the string to be inserted * @param cursorPosition * the cursor position after insertion, relative to the start of the proposal * @param displayString * the proposal's label * @param additionalInfo * the additional information */ private void addBracketProposal(String proposal, int cursorPosition, String displayString, String additionalInfo) { String prolog = this.fExpression.substring(0, this.fDocumentOffset); if (!this.fIsEscape && prolog.endsWith("\\") && proposal.startsWith("\\")) { //$NON-NLS-1$//$NON-NLS-2$ this.fProposals.add(new CompletionProposal(proposal, this.fDocumentOffset, 0, cursorPosition, null, displayString, null, additionalInfo)); return; } for (int i = 1; i <= cursorPosition; i++) { String prefix = proposal.substring(0, i); if (prolog.endsWith(prefix)) { String postfix = proposal.substring(cursorPosition); String epilog = this.fExpression.substring(this.fDocumentOffset); if (epilog.startsWith(postfix)) { this.fPriorityProposals .add(new CompletionProposal(proposal.substring(i, cursorPosition), this.fDocumentOffset, 0, cursorPosition - i, null, displayString, null, additionalInfo)); } else { this.fPriorityProposals.add(new CompletionProposal(proposal.substring(i), this.fDocumentOffset, 0, cursorPosition - i, null, displayString, null, additionalInfo)); } return; } } this.fProposals.add(new CompletionProposal(proposal, this.fDocumentOffset, 0, cursorPosition, null, displayString, null, additionalInfo)); } /** * Adds a proposal that starts with a backslash. * * @param proposal * the string to be inserted * @param displayString * the proposal's label * @param additionalInfo * the additional information */ private void addBsProposal(String proposal, String displayString, String additionalInfo) { if (this.fIsEscape) { this.fPriorityProposals.add(new CompletionProposal(proposal.substring(1), this.fDocumentOffset, 0, proposal.length() - 1, null, displayString, null, additionalInfo)); } else { this.addProposal(proposal, displayString, additionalInfo); } } } /** * The context information validator. */ private IContextInformationValidator fValidator = new SubjectControlContextInformationValidator(this); /** * <code>true</code> iff the processor is for the find field. <code>false</code> iff the processor is for the * replace field. */ private final boolean fIsFind; /** * Create content assistant * @return content assistant */ public static SubjectControlContentAssistant createContentAssistant() { final SubjectControlContentAssistant contentAssistant = new SubjectControlContentAssistant(); contentAssistant.setRestoreCompletionProposalSize(Activator.getDefault().getDialogSettings()); IContentAssistProcessor processor = new LoggingRegExContentAssistProcessor(true); contentAssistant.setContentAssistProcessor(processor, IDocument.DEFAULT_CONTENT_TYPE); contentAssistant.setContextInformationPopupOrientation(IContentAssistant.CONTEXT_INFO_ABOVE); contentAssistant.setInformationControlCreator(new IInformationControlCreator() { /* * @see org.eclipse.jface.text.IInformationControlCreator#createInformationControl(org.eclipse.swt.widgets.Shell) */ public IInformationControl createInformationControl(Shell parent) { return new DefaultInformationControl(parent); } }); return contentAssistant; } /** * @param isFind */ public LoggingRegExContentAssistProcessor(boolean isFind) { this.fIsFind = isFind; } /* * @see IContentAssistProcessor#computeCompletionProposals(ITextViewer, int) */ /** * @see org.eclipse.jface.text.contentassist.IContentAssistProcessor#computeCompletionProposals(org.eclipse.jface.text.ITextViewer, int) */ public ICompletionProposal[] computeCompletionProposals(ITextViewer viewer, int documentOffset) { throw new UnsupportedOperationException("ITextViewer not supported"); //$NON-NLS-1$ } /* * @see IContentAssistProcessor#computeContextInformation(ITextViewer, int) */ /** * @see org.eclipse.jface.text.contentassist.IContentAssistProcessor#computeContextInformation(org.eclipse.jface.text.ITextViewer, int) */ public IContextInformation[] computeContextInformation(ITextViewer viewer, int documentOffset) { throw new UnsupportedOperationException("ITextViewer not supported"); //$NON-NLS-1$ } /* * @see IContentAssistProcessor#getCompletionProposalAutoActivationCharacters() */ /** * @see org.eclipse.jface.text.contentassist.IContentAssistProcessor#getCompletionProposalAutoActivationCharacters() */ public char[] getCompletionProposalAutoActivationCharacters() { if (this.fIsFind) { return new char[] { '\\', '[', '(' }; } return new char[] { '$' }; } /* * @see IContentAssistProcessor#getContextInformationAutoActivationCharacters() */ /** * @see org.eclipse.jface.text.contentassist.IContentAssistProcessor#getContextInformationAutoActivationCharacters() */ public char[] getContextInformationAutoActivationCharacters() { return new char[] {}; } /* * @see IContentAssistProcessor#getContextInformationValidator() */ /** * @see org.eclipse.jface.text.contentassist.IContentAssistProcessor#getContextInformationValidator() */ public IContextInformationValidator getContextInformationValidator() { return this.fValidator; } /* * @see IContentAssistProcessor#getErrorMessage() */ /** * @see org.eclipse.jface.text.contentassist.IContentAssistProcessor#getErrorMessage() */ public String getErrorMessage() { return null; } /* * @see ISubjectControlContentAssistProcessor#computeCompletionProposals(IContentAssistSubjectControl, int) */ /** * @see org.eclipse.jface.contentassist.ISubjectControlContentAssistProcessor#computeCompletionProposals(org.eclipse.jface.contentassist.IContentAssistSubjectControl, int) */ public ICompletionProposal[] computeCompletionProposals(IContentAssistSubjectControl contentAssistSubjectControl, int documentOffset) { if (this.fIsFind) { return new ProposalComputer(contentAssistSubjectControl, documentOffset).computeFindProposals(); } return new ProposalComputer(contentAssistSubjectControl, documentOffset).computeReplaceProposals(); } /* * @see ISubjectControlContentAssistProcessor#computeContextInformation(IContentAssistSubjectControl, int) */ /** * @see org.eclipse.jface.contentassist.ISubjectControlContentAssistProcessor#computeContextInformation(org.eclipse.jface.contentassist.IContentAssistSubjectControl, int) */ public IContextInformation[] computeContextInformation(IContentAssistSubjectControl contentAssistSubjectControl, int documentOffset) { return null; } }