@Override
public WeaselCompilerReturn compile(WeaselCompiler compiler, WeaselKeyWordCompilerHelper compilerHelper, WeaselGenericClass write, WeaselGenericClass expect, WeaselGenericClass elementParent, boolean isVariable) throws WeaselCompilerException {
WeaselInstructionList instructions = new WeaselInstructionList();
if(newClass!=null){
WeaselClass weaselClass = compiler.getWeaselClass(WeaselClass.mapClassNames(newClass));
//TODO
WeaselGenericClass genericClass = new WeaselGenericClass(weaselClass);
if(arraySize==null){
instructions.add(token.line, new WeaselInstructionNew(weaselClass.getByteName()));
instructions.add(token.line, new WeaselInstructionPush(1));
instructions.add(token.line, new WeaselInstructionInvoke(weaselClass.getRealName()+".<preInit>()"));
List<WeaselGenericMethod2> methods = genericClass.getGenericMethods("<init>", true);
instructions.add(token.line, new WeaselInstructionPush(1));
WeaselParameterCompileReturn wcr = WeaselTree.compileParamList(token.line, newClass, compiler, compilerHelper, func, methods);
instructions.addAll(wcr.instructions);
instructions.add(token.line, new WeaselInstructionInvoke(wcr.method.getMethod().getMethod().getClassNameAndDesk()));
}else{
String className = weaselClass.getByteName();
for(int i=0; i<arraySize.size(); i++){
className = "["+className;
}
genericClass = new WeaselGenericClass(compiler.getWeaselClass(className), genericClass.getGenerics());
if(arrayInit==null){
className = weaselClass.getByteName();
int sizes = 0;
for(WeaselTree tree:arraySize){
if(tree==null){
className = "["+className;
}else{
WeaselCompilerReturn wcr = tree.compile(compiler, compilerHelper, null, new WeaselGenericClass(compiler.baseTypes.intClass), null, false);
instructions.addAll(wcr.getInstructions(compiler, compiler.baseTypes.intClass));
sizes++;
}
}
instructions.add(token.line, new WeaselInstructionNewArray(className, sizes));
}else{
instructions.addAll(arrayInit.compile(compiler, compilerHelper, genericClass));
}
}
return new WeaselCompilerReturnInstructionList(instructions, genericClass);
}else if(isFunc){
List<WeaselGenericMethod2> methods;
if(elementParent==null){
methods = compilerHelper.getGenericMethods((String)token.param);
}else{
compiler.compileEasy(elementParent.getBaseClass());
methods = elementParent.getGenericMethods((String)token.param, isVariable);
}
if(methods.isEmpty()){
throw new WeaselCompilerException(token.line, "Method not found %s", token);
}
WeaselParameterCompileReturn wcr = WeaselTree.compileParamList(token.line, (String)token.param, compiler, compilerHelper, func, methods);
if(elementParent==null){
if(!WeaselModifier.isStatic(wcr.method.getMethod().getMethod().getModifier())){
instructions.add(token.line, new WeaselInstructionLoadVariable(compilerHelper.getVariable("this").pos));
}
}
instructions.addAll(wcr.instructions);
if(elementParent==null){
if(WeaselModifier.isStatic(wcr.method.getMethod().getMethod().getModifier())){
instructions.add(token.line, new WeaselInstructionInvokeStatic(wcr.method.getMethod().getMethod().getNameAndDesk()));
}else{
instructions.add(token.line, new WeaselInstructionInvoke(wcr.method.getMethod().getMethod().getClassNameAndDesk()));
}
}else{
if(isVariable){
instructions.add(token.line, new WeaselInstructionInvoke(wcr.method.getMethod().getMethod().getClassNameAndDesk()));
}else{
instructions.add(token.line, new WeaselInstructionInvokeStatic(wcr.method.getMethod().getMethod().getNameAndDesk()));
}
}
return new WeaselCompilerReturnInstructionList(instructions, wcr.method.getGenericReturn());
}else if(isIndex){
WeaselGenericClass arrayClass;
String variable = (String)token.param;
if(elementParent==null){
WeaselVariableInfo wvi = compilerHelper.getVariable(variable);
if(wvi==null){
WeaselGenericField wf = compilerHelper.getGenericField(variable);
if(wf==null){
WeaselClass weaselClass;
try{
weaselClass = compiler.getWeaselClass(WeaselClass.mapClassNames(variable));
}catch(WeaselNativeException e){
throw new WeaselCompilerException(token.line, "Variable not declared bevore %s", variable);
}
return new WeaselCompilerReturnInstructionList(instructions, new WeaselGenericClass(weaselClass), true);
}
if(WeaselModifier.isStatic(compilerHelper.getCompilingMethod().getMethod().getMethod().getModifier())){
if(!WeaselModifier.isStatic(wf.getField().getModifier())){
throw new WeaselCompilerException(token.line, "Variable %s is not static", variable);
}
}
if(WeaselModifier.isStatic(wf.getField().getModifier())){
instructions.add(token.line, new WeaselInstructionReadStaticField(wf.getField().getDesk()));
}else{
wvi = compilerHelper.getVariable("this");
instructions.add(token.line, new WeaselInstructionReadFieldOf(wvi.pos, wf.getField().getDesk()));
}
arrayClass = wf.getGenericType();
}else{
arrayClass = wvi.type;
instructions.add(token.line, new WeaselInstructionLoadVariable(wvi.pos));
}
}else{
WeaselGenericField field = elementParent.getGenericField((String)token.param);
if(field==null)
throw new WeaselCompilerException(token.line, "Variable %s not declared in %s", token, elementParent);
arrayClass = field.getGenericType();
instructions.add(token.line, new WeaselInstructionReadField(field.getField().getDesk()));
}
if(!arrayClass.getBaseClass().isArray()){
throw new WeaselCompilerException(token.line, "%s is not an array", arrayClass);
}
WeaselCompilerReturn wcr = func.compile(compiler, compilerHelper, null, new WeaselGenericClass(compiler.baseTypes.intClass), null, false);
instructions.addAll(wcr.getInstructions(compiler, compiler.baseTypes.intClass));
if(write==null){
instructions.add(token.line, new WeaselInstructionReadIndex(WeaselPrimitive.getPrimitiveID(arrayClass.getBaseClass().getArrayClass())));
}else{
instructions.add(token.line, new WeaselInstructionPlaceHolder());
WeaselTree.autoCast(compiler, write, new WeaselGenericClass(arrayClass.getBaseClass().getArrayClass(), arrayClass.getGenerics()), token.line, instructions, true);
instructions.add(token.line, new WeaselInstructionWriteIndex(WeaselPrimitive.getPrimitiveID(arrayClass.getBaseClass().getArrayClass())));
}
return new WeaselCompilerReturnInstructionList(instructions, new WeaselGenericClass(arrayClass.getBaseClass().getArrayClass(), arrayClass.getGenerics()));
}else if(token==null){
return tree.compile(compiler, compilerHelper, write, expect, elementParent, isVariable);
}else{
switch(token.tokenType){
case BOOL:
case DOUBLE:
case INTEGER:
case NULL:
case STRING:
if(write!=null){
throw new WeaselCompilerException(token.line, "Can't write %s to constant %s", write, token);
}
return new WeaselCompilerReturnConstant(compiler, token.line, token.param);
case KEYWORD:
if(token.param!=WeaselKeyWord.THIS)
throw new WeaselCompilerException(token.line, "Expect ident but got %s", token);
case IDENT:
String variable = token.toString();
if(elementParent==null){
WeaselVariableInfo wvi = compilerHelper.getVariable(variable);
if(wvi==null){
WeaselGenericField wf = compilerHelper.getGenericField(variable);
if(wf==null){
WeaselClass weaselClass;
try{
weaselClass = compiler.getWeaselClass(WeaselClass.mapClassNames(variable));
}catch(WeaselNativeException e){
throw new WeaselCompilerException(token.line, "Variable not declared bevore %s", variable);
}
return new WeaselCompilerReturnInstructionList(instructions, new WeaselGenericClass(weaselClass), true);
}
if(WeaselModifier.isStatic(compilerHelper.getCompilingMethod().getMethod().getMethod().getModifier())){
if(!WeaselModifier.isStatic(wf.getField().getModifier())){
throw new WeaselCompilerException(token.line, "Variable %s is not static", variable);
}
}
if(WeaselModifier.isStatic(wf.getField().getModifier())){
if(write==null){
instructions.add(token.line, new WeaselInstructionReadStaticField(wf.getField().getDesk()));
}else{
instructions.add(token.line, new WeaselInstructionPlaceHolder());
WeaselTree.autoCast(compiler, write, wf.getGenericType(), token.line, instructions, true);
instructions.add(token.line, new WeaselInstructionWriteStaticField(wf.getField().getDesk()));
}
}else{
wvi = compilerHelper.getVariable("this");
if(write==null){
instructions.add(token.line, new WeaselInstructionReadFieldOf(wvi.pos, wf.getField().getDesk()));
}else{
instructions.add(token.line, new WeaselInstructionPlaceHolder());
WeaselTree.autoCast(compiler, write, wf.getGenericType(), token.line, instructions, true);
instructions.add(token.line, new WeaselInstructionWriteFieldOf(wvi.pos, wf.getField().getDesk()));
}
}
return new WeaselCompilerReturnInstructionList(instructions, wf.getGenericType());
}else{
if(write==null){
instructions.add(token.line, new WeaselInstructionLoadVariable(wvi.pos));
}else{
instructions.add(token.line, new WeaselInstructionPlaceHolder());
WeaselTree.autoCast(compiler, write, wvi.type, token.line, instructions, true);
instructions.add(token.line, new WeaselInstructionSaveVariable(wvi.pos));
}
return new WeaselCompilerReturnInstructionList(instructions, wvi.type);
}
}else{
WeaselGenericField wf = elementParent.getGenericField(variable);
if(wf==null){
WeaselClass weaselClass;
try{
weaselClass = compiler.getWeaselClass(WeaselClass.mapClassNames(variable));
}catch(WeaselNativeException e){
throw new WeaselCompilerException(token.line, "Variable not declared bevore %s", variable);
}