2024-02-02 12:06:37 +00:00
|
|
|
package de.dhbwstuttgart.syntaxtree;
|
|
|
|
|
|
|
|
import de.dhbwstuttgart.parser.NullToken;
|
|
|
|
import de.dhbwstuttgart.parser.scope.JavaClassName;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.type.GenericRefType;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.type.RefType;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.type.TypePlaceholder;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.type.RefTypeOrTPHOrWildcardOrGeneric;
|
|
|
|
import org.antlr.v4.runtime.Token;
|
|
|
|
|
|
|
|
import java.lang.reflect.Modifier;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.Collection;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Optional;
|
|
|
|
|
2024-04-02 21:45:56 +00:00
|
|
|
import java.util.function.Predicate;
|
|
|
|
import java.util.stream.Stream;
|
2024-04-27 15:37:15 +00:00
|
|
|
//import de.dhbwstuttgart.syntaxtree.*;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.SyntaxTreeNode;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.Field;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.Constructor;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.Method;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.GenericDeclarationList;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.GenericTypeVar;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.TypeScope;
|
|
|
|
import de.dhbwstuttgart.syntaxtree.ASTVisitor;
|
2024-04-02 21:45:56 +00:00
|
|
|
|
|
|
|
import java.lang.Boolean;
|
|
|
|
import java.lang.String;
|
|
|
|
import java.lang.Integer;
|
|
|
|
|
2024-02-02 12:06:37 +00:00
|
|
|
/**
|
|
|
|
* Stellt jede Art von Klasse dar. Auch abstrakte Klassen und Interfaces
|
|
|
|
*/
|
|
|
|
public class ClassOrInterface extends SyntaxTreeNode implements TypeScope {
|
|
|
|
private Boolean methodAdded = false; // wird benoetigt bei in JavaTXCompiler.getConstraints()
|
|
|
|
protected int modifiers;
|
|
|
|
protected JavaClassName name;
|
2024-03-16 16:16:23 +00:00
|
|
|
private final String fileName;
|
|
|
|
|
2024-02-02 12:06:37 +00:00
|
|
|
private List<Field> fields = new ArrayList<>();
|
|
|
|
private Optional<Constructor> fieldInitializations; // PL 2018-11-24: Noetig, um Bytecode fuer initializators nur einmal zu erzeugen
|
|
|
|
private Optional<Method> staticInitializer;
|
|
|
|
private List<Method> methods = new ArrayList<>();
|
|
|
|
private GenericDeclarationList genericClassParameters;
|
|
|
|
private RefType superClass;
|
|
|
|
protected boolean isInterface;
|
|
|
|
protected boolean isFunctionalInterface;
|
|
|
|
private List<RefType> implementedInterfaces;
|
|
|
|
private List<RefType> permittedSubtypes;
|
|
|
|
private List<Constructor> constructors;
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public ClassOrInterface(modifiers, name, fielddecl, fieldInitializations, staticInitializer, methods, constructors, genericClassParameters, superClass, isInterface, isFunctionalInterface, implementedInterfaces, permittedSubtypes, offset, fileName) {
|
2024-02-02 12:06:37 +00:00
|
|
|
super(offset);
|
|
|
|
if (isInterface) {
|
2024-04-02 21:45:56 +00:00
|
|
|
modifiers = modifiers | Modifier.INTERFACE | Modifier.ABSTRACT;
|
2024-02-02 12:06:37 +00:00
|
|
|
}
|
|
|
|
this.modifiers = modifiers;
|
|
|
|
this.name = name;
|
|
|
|
this.fields = fielddecl;
|
|
|
|
this.fieldInitializations = fieldInitializations;
|
|
|
|
this.staticInitializer = staticInitializer;
|
|
|
|
this.genericClassParameters = genericClassParameters;
|
|
|
|
this.superClass = superClass;
|
|
|
|
this.isInterface = isInterface;
|
|
|
|
this.isFunctionalInterface= isFunctionalInterface;
|
|
|
|
this.implementedInterfaces = implementedInterfaces;
|
|
|
|
this.permittedSubtypes = permittedSubtypes;
|
|
|
|
this.methods = methods;
|
|
|
|
this.constructors = constructors;
|
2024-03-16 16:16:23 +00:00
|
|
|
this.fileName = fileName;
|
2024-02-02 12:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* erzeugt fuer Fields, Konstruktoren und Methoden neue ArrayList-Objekte alle anderen Datenobjekte werden nur kopiert.
|
|
|
|
*/
|
2024-04-21 23:06:03 +00:00
|
|
|
public ClassOrInterface(cl) {
|
2024-02-02 12:06:37 +00:00
|
|
|
super(cl.getOffset());
|
|
|
|
this.modifiers = cl.modifiers;
|
|
|
|
this.name = cl.name;
|
|
|
|
this.fields = new ArrayList<>(cl.fields);
|
|
|
|
this.fieldInitializations = cl.fieldInitializations;
|
|
|
|
this.staticInitializer = cl.staticInitializer;
|
|
|
|
this.genericClassParameters = cl.genericClassParameters;
|
|
|
|
this.superClass = cl.superClass;
|
|
|
|
this.isInterface = cl.isInterface;
|
|
|
|
this.isFunctionalInterface= cl.isFunctionalInterface;
|
|
|
|
this.implementedInterfaces = cl.implementedInterfaces;
|
|
|
|
this.methods = new ArrayList<>(cl.methods);
|
|
|
|
this.constructors = new ArrayList<>(cl.constructors);
|
2024-03-16 16:16:23 +00:00
|
|
|
this.fileName = cl.fileName;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getFileName() {
|
2024-03-16 16:16:23 +00:00
|
|
|
return fileName;
|
2024-02-02 12:06:37 +00:00
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getField(name) {
|
2024-02-02 12:06:37 +00:00
|
|
|
// TODO This should be a map
|
2024-04-21 23:06:03 +00:00
|
|
|
return fields.stream().filter(field -> field.getName().equals(name)).findFirst();
|
2024-02-02 12:06:37 +00:00
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getStaticInitializer() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return staticInitializer;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public isInterface() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return (Modifier.INTERFACE & this.getModifiers()) != 0;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public isFunctionalInterface() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.isFunctionalInterface;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gets if it is added
|
2024-04-21 23:06:03 +00:00
|
|
|
public areMethodsAdded() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return methodAdded;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sets that it is added
|
2024-04-21 23:06:03 +00:00
|
|
|
public setMethodsAdded() {
|
2024-02-02 12:06:37 +00:00
|
|
|
methodAdded = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gets class name
|
2024-04-21 23:06:03 +00:00
|
|
|
public getClassName() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get modifiers
|
2024-04-21 23:06:03 +00:00
|
|
|
public getModifiers() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.modifiers;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getFieldDecl() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.fields;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getfieldInitializations() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.fieldInitializations;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getMethods() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.methods;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* public RefType getType() { return generateTypeOfClass(this.getClassName(), this.getGenerics(), this.getOffset()); }
|
|
|
|
*/
|
|
|
|
// TODO: Das hier ist ein Problem. Je nach Kontext wird hier ein anderer Typ benötigt
|
2024-04-21 23:06:03 +00:00
|
|
|
public static generateTypeOfClass(name, genericsOfClass, offset) {
|
2024-02-02 12:06:37 +00:00
|
|
|
// Hier wird immer ein generischer Typ generiert, also mit Type placeholdern
|
|
|
|
List<RefTypeOrTPHOrWildcardOrGeneric> params = new ArrayList<>();
|
|
|
|
for (GenericTypeVar genericTypeVar : genericsOfClass) {
|
|
|
|
// params.add(genericTypeVar.getTypePlaceholder());
|
|
|
|
params.add(TypePlaceholder.fresh(offset));
|
|
|
|
}
|
|
|
|
return new RefType(name, params, offset);
|
|
|
|
}
|
|
|
|
|
2024-04-10 12:41:05 +00:00
|
|
|
|
2024-02-02 12:06:37 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @return die aktuelle Klasse als RefType
|
|
|
|
*/
|
2024-04-21 23:06:03 +00:00
|
|
|
public generateTypeOfThisClass() {
|
2024-02-02 12:06:37 +00:00
|
|
|
List<RefTypeOrTPHOrWildcardOrGeneric> params = new ArrayList<>();
|
|
|
|
for (GenericTypeVar genericTypeVar : this.getGenerics()) {
|
|
|
|
// params.add(genericTypeVar.getTypePlaceholder());
|
|
|
|
params.add(new GenericRefType(genericTypeVar.getName(), new NullToken()));
|
|
|
|
}
|
|
|
|
return new RefType(name, params, new NullToken());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Die Superklasse im Kontext dieser ClassOrInterface Das bedeutet, dass generische Variablen als GenericRefTypes dargestellt sind
|
|
|
|
*/
|
2024-04-21 23:06:03 +00:00
|
|
|
public getSuperClass() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return superClass;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getGenerics() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.genericClassParameters;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2024-04-21 23:06:03 +00:00
|
|
|
public getReturnType() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getConstructors() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return constructors;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2024-04-21 23:06:03 +00:00
|
|
|
public accept(visitor) {
|
2024-02-02 12:06:37 +00:00
|
|
|
visitor.visit(this);
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public getSuperInterfaces() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return implementedInterfaces;
|
|
|
|
}
|
|
|
|
|
2024-04-21 23:06:03 +00:00
|
|
|
public toString() {
|
2024-02-02 12:06:37 +00:00
|
|
|
return this.name.toString() + this.genericClassParameters.toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|