blob: 9d157a5f08f477cdceae1fb00ccf9d73a9110bfd [file] [log] [blame]
// Copyright (c) 2014, 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 'dart:async';
import 'dart:collection';
import 'package:analysis_server/src/services/search/element_visitors.dart';
import 'package:analysis_server/src/services/search/search_engine.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/dart/element/type.dart';
/**
* Returns direct children of [parent].
*/
List<Element> getChildren(Element parent, [String name]) {
List<Element> children = <Element>[];
visitChildren(parent, (Element element) {
if (name == null || element.displayName == name) {
children.add(element);
}
return false;
});
return children;
}
/**
* Returns direct non-synthetic children of the given [ClassElement].
*
* Includes: fields, accessors and methods.
* Excludes: constructors and synthetic elements.
*/
List<Element> getClassMembers(ClassElement clazz, [String name]) {
List<Element> members = <Element>[];
visitChildren(clazz, (Element element) {
if (element.isSynthetic) {
return false;
}
if (element is ConstructorElement) {
return false;
}
if (name != null && element.displayName != name) {
return false;
}
if (element is ExecutableElement) {
members.add(element);
}
if (element is FieldElement) {
members.add(element);
}
return false;
});
return members;
}
/**
* Returns a [Set] with direct subclasses of [seed].
*/
Future<Set<ClassElement>> getDirectSubClasses(
SearchEngine searchEngine, ClassElement seed) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
List<SearchMatch> matches = await searchEngine.searchSubtypes(seed);
return matches.map((match) => match.element).cast<ClassElement>().toSet();
}
/// Return the non-synthetic children of the given [extension]. This includes
/// fields, accessors and methods, but excludes synthetic elements.
List<Element> getExtensionMembers(ExtensionElement extension, [String name]) {
List<Element> members = <Element>[];
visitChildren(extension, (Element element) {
if (element.isSynthetic) {
return false;
}
if (name != null && element.displayName != name) {
return false;
}
if (element is ExecutableElement) {
members.add(element);
}
if (element is FieldElement) {
members.add(element);
}
return false;
});
return members;
}
/**
* @return all implementations of the given {@link ClassMemberElement} is its superclasses and
* their subclasses.
*/
Future<Set<ClassMemberElement>> getHierarchyMembers(
SearchEngine searchEngine, ClassMemberElement member) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
Set<ClassMemberElement> result = new HashSet<ClassMemberElement>();
// extension member
if (member.enclosingElement is ExtensionElement) {
result.add(member);
return new Future.value(result);
}
// static elements
if (member.isStatic || member is ConstructorElement) {
result.add(member);
return new Future.value(result);
}
// method, field, etc
String name = member.displayName;
ClassElement memberClass = member.enclosingElement;
Set<ClassElement> searchClasses = getSuperClasses(memberClass);
searchClasses.add(memberClass);
for (ClassElement superClass in searchClasses) {
// ignore if super- class does not declare member
if (getClassMembers(superClass, name).isEmpty) {
continue;
}
// check all sub- classes
Set<ClassElement> subClasses =
await searchEngine.searchAllSubtypes(superClass);
subClasses.add(superClass);
for (ClassElement subClass in subClasses) {
List<Element> subClassMembers = getChildren(subClass, name);
for (Element member in subClassMembers) {
if (member is ClassMemberElement) {
result.add(member);
}
}
}
}
return result;
}
/**
* If the [element] is a named parameter in a [MethodElement], return all
* corresponding named parameters in the method hierarchy.
*/
Future<List<ParameterElement>> getHierarchyNamedParameters(
SearchEngine searchEngine, ParameterElement element) async {
// TODO(brianwilkerson) Determine whether this await is necessary.
await null;
if (element.isNamed) {
Element method = element.enclosingElement;
if (method is MethodElement) {
var hierarchyParameters = <ParameterElement>[];
var hierarchyMembers = await getHierarchyMembers(searchEngine, method);
for (ClassMemberElement hierarchyMethod in hierarchyMembers) {
if (hierarchyMethod is MethodElement) {
for (var hierarchyParameter in hierarchyMethod.parameters) {
if (hierarchyParameter.isNamed &&
hierarchyParameter.name == element.name) {
hierarchyParameters.add(hierarchyParameter);
break;
}
}
}
}
return hierarchyParameters;
}
}
return [element];
}
/**
* Returns non-synthetic members of the given [ClassElement] and its super
* classes.
*
* Includes: fields, accessors and methods.
* Excludes: constructors and synthetic elements.
*/
List<Element> getMembers(ClassElement clazz) {
List<Element> members = <Element>[];
members.addAll(getClassMembers(clazz));
Set<ClassElement> superClasses = getSuperClasses(clazz);
for (ClassElement superClass in superClasses) {
members.addAll(getClassMembers(superClass));
}
return members;
}
/**
* Returns a [Set] with all direct and indirect superclasses of [seed].
*/
Set<ClassElement> getSuperClasses(ClassElement seed) {
Set<ClassElement> result = new HashSet<ClassElement>();
// prepare queue
List<ClassElement> queue = new List<ClassElement>();
queue.add(seed);
// process queue
while (queue.isNotEmpty) {
ClassElement current = queue.removeLast();
// add if not checked already
if (!result.add(current)) {
continue;
}
// append supertype
{
InterfaceType superType = current.supertype;
if (superType != null) {
queue.add(superType.element);
}
}
// append superclass constraints
for (InterfaceType interface in current.superclassConstraints) {
queue.add(interface.element);
}
// append interfaces
for (InterfaceType interface in current.interfaces) {
queue.add(interface.element);
}
}
// we don't need "seed" itself
result.remove(seed);
return result;
}
/**
* If the given [element] is a synthetic [PropertyAccessorElement] returns
* its variable, otherwise returns [element].
*/
Element getSyntheticAccessorVariable(Element element) {
if (element is PropertyAccessorElement) {
if (element.isSynthetic) {
return element.variable;
}
}
return element;
}