/******************************************************************************* * Copyright (c) 2005, 2009 Spring IDE Developers * 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: * Spring IDE Developers - initial API and implementation *******************************************************************************/ package org.springframework.ide.eclipse.beans.ui.editor.contentassist; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.NullProgressMonitor; import org.eclipse.jface.text.BadLocationException; import org.eclipse.jface.text.DocumentEvent; import org.eclipse.jface.text.IDocument; import org.eclipse.jface.text.ITextViewer; import org.eclipse.jface.text.ITextViewerExtension5; import org.eclipse.jface.text.contentassist.CompletionProposal; import org.eclipse.jface.text.contentassist.ICompletionProposal; import org.eclipse.jface.text.contentassist.ICompletionProposalExtension; import org.eclipse.jface.text.contentassist.ICompletionProposalExtension2; import org.eclipse.jface.text.contentassist.ICompletionProposalExtension4; import org.eclipse.jface.text.contentassist.ICompletionProposalExtension5; import org.eclipse.jface.text.contentassist.IContextInformation; import org.eclipse.swt.graphics.Image; import org.eclipse.swt.graphics.Point; import org.eclipse.wst.sse.core.internal.util.Debug; import org.eclipse.wst.sse.ui.internal.contentassist.IRelevanceCompletionProposal; import org.eclipse.wst.sse.ui.internal.contentassist.IRelevanceConstants; import org.springframework.ide.eclipse.beans.ui.editor.util.BeansEditorUtils; /** * An implementation of ICompletionProposal whose values can be read after * creation. * @author Christian Dupuis */ @SuppressWarnings("restriction") public class BeansJavaCompletionProposal implements ICompletionProposal, ICompletionProposalExtension, ICompletionProposalExtension2, ICompletionProposalExtension4, ICompletionProposalExtension5, IRelevanceCompletionProposal { private IContextInformation fContextInformation; private int fCursorPosition = 0; private String fDisplayString; private Image fImage; private int fOriginalReplacementLength; private int fRelevance = IRelevanceConstants.R_NONE; private int fReplacementLength = 0; private int fReplacementOffset = 0; private String fReplacementString = null; private char[] fTriggers; private boolean fUpdateLengthOnValidate; private Object proposedObject = null; /** * Constructor with relevance and replacement length update flag. If the * <code>updateReplacementLengthOnValidate</code> flag is true, then when * the user types, the replacement length will be incremented by the number * of new characters inserted from the original position. Otherwise the * replacement length will not change on validate. ex. <tag |name="attr"> - * the replacement length is 4 <tag i|name="attr"> - the replacement length * is now 5 <tag id|name="attr"> - the replacement length is now 6 <tag * |name="attr"> - the replacementlength is now 4 again <tag |name="attr"> - * the replacment length remains 4 */ public BeansJavaCompletionProposal(String replacementString, int replacementOffset, int replacementLength, int cursorPosition, Image image, String displayString, IContextInformation contextInformation, int relevance, boolean updateReplacementLengthOnValidate, Object proposedObject) { fReplacementString = "\"" + replacementString; fReplacementOffset = replacementOffset; fReplacementLength = replacementLength; fCursorPosition = cursorPosition + 1; fImage = image; fDisplayString = displayString; fContextInformation = contextInformation; fRelevance = relevance; fUpdateLengthOnValidate = updateReplacementLengthOnValidate; fOriginalReplacementLength = fReplacementLength; this.proposedObject = proposedObject; } public BeansJavaCompletionProposal(String replacementString, int replacementOffset, int replacementLength, int cursorPosition, Image image, String displayString, IContextInformation contextInformation, int relevance, Object proposedObject) { this(replacementString, replacementOffset, replacementLength, cursorPosition, image, displayString, contextInformation, relevance, true, proposedObject); } public void apply(IDocument document) { String charBeforeCursor = ""; try { charBeforeCursor = document.get(getReplacementOffset() - 1, 1); } catch (BadLocationException e) { // do nothinh } if ("\"".equals(charBeforeCursor)) { CompletionProposal proposal = new CompletionProposal( getReplacementString(), getReplacementOffset() - 1, getReplacementLength(), getCursorPosition() + 1, getImage(), getDisplayString(), getContextInformation(), getAdditionalProposalInfo()); proposal.apply(document); } else { CompletionProposal proposal = new CompletionProposal( getReplacementString() + "\"", getReplacementOffset(), getReplacementLength(), getCursorPosition() + 1, getImage(), getDisplayString(), getContextInformation(), getAdditionalProposalInfo()); proposal.apply(document); } } /* * (non-Javadoc) * @see org.eclipse.jface.text.contentassist.ICompletionProposalExtension#apply(org.eclipse.jface.text.IDocument, * char, int) */ public void apply(IDocument document, char trigger, int offset) { try { String charBeforeCursor = document.get(getReplacementOffset(), 1); } catch (BadLocationException e) { } CompletionProposal proposal = new CompletionProposal( getReplacementString() + "\"", getReplacementOffset(), getReplacementLength(), getCursorPosition() + 1, getImage(), getDisplayString(), getContextInformation(), getAdditionalProposalInfo()); proposal.apply(document); // we want to ContextInformationPresenter.updatePresentation() here } public void apply(ITextViewer viewer, char trigger, int stateMask, int offset) { IDocument document = viewer.getDocument(); // CMVC 252634 to compensate for "invisible" initial region int caretOffset = viewer.getTextWidget().getCaretOffset(); if (viewer instanceof ITextViewerExtension5) { ITextViewerExtension5 extension = (ITextViewerExtension5) viewer; caretOffset = extension.widgetOffset2ModelOffset(caretOffset); } else { caretOffset = viewer.getTextWidget().getCaretOffset() + viewer.getVisibleRegion().getOffset(); } if (caretOffset == getReplacementOffset()) { apply(document); } else { // replace the text without affecting the caret Position as this // causes the cursor to move on its own try { int endOffsetOfChanges = getReplacementString().length() + getReplacementOffset(); // Insert the portion of the new text that comes after the // current caret position if (endOffsetOfChanges >= caretOffset) { int postCaretReplacementLength = getReplacementOffset() + getReplacementLength() - caretOffset; int preCaretReplacementLength = getReplacementString() .length() - (endOffsetOfChanges - caretOffset); if (postCaretReplacementLength < 0) { if (Debug.displayWarnings) { System.out .println("** postCaretReplacementLength was negative: " + postCaretReplacementLength); //$NON-NLS-1$ } // This is just a quick fix while I figure out what // replacement length is supposed to be // in each case, otherwise we'll get negative // replacment length sometimes postCaretReplacementLength = 0; } String charAfterCursor = document.get(caretOffset, 1); if ("\"".equals(charAfterCursor)) { document .replace( caretOffset, ((postCaretReplacementLength - 1) < 0 ? postCaretReplacementLength : postCaretReplacementLength - 1), getReplacementString().substring( preCaretReplacementLength)); } else { document.replace(caretOffset, postCaretReplacementLength, getReplacementString().substring( preCaretReplacementLength) + "\""); } } // Insert the portion of the new text that comes before the // current caret position // Done second since offsets would change for the post text // otherwise // Outright insertions are handled here if (caretOffset > getReplacementOffset()) { int preCaretTextLength = caretOffset - getReplacementOffset(); document.replace(getReplacementOffset(), preCaretTextLength, getReplacementString() .substring(0, preCaretTextLength)); } } catch (BadLocationException x) { apply(document); } catch (StringIndexOutOfBoundsException e) { apply(document); } } } public String getAdditionalProposalInfo() { return BeansEditorUtils.createAdditionalProposalInfo( getProposedObject(), new NullProgressMonitor()); } public Object getAdditionalProposalInfo(IProgressMonitor monitor) { return BeansEditorUtils.createAdditionalProposalInfo( getProposedObject(), monitor); } public IContextInformation getContextInformation() { return fContextInformation; } public int getContextInformationPosition() { return getCursorPosition(); } public int getCursorPosition() { return fCursorPosition; } public String getDisplayString() { return fDisplayString; } public Image getImage() { return fImage; } public Object getProposedObject() { return proposedObject; } public int getRelevance() { return fRelevance; } public int getReplacementLength() { return fReplacementLength; } public int getReplacementOffset() { return fReplacementOffset; } public String getReplacementString() { return fReplacementString; } public Point getSelection(IDocument document) { CompletionProposal proposal = new CompletionProposal( getReplacementString(), getReplacementOffset(), getReplacementLength(), getCursorPosition(), getImage(), getDisplayString(), getContextInformation(), null); return proposal.getSelection(document); } /** * @see org.eclipse.jface.text.contentassist.ICompletionProposalExtension#getTriggerCharacters() */ public char[] getTriggerCharacters() { return fTriggers; } public boolean isAutoInsertable() { return true; } public boolean isValidFor(IDocument document, int offset) { return validate(document, offset, null); } public void selected(ITextViewer viewer, boolean smartToggle) { } public void setContextInformation(IContextInformation contextInfo) { fContextInformation = contextInfo; } public void setCursorPosition(int pos) { fCursorPosition = pos; } public void setRelevance(int relevance) { fRelevance = relevance; } public void setReplacementOffset(int replacementOffset) { fReplacementOffset = replacementOffset; } public void setReplacementString(String replacementString) { fReplacementString = replacementString; } public void setTriggerCharacters(char[] triggers) { fTriggers = triggers; } // code is borrowed from JavaCompletionProposal protected boolean startsWith(IDocument document, int offset, String word) { int wordLength = word == null ? 0 : word.length(); if (offset > fReplacementOffset + wordLength) return false; try { int length = offset - fReplacementOffset; String start = document.get(fReplacementOffset, length); // Remove " for comparison start = start.replaceAll("\"", ""); String wordTemp = word.replaceAll("\"", "").substring(0, start.length()); return wordTemp.equalsIgnoreCase(start); } catch (BadLocationException x) { } return false; } public void unselected(ITextViewer viewer) { } public boolean validate(IDocument document, int offset, DocumentEvent event) { if (offset < fReplacementOffset) return false; boolean validated = startsWith(document, offset, fReplacementString); boolean validatedClass = startsWith(document, offset, fDisplayString); // CMVC 269884 if (fUpdateLengthOnValidate) { int newLength = offset - getReplacementOffset(); int delta = newLength - fOriginalReplacementLength; fReplacementLength = delta + fOriginalReplacementLength; } return validated || validatedClass; } }