blob: c0124a7a8bb7f4853cdc791938084509d16ce039 [file] [log] [blame]
// Copyright (c) 2019, 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:analysis_server/lsp_protocol/protocol.dart';
import 'package:analysis_server/src/domains/analysis/occurrences.dart';
import 'package:analysis_server/src/domains/analysis/occurrences_dart.dart';
import 'package:analysis_server/src/lsp/error_or.dart';
import 'package:analysis_server/src/lsp/handlers/handlers.dart';
import 'package:analysis_server/src/lsp/mapping.dart';
import 'package:analysis_server/src/lsp/registration/feature_registration.dart';
typedef StaticOptions = Either2<bool, DocumentHighlightOptions>;
class DocumentHighlightsHandler extends SharedMessageHandler<
TextDocumentPositionParams, List<DocumentHighlight>?> {
DocumentHighlightsHandler(super.server);
@override
Method get handlesMessage => Method.textDocument_documentHighlight;
@override
LspJsonHandler<TextDocumentPositionParams> get jsonHandler =>
TextDocumentPositionParams.jsonHandler;
@override
Future<ErrorOr<List<DocumentHighlight>?>> handle(
TextDocumentPositionParams params,
MessageInfo message,
CancellationToken token) async {
if (!isDartDocument(params.textDocument)) {
return success(const []);
}
final pos = params.position;
final path = pathOfDoc(params.textDocument);
final unit = await path.mapResult(requireResolvedUnit);
final offset = await unit.mapResult((unit) => toOffset(unit.lineInfo, pos));
return offset.mapResult((requestedOffset) {
final collector = OccurrencesCollectorImpl();
addDartOccurrences(collector, unit.result.unit);
/// Checks whether an Occurrence offset/length spans the requested
/// offset.
///
/// It's possible multiple occurrences might match because some nodes
/// such as object destructuring might match multiple elements (for
/// example the object getter and a declared variable).
bool spansRequestedPosition(int offset, int length) {
return offset <= requestedOffset && offset + length >= requestedOffset;
}
// Find an occurrence that has an instance that spans the position.
final occurrences = collector.allOccurrences
.where((occurrence) => occurrence.offsets.any(
(offset) => spansRequestedPosition(offset, occurrence.length)))
.toList();
if (occurrences.isNotEmpty) {
return success(toHighlights(unit.result.lineInfo, occurrences));
}
// No matches.
return success(null);
});
}
}
class DocumentHighlightsRegistrations extends FeatureRegistration
with SingleDynamicRegistration, StaticRegistration<StaticOptions> {
DocumentHighlightsRegistrations(super.info);
@override
ToJsonable? get options =>
TextDocumentRegistrationOptions(documentSelector: fullySupportedTypes);
@override
Method get registrationMethod => Method.textDocument_documentHighlight;
@override
StaticOptions get staticOptions => Either2.t1(true);
@override
bool get supportsDynamic => clientDynamic.documentHighlights;
}