structurizeConstructor function

Future<StructurizeResult> structurizeConstructor(
  1. DartType type,
  2. ConstructorElement constructorElement,
  3. SubjectGenContext<Element> context,
  4. CachedAliasCounter counter,
)

Implementation

Future<StructurizeResult> structurizeConstructor(
    DartType type,
    ConstructorElement constructorElement,
    SubjectGenContext<Element> context,
    CachedAliasCounter counter) async {
  List<AliasImport> imports = [];
  List<IRStructureField> fields = [];
  var element = type.element! as ClassElement;
  var serialName = element.name;

  // Determine used constructor
  var constructorName = "";
  if (element.getNamedConstructor("dog") != null) {
    constructorElement = element.getNamedConstructor("dog")!;
    constructorName = ".dog";
  }

  for (var e in constructorElement.parameters) {
    var fieldName = e.name.replaceFirst("this.", "");
    var field = element.getField(fieldName);
    if (field == null) {
      throw Exception(
          "Serializable constructors must only reference instance fields");
    }
    var fieldType = field.type;
    var serialType = await getSerialType(fieldType, context);
    var iterableType = await getIterableType(fieldType, context);

    var optional = field.type.nullabilitySuffix == NullabilitySuffix.question;
    if (fieldType.isDynamic) optional = true;

    var propertyName = fieldName;
    if (propertyNameChecker.hasAnnotationOf(field)) {
      var annotation = propertyNameChecker.annotationsOf(field).first;
      propertyName = annotation.getField("name")!.toStringValue()!;
    }

    var propertySerializer = "null";
    if (propertySerializerChecker.hasAnnotationOf(field)) {
      var serializerAnnotation =
          propertySerializerChecker.annotationsOf(field).first;
      propertySerializer =
          counter.get(serializerAnnotation.getField("type")!.toTypeValue()!);
    }

    fields.add(IRStructureField(
        fieldName,
        counter.get(fieldType),
        propertySerializer,
        counter.get(serialType),
        iterableType,
        propertyName,
        optional,
        !isDogPrimitiveType(serialType),
        getRetainedAnnotationSourceArray(field, counter),
        mapChecker.isAssignableFrom(field)));
  }

  // Create proxy arguments
  var getters = fields.map((e) => e.accessor).toList();
  var activator =
      "return ${element.name}$constructorName(${constructorElement.parameters.mapIndexed((i, e) {
    var y = fields[i];
    if (e.isNamed) {
      if (y.iterableKind == IterableKind.none) return "${e.name}: list[$i]";
      if (y.optional) return "${e.name}: list[$i]?.cast<${y.serialType}>()";
      return "${e.name}: list[$i].cast<${y.serialType}>()";
    } else {
      if (y.iterableKind == IterableKind.none) return "list[$i]";
      if (y.optional) return "list[$i]?.cast<${y.serialType}>()";
      return "list[$i].cast<${y.serialType}>()";
    }
  }).join(", ")});";
  var isDataclass = dataclassChecker.isAssignableFromType(element.thisType);
  var structure = IRStructure(
      counter.get(type),
      isDataclass ? StructureConformity.dataclass : StructureConformity.basic,
      serialName,
      fields,
      getRetainedAnnotationSourceArray(element, counter));
  return StructurizeResult(imports, structure, getters, activator);
}