blob: d03012675f0205b60f89c1dfe936eda00bb0e4d4 [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.
library services.completion.computer.dart.toplevel;
import 'dart:async';
import 'package:analysis_server/src/protocol.dart' hide Element;
import 'package:analysis_server/src/services/completion/dart_completion_manager.dart';
import 'package:analysis_server/src/services/completion/suggestion_builder.dart';
import 'package:analysis_server/src/services/search/search_engine.dart';
import 'package:analyzer/src/generated/ast.dart';
import 'package:analyzer/src/generated/element.dart';
/**
* A computer for calculating imported class and top level variable
* `completion.getSuggestions` request results.
*/
class ImportedTypeComputer extends DartCompletionComputer {
@override
bool computeFast(DartCompletionRequest request) {
// TODO: implement computeFast
// - compute results based upon current search, then replace those results
// during the full compute phase
// - filter results based upon completion offset
return false;
}
@override
Future<bool> computeFull(DartCompletionRequest request) {
return request.node.accept(new _ImportedTypeVisitor(request));
}
}
/**
* A visitor for determining which imported class and top level variable
* should be suggested and building those suggestions.
*/
class _ImportedTypeVisitor extends GeneralizingAstVisitor<Future<bool>> {
final DartCompletionRequest request;
_ImportedTypeVisitor(this.request);
@override
Future<bool> visitCombinator(Combinator node) {
NamespaceDirective directive =
node.getAncestor((parent) => parent is NamespaceDirective);
if (directive != null) {
LibraryElement library = directive.uriElement;
LibraryElementSuggestionBuilder.suggestionsFor(request, library);
return new Future.value(true);
}
return new Future.value(false);
}
@override
Future<bool> visitNode(AstNode node) {
return _addImportedElements();
}
@override
Future<bool> visitSimpleIdentifier(SimpleIdentifier node) {
return node.parent.accept(this);
}
@override
Future<bool> visitVariableDeclaration(VariableDeclaration node) {
// Do not add suggestions if editing the name in a var declaration
SimpleIdentifier name = node.name;
if (name == null ||
name.offset < request.offset ||
request.offset > name.end) {
return visitNode(node);
}
return new Future.value(false);
}
Future<bool> _addImportedElements() {
var future = request.searchEngine.searchTopLevelDeclarations('');
return future.then((List<SearchMatch> matches) {
Set<LibraryElement> visibleLibs = new Set<LibraryElement>();
Set<LibraryElement> excludedLibs = new Set<LibraryElement>();
Map<LibraryElement, Set<String>> showNames =
new Map<LibraryElement, Set<String>>();
Map<LibraryElement, Set<String>> hideNames =
new Map<LibraryElement, Set<String>>();
// Exclude elements from the local library
// as they will be included by the LocalComputer
excludedLibs.add(request.unit.element.library);
// Build the set of visible and excluded libraries
// and the list of names that should be shown or hidden
request.unit.directives.forEach((Directive directive) {
if (directive is ImportDirective) {
LibraryElement lib = directive.element.importedLibrary;
if (directive.prefix == null) {
visibleLibs.add(lib);
directive.combinators.forEach((Combinator combinator) {
if (combinator is ShowCombinator) {
showNames[lib] = combinator.shownNames.map(
(SimpleIdentifier id) => id.name).toSet();
} else if (combinator is HideCombinator) {
hideNames[lib] = combinator.hiddenNames.map(
(SimpleIdentifier id) => id.name).toSet();
}
});
} else {
excludedLibs.add(lib);
}
}
});
// Compute the set of possible classes, functions, and top level variables
matches.forEach((SearchMatch match) {
if (match.kind == MatchKind.DECLARATION) {
Element element = match.element;
LibraryElement lib = element.library;
if (element.isPublic && !excludedLibs.contains(lib)) {
String completion = element.displayName;
Set<String> show = showNames[lib];
Set<String> hide = hideNames[lib];
if ((show == null || show.contains(completion)) &&
(hide == null || !hide.contains(completion))) {
request.suggestions.add(
new CompletionSuggestion(
new CompletionSuggestionKind.fromElementKind(element.kind),
visibleLibs.contains(lib) || lib.isDartCore ?
CompletionRelevance.DEFAULT :
CompletionRelevance.LOW,
completion,
completion.length,
0,
element.isDeprecated,
false));
}
}
}
});
return true;
});
}
}