/*******************************************************************************
* Copyright (c) 2009-2013 CWI
* 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:
* * Arnold Lankamp - Arnold.Lankamp@cwi.nl
*******************************************************************************/
package org.rascalmpl.parser.gtd.stack;
import org.rascalmpl.parser.gtd.stack.filter.ICompletionFilter;
import org.rascalmpl.parser.gtd.stack.filter.IEnterFilter;
@SuppressWarnings("cast")
public final class SeparatedListStackNode<P> extends AbstractExpandableStackNode<P>{
private final P production;
private final String name;
private final AbstractStackNode<P>[] children;
private final AbstractStackNode<P> emptyChild;
public SeparatedListStackNode(int id, int dot, P production, AbstractStackNode<P> child, AbstractStackNode<P>[] separators, boolean isPlusList){
super(id, dot);
this.production = production;
this.name = String.valueOf(id); // Add the id to make it unique.
this.children = generateChildren(child, separators);
this.emptyChild = isPlusList ? null : generateEmptyChild();
}
public SeparatedListStackNode(int id, int dot, P production, AbstractStackNode<P> child, AbstractStackNode<P>[] separators, boolean isPlusList, IEnterFilter[] enterFilters, ICompletionFilter[] completionFilters){
super(id, dot, enterFilters, completionFilters);
this.production = production;
this.name = String.valueOf(id); // Add the id to make it unique.
this.children = generateChildren(child, separators);
this.emptyChild = isPlusList ? null : generateEmptyChild();
}
private SeparatedListStackNode(SeparatedListStackNode<P> original, int startLocation){
super(original, startLocation);
production = original.production;
name = original.name;
children = original.children;
emptyChild = original.emptyChild;
}
/**
* Generates and initializes the alternative for this separated list.
*/
@SuppressWarnings("unchecked")
private AbstractStackNode<P>[] generateChildren(AbstractStackNode<P> child, AbstractStackNode<P>[] separators){
AbstractStackNode<P> listNode = child.getCleanCopy(DEFAULT_START_LOCATION);
listNode.setAlternativeProduction(production);
int numberOfSeparators = separators.length;
AbstractStackNode<P>[] prod = (AbstractStackNode<P>[]) new AbstractStackNode[numberOfSeparators + 2];
listNode.setProduction(prod);
prod[0] = listNode; // Start
for(int i = numberOfSeparators - 1; i >= 0; --i){
AbstractStackNode<P> separator = separators[i];
separator.setProduction(prod);
separator.markAsSeparator();
prod[i + 1] = separator;
}
prod[numberOfSeparators + 1] = listNode; // End
return (AbstractStackNode<P>[]) new AbstractStackNode[]{listNode};
}
/**
* Generates and initializes the empty child for this separated list (if it's a star list).
*/
@SuppressWarnings("unchecked")
private AbstractStackNode<P> generateEmptyChild(){
AbstractStackNode<P> empty = (AbstractStackNode<P>) EMPTY.getCleanCopy(DEFAULT_START_LOCATION);
empty.setAlternativeProduction(production);
return empty;
}
public String getName(){
return name;
}
public AbstractStackNode<P> getCleanCopy(int startLocation){
return new SeparatedListStackNode<P>(this, startLocation);
}
public AbstractStackNode<P>[] getChildren(){
return children;
}
public boolean canBeEmpty(){
return emptyChild != null;
}
public AbstractStackNode<P> getEmptyChild(){
return emptyChild;
}
public String toString(){
StringBuilder sb = new StringBuilder();
sb.append(name);
sb.append('(');
sb.append(startLocation);
sb.append(')');
return sb.toString();
}
public int hashCode(){
return production.hashCode();
}
public boolean isEqual(AbstractStackNode<P> stackNode){
if(!(stackNode instanceof SeparatedListStackNode)) return false;
SeparatedListStackNode<P> otherNode = (SeparatedListStackNode<P>) stackNode;
if(!production.equals(otherNode.production)) return false;
return hasEqualFilters(stackNode);
}
}