| //===-- CompletionRequest.cpp ---------------------------------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "lldb/Utility/CompletionRequest.h" |
| |
| using namespace lldb; |
| using namespace lldb_private; |
| |
| CompletionRequest::CompletionRequest(llvm::StringRef command_line, |
| unsigned raw_cursor_pos, |
| CompletionResult &result) |
| : m_command(command_line), m_raw_cursor_pos(raw_cursor_pos), |
| m_result(result) { |
| assert(raw_cursor_pos <= command_line.size() && "Out of bounds cursor?"); |
| |
| // We parse the argument up to the cursor, so the last argument in |
| // parsed_line is the one containing the cursor, and the cursor is after the |
| // last character. |
| llvm::StringRef partial_command(command_line.substr(0, raw_cursor_pos)); |
| m_parsed_line = Args(partial_command); |
| |
| if (GetParsedLine().GetArgumentCount() == 0) { |
| m_cursor_index = 0; |
| m_cursor_char_position = 0; |
| } else { |
| m_cursor_index = GetParsedLine().GetArgumentCount() - 1U; |
| m_cursor_char_position = |
| strlen(GetParsedLine().GetArgumentAtIndex(m_cursor_index)); |
| } |
| |
| // The cursor is after a space but the space is not part of the argument. |
| // Let's add an empty fake argument to the end to make sure the completion |
| // code. Note: The space could be part of the last argument when it's quoted. |
| if (partial_command.endswith(" ") && |
| !GetCursorArgumentPrefix().endswith(" ")) |
| AppendEmptyArgument(); |
| } |
| |
| std::string CompletionResult::Completion::GetUniqueKey() const { |
| |
| // We build a unique key for this pair of completion:description. We |
| // prefix the key with the length of the completion string. This prevents |
| // that we could get any collisions from completions pairs such as these: |
| // "foo:", "bar" would be "foo:bar", but will now be: "4foo:bar" |
| // "foo", ":bar" would be "foo:bar", but will now be: "3foo:bar" |
| |
| std::string result; |
| result.append(std::to_string(m_completion.size())); |
| result.append(m_completion); |
| result.append(std::to_string(static_cast<int>(m_mode))); |
| result.append(":"); |
| result.append(m_descripton); |
| return result; |
| } |
| |
| void CompletionResult::AddResult(llvm::StringRef completion, |
| llvm::StringRef description, |
| CompletionMode mode) { |
| Completion r(completion, description, mode); |
| |
| // Add the completion if we haven't seen the same value before. |
| if (m_added_values.insert(r.GetUniqueKey()).second) |
| m_results.push_back(r); |
| } |
| |
| void CompletionResult::GetMatches(StringList &matches) const { |
| matches.Clear(); |
| for (const Completion &completion : m_results) |
| matches.AppendString(completion.GetCompletion()); |
| } |
| |
| void CompletionResult::GetDescriptions(StringList &descriptions) const { |
| descriptions.Clear(); |
| for (const Completion &completion : m_results) |
| descriptions.AppendString(completion.GetDescription()); |
| } |