/******************************************************************************* * Copyright (c) 2001, 2012 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 * Jens Lukowski/Innoopract - initial renaming/restructuring * *******************************************************************************/ package org.eclipse.wst.xml.ui.internal.autoedit; import org.eclipse.jface.text.BadLocationException; import org.eclipse.jface.text.DocumentCommand; import org.eclipse.jface.text.IDocument; import org.eclipse.wst.sse.core.StructuredModelManager; import org.eclipse.wst.sse.core.internal.provisional.IStructuredModel; import org.eclipse.wst.sse.core.internal.provisional.text.IStructuredDocumentRegion; import org.eclipse.wst.sse.core.internal.provisional.text.ITextRegion; import org.eclipse.wst.sse.ui.internal.text.StructuredAutoEditStrategy; import org.eclipse.wst.xml.core.internal.provisional.document.IDOMElement; import org.eclipse.wst.xml.core.internal.provisional.document.IDOMNode; import org.eclipse.wst.xml.core.internal.regions.DOMRegionContext; import org.eclipse.wst.xml.ui.internal.Logger; import org.eclipse.wst.xml.ui.internal.XMLUIPlugin; import org.eclipse.wst.xml.ui.internal.preferences.XMLUIPreferenceNames; import org.w3c.dom.Node; /** * Automatically inserts closing comment tag or end tag when appropriate. */ public class StructuredAutoEditStrategyXML extends StructuredAutoEditStrategy { /* * NOTE: copies of this class exists in * org.eclipse.wst.xml.ui.internal.autoedit * org.eclipse.wst.html.ui.internal.autoedit */ public void customizeDocumentCommand(IDocument document, DocumentCommand command) { if (!supportsSmartInsert(document)) { return; } IStructuredModel model = null; try { model = StructuredModelManager.getModelManager().getExistingModelForRead(document); if (model != null) { if (command.text != null) { smartInsertCloseElement(command, document, model); smartInsertForComment(command, document, model); smartInsertForEndTag(command, document, model); smartRemoveEndTag(command, document, model); } } } finally { if (model != null) { model.releaseFromRead(); } } } private boolean isPreferenceEnabled(String key) { return (key != null && XMLUIPlugin.getDefault().getPreferenceStore().getBoolean(key)); } private boolean isCommentNode(IDOMNode node) { return ((node != null) && (node instanceof IDOMElement) && ((IDOMElement) node).isCommentTag()); } private boolean isDocumentNode(IDOMNode node) { return ((node != null) && (node.getNodeType() == Node.DOCUMENT_NODE)); } /** * Attempts to clean up an end-tag if a start-tag is converted into an empty-element * tag (e.g., <node />) and the original element was empty. * * @param command the document command describing the change * @param document the document that will be changed * @param model the model based on the document */ private void smartRemoveEndTag(DocumentCommand command, IDocument document, IStructuredModel model) { try { // An opening tag is now a self-terminated end-tag if ("/".equals(command.text) && ">".equals(document.get(command.offset, 1)) && command.length == 0 && isPreferenceEnabled(XMLUIPreferenceNames.TYPING_REMOVE_END_TAGS)) { //$NON-NLS-1$ //$NON-NLS-2$ IDOMNode node = (IDOMNode) model.getIndexedRegion(command.offset); if (node != null && !node.hasChildNodes()) { IStructuredDocumentRegion region = node.getFirstStructuredDocumentRegion(); if(region.getFirstRegion().getType() == DOMRegionContext.XML_TAG_OPEN && command.offset <= region.getEnd()) { /* if the region before the command offset is a an attribute value region * check to see if it has both and opening and closing quote */ ITextRegion prevTextRegion = region.getRegionAtCharacterOffset(command.offset-1); boolean inUnclosedAttValueRegion = false; if(prevTextRegion.getType() == DOMRegionContext.XML_TAG_ATTRIBUTE_VALUE) { //get the text of the attribute value region String prevText = region.getText(prevTextRegion); inUnclosedAttValueRegion = (prevText.startsWith("'") && ((prevText.length() == 1) || !prevText.endsWith("'"))) || (prevText.startsWith("\"") && ((prevText.length() == 1) || !prevText.endsWith("\""))); if (!inUnclosedAttValueRegion) { // Check if action is taking place within the paired quotes. This means quotes are actually mismatched and attribute is not properly closed inUnclosedAttValueRegion = prevTextRegion == region.getRegionAtCharacterOffset(command.offset); } } //if command offset is in an unclosed attribute value region then done remove the end tag if(!inUnclosedAttValueRegion) { region = node.getEndStructuredDocumentRegion(); if (region != null && region.isEnded()) { document.replace(region.getStartOffset(), region.getLength(), ""); //$NON-NLS-1$ } } } } } } catch (BadLocationException e) { Logger.logException(e); } } private void smartInsertForComment(DocumentCommand command, IDocument document, IStructuredModel model) { try { if (command.text.equals("-") && (document.getLength() >= 3) && document.get(command.offset - 3, 3).equals("<!-") && isPreferenceEnabled(XMLUIPreferenceNames.TYPING_COMPLETE_COMMENTS)) { //$NON-NLS-1$ //$NON-NLS-2$ command.text += " -->"; //$NON-NLS-1$ command.shiftsCaret = false; command.caretOffset = command.offset + 2; command.doit = false; } } catch (BadLocationException e) { Logger.logException(e); } } /** * Attempts to insert the end tag when completing a start-tag with the '>' character. * * @param command * @param document * @param model */ private void smartInsertCloseElement(DocumentCommand command, IDocument document, IStructuredModel model) { try { // Check terminating start tag, but ignore empty-element tags if (command.text.equals(">") && document.getLength() > 0 && document.getChar(command.offset - 1) != '/' && isPreferenceEnabled(XMLUIPreferenceNames.TYPING_COMPLETE_ELEMENTS)) { //$NON-NLS-1$ IDOMNode node = (IDOMNode) model.getIndexedRegion(command.offset - 1); boolean isClosedByParent = false; // Only insert an end-tag if necessary. Because of the way the document is parsed, it is possible for a child tag with the same // name as an ancestor to be paired with the end-tag of an ancestor, so the ancestors must be checked for an unclosed tag. if (node != null && node.getNodeType() == Node.ELEMENT_NODE && (!node.isClosed() || (isClosedByParent = hasUnclosedAncestor(node)))) { IStructuredDocumentRegion region = node.getEndStructuredDocumentRegion(); if (region != null && region.getRegions().size() > 0 && region.getRegions().get(0).getType() == DOMRegionContext.XML_END_TAG_OPEN && !isClosedByParent) return; region = node.getStartStructuredDocumentRegion(); if (region != null) { ITextRegion textRegion = region.getRegionAtCharacterOffset(command.offset - 1); if (textRegion != null && DOMRegionContext.XML_TAG_ATTRIBUTE_VALUE.equals(textRegion.getType())) { // Check that the command is not within a quoted attribute value final String text = region.getText(textRegion); final char first = text.charAt(0); if (first == '\'' || first == '"') { if (text.length() < 2) // Just an opening quote return; final char last = text.charAt(text.length() - 1); if (last == first) { // Quote is paired if (command.offset < region.getTextEndOffset(textRegion)) // Command is within the value return; } else // Unpaired quote return; } } } command.text += "</" + getElementName(node, command.offset) + ">"; //$NON-NLS-1$ //$NON-NLS-2$ command.shiftsCaret = false; command.caretOffset = command.offset + 1; } } } catch (BadLocationException e) { e.printStackTrace(); } } /** * Get the element name that will be created by closing the start tag. Defaults * to the node's nodeName. * @param node the node that is being edited * @param offset the offset in the document where the start tag is closed * @return The element name of the tag */ private String getElementName(IDOMNode node, int offset) { String name = null; IStructuredDocumentRegion region = node.getFirstStructuredDocumentRegion(); ITextRegion textRegion = region.getRegionAtCharacterOffset(offset); if (textRegion != null && textRegion.getType() == DOMRegionContext.XML_TAG_NAME) { int nameStart = region.getStartOffset(textRegion); String regionText = region.getText(textRegion); int length = offset - nameStart; if (length <= regionText.length()) name = regionText.substring(0, length); } // Default to the node name if (name == null) name = node.getNodeName(); return name; } private void smartInsertForEndTag(DocumentCommand command, IDocument document, IStructuredModel model) { try { if (command.text.equals("/") && (document.getLength() >= 1) && document.get(command.offset - 1, 1).equals("<") && isPreferenceEnabled(XMLUIPreferenceNames.TYPING_COMPLETE_END_TAGS)) { //$NON-NLS-1$ //$NON-NLS-2$ IDOMNode parentNode = (IDOMNode) ((IDOMNode) model.getIndexedRegion(command.offset - 1)).getParentNode(); if (isCommentNode(parentNode)) { // loop and find non comment node parent while ((parentNode != null) && isCommentNode(parentNode)) { parentNode = (IDOMNode) parentNode.getParentNode(); } } if (!isDocumentNode(parentNode)) { // only add end tag if one does not already exist or if // add '/' does not create one already IStructuredDocumentRegion endTagStructuredDocumentRegion = parentNode.getEndStructuredDocumentRegion(); IDOMNode ancestor = parentNode; boolean smartInsertForEnd = false; if(endTagStructuredDocumentRegion != null) { // Look for ancestors by the same name that are missing end tags while((ancestor = (IDOMNode) ancestor.getParentNode()) != null) { if(ancestor.getEndStructuredDocumentRegion() == null && parentNode.getNodeName().equals(ancestor.getNodeName())) { smartInsertForEnd = true; break; } } } if (endTagStructuredDocumentRegion == null || smartInsertForEnd) { StringBuffer toAdd = new StringBuffer(parentNode.getNodeName()); if (toAdd.length() > 0) { toAdd.append(">"); //$NON-NLS-1$ String suffix = toAdd.toString(); if ((document.getLength() < command.offset + suffix.length()) || (!suffix.equals(document.get(command.offset, suffix.length())))) { command.text += suffix; } } } } } } catch (BadLocationException e) { Logger.logException(e); } } /** * Checks if <code>node</code> has an unclosed ancestor by the same name * * @param node the node to check * @return true if <code>node</code> has an unclosed parent with the same node name */ private boolean hasUnclosedAncestor(IDOMNode node) { IDOMNode parent = (IDOMNode) node.getParentNode(); while (parent != null && parent.getNodeType() != Node.DOCUMENT_NODE && parent.getNodeName().equals(node.getNodeName())) { if (!parent.isClosed()) return true; parent = (IDOMNode) parent.getParentNode(); } return false; } }