/**
* Implements the CFML Function arraysort
*/
package railo.runtime.functions.arrays;
import java.util.Comparator;
import railo.runtime.PageContext;
import railo.runtime.exp.CasterException;
import railo.runtime.exp.ExpressionException;
import railo.runtime.exp.FunctionException;
import railo.runtime.exp.PageException;
import railo.runtime.exp.PageRuntimeException;
import railo.runtime.functions.BIF;
import railo.runtime.op.Caster;
import railo.runtime.type.Array;
import railo.runtime.type.Closure;
import railo.runtime.type.UDF;
import railo.runtime.type.util.ArrayUtil;
public final class ArraySort extends BIF {
private static final long serialVersionUID = -747941236369495141L;
public static boolean call(PageContext pc , Array array, Object sortTypeOrClosure) throws PageException {
return call(pc , array, sortTypeOrClosure, "asc",false);
}
public static boolean call(PageContext pc , Array array, Object sortTypeOrClosure, String sortorder) throws PageException {
return call(pc , array, sortTypeOrClosure, sortorder,false);
}
public static boolean call(PageContext pc , Array array, Object sortTypeOrClosure, String sortorder, boolean localeSensitive) throws PageException {
if(array.getDimension()>1)
throw new ExpressionException("only 1 dimensional arrays can be sorted");
if(sortTypeOrClosure instanceof UDF){
UDFComparator comp=new UDFComparator(pc, (UDF)sortTypeOrClosure);
array.sort(comp);
}
else {
array.sort(ArrayUtil.toComparator(pc,Caster.toString(sortTypeOrClosure), sortorder,localeSensitive));
}
return true;
}
@Override
public Object invoke(PageContext pc, Object[] args) throws PageException {
if(args.length==2)return call(pc,Caster.toArray(args[0]),args[1]);
if(args.length==3)return call(pc,Caster.toArray(args[0]),args[1],Caster.toString(args[2]));
return call(pc,Caster.toArray(args[0]),args[1],Caster.toString(args[2]),Caster.toBooleanValue(args[3]));
}
}
class UDFComparator implements Comparator<Object> {
private UDF udf;
private Object[] args=new Object[2];
private PageContext pc;
public UDFComparator(PageContext pc,UDF udf){
this.pc=pc;
this.udf=udf;
}
@Override
public int compare(Object oLeft, Object oRight) {
try {
args[0]=oLeft;
args[1]=oRight;
Object res = udf.call(pc, args, false);
Integer i = Caster.toInteger(res,null);
if(i==null) throw new FunctionException(pc,"ArraySort",2,"function","return value of the "+(udf instanceof Closure?"closure":"function ["+udf.getFunctionName()+"]")+" cannot be casted to a integer.",CasterException.createMessage(res, "integer"));
return i.intValue();
}
catch (PageException pe) {
throw new PageRuntimeException(pe);
}
}
}