# typed: strict # frozen_string_literal: true module RubyLsp module Requests # ![Completion demo](../../completion.gif) # # The [completion](https://microsoft.github.io/language-server-protocol/specification#textDocument_completion) # suggests possible completions according to what the developer is typing. # # Currently supported targets: # - Classes # - Modules # - Constants # - Require paths # - Methods invoked on self only # # # Example # # ```ruby # require "ruby_lsp/requests" # --> completion: suggests `base_request`, `code_actions`, ... # # RubyLsp::Requests:: # --> completion: suggests `Completion`, `Hover`, ... # ``` class Completion < Listener extend T::Sig extend T::Generic ResponseType = type_member { { fixed: T::Array[Interface::CompletionItem] } } sig { override.returns(ResponseType) } attr_reader :_response sig do params( index: RubyIndexer::Index, nesting: T::Array[String], dispatcher: Prism::Dispatcher, message_queue: Thread::Queue, ).void end def initialize(index, nesting, dispatcher, message_queue) super(dispatcher, message_queue) @_response = T.let([], ResponseType) @index = index @nesting = nesting dispatcher.register( self, :on_string_node_enter, :on_constant_path_node_enter, :on_constant_read_node_enter, :on_call_node_enter, ) end sig { params(node: Prism::StringNode).void } def on_string_node_enter(node) @index.search_require_paths(node.content).map!(&:require_path).sort!.each do |path| @_response << build_completion(T.must(path), node) end end # Handle completion on regular constant references (e.g. `Bar`) sig { params(node: Prism::ConstantReadNode).void } def on_constant_read_node_enter(node) return if DependencyDetector.instance.typechecker name = node.slice candidates = @index.prefix_search(name, @nesting) candidates.each do |entries| complete_name = T.must(entries.first).name @_response << build_entry_completion( complete_name, name, node, entries, top_level?(complete_name), ) end end # Handle completion on namespaced constant references (e.g. `Foo::Bar`) sig { params(node: Prism::ConstantPathNode).void } def on_constant_path_node_enter(node) return if DependencyDetector.instance.typechecker name = node.slice top_level_reference = if name.start_with?("::") name = name.delete_prefix("::") true else false end # If we're trying to provide completion for an aliased namespace, we need to first discover it's real name in # order to find which possible constants match the desired search *namespace, incomplete_name = name.split("::") aliased_namespace = T.must(namespace).join("::") namespace_entries = @index.resolve(aliased_namespace, @nesting) return unless namespace_entries real_namespace = @index.follow_aliased_namespace(T.must(namespace_entries.first).name) candidates = @index.prefix_search("#{real_namespace}::#{incomplete_name}", top_level_reference ? [] : @nesting) candidates.each do |entries| # The only time we may have a private constant reference from outside of the namespace is if we're dealing # with ConstantPath and the entry name doesn't start with the current nesting first_entry = T.must(entries.first) next if first_entry.visibility == :private && !first_entry.name.start_with?("#{@nesting}::") constant_name = T.must(first_entry.name.split("::").last) full_name = aliased_namespace.empty? ? constant_name : "#{aliased_namespace}::#{constant_name}" @_response << build_entry_completion( full_name, name, node, entries, top_level_reference || top_level?(T.must(entries.first).name), ) end end sig { params(node: Prism::CallNode).void } def on_call_node_enter(node) return if DependencyDetector.instance.typechecker return unless self_receiver?(node) name = node.message return unless name receiver_entries = @index[@nesting.join("::")] return unless receiver_entries receiver = T.must(receiver_entries.first) candidates = T.cast(@index.prefix_search(name), T::Array[T::Array[RubyIndexer::Entry::Method]]) candidates.each do |entries| entry = entries.find { |e| e.owner&.name == receiver.name } next unless entry @_response << build_method_completion(entry, node) end end private sig { params(entry: RubyIndexer::Entry::Method, node: Prism::CallNode).returns(Interface::CompletionItem) } def build_method_completion(entry, node) name = entry.name parameters = entry.parameters new_text = parameters.empty? ? name : "#{name}(#{parameters.map(&:name).join(", ")})" Interface::CompletionItem.new( label: name, filter_text: name, text_edit: Interface::TextEdit.new(range: range_from_node(node), new_text: new_text), kind: Constant::CompletionItemKind::METHOD, label_details: Interface::CompletionItemLabelDetails.new( description: entry.file_name, ), documentation: markdown_from_index_entries(name, entry), ) end sig { params(label: String, node: Prism::StringNode).returns(Interface::CompletionItem) } def build_completion(label, node) Interface::CompletionItem.new( label: label, text_edit: Interface::TextEdit.new( range: range_from_node(node), new_text: label, ), kind: Constant::CompletionItemKind::REFERENCE, ) end sig do params( real_name: String, incomplete_name: String, node: Prism::Node, entries: T::Array[RubyIndexer::Entry], top_level: T::Boolean, ).returns(Interface::CompletionItem) end def build_entry_completion(real_name, incomplete_name, node, entries, top_level) first_entry = T.must(entries.first) kind = case first_entry when RubyIndexer::Entry::Class Constant::CompletionItemKind::CLASS when RubyIndexer::Entry::Module Constant::CompletionItemKind::MODULE when RubyIndexer::Entry::Constant Constant::CompletionItemKind::CONSTANT else Constant::CompletionItemKind::REFERENCE end insertion_text = real_name.dup filter_text = real_name.dup # If we have two entries with the same name inside the current namespace and the user selects the top level # option, we have to ensure it's prefixed with `::` or else we're completing the wrong constant. For example: # If we have the index with ["Foo::Bar", "Bar"], and we're providing suggestions for `B` inside a `Foo` module, # then selecting the `Foo::Bar` option needs to complete to `Bar` and selecting the top level `Bar` option needs # to complete to `::Bar`. if top_level insertion_text.prepend("::") filter_text.prepend("::") end # If the user is searching for a constant inside the current namespace, then we prefer completing the short name # of that constant. E.g.: # # module Foo # class Bar # end # # Foo::B # --> completion inserts `Bar` instead of `Foo::Bar` # end @nesting.each do |namespace| prefix = "#{namespace}::" shortened_name = insertion_text.delete_prefix(prefix) # If a different entry exists for the shortened name, then there's a conflict and we should not shorten it conflict_name = "#{@nesting.join("::")}::#{shortened_name}" break if real_name != conflict_name && @index[conflict_name] insertion_text = shortened_name # If the user is typing a fully qualified name `Foo::Bar::Baz`, then we should not use the short name (e.g.: # `Baz`) as filtering. So we only shorten the filter text if the user is not including the namespaces in their # typing filter_text.delete_prefix!(prefix) unless incomplete_name.start_with?(prefix) end # When using a top level constant reference (e.g.: `::Bar`), the editor includes the `::` as part of the filter. # For these top level references, we need to include the `::` as part of the filter text or else it won't match # the right entries in the index Interface::CompletionItem.new( label: real_name, filter_text: filter_text, text_edit: Interface::TextEdit.new( range: range_from_node(node), new_text: insertion_text, ), kind: kind, label_details: Interface::CompletionItemLabelDetails.new( description: entries.map(&:file_name).join(","), ), documentation: markdown_from_index_entries(real_name, entries), ) end # Check if there are any conflicting names for `entry_name`, which would require us to use a top level reference. # For example: # # ```ruby # class Bar; end # # module Foo # class Bar; end # # # in this case, the completion for `Bar` conflicts with `Foo::Bar`, so we can't suggest `Bar` as the # # completion, but instead need to suggest `::Bar` # B # end # ``` sig { params(entry_name: String).returns(T::Boolean) } def top_level?(entry_name) @nesting.length.downto(0).each do |i| prefix = T.must(@nesting[0...i]).join("::") full_name = prefix.empty? ? entry_name : "#{prefix}::#{entry_name}" next if full_name == entry_name return true if @index[full_name] end false end end end end