Skip to content
This repository has been archived by the owner on Nov 1, 2024. It is now read-only.

Commit

Permalink
Add support for extension types
Browse files Browse the repository at this point in the history
  • Loading branch information
srujzs committed Nov 20, 2023
1 parent 006790d commit 180d067
Show file tree
Hide file tree
Showing 9 changed files with 999 additions and 22 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
* Add `Expression.bitwiseXorAssign`
* Add `Expression.bitwiseOrAssign`
* Allow passing an `Expression` through `literal` without an exception.
* Add support for extension types.

## 4.7.0

Expand Down
1 change: 1 addition & 0 deletions lib/code_builder.dart
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@ export 'src/specs/expression.dart'
literalString,
literalTrue;
export 'src/specs/extension.dart' show Extension, ExtensionBuilder;
export 'src/specs/extension_type.dart' show ExtensionType, ExtensionTypeBuilder;
export 'src/specs/field.dart' show Field, FieldBuilder, FieldModifier;
export 'src/specs/library.dart' show Library, LibraryBuilder;
export 'src/specs/method.dart'
Expand Down
57 changes: 57 additions & 0 deletions lib/src/emitter.dart
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import 'specs/directive.dart';
import 'specs/enum.dart';
import 'specs/expression.dart';
import 'specs/extension.dart';
import 'specs/extension_type.dart';
import 'specs/field.dart';
import 'specs/library.dart';
import 'specs/method.dart';
Expand Down Expand Up @@ -336,6 +337,62 @@ class DartEmitter extends Object
return out;
}

@override
StringSink visitExtensionType(ExtensionType spec, [StringSink? output]) {
final out = output ??= StringBuffer();
spec.docs.forEach(out.writeln);
for (var a in spec.annotations) {
visitAnnotation(a, out);
}

out.write('extension type ');
if (spec.constant) out.write('const ');
out.write(spec.name);
visitTypeParameters(spec.types.map((r) => r.type), out);
if (spec.primaryConstructorName.isNotEmpty) {
out.write('.${spec.primaryConstructorName}');
}
out.write('(');
_visitRepresentationDeclaration(spec.representationDeclaration, out);
out.write(')');

if (spec.implements.isNotEmpty) {
out
..write(' implements ')
..writeAll(
spec.implements.map<StringSink>((m) => m.type.accept(this)), ',');
}

out.writeln(' {');
for (var c in spec.constructors) {
visitConstructor(c, spec.name, out);
out.writeln();
}
for (var f in spec.fields) {
visitField(f, out);
out.writeln();
}
for (var m in spec.methods) {
visitMethod(m, out);
if (_isLambdaMethod(m)) {
out.writeln(';');
}
out.writeln();
}
out.writeln('}');
return out;
}

void _visitRepresentationDeclaration(
RepresentationDeclaration spec, StringSink out) {
spec.docs.forEach(out.writeln);
for (var a in spec.annotations) {
visitAnnotation(a, out);
}
spec.declaredRepresentationType.accept(this, out);
out.write(' ${spec.name}');
}

@override
StringSink visitDirective(Directive spec, [StringSink? output]) {
output ??= StringBuffer();
Expand Down
152 changes: 152 additions & 0 deletions lib/src/specs/extension_type.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
// Copyright (c) 2023, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.

import 'package:built_collection/built_collection.dart';
import 'package:built_value/built_value.dart';
import 'package:meta/meta.dart';

import '../base.dart';
import '../mixins/annotations.dart';
import '../mixins/dartdoc.dart';
import '../mixins/generics.dart';
import '../visitors.dart';
import 'constructor.dart';
import 'expression.dart';
import 'field.dart';
import 'method.dart';
import 'reference.dart';

part 'extension_type.g.dart';

@immutable
abstract class ExtensionType extends Object
with HasAnnotations, HasDartDocs, HasGenerics
implements Built<ExtensionType, ExtensionTypeBuilder>, Spec {
factory ExtensionType([void Function(ExtensionTypeBuilder)? updates]) =
_$ExtensionType;

ExtensionType._();

@override
BuiltList<Expression> get annotations;

@override
BuiltList<String> get docs;

/// Whether this extension type is declared as `const`.
bool get constant;

String get name;

@override
BuiltList<Reference> get types;

/// Name of the extension type's primary constructor. An empty string
/// will make it unnamed.
String get primaryConstructorName;

RepresentationDeclaration get representationDeclaration;

BuiltList<Reference> get implements;

BuiltList<Constructor> get constructors;

BuiltList<Field> get fields;

BuiltList<Method> get methods;

@override
R accept<R>(
SpecVisitor<R> visitor, [
R? context,
]) =>
visitor.visitExtensionType(this, context);
}

abstract class ExtensionTypeBuilder extends Object
with HasAnnotationsBuilder, HasDartDocsBuilder, HasGenericsBuilder
implements Builder<ExtensionType, ExtensionTypeBuilder> {
factory ExtensionTypeBuilder() = _$ExtensionTypeBuilder;

ExtensionTypeBuilder._();

@override
void update(void Function(ExtensionTypeBuilder)? updates) {
updates?.call(this);
}

@override
ListBuilder<Expression> annotations = ListBuilder<Expression>();

@override
ListBuilder<String> docs = ListBuilder<String>();

/// Whether this extension type is declared as `const`.
bool constant = false;

String? name;

@override
ListBuilder<Reference> types = ListBuilder<Reference>();

/// Name of the extension type's primary constructor. An empty string
/// will make it unnamed.
String primaryConstructorName = '';

RepresentationDeclaration? representationDeclaration;

ListBuilder<Reference> implements = ListBuilder<Reference>();

ListBuilder<Constructor> constructors = ListBuilder<Constructor>();

ListBuilder<Field> fields = ListBuilder<Field>();

ListBuilder<Method> methods = ListBuilder<Method>();
}

abstract class RepresentationDeclaration extends Object
with HasAnnotations, HasDartDocs
implements
Built<RepresentationDeclaration, RepresentationDeclarationBuilder> {
factory RepresentationDeclaration(
[void Function(RepresentationDeclarationBuilder)? updates]) =
_$RepresentationDeclaration;

RepresentationDeclaration._();

@override
BuiltList<Expression> get annotations;

@override
BuiltList<String> get docs;

Reference get declaredRepresentationType;

String get name;
}

abstract class RepresentationDeclarationBuilder extends Object
with HasAnnotationsBuilder, HasDartDocsBuilder
implements
Builder<RepresentationDeclaration, RepresentationDeclarationBuilder> {
factory RepresentationDeclarationBuilder() =
_$RepresentationDeclarationBuilder;

RepresentationDeclarationBuilder._();

@override
void update(void Function(RepresentationDeclarationBuilder)? updates) {
updates?.call(this);
}

@override
ListBuilder<Expression> annotations = ListBuilder<Expression>();

@override
ListBuilder<String> docs = ListBuilder<String>();

Reference? declaredRepresentationType;

String? name;
}
Loading

0 comments on commit 180d067

Please sign in to comment.