blob: 56c928769cd90cff501561c84c900d691e8c9b91 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2004, 2008 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.css.core.internal.formatter;
import org.eclipse.core.runtime.Preferences;
import org.eclipse.jface.text.IRegion;
import org.eclipse.wst.css.core.internal.CSSCorePlugin;
import org.eclipse.wst.css.core.internal.cleanup.CSSCleanupStrategy;
import org.eclipse.wst.css.core.internal.parserz.CSSRegionContexts;
import org.eclipse.wst.css.core.internal.preferences.CSSCorePreferenceNames;
import org.eclipse.wst.css.core.internal.provisional.document.ICSSModel;
import org.eclipse.wst.css.core.internal.provisional.document.ICSSNode;
import org.eclipse.wst.sse.core.internal.provisional.IndexedRegion;
import org.eclipse.wst.sse.core.internal.provisional.text.IStructuredDocument;
import org.eclipse.wst.sse.core.internal.provisional.text.IStructuredDocumentRegion;
import org.eclipse.wst.sse.core.internal.provisional.text.ITextRegion;
/**
*
*/
public class StyleDeclarationFormatter extends DefaultCSSSourceFormatter {
private static StyleDeclarationFormatter instance;
/**
*
*/
StyleDeclarationFormatter() {
super();
}
/**
*
*/
protected void formatBefore(ICSSNode node, ICSSNode child, String toAppend, StringBuffer source, IRegion exceptFor) {
CSSCleanupStrategy stgy = getCleanupStrategy(node);
ICSSNode prev = (child != null) ? child.getPreviousSibling() : node.getLastChild();
int start = (prev != null) ? ((IndexedRegion) prev).getEndOffset() : 0;
int end = (child != null) ? ((IndexedRegion) child).getStartOffset() : 0;
// check no child
if (child == null && prev == null)
return;
if (start > 0 && start < end) { // format source
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration with no
// model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
// get meaning regions
CompoundRegion[] regions = null;
if (exceptFor == null)
regions = getRegionsWithoutWhiteSpaces(structuredDocument, new FormatRegion(start, end - start), stgy);
else {
String pickupType = CSSRegionContexts.CSS_DECLARATION_DELIMITER;
if (prev == null || child == null)
pickupType = null;
regions = getRegions(structuredDocument, new FormatRegion(start, end - start), exceptFor, pickupType);
}
// extract source
for (int i = 0; i < regions.length; i++) {
appendSpaceBefore(node, regions[i], source);
source.append(decoratedRegion(regions[i], 0, stgy)); // must
// be comments
}
}
}
} else if (prev != null && child != null) { // generate source :
// between two declarations
// BUG93037-properties view adds extra ; when add new property
boolean semicolonFound = false;
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration with no
// model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
int prevStart = (prev != null) ? ((IndexedRegion) prev).getStartOffset() : 0;
int prevEnd = (prev != null) ? ((IndexedRegion) prev).getEndOffset() : 0;
CompoundRegion[] regions = getRegionsWithoutWhiteSpaces(structuredDocument, new FormatRegion(prevStart, prevEnd - prevStart), stgy);
int i = regions.length - 1;
while (i >= 0 && !semicolonFound) {
if (regions[i].getType() == CSSRegionContexts.CSS_DECLARATION_DELIMITER)
semicolonFound = true;
--i;
}
}
}
if (!semicolonFound)
source.append(";");//$NON-NLS-1$
} else if (prev == null) { // generate source : before the first
// declaration
org.eclipse.wst.css.core.internal.util.RegionIterator it = null;
if (end > 0) {
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration with
// no model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
it = new org.eclipse.wst.css.core.internal.util.RegionIterator(structuredDocument, end - 1);
}
}
} else {
int pos = getChildInsertPos(node);
if (pos >= 0) {
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration
// with no model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
it = new org.eclipse.wst.css.core.internal.util.RegionIterator(structuredDocument, pos - 1);
}
}
}
}
if (it != null) {
int limit = ((IndexedRegion) ((node.getParentNode() != null) ? node.getParentNode() : node)).getStartOffset();
while (it.hasPrev()) {
ITextRegion curReg = it.prev();
if (curReg.getType() == CSSRegionContexts.CSS_LBRACE || curReg.getType() == CSSRegionContexts.CSS_DECLARATION_DELIMITER)
break;
if (curReg.getType() != CSSRegionContexts.CSS_S && curReg.getType() != CSSRegionContexts.CSS_COMMENT) {
source.append(";");//$NON-NLS-1$
break;
}
if (it.getStructuredDocumentRegion().getStartOffset(curReg) <= limit)
break;
}
}
} else if (child == null) { // generate source : after the last
// declaration
org.eclipse.wst.css.core.internal.util.RegionIterator it = null;
if (start > 0) {
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration with
// no model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
it = new org.eclipse.wst.css.core.internal.util.RegionIterator(structuredDocument, start);
}
}
} else {
int pos = getChildInsertPos(node);
if (pos >= 0) {
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration
// with no model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
it = new org.eclipse.wst.css.core.internal.util.RegionIterator(structuredDocument, pos);
}
}
}
}
if (it != null) {
int limit = ((IndexedRegion) ((node.getParentNode() != null) ? node.getParentNode() : node)).getEndOffset();
while (it.hasNext()) {
ITextRegion curReg = it.next();
if (curReg.getType() == CSSRegionContexts.CSS_RBRACE || curReg.getType() == CSSRegionContexts.CSS_DECLARATION_DELIMITER)
break;
if (curReg.getType() != CSSRegionContexts.CSS_S && curReg.getType() != CSSRegionContexts.CSS_COMMENT) {
// Bug 219004 - Before appending a ;, make sure that there
// isn't one already
boolean semicolonFound = false;
while(it.hasNext() && !semicolonFound) {
if(it.next().getType() == CSSRegionContexts.CSS_DECLARATION_DELIMITER)
semicolonFound = true;
}
if(!semicolonFound)
source.append(";");//$NON-NLS-1$
break;
}
if (limit <= it.getStructuredDocumentRegion().getEndOffset(curReg))
break;
}
}
}
if (child == null) {
if (((IndexedRegion) node).getEndOffset() <= 0) {
// get next region
int pos = getChildInsertPos(node);
CompoundRegion toAppendRegion = null;
if (pos >= 0) {
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration
// with no model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
IStructuredDocumentRegion flatNode = structuredDocument.getRegionAtCharacterOffset(pos);
toAppendRegion = new CompoundRegion(flatNode, flatNode.getRegionAtCharacterOffset(pos));
}
}
}
appendDelimBefore(node.getParentNode(), toAppendRegion, source);
}
} else if ((prev != null || ((IndexedRegion) node).getEndOffset() <= 0)) {
Preferences preferences = CSSCorePlugin.getDefault().getPluginPreferences();
if (preferences.getBoolean(CSSCorePreferenceNames.WRAPPING_ONE_PER_LINE) && (node.getOwnerDocument() != node || !preferences.getBoolean(CSSCorePreferenceNames.WRAPPING_PROHIBIT_WRAP_ON_ATTR)))
appendDelimBefore(node, null, source);
else if (prev != null || node.getOwnerDocument() != node)
appendSpaceBefore(node, toAppend, source);
}
}
/**
*
*/
protected void formatBefore(ICSSNode node, ICSSNode child, IRegion region, String toAppend, StringBuffer source) {
CSSCleanupStrategy stgy = getCleanupStrategy(node);
ICSSModel cssModel = node.getOwnerDocument().getModel();
// BUG202615 - it is possible to have a style declaration
// with no model associated with it
if (cssModel != null) {
IStructuredDocument structuredDocument = cssModel.getStructuredDocument();
if (structuredDocument != null) {
CompoundRegion[] regions = getRegionsWithoutWhiteSpaces(structuredDocument, region, stgy);
CompoundRegion[] outside = getOutsideRegions(structuredDocument, region);
for (int i = 0; i < regions.length; i++) {
if (i != 0 || needS(outside[0]))
appendSpaceBefore(node, regions[i], source);
source.append(decoratedRegion(regions[i], 0, stgy)); // must
// be
// comments
}
Preferences preferences = CSSCorePlugin.getDefault().getPluginPreferences();
if (needS(outside[1])) {
if (((IndexedRegion) child).getStartOffset() == region.getOffset() + region.getLength() && preferences.getBoolean(CSSCorePreferenceNames.WRAPPING_ONE_PER_LINE) && (node.getOwnerDocument() != node || !preferences.getBoolean(CSSCorePreferenceNames.WRAPPING_PROHIBIT_WRAP_ON_ATTR))) {
appendDelimBefore(node, null, source);
} else
appendSpaceBefore(node, toAppend, source);
}
}
}
}
/**
* Insert the method's description here.
*/
public int getChildInsertPos(ICSSNode node) {
if (node == null)
return -1;
int pos = super.getChildInsertPos(node);
if (pos < 0) {
CSSSourceGenerator formatter = getParentFormatter(node);
return (formatter != null) ? formatter.getChildInsertPos(node.getParentNode()) : -1;
}
return pos;
}
/**
*
*/
public synchronized static StyleDeclarationFormatter getInstance() {
if (instance == null)
instance = new StyleDeclarationFormatter();
return instance;
}
/**
*
* @return int
* @param node
* org.eclipse.wst.css.core.model.interfaces.ICSSNode
* @param insertPos
* int
*/
public int getLengthToReformatAfter(ICSSNode node, int insertPos) {
if (node == null)
return 0;
IndexedRegion nnode = (IndexedRegion) node;
if (insertPos < 0 || !nnode.contains(insertPos)) {
if (node.getParentNode() != null && nnode.getEndOffset() <= 0) {
CSSSourceGenerator pntFormatter = getParentFormatter(node);
if (pntFormatter != null)
return pntFormatter.getLengthToReformatAfter(node.getParentNode(), insertPos);
}
return 0;
}
return super.getLengthToReformatAfter(node, insertPos);
}
/**
*
* @return int
* @param node
* org.eclipse.wst.css.core.model.interfaces.ICSSNode
* @param insertPos
* int
*/
public int getLengthToReformatBefore(ICSSNode node, int insertPos) {
if (node == null)
return 0;
IndexedRegion nnode = (IndexedRegion) node;
if (insertPos <= 0 || !nnode.contains(insertPos - 1)) {
if (node.getParentNode() != null && nnode.getEndOffset() <= 0) {
CSSSourceGenerator pntFormatter = getParentFormatter(node);
if (pntFormatter != null)
return pntFormatter.getLengthToReformatBefore(node.getParentNode(), insertPos);
}
return 0;
}
return super.getLengthToReformatBefore(node, insertPos);
}
}