| // 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); |
| } |
| } |
| } |