toTupleClass function
Implementation
Spec toTupleClass(
String className,
ts.ObjectType obj,
GenOption option,
) {
final constructorParameters = <Parameter>[];
final fromJson = StringBuffer();
final fromDeserializable = StringBuffer();
final toJson = StringBuffer();
final toJsonFields = StringBuffer();
final toSerializableFields = StringBuffer();
final hashes = <String>[];
final equals = <String>[];
final fields = <Field>[];
final copyWithParameters = <Parameter>[];
final copyWithAssign = StringBuffer();
obj.children.forEachIndexed((index, e) {
final child = e.child;
final fieldName = 'item${index + 1}';
final isOpt = child is ts.OptType;
var dartType = child.dartType();
if (isOpt && !dartType.endsWith('?')) {
dartType += '?';
}
final isList = dartType.startsWith('List<');
if (isList) {
hashes.add('const DeepCollectionEquality().hash($fieldName)');
equals.add(
'const DeepCollectionEquality().equals(other.$fieldName, $fieldName)',
);
} else {
hashes.add(fieldName);
equals.add(
'(identical(other.$fieldName, $fieldName) || other.$fieldName == $fieldName)',
);
}
final type = Reference(dartType);
constructorParameters.add(
Parameter(
(b) => b
..toThis = true
..name = fieldName,
),
);
copyWithParameters.add(
Parameter(
(b) => b
..name = fieldName
..named = true
..type = Reference(dartType.endsWith('?') ? dartType : '$dartType?'),
),
);
copyWithAssign.writeln('$fieldName ?? this.$fieldName,');
fields.add(
Field(
(b) => b
..name = fieldName
..type = type
..modifier = FieldModifier.final$
..docs = ListBuilder(
['/// [$fieldName] defined in Candid: `${child.did}`'],
),
),
);
final deserJson = child.deserializeAndReplace(
replace: 'tuple[$index]',
fromIDL: false,
);
fromJson.writeln('$deserJson,');
final deserIDL = child.deserializeAndReplace(
replace: 'tuple[$index]',
fromIDL: true,
);
fromDeserializable.writeln('$deserIDL,');
final ser = child.serialize(fromIDL: !option.explicitSerializationMethods);
final arg =
ser == null ? fieldName : ser.replaceAll(ts.IDLType.ph, fieldName);
toJson.writeln('$arg,');
toSerializableFields.writeln('final $fieldName = this.$fieldName;');
final toJsonField = _typeToJsonField(option, obj, e, fieldName);
toJsonFields.writeln('final $fieldName = this.$toJsonField;');
});
return Class(
(b) => b
..name = className
..annotations = ListBuilder([const CodeExpression(Code('immutable'))])
..constructors = ListBuilder([
Constructor(
(b) => b
..requiredParameters = ListBuilder(constructorParameters)
..constant = true,
),
if (option.explicitSerializationMethods)
Constructor(
(b) => b
..docs = ListBuilder([
'/// An extra method for the deserialization with `packages:agent_dart`.',
])
..name = 'fromIDLDeserializable'
..factory = true
..body = Code('return $className($fromDeserializable);')
..requiredParameters = ListBuilder([
Parameter(
(b) => b
..type = const Reference('List<dynamic>')
..name = 'tuple',
),
]),
),
Constructor(
(b) => b
..name = 'fromJson'
..factory = true
..body = Code('return $className($fromJson);')
..requiredParameters = ListBuilder([
Parameter(
(b) => b
..type = const Reference('List<dynamic>')
..name = 'tuple',
),
]),
),
])
..fields = ListBuilder(fields)
..methods = ListBuilder([
if (option.explicitSerializationMethods)
Method(
(b) => b
..docs = ListBuilder([
'/// An extra method for the serialization with `packages:agent_dart`.',
])
..name = 'toIDLSerializable'
..body = Code('${toSerializableFields}return [$toJson];')
..returns = const Reference('List<dynamic>'),
),
Method(
(b) => b
..name = 'toJson'
..body = Code('${toJsonFields}return [$toJson];')
..returns = Reference(
'${option.explicitSerializationMethods ? '' : '\n'}'
'List<dynamic>',
),
),
if (option.copyWith)
Method(
(b) => b
..name = 'copyWith'
..optionalParameters = ListBuilder(copyWithParameters)
..body = Code('return $className($copyWithAssign);')
..returns = Reference(className),
),
if (option.equal) ...[
Method(
(b) => b
..name = 'operator =='
..returns = const Reference('bool')
..requiredParameters = ListBuilder([
Parameter(
(b) => b
..name = 'other'
..type = const Reference('Object'),
),
])
..annotations =
ListBuilder([const CodeExpression(Code('override'))])
..body = Code(
'return identical(this, other) '
'|| (other.runtimeType == runtimeType && other is $className ${fields.isEmpty ? '' : '&&'} '
'${equals.join("&&")});',
),
),
Method(
(b) => b
..type = MethodType.getter
..name = 'hashCode'
..returns = const Reference('int')
..lambda = true
..annotations =
ListBuilder([const CodeExpression(Code('override'))])
..body =
Code('Object.hashAll([runtimeType,${hashes.join(",")}])'),
),
],
toStringMethod,
])
..docs = ListBuilder(['/// [$className] defined in Candid', obj.doc]),
);
}