Index: pkg/analysis_server/lib/src/services/completion/dart/inherited_contributor.dart |
diff --git a/pkg/analysis_server/lib/src/services/completion/dart/inherited_contributor.dart b/pkg/analysis_server/lib/src/services/completion/dart/inherited_contributor.dart |
new file mode 100644 |
index 0000000000000000000000000000000000000000..3b56e40881498c36934a5c9bb6dfec90b8f66712 |
--- /dev/null |
+++ b/pkg/analysis_server/lib/src/services/completion/dart/inherited_contributor.dart |
@@ -0,0 +1,175 @@ |
+// Copyright (c) 2015, 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. |
+ |
+library services.completion.computer.dart.invocation; |
+ |
+import 'dart:async'; |
+ |
+import 'package:analysis_server/src/protocol_server.dart' |
+ show CompletionSuggestion, CompletionSuggestionKind, SourceChange; |
+import 'package:analysis_server/src/protocol_server.dart' as protocol |
+ hide CompletionSuggestion, CompletionSuggestionKind; |
+import 'package:analysis_server/src/provisional/completion/completion_core.dart'; |
+import 'package:analysis_server/src/provisional/completion/completion_dart.dart'; |
+import 'package:analysis_server/src/provisional/completion/dart/completion_target.dart'; |
+import 'package:analysis_server/src/services/completion/dart_completion_manager.dart' |
+ show DART_RELEVANCE_HIGH; |
+import 'package:analyzer/src/generated/ast.dart'; |
+import 'package:analyzer/src/generated/element.dart'; |
+import 'package:analyzer/src/generated/resolver.dart'; |
+import 'package:analyzer/src/generated/source.dart'; |
+// import 'package:analysis_server/plugin/edit/utilities/change_builder_dart.dart'; |
+// import 'package:analyzer/src/generated/engine.dart'; |
+ |
+/** |
+ * A completion contributor used to suggest replacing partial identifiers inside |
+ * a class declaration with templates for inherited members. |
+ */ |
+class InheritedContributor implements DartCompletionContributor { |
+ @override |
+ Future<List<CompletionSuggestion>> computeSuggestions( |
+ DartCompletionRequest request) async { |
+ // Determine if the target looks like a partial identifier |
+ // inside a class declaration |
+ SimpleIdentifier targetId = _getTargetId(request.target); |
+ if (targetId == null) { |
+ return EMPTY_LIST; |
+ } |
+ |
+ // Partially resolve the compilation unit |
+ CompilationUnit unit = await request.resolveDeclarationsInScope(); |
+ // Gracefully degrade if the compilation unit could not be resolved |
+ // e.g. detached part file or source change |
+ if (unit == null) { |
+ return EMPTY_LIST; |
+ } |
+ |
+ // Recompute the target since resolution may have changed it |
+ targetId = _getTargetId(request.target); |
+ if (targetId == null) { |
+ return EMPTY_LIST; |
+ } |
+ ClassDeclaration classDecl = |
+ targetId.getAncestor((p) => p is ClassDeclaration); |
+ if (classDecl == null) { |
+ return EMPTY_LIST; |
+ } |
+ |
+ // Generate a collection of inherited members |
+ ClassElement classElem = classDecl.element; |
+ InheritanceManager manager = new InheritanceManager(classElem.library); |
+ MemberMap map = manager.getMapOfMembersInheritedFromInterfaces(classElem); |
+ List<String> memberNames = _computeMemberNames(map, classElem); |
+ |
+ // Build suggestions |
+ List<CompletionSuggestion> suggestions = <CompletionSuggestion>[]; |
+ for (String memberName in memberNames) { |
+ ExecutableElement element = map.get(memberName); |
+ // Gracefully degrade if the overridden element has not been resolved. |
+ if (element.returnType != null) { |
+ CompletionSuggestion suggestion = |
+ _buildSuggestion(request, targetId, unit, element); |
+ if (suggestion != null) { |
+ suggestions.add(suggestion); |
+ } |
+ } |
+ } |
+ return suggestions; |
+ } |
+ |
+ /** |
+ * If the target looks like a partial identifier inside a class declaration |
+ * then return that identifier, otherwise return `null`. |
+ */ |
+ SimpleIdentifier _getTargetId(CompletionTarget target) { |
+ AstNode node = target.containingNode; |
+ if (node is ClassDeclaration) { |
+ Object entity = target.entity; |
+ if (entity is FieldDeclaration) { |
+ NodeList<VariableDeclaration> variables = entity.fields.variables; |
+ if (variables.length == 1) { |
+ SimpleIdentifier targetId = variables[0].name; |
+ if (targetId.name.isEmpty) { |
+ return targetId; |
+ } |
+ } |
+ } |
+ } |
+ return null; |
+ } |
+ |
+ /** |
+ * Return a template for an override of the given [element] in the given |
+ * [source]. If selected, the template will replace [targetId]. |
+ */ |
+ String _buildRepacementText(Source source, SimpleIdentifier targetId, |
+ CompilationUnit unit, ExecutableElement element) { |
+ // AnalysisContext context = element.context; |
+ // Inject partially resolved unit for use by change builder |
+ // DartChangeBuilder builder = new DartChangeBuilder(context, unit); |
+ // builder.addFileEdit(source, context.getModificationStamp(source), |
+ // (DartFileEditBuilder builder) { |
+ // builder.addReplacement(targetId.offset, targetId.length, |
+ // (DartEditBuilder builder) { |
+ // builder.writeOverrideOfInheritedMember(element); |
+ // }); |
+ // }); |
+ // return builder.sourceChange.edits[0].edits[0].replacement.trim(); |
+ return ''; |
+ } |
+ |
+ /** |
+ * Build a suggestion to replace [targetId] in the given [unit] |
+ * with an override of the given [element]. |
+ */ |
+ CompletionSuggestion _buildSuggestion( |
+ DartCompletionRequest request, |
+ SimpleIdentifier targetId, |
+ CompilationUnit unit, |
+ ExecutableElement element) { |
+ String completion = |
+ _buildRepacementText(request.source, targetId, unit, element); |
+ if (completion == null || completion.length == 0) { |
+ return null; |
+ } |
+ CompletionSuggestion suggestion = new CompletionSuggestion( |
+ CompletionSuggestionKind.IDENTIFIER, |
+ DART_RELEVANCE_HIGH, |
+ completion, |
+ targetId.offset, |
+ 0, |
+ element.isDeprecated, |
+ false); |
+ suggestion.element = protocol.convertElement(element); |
+ return suggestion; |
+ } |
+ |
+ /** |
+ * Return a list containing the names of all of the inherited but not |
+ * implemented members of the class represented by the given [element]. |
+ * The [map] is used to find all of the members that are inherited. |
+ */ |
+ List<String> _computeMemberNames(MemberMap map, ClassElement element) { |
+ List<String> memberNames = <String>[]; |
+ int count = map.size; |
+ for (int i = 0; i < count; i++) { |
+ String memberName = map.getKey(i); |
+ if (!_hasMember(element, memberName)) { |
+ memberNames.add(memberName); |
+ } |
+ } |
+ return memberNames; |
+ } |
+ |
+ /** |
+ * Return `true` if the given [classElement] directly declares a member with |
+ * the given [memberName]. |
+ */ |
+ bool _hasMember(ClassElement classElement, String memberName) { |
+ return classElement.getField(memberName) != null || |
+ classElement.getGetter(memberName) != null || |
+ classElement.getMethod(memberName) != null || |
+ classElement.getSetter(memberName) != null; |
+ } |
+} |