blob: 92f87fb131a780d1a0c91199f32f4bcb136f8b73 [file] [log] [blame]
// Copyright (c) 2019, 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:analyzer/dart/element/element.dart';
import 'package:dartdoc/src/element_type.dart';
import 'package:dartdoc/src/model/comment_referable.dart';
import 'package:dartdoc/src/model/extension_target.dart';
import 'package:dartdoc/src/model/model.dart';
import 'package:dartdoc/src/quiver.dart' as quiver;
/// Extension methods
class Extension extends Container implements EnclosedElement {
ElementType extendedType;
Extension(
ExtensionElement element, Library library, PackageGraph packageGraph)
: super(element, library, packageGraph) {
extendedType = modelBuilder.typeFrom(_extension.extendedType, library);
}
/// Detect if this extension applies to every object.
bool get alwaysApplies =>
extendedType.instantiatedType.isDynamic ||
extendedType.instantiatedType.isVoid ||
extendedType.instantiatedType.isDartCoreObject;
bool couldApplyTo<T extends ExtensionTarget>(T c) =>
_couldApplyTo(c.modelType);
/// Return true if this extension could apply to [t].
bool _couldApplyTo(DefinedElementType t) {
if (extendedType.instantiatedType.isDynamic ||
extendedType.instantiatedType.isVoid) {
return true;
}
return t.instantiatedType == extendedType.instantiatedType ||
(t.instantiatedType.element == extendedType.instantiatedType.element &&
extendedType.isSubtypeOf(t)) ||
extendedType.isBoundSupertypeTo(t);
}
/// Returns the library that encloses this element.
@override
ModelElement get enclosingElement => library;
ExtensionElement get _extension => (element as ExtensionElement);
@override
String get kind => 'extension';
List<Method> _methods;
@override
List<Method> get declaredMethods {
_methods ??= _extension.methods.map((e) {
return modelBuilder.from(e, library) as Method;
}).toList(growable: false);
return _methods;
}
@override
String get name => super.name ?? '';
List<Field> _declaredFields;
@override
List<Field> get declaredFields {
_declaredFields ??= _extension.fields.map((f) {
Accessor getter, setter;
if (f.getter != null) {
getter = ContainerAccessor(f.getter, library, packageGraph);
}
if (f.setter != null) {
setter = ContainerAccessor(f.setter, library, packageGraph);
}
return modelBuilder.fromPropertyInducingElement(f, library,
getter: getter, setter: setter) as Field;
}).toList(growable: false);
return _declaredFields;
}
List<TypeParameter> _typeParameters;
// a stronger hash?
@override
List<TypeParameter> get typeParameters {
_typeParameters ??= _extension.typeParameters.map((f) {
var lib = modelBuilder.fromElement(f.enclosingElement.library);
return modelBuilder.from(f, lib) as TypeParameter;
}).toList();
return _typeParameters;
}
List<ModelElement> _allModelElements;
@override
List<ModelElement> get allModelElements {
_allModelElements ??= List.from(
quiver.concat<ModelElement>([
super.allModelElements,
typeParameters,
]),
growable: false);
return _allModelElements;
}
@override
String get filePath => '${library.dirName}/$fileName';
@override
String get href {
if (!identical(canonicalModelElement, this)) {
return canonicalModelElement?.href;
}
assert(canonicalLibrary != null);
assert(canonicalLibrary == library);
return '${package.baseHref}$filePath';
}
Map<String, CommentReferable> _referenceChildren;
@override
Map<String, CommentReferable> get referenceChildren {
return _referenceChildren ??= {
...extendedType.referenceChildren,
// Override extendedType entries with local items.
...super.referenceChildren,
};
}
}