blob: c841ea9ccbb26efc5ee616c62d7a6b2a43b74b12 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2000, 2008 IBM Corporation and others.
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* IBM Corporation - initial API and implementation
*******************************************************************************/
package org.eclipse.dltk.internal.javascript.corext.refactoring.util;
import java.util.ArrayList;
import java.util.List;
import org.eclipse.core.runtime.Assert;
import org.eclipse.dltk.javascript.core.dom.Expression;
import org.eclipse.dltk.javascript.core.dom.Node;
import org.eclipse.dltk.javascript.core.dom.util.DomSwitch;
import org.eclipse.emf.ecore.EObject;
import org.eclipse.jface.text.IRegion;
import org.eclipse.jface.text.Region;
/**
* Maps a selection to a set of AST nodes.
*/
public class SelectionAnalyzer extends DomSwitch<Boolean>{
private Selection fSelection;
private boolean fTraverseSelectedNode;
private Node fLastCoveringNode;
// Selected nodes
private List<Node> fSelectedNodes;
public SelectionAnalyzer(Selection selection, boolean traverseSelectedNode) {
//super(true);
Assert.isNotNull(selection);
fSelection= selection;
fTraverseSelectedNode= traverseSelectedNode;
}
public boolean hasSelectedNodes() {
return fSelectedNodes != null && !fSelectedNodes.isEmpty();
}
public Node[] getSelectedNodes() {
if (fSelectedNodes == null || fSelectedNodes.isEmpty())
return new Node[0];
return (Node[]) fSelectedNodes.toArray(new Node[fSelectedNodes.size()]);
}
public Node getFirstSelectedNode() {
if (fSelectedNodes == null || fSelectedNodes.isEmpty())
return null;
return fSelectedNodes.get(0);
}
/*public ASTNode getLastSelectedNode() {
if (fSelectedNodes == null || fSelectedNodes.isEmpty())
return null;
return (ASTNode)fSelectedNodes.get(fSelectedNodes.size() - 1);
}*/
public boolean isExpressionSelected() {
if (!hasSelectedNodes())
return false;
return fSelectedNodes.get(0) instanceof Expression;
}
public IRegion getSelectedNodeRange() {
if (fSelectedNodes == null || fSelectedNodes.isEmpty())
return null;
Node firstNode= fSelectedNodes.get(0);
Node lastNode= fSelectedNodes.get(fSelectedNodes.size() - 1);
int start= firstNode.getBegin();
return new Region(start, lastNode.getEnd() - start);
}
public Node getLastCoveringNode() {
return fLastCoveringNode;
}
protected Selection getSelection() {
return fSelection;
}
//--- node management ---------------------------------------------------------*/
public void traverse(Node node) {
if (isTraverseNeeded(node))
for(EObject item : node.eContents())
traverse((Node)item);
doSwitch(node);
}
public Boolean isTraverseNeeded(Node node) {
// The selection lies behind the node.
if (fSelection.liesOutside(node)) {
return false;
} else if (fSelection.covers(node)) {
if (isFirstNode()) {
handleFirstSelectedNode(node);
} else {
handleNextSelectedNode(node);
}
return fTraverseSelectedNode;
} else if (fSelection.coveredBy(node)) {
fLastCoveringNode= node;
return true;
} else if (fSelection.endsIn(node)) {
return handleSelectionEndsIn(node);
}
// There is a possibility that the user has selected trailing semicolons that don't belong
// to the statement. So dive into it to check if sub nodes are fully covered.
return true;
}
protected void reset() {
fSelectedNodes= null;
}
protected void handleFirstSelectedNode(Node node) {
fSelectedNodes= new ArrayList<Node>(5);
fSelectedNodes.add(node);
}
protected void handleNextSelectedNode(Node node) {
if (getFirstSelectedNode().eContainer() == node.eContainer()) {
fSelectedNodes.add(node);
}
}
protected boolean handleSelectionEndsIn(Node node) {
return false;
}
/*protected List internalGetSelectedNodes() {
return fSelectedNodes;
}*/
private boolean isFirstNode() {
return fSelectedNodes == null;
}
}