blob: bb3cc963a60386561bab33c2b58d8b2aef03a853 [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.
// ignore_for_file: analyzer_use_new_elements
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/source/line_info.dart';
import 'package:dartdoc/src/element_type.dart';
import 'package:dartdoc/src/model/comment_referable.dart';
import 'package:dartdoc/src/model/kind.dart';
import 'package:dartdoc/src/model/model.dart';
import 'package:dartdoc/src/model_utils.dart';
class Constructor extends ModelElement with ContainerMember, TypeParameters {
@override
final ConstructorElement element;
Constructor(this.element, super.library, super.packageGraph);
@override
CharacterLocation? get characterLocation {
if (element.isSynthetic) {
// Make warnings for a synthetic constructor refer to somewhere reasonable
// since a synthetic constructor has no definition independent of the
// parent class.
return enclosingElement.characterLocation;
}
return super.characterLocation;
}
@override
bool get isPublic {
if (!super.isPublic) return false;
if (element2.hasPrivateName) return false;
var class_ = element.enclosingElement3;
// Enums cannot be explicitly constructed or extended.
if (class_ is EnumElement) return false;
if (class_ is ClassElement) {
if (element.isFactory) return true;
if (class_.isSealed ||
(class_.isAbstract && class_.isFinal) ||
(class_.isAbstract && class_.isInterface)) {
/// Sealed classes, abstract final classes, and abstract interface
/// classes, cannot be instantiated nor extended, from outside the
/// declaring library. Avoid documenting them.
return false;
}
}
return true;
}
@override
List<TypeParameter> get typeParameters =>
(enclosingElement as Constructable).typeParameters;
@override
Container get enclosingElement =>
getModelFor(element.enclosingElement3, library) as Container;
@override
String get aboveSidebarPath => enclosingElement.sidebarPath;
@override
String? get belowSidebarPath => null;
String get fullKind {
if (isConst) return 'const $kind';
if (isFactory) return 'factory $kind';
return kind.toString();
}
@override
String get fullyQualifiedName {
if (isUnnamedConstructor) return super.fullyQualifiedName;
return '${library.name}.$name';
}
@override
bool get isConst => element.isConst;
bool get isUnnamedConstructor => name == enclosingElement.name;
bool get isFactory => element.isFactory;
@override
Kind get kind => Kind.constructor;
late final Callable modelType = getTypeFor(element.type, library) as Callable;
@override
String get name {
// TODO(jcollins-g): After the old lookup code is retired, rationalize
// [name] around the conventions used in referenceChildren and replace
// code there and elsewhere with simple references to the name.
var constructorName = element.name;
if (constructorName.isEmpty) {
return enclosingElement.name;
}
return '${enclosingElement.name}.$constructorName';
}
@override
String get nameWithGenerics {
var constructorName = element.name;
if (constructorName.isEmpty) {
return '${enclosingElement.name}$genericParameters';
}
return '${enclosingElement.name}$genericParameters.$constructorName';
}
String? get shortName {
if (name.contains('.')) {
return name.substring(element.enclosingElement3.name.length + 1);
} else {
return name;
}
}
@override
late final Map<String, CommentReferable> referenceChildren = () {
// Find the element that [parameter] is _really_ referring to.
Element? dereferenceParameter(ParameterElement? parameter) =>
switch (parameter) {
FieldFormalParameterElement() => parameter.field,
SuperFormalParameterElement() =>
dereferenceParameter(parameter.superConstructorParameter),
_ => parameter
};
var parameterElements = parameters.map((parameter) {
var element = dereferenceParameter(parameter.element);
return element == null ? parameter : getModelForElement(element);
});
return {
for (var element in parameterElements) element.referenceName: element,
for (var tp in typeParameters) tp.referenceName: tp,
};
}();
@override
String get referenceName =>
isUnnamedConstructor ? enclosingElement.name : element.name;
}