blob: 560edbb7d225820618ee02b23c6315b9bf9a67ed [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 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/dart/element/visitor.dart';
/**
* Return the [Element] that is either [root], or one of its direct or
* indirect children, and has the given [nameOffset].
*/
Element findElementByNameOffset(Element root, int nameOffset) {
if (root == null) {
return null;
}
try {
var visitor = new _ElementByNameOffsetVisitor(nameOffset);
root.accept(visitor);
} on Element catch (result) {
return result;
}
return null;
}
/**
* Uses [processor] to visit all of the children of [element].
* If [processor] returns `true`, then children of a child are visited too.
*/
void visitChildren(Element element, BoolElementProcessor processor) {
element.visitChildren(new _ElementVisitorAdapter(processor));
}
/**
* Uses [processor] to visit all of the top-level elements of [library].
*/
void visitLibraryTopLevelElements(
LibraryElement library, VoidElementProcessor processor) {
library.visitChildren(new _TopLevelElementsVisitor(processor));
}
/**
* An [Element] processor function type.
* If `true` is returned, children of [element] will be visited.
*/
typedef bool BoolElementProcessor(Element element);
/**
* An [Element] processor function type.
*/
typedef void VoidElementProcessor(Element element);
/**
* A visitor that finds the deep-most [Element] that contains the [nameOffset].
*/
class _ElementByNameOffsetVisitor extends GeneralizingElementVisitor {
final int nameOffset;
_ElementByNameOffsetVisitor(this.nameOffset);
visitElement(Element element) {
if (element.nameOffset != -1 &&
!element.isSynthetic &&
element.nameOffset == nameOffset) {
throw element;
}
super.visitElement(element);
}
}
/**
* A [GeneralizingElementVisitor] adapter for [ElementProcessor].
*/
class _ElementVisitorAdapter extends GeneralizingElementVisitor {
final BoolElementProcessor processor;
_ElementVisitorAdapter(this.processor);
@override
void visitElement(Element element) {
bool visitChildren = processor(element);
if (visitChildren == true) {
element.visitChildren(this);
}
}
}
/**
* A [GeneralizingElementVisitor] for visiting top-level elements.
*/
class _TopLevelElementsVisitor extends GeneralizingElementVisitor {
final VoidElementProcessor processor;
_TopLevelElementsVisitor(this.processor);
@override
void visitElement(Element element) {
if (element is CompilationUnitElement) {
element.visitChildren(this);
} else {
processor(element);
}
}
}