blob: 8026a69e25866bbd8c7b64611e5f21ca2163ed12 [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
bool get requiresTrustedCaller => false;
@override
Future<ErrorOr<List<DocumentHighlight>>> handle(
TextDocumentPositionParams params,
MessageInfo message,
CancellationToken token,
) async {
if (!isDartDocument(params.textDocument)) {
return success(const []);
}
var pos = params.position;
var path = pathOfDoc(params.textDocument);
var unit = await path.mapResult(requireResolvedUnit);
var offset = unit.mapResultSync((unit) => toOffset(unit.lineInfo, pos));
return (unit, offset).mapResults((unit, requestedOffset) async {
var collector = OccurrencesCollectorImpl();
addDartOccurrences(collector, unit.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.
var occurrences =
collector.allOccurrences
.where(
(occurrence) => occurrence.offsets.any(
(offset) => spansRequestedPosition(offset, occurrence.length),
),
)
.toList();
// No matches will return an empty list (not null) because that prevents
// the editor falling back to a text search.
return success(toHighlights(unit.lineInfo, occurrences));
});
}
}
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;
}