|
4 | 4 |
|
5 | 5 | import 'package:analysis_server/lsp_protocol/protocol.dart' hide Element; |
6 | 6 | import 'package:analysis_server/src/lsp/constants.dart'; |
7 | | -import 'package:analysis_server/src/lsp/handlers/custom/abstract_go_to.dart'; |
8 | | -import 'package:analyzer/dart/analysis/results.dart'; |
9 | | -import 'package:analyzer/dart/element/element.dart'; |
| 7 | +import 'package:analysis_server/src/lsp/error_or.dart'; |
| 8 | +import 'package:analysis_server/src/lsp/handlers/handlers.dart'; |
| 9 | +import 'package:analysis_server/src/lsp/mapping.dart'; |
| 10 | +import 'package:analyzer/dart/ast/ast.dart'; |
| 11 | +import 'package:analyzer/dart/element/element2.dart'; |
10 | 12 | import 'package:analyzer/src/dart/analysis/session.dart'; |
| 13 | +import 'package:analyzer/src/dart/ast/element_locator.dart'; |
11 | 14 | import 'package:analyzer/src/dart/element/inheritance_manager3.dart'; |
12 | | -import 'package:analyzer/src/utilities/extensions/element.dart'; |
| 15 | +import 'package:analyzer/src/utilities/extensions/ast.dart'; |
13 | 16 |
|
14 | | -class SuperHandler extends AbstractGoToHandler { |
| 17 | +class SuperHandler |
| 18 | + extends SharedMessageHandler<TextDocumentPositionParams, Location?> { |
15 | 19 | SuperHandler(super.server); |
16 | 20 |
|
17 | 21 | @override |
18 | 22 | Method get handlesMessage => CustomMethods.super_; |
19 | 23 |
|
| 24 | + @override |
| 25 | + LspJsonHandler<TextDocumentPositionParams> get jsonHandler => |
| 26 | + TextDocumentPositionParams.jsonHandler; |
| 27 | + |
20 | 28 | @override |
21 | 29 | bool get requiresTrustedCaller => false; |
22 | 30 |
|
23 | 31 | @override |
24 | | - Either2<Location?, List<Location>> findRelatedLocations( |
25 | | - Element element, |
26 | | - ResolvedLibraryResult libraryResult, |
27 | | - ResolvedUnitResult unit, |
28 | | - String? prefix) { |
29 | | - // Although the base class supports returning multiple elements, this |
30 | | - // handler is documented to only return a single element. |
31 | | - // Changing this to return a list could be a breaking change for |
32 | | - // clients. |
33 | | - return Either2.t1( |
34 | | - elementToLocation(_SuperComputer().computeSuper(element))); |
| 32 | + Future<ErrorOr<Location?>> handle( |
| 33 | + TextDocumentPositionParams params, |
| 34 | + MessageInfo message, |
| 35 | + CancellationToken token, |
| 36 | + ) async { |
| 37 | + if (!isDartDocument(params.textDocument)) { |
| 38 | + return success(null); |
| 39 | + } |
| 40 | + |
| 41 | + var pos = params.position; |
| 42 | + var path = pathOfDoc(params.textDocument); |
| 43 | + var unit = await path.mapResult(requireResolvedUnit); |
| 44 | + var offset = unit.mapResultSync((unit) => toOffset(unit.lineInfo, pos)); |
| 45 | + |
| 46 | + return (unit, offset).mapResultsSync((unit, offset) { |
| 47 | + // Find the nearest node that could have a super. |
| 48 | + var node = unit.unit |
| 49 | + .nodeCovering(offset: offset) |
| 50 | + ?.thisOrAncestorMatching(_canHaveSuper); |
| 51 | + if (node == null) { |
| 52 | + return success(null); |
| 53 | + } |
| 54 | + |
| 55 | + var element = ElementLocator.locate2(node); |
| 56 | + if (element == null) { |
| 57 | + return success(null); |
| 58 | + } |
| 59 | + |
| 60 | + var targetFragment = _SuperComputer().computeSuper(element); |
| 61 | + var location = fragmentToLocation(uriConverter, targetFragment); |
| 62 | + return success(location); |
| 63 | + }); |
35 | 64 | } |
| 65 | + |
| 66 | + /// Returns whether [node] is something that can be considered to have a |
| 67 | + /// "super" (a class or a class member). |
| 68 | + bool _canHaveSuper(AstNode node) => |
| 69 | + node is ClassDeclaration || node is ClassMember; |
36 | 70 | } |
37 | 71 |
|
38 | 72 | class _SuperComputer { |
39 | | - Element? computeSuper(Element element) { |
| 73 | + Fragment? computeSuper(Element2 element) { |
40 | 74 | return switch (element) { |
41 | | - ConstructorElement element => _findSuperConstructor(element), |
42 | | - InterfaceElement element => _findSuperClass(element), |
| 75 | + ConstructorElement2 element => _findSuperConstructor(element), |
| 76 | + InterfaceElement2 element => _findSuperClass(element), |
43 | 77 | _ => _findSuperMember(element), |
44 | 78 | }; |
45 | 79 | } |
46 | 80 |
|
47 | | - Element? _findSuperClass(InterfaceElement element) { |
48 | | - return element.supertype?.element; |
| 81 | + Fragment? _findSuperClass(InterfaceElement2 element) { |
| 82 | + // For super classes, we use the first fragment (the original declaration). |
| 83 | + // This differs from methods/getters because we jump to the end of the |
| 84 | + // augmentation chain for those. |
| 85 | + return element.supertype?.element3.firstFragment; |
49 | 86 | } |
50 | 87 |
|
51 | | - Element? _findSuperConstructor(ConstructorElement element) { |
52 | | - return element.superConstructor?.withAugmentations.last; |
| 88 | + Fragment? _findSuperConstructor(ConstructorElement2 element) { |
| 89 | + return _lastFragment(element.superConstructor2); |
53 | 90 | } |
54 | 91 |
|
55 | | - Element? _findSuperMember(Element element) { |
| 92 | + Fragment? _findSuperMember(Element2 element) { |
56 | 93 | var session = element.session; |
57 | 94 | if (session is! AnalysisSessionImpl) { |
58 | 95 | return null; |
59 | 96 | } |
60 | 97 |
|
61 | 98 | var inheritanceManager = session.inheritanceManager; |
62 | | - var elementName = element.name; |
63 | | - var interfaceElement = element.thisOrAncestorOfType<InterfaceElement>(); |
| 99 | + var elementName = element.name3; |
| 100 | + var interfaceElement = element.thisOrAncestorOfType2<InterfaceElement2>(); |
64 | 101 |
|
65 | 102 | if (elementName == null || interfaceElement == null) { |
66 | 103 | return null; |
67 | 104 | } |
68 | 105 |
|
69 | | - var name = Name(interfaceElement.library.source.uri, elementName); |
70 | | - var member = inheritanceManager.getInherited2(interfaceElement, name); |
| 106 | + var name = Name( |
| 107 | + interfaceElement.library2.firstFragment.source.uri, |
| 108 | + elementName, |
| 109 | + ); |
| 110 | + var member = inheritanceManager.getInherited4(interfaceElement, name); |
71 | 111 |
|
72 | | - return member; |
| 112 | + return _lastFragment(member); |
| 113 | + } |
| 114 | + |
| 115 | + Fragment? _lastFragment(FragmentedElement? element) { |
| 116 | + Fragment? fragment = element?.firstFragment; |
| 117 | + while (fragment?.nextFragment != null) { |
| 118 | + fragment = fragment?.nextFragment; |
| 119 | + } |
| 120 | + return fragment; |
73 | 121 | } |
74 | 122 | } |
0 commit comments