| // 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/ast/ast.dart'; |
| import 'package:analyzer/src/dart/ast/utilities.dart'; |
| import 'package:test/test.dart'; |
| import 'package:test_reflective_loader/test_reflective_loader.dart'; |
| |
| import '../../../generated/parser_test_base.dart' show ParserTestCase; |
| import '../../../util/ast_type_matchers.dart'; |
| |
| main() { |
| defineReflectiveSuite(() { |
| defineReflectiveTests(NodeLocatorTest); |
| defineReflectiveTests(NodeLocator2Test); |
| }); |
| } |
| |
| @reflectiveTest |
| class NodeLocator2Test extends _SharedNodeLocatorTests { |
| @override |
| AstNode? locate( |
| CompilationUnit unit, |
| int start, [ |
| int? end, |
| ]) { |
| var locator = NodeLocator2(start, end); |
| var node = locator.searchWithin(unit)!; |
| return node; |
| } |
| |
| void test_onlyStartOffset() { |
| String code = ' f() {} '; |
| // 01234567 |
| CompilationUnit unit = parseCompilationUnit(code); |
| var function = unit.declarations.single as FunctionDeclaration; |
| var expression = function.functionExpression; |
| var body = expression.body as BlockFunctionBody; |
| expect(NodeLocator2(0).searchWithin(unit), same(unit)); |
| expect(NodeLocator2(1).searchWithin(unit), same(function)); |
| expect(NodeLocator2(2).searchWithin(unit), same(function)); |
| expect(NodeLocator2(3).searchWithin(unit), same(expression.parameters)); |
| expect(NodeLocator2(4).searchWithin(unit), same(expression)); |
| expect(NodeLocator2(5).searchWithin(unit), same(body.block)); |
| expect(NodeLocator2(6).searchWithin(unit), same(body.block)); |
| expect(NodeLocator2(7).searchWithin(unit), same(unit)); |
| expect(NodeLocator2(100).searchWithin(unit), isNull); |
| } |
| |
| void test_startEndOffset() { |
| String code = ' f() {} '; |
| // 01234567 |
| CompilationUnit unit = parseCompilationUnit(code); |
| var function = unit.declarations.single as FunctionDeclaration; |
| expect(NodeLocator2(-1, 2).searchWithin(unit), isNull); |
| expect(NodeLocator2(0, 2).searchWithin(unit), same(unit)); |
| expect(NodeLocator2(1, 2).searchWithin(unit), same(function)); |
| expect(NodeLocator2(1, 3).searchWithin(unit), same(function)); |
| expect(NodeLocator2(1, 4).searchWithin(unit), same(function)); |
| expect(NodeLocator2(5, 7).searchWithin(unit), same(unit)); |
| expect(NodeLocator2(5, 100).searchWithin(unit), isNull); |
| expect(NodeLocator2(100, 200).searchWithin(unit), isNull); |
| } |
| } |
| |
| @reflectiveTest |
| class NodeLocatorTest extends _SharedNodeLocatorTests { |
| @override |
| AstNode? locate( |
| CompilationUnit unit, |
| int start, [ |
| int? end, |
| ]) { |
| var locator = NodeLocator(start, end); |
| var node = locator.searchWithin(unit)!; |
| expect(locator.foundNode, same(node)); |
| return node; |
| } |
| |
| void test_range() { |
| CompilationUnit unit = parseCompilationUnit("library myLib;"); |
| var node = _assertLocate(unit, 4, 10); |
| expect(node, isLibraryDirective); |
| } |
| |
| void test_searchWithin_null() { |
| NodeLocator locator = NodeLocator(0, 0); |
| expect(locator.searchWithin(null), isNull); |
| } |
| |
| void test_searchWithin_offset() { |
| CompilationUnit unit = parseCompilationUnit("library myLib;"); |
| var node = _assertLocate(unit, 10, 10); |
| expect(node, isSimpleIdentifier); |
| } |
| |
| void test_searchWithin_offsetAfterNode() { |
| CompilationUnit unit = parseCompilationUnit(r''' |
| class A {} |
| class B {}'''); |
| NodeLocator locator = NodeLocator(1024, 1024); |
| var node = locator.searchWithin(unit.declarations[0]); |
| expect(node, isNull); |
| } |
| |
| void test_searchWithin_offsetBeforeNode() { |
| CompilationUnit unit = parseCompilationUnit(r''' |
| class A {} |
| class B {}'''); |
| NodeLocator locator = NodeLocator(0, 0); |
| var node = locator.searchWithin(unit.declarations[1]); |
| expect(node, isNull); |
| } |
| } |
| |
| abstract class _SharedNodeLocatorTests extends ParserTestCase { |
| AstNode? locate( |
| CompilationUnit unit, |
| int start, [ |
| int? end, |
| ]); |
| |
| void test_searchWithin_class_afterName_beforeTypeParameters() { |
| var source = r''' |
| class A<T> {} |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('<T> {}')); |
| expect(node, isClassDeclaration); |
| } |
| |
| void test_searchWithin_constructor_afterName_beforeParameters() { |
| var source = r''' |
| class A { |
| A() {} |
| } |
| '''; |
| var unit = parseCompilationUnit(source); |
| // TODO(dantup): Update these tests to use markers. |
| var node = _assertLocate(unit, source.indexOf('() {}')); |
| expect(node, isConstructorDeclaration); |
| } |
| |
| void test_searchWithin_function_afterName_beforeParameters() { |
| var source = r''' |
| void f() {} |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('() {}')); |
| expect(node, isFunctionDeclaration); |
| } |
| |
| void test_searchWithin_function_afterName_beforeTypeParameters() { |
| var source = r''' |
| void f<T>() {} |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('<T>() {}')); |
| expect(node, isFunctionDeclaration); |
| } |
| |
| void test_searchWithin_method_afterName_beforeParameters() { |
| var source = r''' |
| class A { |
| void m() {} |
| } |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('() {}')); |
| expect(node, isMethodDeclaration); |
| } |
| |
| void test_searchWithin_method_afterName_beforeTypeParameters() { |
| var source = r''' |
| class A { |
| void m<T>() {} |
| } |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('<T>() {}')); |
| expect(node, isMethodDeclaration); |
| } |
| |
| void test_searchWithin_namedConstructor_afterName_beforeParameters() { |
| var source = r''' |
| class A { |
| A.c() {} |
| } |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('() {}')); |
| expect(node, isConstructorDeclaration); |
| } |
| |
| void test_searchWithin_setter_afterName_beforeParameters() { |
| var source = r''' |
| set s(int i) {} |
| '''; |
| var unit = parseCompilationUnit(source); |
| var node = _assertLocate(unit, source.indexOf('(int i)')); |
| expect(node, isFunctionDeclaration); |
| } |
| |
| AstNode _assertLocate( |
| CompilationUnit unit, |
| int start, [ |
| int? end, |
| ]) { |
| end ??= start; |
| var node = locate(unit, start, end)!; |
| expect(node.offset <= start, isTrue, reason: "Node starts after range"); |
| expect(node.offset + node.length > end, isTrue, |
| reason: "Node ends before range"); |
| return node; |
| } |
| } |