144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
644d93782SGreg Clayton //
744d93782SGreg Clayton //===----------------------------------------------------------------------===//
844d93782SGreg Clayton 
92f3df613SZachary Turner #include "lldb/Core/IOHandler.h"
102f3df613SZachary Turner 
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
167c9aa073STodd Fiala #if defined(__APPLE__)
177c9aa073STodd Fiala #include <deque>
187c9aa073STodd Fiala #endif
1944d93782SGreg Clayton #include <string>
2044d93782SGreg Clayton 
2144d93782SGreg Clayton #include "lldb/Core/Debugger.h"
2244d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
23672d2c12SJonas Devlieghere #include "lldb/Host/File.h"
24672d2c12SJonas Devlieghere #include "lldb/Utility/Predicate.h"
25672d2c12SJonas Devlieghere #include "lldb/Utility/Status.h"
26672d2c12SJonas Devlieghere #include "lldb/Utility/StreamString.h"
27672d2c12SJonas Devlieghere #include "lldb/Utility/StringList.h"
28672d2c12SJonas Devlieghere #include "lldb/lldb-forward.h"
292f3df613SZachary Turner 
30cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3144d93782SGreg Clayton #include "lldb/Host/Editline.h"
32cacde7dfSTodd Fiala #endif
3344d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
352f3df613SZachary Turner #ifndef LLDB_DISABLE_CURSES
362f3df613SZachary Turner #include "lldb/Breakpoint/BreakpointLocation.h"
372f3df613SZachary Turner #include "lldb/Core/Module.h"
382f3df613SZachary Turner #include "lldb/Core/ValueObject.h"
392f3df613SZachary Turner #include "lldb/Core/ValueObjectRegister.h"
4044d93782SGreg Clayton #include "lldb/Symbol/Block.h"
4144d93782SGreg Clayton #include "lldb/Symbol/Function.h"
4244d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
43c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
44c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
452f3df613SZachary Turner #include "lldb/Target/RegisterContext.h"
46c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
472f3df613SZachary Turner #include "lldb/Target/StopInfo.h"
48b9c1b51eSKate Stone #include "lldb/Target/Target.h"
49b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
50d821c997SPavel Labath #include "lldb/Utility/State.h"
51c5dac77aSEugene Zelenko #endif
527c9aa073STodd Fiala 
53672d2c12SJonas Devlieghere #include "llvm/ADT/StringRef.h"
542f3df613SZachary Turner 
55fab31220STed Woodward #ifdef _MSC_VER
56aaea8ee6SZachary Turner #include "lldb/Host/windows/windows.h"
57fab31220STed Woodward #endif
58fab31220STed Woodward 
59672d2c12SJonas Devlieghere #include <memory>
60672d2c12SJonas Devlieghere #include <mutex>
612f3df613SZachary Turner 
62672d2c12SJonas Devlieghere #include <assert.h>
63672d2c12SJonas Devlieghere #include <ctype.h>
64672d2c12SJonas Devlieghere #include <errno.h>
65672d2c12SJonas Devlieghere #include <locale.h>
66672d2c12SJonas Devlieghere #include <stdint.h>
67672d2c12SJonas Devlieghere #include <stdio.h>
68672d2c12SJonas Devlieghere #include <string.h>
69672d2c12SJonas Devlieghere #include <type_traits>
702f3df613SZachary Turner 
7144d93782SGreg Clayton using namespace lldb;
7244d93782SGreg Clayton using namespace lldb_private;
7344d93782SGreg Clayton 
74b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type)
75b9c1b51eSKate Stone     : IOHandler(debugger, type,
7644d93782SGreg Clayton                 StreamFileSP(), // Adopt STDIN from top input reader
7744d93782SGreg Clayton                 StreamFileSP(), // Adopt STDOUT from top input reader
78340b0309SGreg Clayton                 StreamFileSP(), // Adopt STDERR from top input reader
79d77c2e09SJonas Devlieghere                 0,              // Flags
80d77c2e09SJonas Devlieghere                 nullptr         // Shadow file recorder
81d77c2e09SJonas Devlieghere       ) {}
8244d93782SGreg Clayton 
83b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
8444d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
8544d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
86d77c2e09SJonas Devlieghere                      const lldb::StreamFileSP &error_sp, uint32_t flags,
87d77c2e09SJonas Devlieghere                      repro::DataRecorder *data_recorder)
88b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
89d77c2e09SJonas Devlieghere       m_error_sp(error_sp), m_data_recorder(data_recorder), m_popped(false),
90d77c2e09SJonas Devlieghere       m_flags(flags), m_type(type), m_user_data(nullptr), m_done(false),
91d77c2e09SJonas Devlieghere       m_active(false) {
9244d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
9344d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
94b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
9544d93782SGreg Clayton                                              m_error_sp);
9644d93782SGreg Clayton }
9744d93782SGreg Clayton 
98315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
9944d93782SGreg Clayton 
100b9c1b51eSKate Stone int IOHandler::GetInputFD() {
101c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
10244d93782SGreg Clayton }
10344d93782SGreg Clayton 
104b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
105c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10644d93782SGreg Clayton }
10744d93782SGreg Clayton 
108b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
109c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
11044d93782SGreg Clayton }
11144d93782SGreg Clayton 
112b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
113c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11444d93782SGreg Clayton }
11544d93782SGreg Clayton 
116b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
117c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
11844d93782SGreg Clayton }
11944d93782SGreg Clayton 
120b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
121c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
12244d93782SGreg Clayton }
12344d93782SGreg Clayton 
124b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
12544d93782SGreg Clayton 
126b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
12744d93782SGreg Clayton 
128b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
12944d93782SGreg Clayton 
130b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
131340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
132340b0309SGreg Clayton }
133340b0309SGreg Clayton 
134b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
135340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
136340b0309SGreg Clayton }
13744d93782SGreg Clayton 
138b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
139e30f11d9SKate Stone 
140b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
141e30f11d9SKate Stone 
142b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
143b9c1b51eSKate Stone   if (stream) {
14416ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1454446487dSPavel Labath     if (m_top)
1464446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1474446487dSPavel Labath   }
1484446487dSPavel Labath }
1494446487dSPavel Labath 
1507a120c8bSZachary Turner IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, llvm::StringRef prompt,
151b9c1b51eSKate Stone                                    bool default_response)
152b9c1b51eSKate Stone     : IOHandlerEditline(
153b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
154c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
155514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
156514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
15744d93782SGreg Clayton           false,             // Multi-line
158e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
159d77c2e09SJonas Devlieghere           0, *this, nullptr),
160b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
16144d93782SGreg Clayton   StreamString prompt_stream;
16244d93782SGreg Clayton   prompt_stream.PutCString(prompt);
16344d93782SGreg Clayton   if (m_default_response)
16444d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
16544d93782SGreg Clayton   else
16644d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
16744d93782SGreg Clayton 
168514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
16944d93782SGreg Clayton }
17044d93782SGreg Clayton 
171315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
17244d93782SGreg Clayton 
1737f88829cSRaphael Isemann int IOHandlerConfirm::IOHandlerComplete(
1747f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
1757f88829cSRaphael Isemann     const char *last_char, int skip_first_n_matches, int max_matches,
1767f88829cSRaphael Isemann     StringList &matches, StringList &descriptions) {
177b9c1b51eSKate Stone   if (current_line == cursor) {
178b9c1b51eSKate Stone     if (m_default_response) {
17944d93782SGreg Clayton       matches.AppendString("y");
180b9c1b51eSKate Stone     } else {
18144d93782SGreg Clayton       matches.AppendString("n");
18244d93782SGreg Clayton     }
18344d93782SGreg Clayton   }
18444d93782SGreg Clayton   return matches.GetSize();
18544d93782SGreg Clayton }
18644d93782SGreg Clayton 
187b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
188b9c1b51eSKate Stone                                               std::string &line) {
189b9c1b51eSKate Stone   if (line.empty()) {
19044d93782SGreg Clayton     // User just hit enter, set the response to the default
19144d93782SGreg Clayton     m_user_response = m_default_response;
19244d93782SGreg Clayton     io_handler.SetIsDone(true);
19344d93782SGreg Clayton     return;
19444d93782SGreg Clayton   }
19544d93782SGreg Clayton 
196b9c1b51eSKate Stone   if (line.size() == 1) {
197b9c1b51eSKate Stone     switch (line[0]) {
19844d93782SGreg Clayton     case 'y':
19944d93782SGreg Clayton     case 'Y':
20044d93782SGreg Clayton       m_user_response = true;
20144d93782SGreg Clayton       io_handler.SetIsDone(true);
20244d93782SGreg Clayton       return;
20344d93782SGreg Clayton     case 'n':
20444d93782SGreg Clayton     case 'N':
20544d93782SGreg Clayton       m_user_response = false;
20644d93782SGreg Clayton       io_handler.SetIsDone(true);
20744d93782SGreg Clayton       return;
20844d93782SGreg Clayton     default:
20944d93782SGreg Clayton       break;
21044d93782SGreg Clayton     }
21144d93782SGreg Clayton   }
21244d93782SGreg Clayton 
213b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
21444d93782SGreg Clayton     m_user_response = true;
21544d93782SGreg Clayton     io_handler.SetIsDone(true);
216b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
21744d93782SGreg Clayton     m_user_response = false;
21844d93782SGreg Clayton     io_handler.SetIsDone(true);
21944d93782SGreg Clayton   }
22044d93782SGreg Clayton }
22144d93782SGreg Clayton 
2227f88829cSRaphael Isemann int IOHandlerDelegate::IOHandlerComplete(
2237f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
2247f88829cSRaphael Isemann     const char *last_char, int skip_first_n_matches, int max_matches,
2257f88829cSRaphael Isemann     StringList &matches, StringList &descriptions) {
226b9c1b51eSKate Stone   switch (m_completion) {
22744d93782SGreg Clayton   case Completion::None:
22844d93782SGreg Clayton     break;
22944d93782SGreg Clayton 
23044d93782SGreg Clayton   case Completion::LLDBCommand:
231b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
232b9c1b51eSKate Stone         current_line, cursor, last_char, skip_first_n_matches, max_matches,
2337f88829cSRaphael Isemann         matches, descriptions);
234b9c1b51eSKate Stone   case Completion::Expression: {
2357f88829cSRaphael Isemann     CompletionResult result;
236e010f6baSRaphael Isemann     CompletionRequest request(current_line, cursor - current_line,
2377f88829cSRaphael Isemann                               skip_first_n_matches, max_matches, result);
238b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
239b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
240a2e76c0bSRaphael Isemann         CommandCompletions::eVariablePathCompletion, request, nullptr);
2417f88829cSRaphael Isemann     result.GetMatches(matches);
2427f88829cSRaphael Isemann     result.GetDescriptions(descriptions);
24344d93782SGreg Clayton 
2441a6d7ab5SRaphael Isemann     size_t num_matches = request.GetNumberOfMatches();
245b9c1b51eSKate Stone     if (num_matches > 0) {
246*175f0930SJonas Devlieghere       std::string common_prefix = matches.LongestCommonPrefix();
247a2e76c0bSRaphael Isemann       const size_t partial_name_len = request.GetCursorArgumentPrefix().size();
24844d93782SGreg Clayton 
24905097246SAdrian Prantl       // If we matched a unique single command, add a space... Only do this if
25005097246SAdrian Prantl       // the completer told us this was a complete word, however...
251a2e76c0bSRaphael Isemann       if (num_matches == 1 && request.GetWordComplete()) {
25244d93782SGreg Clayton         common_prefix.push_back(' ');
25344d93782SGreg Clayton       }
25444d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
25544d93782SGreg Clayton       matches.InsertStringAtIndex(0, std::move(common_prefix));
25644d93782SGreg Clayton     }
25744d93782SGreg Clayton     return num_matches;
258b9c1b51eSKate Stone   } break;
25944d93782SGreg Clayton   }
26044d93782SGreg Clayton 
26144d93782SGreg Clayton   return 0;
26244d93782SGreg Clayton }
26344d93782SGreg Clayton 
264b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
265b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
26644d93782SGreg Clayton     const char *editline_name, // Used for saving history files
267514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
268514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
269d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
270b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
27144d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
27244d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
27344d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
274340b0309SGreg Clayton                         0,              // Flags
27544d93782SGreg Clayton                         editline_name,  // Used for saving history files
276b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
277d77c2e09SJonas Devlieghere                         line_number_start, delegate, data_recorder) {}
27844d93782SGreg Clayton 
279b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
280b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
281b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
282b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
28344d93782SGreg Clayton     const char *editline_name, // Used for saving history files
284514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
285514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
286d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
287d77c2e09SJonas Devlieghere     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags,
288d77c2e09SJonas Devlieghere                 data_recorder),
289cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
290d5b44036SJonas Devlieghere       m_editline_up(),
291cacde7dfSTodd Fiala #endif
292b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
293b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
294b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
295b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
296b9c1b51eSKate Stone       m_editing(false) {
29744d93782SGreg Clayton   SetPrompt(prompt);
29844d93782SGreg Clayton 
299cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
300914b8d98SDeepak Panickal   bool use_editline = false;
301340b0309SGreg Clayton 
302340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
30344d93782SGreg Clayton 
304b9c1b51eSKate Stone   if (use_editline) {
305d5b44036SJonas Devlieghere     m_editline_up.reset(new Editline(editline_name, GetInputFILE(),
306b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
307e30f11d9SKate Stone                                      m_color_prompts));
308d5b44036SJonas Devlieghere     m_editline_up->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
309d5b44036SJonas Devlieghere     m_editline_up->SetAutoCompleteCallback(AutoCompleteCallback, this);
310e30f11d9SKate Stone     // See if the delegate supports fixing indentation
311e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
312b9c1b51eSKate Stone     if (indent_chars) {
313b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
31405097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
315d5b44036SJonas Devlieghere       m_editline_up->SetFixIndentationCallback(FixIndentationCallback, this,
316b9c1b51eSKate Stone                                                indent_chars);
317e30f11d9SKate Stone     }
31844d93782SGreg Clayton   }
319cacde7dfSTodd Fiala #endif
320e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
321514d8cd8SZachary Turner   SetPrompt(prompt);
322e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
32344d93782SGreg Clayton }
32444d93782SGreg Clayton 
325b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
326cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
327d5b44036SJonas Devlieghere   m_editline_up.reset();
328cacde7dfSTodd Fiala #endif
32944d93782SGreg Clayton }
33044d93782SGreg Clayton 
331b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
332e30f11d9SKate Stone   IOHandler::Activate();
3330affb582SDave Lee   m_delegate.IOHandlerActivated(*this, GetIsInteractive());
334e30f11d9SKate Stone }
335e30f11d9SKate Stone 
336b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
337e30f11d9SKate Stone   IOHandler::Deactivate();
338e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
339e30f11d9SKate Stone }
340e30f11d9SKate Stone 
341b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
342cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
343d5b44036SJonas Devlieghere   if (m_editline_up) {
344d77c2e09SJonas Devlieghere     bool b = m_editline_up->GetLine(line, interrupted);
345d77c2e09SJonas Devlieghere     if (m_data_recorder)
346d77c2e09SJonas Devlieghere       m_data_recorder->Record(line, true);
347d77c2e09SJonas Devlieghere     return b;
348b9c1b51eSKate Stone   } else {
349cacde7dfSTodd Fiala #endif
35044d93782SGreg Clayton     line.clear();
35144d93782SGreg Clayton 
35244d93782SGreg Clayton     FILE *in = GetInputFILE();
353b9c1b51eSKate Stone     if (in) {
354b9c1b51eSKate Stone       if (GetIsInteractive()) {
355c5dac77aSEugene Zelenko         const char *prompt = nullptr;
356e30f11d9SKate Stone 
357e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
358e30f11d9SKate Stone           prompt = GetContinuationPrompt();
359e30f11d9SKate Stone 
360c5dac77aSEugene Zelenko         if (prompt == nullptr)
361e30f11d9SKate Stone           prompt = GetPrompt();
362e30f11d9SKate Stone 
363b9c1b51eSKate Stone         if (prompt && prompt[0]) {
36444d93782SGreg Clayton           FILE *out = GetOutputFILE();
365b9c1b51eSKate Stone           if (out) {
36644d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
36744d93782SGreg Clayton             ::fflush(out);
36844d93782SGreg Clayton           }
36944d93782SGreg Clayton         }
37044d93782SGreg Clayton       }
37144d93782SGreg Clayton       char buffer[256];
37244d93782SGreg Clayton       bool done = false;
3730f86e6e7SGreg Clayton       bool got_line = false;
374e034a04eSGreg Clayton       m_editing = true;
375b9c1b51eSKate Stone       while (!done) {
376e7167908SNathan Lanza #ifdef _WIN32
377e7167908SNathan Lanza         // ReadFile on Windows is supposed to set ERROR_OPERATION_ABORTED
378e7167908SNathan Lanza         // according to the docs on MSDN. However, this has evidently been a
379e7167908SNathan Lanza         // known bug since Windows 8. Therefore, we can't detect if a signal
380e7167908SNathan Lanza         // interrupted in the fgets. So pressing ctrl-c causes the repl to end
381e7167908SNathan Lanza         // and the process to exit. A temporary workaround is just to attempt to
382e7167908SNathan Lanza         // fgets twice until this bug is fixed.
383e7167908SNathan Lanza         if (fgets(buffer, sizeof(buffer), in) == nullptr &&
384e7167908SNathan Lanza             fgets(buffer, sizeof(buffer), in) == nullptr) {
385cb305205SNathan Lanza           // this is the equivalent of EINTR for Windows
386cb305205SNathan Lanza           if (GetLastError() == ERROR_OPERATION_ABORTED)
387cb305205SNathan Lanza             continue;
388e7167908SNathan Lanza #else
389b9c1b51eSKate Stone         if (fgets(buffer, sizeof(buffer), in) == nullptr) {
390e7167908SNathan Lanza #endif
391c7797accSGreg Clayton           const int saved_errno = errno;
392c9cf5798SGreg Clayton           if (feof(in))
39344d93782SGreg Clayton             done = true;
394b9c1b51eSKate Stone           else if (ferror(in)) {
395c7797accSGreg Clayton             if (saved_errno != EINTR)
396c7797accSGreg Clayton               done = true;
397c7797accSGreg Clayton           }
398b9c1b51eSKate Stone         } else {
3990f86e6e7SGreg Clayton           got_line = true;
40044d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
40144d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
40244d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
403b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
40444d93782SGreg Clayton             done = true;
40544d93782SGreg Clayton             // Strip trailing newlines
406b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
40744d93782SGreg Clayton               --buffer_len;
40844d93782SGreg Clayton               if (buffer_len == 0)
40944d93782SGreg Clayton                 break;
41044d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
41144d93782SGreg Clayton             }
41244d93782SGreg Clayton           }
41344d93782SGreg Clayton           line.append(buffer, buffer_len);
41444d93782SGreg Clayton         }
41544d93782SGreg Clayton       }
416e034a04eSGreg Clayton       m_editing = false;
417d77c2e09SJonas Devlieghere       if (m_data_recorder && got_line)
418d77c2e09SJonas Devlieghere         m_data_recorder->Record(line, true);
41905097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
42005097246SAdrian Prantl       // true in this case.
4210f86e6e7SGreg Clayton       return got_line;
422b9c1b51eSKate Stone     } else {
42344d93782SGreg Clayton       // No more input file, we are done...
42444d93782SGreg Clayton       SetIsDone(true);
42544d93782SGreg Clayton     }
426340b0309SGreg Clayton     return false;
427cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
42844d93782SGreg Clayton   }
429cacde7dfSTodd Fiala #endif
43044d93782SGreg Clayton }
43144d93782SGreg Clayton 
432cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
433b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
43444d93782SGreg Clayton                                                 StringList &lines,
435b9c1b51eSKate Stone                                                 void *baton) {
43644d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
437b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
438b9c1b51eSKate Stone                                                               lines);
439e30f11d9SKate Stone }
440e30f11d9SKate Stone 
441b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
442e30f11d9SKate Stone                                               const StringList &lines,
443e30f11d9SKate Stone                                               int cursor_position,
444b9c1b51eSKate Stone                                               void *baton) {
445e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
446b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
447b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
44844d93782SGreg Clayton }
44944d93782SGreg Clayton 
4507f88829cSRaphael Isemann int IOHandlerEditline::AutoCompleteCallback(
4517f88829cSRaphael Isemann     const char *current_line, const char *cursor, const char *last_char,
4527f88829cSRaphael Isemann     int skip_first_n_matches, int max_matches, StringList &matches,
4537f88829cSRaphael Isemann     StringList &descriptions, void *baton) {
45444d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
45544d93782SGreg Clayton   if (editline_reader)
456b9c1b51eSKate Stone     return editline_reader->m_delegate.IOHandlerComplete(
457b9c1b51eSKate Stone         *editline_reader, current_line, cursor, last_char, skip_first_n_matches,
4587f88829cSRaphael Isemann         max_matches, matches, descriptions);
45944d93782SGreg Clayton   return 0;
46044d93782SGreg Clayton }
461cacde7dfSTodd Fiala #endif
46244d93782SGreg Clayton 
463b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
464cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
465d5b44036SJonas Devlieghere   if (m_editline_up) {
466d5b44036SJonas Devlieghere     return m_editline_up->GetPrompt();
467b9c1b51eSKate Stone   } else {
468cacde7dfSTodd Fiala #endif
469cacde7dfSTodd Fiala     if (m_prompt.empty())
470c5dac77aSEugene Zelenko       return nullptr;
471cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
472cacde7dfSTodd Fiala   }
473cacde7dfSTodd Fiala #endif
47444d93782SGreg Clayton   return m_prompt.c_str();
47544d93782SGreg Clayton }
47644d93782SGreg Clayton 
477514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
478514d8cd8SZachary Turner   m_prompt = prompt;
479514d8cd8SZachary Turner 
480cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
481d5b44036SJonas Devlieghere   if (m_editline_up)
482d5b44036SJonas Devlieghere     m_editline_up->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
483cacde7dfSTodd Fiala #endif
48444d93782SGreg Clayton   return true;
48544d93782SGreg Clayton }
48644d93782SGreg Clayton 
487b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
488b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
489b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
490e30f11d9SKate Stone }
491e30f11d9SKate Stone 
492514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
493514d8cd8SZachary Turner   m_continuation_prompt = prompt;
494e30f11d9SKate Stone 
495d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
496d5b44036SJonas Devlieghere   if (m_editline_up)
497d5b44036SJonas Devlieghere     m_editline_up->SetContinuationPrompt(m_continuation_prompt.empty()
498b9c1b51eSKate Stone                                              ? nullptr
499b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
500d553d00cSZachary Turner #endif
501e30f11d9SKate Stone }
502e30f11d9SKate Stone 
503b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
504f6913cd7SGreg Clayton   m_base_line_number = line;
505f6913cd7SGreg Clayton }
506e30f11d9SKate Stone 
507b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
508d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
509d5b44036SJonas Devlieghere   if (m_editline_up)
510d5b44036SJonas Devlieghere     return m_editline_up->GetCurrentLine();
511e30f11d9SKate Stone #endif
512e30f11d9SKate Stone   return m_curr_line_idx;
513e30f11d9SKate Stone }
514e30f11d9SKate Stone 
515b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
516e30f11d9SKate Stone   m_current_lines_ptr = &lines;
517e30f11d9SKate Stone 
51844d93782SGreg Clayton   bool success = false;
519cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
520d5b44036SJonas Devlieghere   if (m_editline_up) {
521d5b44036SJonas Devlieghere     return m_editline_up->GetLines(m_base_line_number, lines, interrupted);
522b9c1b51eSKate Stone   } else {
523cacde7dfSTodd Fiala #endif
524e30f11d9SKate Stone     bool done = false;
52597206d57SZachary Turner     Status error;
52644d93782SGreg Clayton 
527b9c1b51eSKate Stone     while (!done) {
528f6913cd7SGreg Clayton       // Show line numbers if we are asked to
52944d93782SGreg Clayton       std::string line;
530b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
531f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
532f6913cd7SGreg Clayton         if (out)
533b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
534b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
535f6913cd7SGreg Clayton       }
536f6913cd7SGreg Clayton 
537e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
538e30f11d9SKate Stone 
539f0066ad0SGreg Clayton       bool interrupted = false;
540b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
54144d93782SGreg Clayton         lines.AppendString(line);
542e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
543b9c1b51eSKate Stone       } else {
544e30f11d9SKate Stone         done = true;
54544d93782SGreg Clayton       }
54644d93782SGreg Clayton     }
54744d93782SGreg Clayton     success = lines.GetSize() > 0;
548cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
54944d93782SGreg Clayton   }
550cacde7dfSTodd Fiala #endif
55144d93782SGreg Clayton   return success;
55244d93782SGreg Clayton }
55344d93782SGreg Clayton 
55405097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
55505097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
556b9c1b51eSKate Stone void IOHandlerEditline::Run() {
55744d93782SGreg Clayton   std::string line;
558b9c1b51eSKate Stone   while (IsActive()) {
559f0066ad0SGreg Clayton     bool interrupted = false;
560b9c1b51eSKate Stone     if (m_multi_line) {
56144d93782SGreg Clayton       StringList lines;
562b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
563b9c1b51eSKate Stone         if (interrupted) {
564e30f11d9SKate Stone           m_done = m_interrupt_exits;
565e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
566e30f11d9SKate Stone 
567b9c1b51eSKate Stone         } else {
56844d93782SGreg Clayton           line = lines.CopyList();
56944d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
57044d93782SGreg Clayton         }
571b9c1b51eSKate Stone       } else {
57244d93782SGreg Clayton         m_done = true;
57344d93782SGreg Clayton       }
574b9c1b51eSKate Stone     } else {
575b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
576e30f11d9SKate Stone         if (interrupted)
577e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
578e30f11d9SKate Stone         else
57944d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
580b9c1b51eSKate Stone       } else {
58144d93782SGreg Clayton         m_done = true;
58244d93782SGreg Clayton       }
58344d93782SGreg Clayton     }
58444d93782SGreg Clayton   }
58544d93782SGreg Clayton }
58644d93782SGreg Clayton 
587b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
588cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
589d5b44036SJonas Devlieghere   if (m_editline_up)
590d5b44036SJonas Devlieghere     m_editline_up->Cancel();
591cacde7dfSTodd Fiala #endif
592e68f5d6bSGreg Clayton }
593e68f5d6bSGreg Clayton 
594b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
595f0066ad0SGreg Clayton   // Let the delgate handle it first
596f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
597f0066ad0SGreg Clayton     return true;
598f0066ad0SGreg Clayton 
599cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
600d5b44036SJonas Devlieghere   if (m_editline_up)
601d5b44036SJonas Devlieghere     return m_editline_up->Interrupt();
602cacde7dfSTodd Fiala #endif
603f0066ad0SGreg Clayton   return false;
60444d93782SGreg Clayton }
60544d93782SGreg Clayton 
606b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
607cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
608d5b44036SJonas Devlieghere   if (m_editline_up)
609d5b44036SJonas Devlieghere     m_editline_up->Interrupt();
610cacde7dfSTodd Fiala #endif
61144d93782SGreg Clayton }
61244d93782SGreg Clayton 
613b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
6144446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
615d5b44036SJonas Devlieghere   if (m_editline_up)
616d5b44036SJonas Devlieghere     m_editline_up->PrintAsync(stream, s, len);
6174446487dSPavel Labath   else
6184446487dSPavel Labath #endif
619fab31220STed Woodward   {
620fab31220STed Woodward #ifdef _MSC_VER
621341e4789SDawn Perchik     const char *prompt = GetPrompt();
622b9c1b51eSKate Stone     if (prompt) {
623fab31220STed Woodward       // Back up over previous prompt using Windows API
624fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
625fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
626fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
627fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
628fab31220STed Woodward       coord.X -= strlen(prompt);
629fab31220STed Woodward       if (coord.X < 0)
630fab31220STed Woodward         coord.X = 0;
631fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
632fab31220STed Woodward     }
633fab31220STed Woodward #endif
6344446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
635341e4789SDawn Perchik #ifdef _MSC_VER
636fab31220STed Woodward     if (prompt)
637b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
638b9c1b51eSKate Stone                             strlen(prompt));
639341e4789SDawn Perchik #endif
640fab31220STed Woodward   }
6414446487dSPavel Labath }
6424446487dSPavel Labath 
64305097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
644914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
645914b8d98SDeepak Panickal 
64644d93782SGreg Clayton #define KEY_RETURN 10
64744d93782SGreg Clayton #define KEY_ESCAPE 27
64844d93782SGreg Clayton 
649b9c1b51eSKate Stone namespace curses {
65044d93782SGreg Clayton class Menu;
65144d93782SGreg Clayton class MenuDelegate;
65244d93782SGreg Clayton class Window;
65344d93782SGreg Clayton class WindowDelegate;
65444d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
65544d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
65644d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
65744d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
65844d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
65944d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
66044d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
66144d93782SGreg Clayton 
66244d93782SGreg Clayton #if 0
66344d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
66444d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
66544d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
66644d93782SGreg Clayton #endif
667315b6884SEugene Zelenko 
668b9c1b51eSKate Stone struct Point {
66944d93782SGreg Clayton   int x;
67044d93782SGreg Clayton   int y;
67144d93782SGreg Clayton 
672b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
67344d93782SGreg Clayton 
674b9c1b51eSKate Stone   void Clear() {
67544d93782SGreg Clayton     x = 0;
67644d93782SGreg Clayton     y = 0;
67744d93782SGreg Clayton   }
67844d93782SGreg Clayton 
679b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
68044d93782SGreg Clayton     x += rhs.x;
68144d93782SGreg Clayton     y += rhs.y;
68244d93782SGreg Clayton     return *this;
68344d93782SGreg Clayton   }
68444d93782SGreg Clayton 
685b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
68644d93782SGreg Clayton };
68744d93782SGreg Clayton 
688b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
68944d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
69044d93782SGreg Clayton }
691315b6884SEugene Zelenko 
692b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
69344d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
69444d93782SGreg Clayton }
69544d93782SGreg Clayton 
696b9c1b51eSKate Stone struct Size {
69744d93782SGreg Clayton   int width;
69844d93782SGreg Clayton   int height;
699b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
70044d93782SGreg Clayton 
701b9c1b51eSKate Stone   void Clear() {
70244d93782SGreg Clayton     width = 0;
70344d93782SGreg Clayton     height = 0;
70444d93782SGreg Clayton   }
70544d93782SGreg Clayton 
706b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
70744d93782SGreg Clayton };
70844d93782SGreg Clayton 
709b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
71044d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
71144d93782SGreg Clayton }
712315b6884SEugene Zelenko 
713b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
71444d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
71544d93782SGreg Clayton }
71644d93782SGreg Clayton 
717b9c1b51eSKate Stone struct Rect {
71844d93782SGreg Clayton   Point origin;
71944d93782SGreg Clayton   Size size;
72044d93782SGreg Clayton 
721b9c1b51eSKate Stone   Rect() : origin(), size() {}
72244d93782SGreg Clayton 
723b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
72444d93782SGreg Clayton 
725b9c1b51eSKate Stone   void Clear() {
72644d93782SGreg Clayton     origin.Clear();
72744d93782SGreg Clayton     size.Clear();
72844d93782SGreg Clayton   }
72944d93782SGreg Clayton 
730b9c1b51eSKate Stone   void Dump() {
731b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
732b9c1b51eSKate Stone            size.height);
73344d93782SGreg Clayton   }
73444d93782SGreg Clayton 
735b9c1b51eSKate Stone   void Inset(int w, int h) {
73644d93782SGreg Clayton     if (size.width > w * 2)
73744d93782SGreg Clayton       size.width -= w * 2;
73844d93782SGreg Clayton     origin.x += w;
73944d93782SGreg Clayton 
74044d93782SGreg Clayton     if (size.height > h * 2)
74144d93782SGreg Clayton       size.height -= h * 2;
74244d93782SGreg Clayton     origin.y += h;
74344d93782SGreg Clayton   }
744315b6884SEugene Zelenko 
74505097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
74605097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
747b9c1b51eSKate Stone   Rect MakeStatusBar() {
74844d93782SGreg Clayton     Rect status_bar;
749b9c1b51eSKate Stone     if (size.height > 1) {
75044d93782SGreg Clayton       status_bar.origin.x = origin.x;
75144d93782SGreg Clayton       status_bar.origin.y = size.height;
75244d93782SGreg Clayton       status_bar.size.width = size.width;
75344d93782SGreg Clayton       status_bar.size.height = 1;
75444d93782SGreg Clayton       --size.height;
75544d93782SGreg Clayton     }
75644d93782SGreg Clayton     return status_bar;
75744d93782SGreg Clayton   }
75844d93782SGreg Clayton 
75905097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
76005097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
761b9c1b51eSKate Stone   Rect MakeMenuBar() {
76244d93782SGreg Clayton     Rect menubar;
763b9c1b51eSKate Stone     if (size.height > 1) {
76444d93782SGreg Clayton       menubar.origin.x = origin.x;
76544d93782SGreg Clayton       menubar.origin.y = origin.y;
76644d93782SGreg Clayton       menubar.size.width = size.width;
76744d93782SGreg Clayton       menubar.size.height = 1;
76844d93782SGreg Clayton       ++origin.y;
76944d93782SGreg Clayton       --size.height;
77044d93782SGreg Clayton     }
77144d93782SGreg Clayton     return menubar;
77244d93782SGreg Clayton   }
77344d93782SGreg Clayton 
774b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
775b9c1b51eSKate Stone                                  Rect &bottom) const {
77644d93782SGreg Clayton     float top_height = top_percentage * size.height;
77744d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
77844d93782SGreg Clayton   }
77944d93782SGreg Clayton 
780b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
78144d93782SGreg Clayton     top = *this;
782b9c1b51eSKate Stone     if (top_height < size.height) {
78344d93782SGreg Clayton       top.size.height = top_height;
78444d93782SGreg Clayton       bottom.origin.x = origin.x;
78544d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
78644d93782SGreg Clayton       bottom.size.width = size.width;
78744d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
788b9c1b51eSKate Stone     } else {
78944d93782SGreg Clayton       bottom.Clear();
79044d93782SGreg Clayton     }
79144d93782SGreg Clayton   }
79244d93782SGreg Clayton 
793b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
794b9c1b51eSKate Stone                                Rect &right) const {
79544d93782SGreg Clayton     float left_width = left_percentage * size.width;
79644d93782SGreg Clayton     VerticalSplit(left_width, left, right);
79744d93782SGreg Clayton   }
79844d93782SGreg Clayton 
799b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
80044d93782SGreg Clayton     left = *this;
801b9c1b51eSKate Stone     if (left_width < size.width) {
80244d93782SGreg Clayton       left.size.width = left_width;
80344d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
80444d93782SGreg Clayton       right.origin.y = origin.y;
80544d93782SGreg Clayton       right.size.width = size.width - left.size.width;
80644d93782SGreg Clayton       right.size.height = size.height;
807b9c1b51eSKate Stone     } else {
80844d93782SGreg Clayton       right.Clear();
80944d93782SGreg Clayton     }
81044d93782SGreg Clayton   }
81144d93782SGreg Clayton };
81244d93782SGreg Clayton 
813b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
81444d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
81544d93782SGreg Clayton }
816315b6884SEugene Zelenko 
817b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
81844d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
81944d93782SGreg Clayton }
82044d93782SGreg Clayton 
821b9c1b51eSKate Stone enum HandleCharResult {
82244d93782SGreg Clayton   eKeyNotHandled = 0,
82344d93782SGreg Clayton   eKeyHandled = 1,
82444d93782SGreg Clayton   eQuitApplication = 2
82544d93782SGreg Clayton };
82644d93782SGreg Clayton 
827b9c1b51eSKate Stone enum class MenuActionResult {
82844d93782SGreg Clayton   Handled,
82944d93782SGreg Clayton   NotHandled,
83044d93782SGreg Clayton   Quit // Exit all menus and quit
83144d93782SGreg Clayton };
83244d93782SGreg Clayton 
833b9c1b51eSKate Stone struct KeyHelp {
83444d93782SGreg Clayton   int ch;
83544d93782SGreg Clayton   const char *description;
83644d93782SGreg Clayton };
83744d93782SGreg Clayton 
838b9c1b51eSKate Stone class WindowDelegate {
83944d93782SGreg Clayton public:
840b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
84144d93782SGreg Clayton 
842b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
84344d93782SGreg Clayton     return false; // Drawing not handled
84444d93782SGreg Clayton   }
84544d93782SGreg Clayton 
846b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
84744d93782SGreg Clayton     return eKeyNotHandled;
84844d93782SGreg Clayton   }
84944d93782SGreg Clayton 
850b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
85144d93782SGreg Clayton 
852b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
85344d93782SGreg Clayton };
85444d93782SGreg Clayton 
855b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
85644d93782SGreg Clayton public:
85744d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
85844d93782SGreg Clayton 
859bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
86044d93782SGreg Clayton 
861b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
86244d93782SGreg Clayton 
863b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
86444d93782SGreg Clayton 
865b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
86644d93782SGreg Clayton 
867b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
86844d93782SGreg Clayton 
86944d93782SGreg Clayton protected:
87044d93782SGreg Clayton   StringList m_text;
87144d93782SGreg Clayton   int m_first_visible_line;
87244d93782SGreg Clayton };
87344d93782SGreg Clayton 
874b9c1b51eSKate Stone class Window {
87544d93782SGreg Clayton public:
876b9c1b51eSKate Stone   Window(const char *name)
877b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
878b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
879b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
880b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
88144d93782SGreg Clayton 
882b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
883b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
884b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
885b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
886b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
88744d93782SGreg Clayton     if (w)
88844d93782SGreg Clayton       Reset(w);
88944d93782SGreg Clayton   }
89044d93782SGreg Clayton 
891b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
892b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
893b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
894b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
895b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
896b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
897b9c1b51eSKate Stone                    bounds.origin.y));
89844d93782SGreg Clayton   }
89944d93782SGreg Clayton 
900b9c1b51eSKate Stone   virtual ~Window() {
90144d93782SGreg Clayton     RemoveSubWindows();
90244d93782SGreg Clayton     Reset();
90344d93782SGreg Clayton   }
90444d93782SGreg Clayton 
905b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
90644d93782SGreg Clayton     if (m_window == w)
90744d93782SGreg Clayton       return;
90844d93782SGreg Clayton 
909b9c1b51eSKate Stone     if (m_panel) {
91044d93782SGreg Clayton       ::del_panel(m_panel);
911c5dac77aSEugene Zelenko       m_panel = nullptr;
91244d93782SGreg Clayton     }
913b9c1b51eSKate Stone     if (m_window && m_delete) {
91444d93782SGreg Clayton       ::delwin(m_window);
915c5dac77aSEugene Zelenko       m_window = nullptr;
91644d93782SGreg Clayton       m_delete = false;
91744d93782SGreg Clayton     }
918b9c1b51eSKate Stone     if (w) {
91944d93782SGreg Clayton       m_window = w;
92044d93782SGreg Clayton       m_panel = ::new_panel(m_window);
92144d93782SGreg Clayton       m_delete = del;
92244d93782SGreg Clayton     }
92344d93782SGreg Clayton   }
92444d93782SGreg Clayton 
92544d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
92644d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
927b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
928b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
929b9c1b51eSKate Stone   }
93044d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
93144d93782SGreg Clayton   void Erase() { ::werase(m_window); }
932b9c1b51eSKate Stone   Rect GetBounds() {
933b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
934b9c1b51eSKate Stone   } // Get the rectangle in our parent window
93544d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
93644d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
93744d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
938b9c1b51eSKate Stone   Rect GetFrame() {
939b9c1b51eSKate Stone     return Rect(Point(), GetSize());
940b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
94144d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
94244d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
94344d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
94444d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
94544d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
94644d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
94744d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
94844d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
94944d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
95044d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
95144d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
952b9c1b51eSKate Stone   void Resize(const Size &size) {
953b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
954b9c1b51eSKate Stone   }
95544d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
95644d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
95744d93782SGreg Clayton   void Refresh() { ::wrefresh(m_window); }
958b9c1b51eSKate Stone   void DeferredRefresh() {
95944d93782SGreg Clayton     // We are using panels, so we don't need to call this...
96044d93782SGreg Clayton     //::wnoutrefresh(m_window);
96144d93782SGreg Clayton   }
962b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
963b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
964b9c1b51eSKate Stone   }
96544d93782SGreg Clayton   void UnderlineOn() { AttributeOn(A_UNDERLINE); }
96644d93782SGreg Clayton   void UnderlineOff() { AttributeOff(A_UNDERLINE); }
96744d93782SGreg Clayton 
968b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
96944d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
970b9c1b51eSKate Stone     if (bytes_left > right_pad) {
97144d93782SGreg Clayton       bytes_left -= right_pad;
97244d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
97344d93782SGreg Clayton     }
97444d93782SGreg Clayton   }
97544d93782SGreg Clayton 
976b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
97744d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
978b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
97944d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
98044d93782SGreg Clayton       Size size = GetSize();
981b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
982b9c1b51eSKate Stone                      origin.x),
983b9c1b51eSKate Stone             true);
984b9c1b51eSKate Stone     } else {
98544d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
98644d93782SGreg Clayton     }
98744d93782SGreg Clayton   }
98844d93782SGreg Clayton 
989b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
99044d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
991b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
99244d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
993b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
994b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
995b9c1b51eSKate Stone             true);
996b9c1b51eSKate Stone     } else {
99744d93782SGreg Clayton       if (moving_window)
99844d93782SGreg Clayton         MoveWindow(bounds.origin);
99944d93782SGreg Clayton       Resize(bounds.size);
100044d93782SGreg Clayton     }
100144d93782SGreg Clayton   }
100244d93782SGreg Clayton 
1003b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
100444d93782SGreg Clayton     va_list args;
100544d93782SGreg Clayton     va_start(args, format);
100644d93782SGreg Clayton     vwprintw(m_window, format, args);
100744d93782SGreg Clayton     va_end(args);
100844d93782SGreg Clayton   }
100944d93782SGreg Clayton 
1010b9c1b51eSKate Stone   void Touch() {
101144d93782SGreg Clayton     ::touchwin(m_window);
101244d93782SGreg Clayton     if (m_parent)
101344d93782SGreg Clayton       m_parent->Touch();
101444d93782SGreg Clayton   }
101544d93782SGreg Clayton 
1016b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
1017b9c1b51eSKate Stone                            bool make_active) {
1018c6091d2bSJonas Devlieghere     auto get_window = [this, &bounds]() {
1019c6091d2bSJonas Devlieghere       return m_window
1020c6091d2bSJonas Devlieghere                  ? ::subwin(m_window, bounds.size.height, bounds.size.width,
1021c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x)
1022c6091d2bSJonas Devlieghere                  : ::newwin(bounds.size.height, bounds.size.width,
1023c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x);
1024c6091d2bSJonas Devlieghere     };
1025c6091d2bSJonas Devlieghere     WindowSP subwindow_sp = std::make_shared<Window>(name, get_window(), true);
1026c6091d2bSJonas Devlieghere     subwindow_sp->m_is_subwin = subwindow_sp.operator bool();
102744d93782SGreg Clayton     subwindow_sp->m_parent = this;
1028b9c1b51eSKate Stone     if (make_active) {
102944d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
103044d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
103144d93782SGreg Clayton     }
103244d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
103344d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
103444d93782SGreg Clayton     m_needs_update = true;
103544d93782SGreg Clayton     return subwindow_sp;
103644d93782SGreg Clayton   }
103744d93782SGreg Clayton 
1038b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
103944d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
104044d93782SGreg Clayton     size_t i = 0;
1041b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1042b9c1b51eSKate Stone       if ((*pos).get() == window) {
104344d93782SGreg Clayton         if (m_prev_active_window_idx == i)
104444d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1045b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1046b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
104744d93782SGreg Clayton           --m_prev_active_window_idx;
104844d93782SGreg Clayton 
104944d93782SGreg Clayton         if (m_curr_active_window_idx == i)
105044d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1051b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1052b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
105344d93782SGreg Clayton           --m_curr_active_window_idx;
105444d93782SGreg Clayton         window->Erase();
105544d93782SGreg Clayton         m_subwindows.erase(pos);
105644d93782SGreg Clayton         m_needs_update = true;
105744d93782SGreg Clayton         if (m_parent)
105844d93782SGreg Clayton           m_parent->Touch();
105944d93782SGreg Clayton         else
106044d93782SGreg Clayton           ::touchwin(stdscr);
106144d93782SGreg Clayton         return true;
106244d93782SGreg Clayton       }
106344d93782SGreg Clayton     }
106444d93782SGreg Clayton     return false;
106544d93782SGreg Clayton   }
106644d93782SGreg Clayton 
1067b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
106844d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
106944d93782SGreg Clayton     size_t i = 0;
1070b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
10718d20cfdfSJonas Devlieghere       if ((*pos)->m_name == name)
107244d93782SGreg Clayton         return *pos;
107344d93782SGreg Clayton     }
107444d93782SGreg Clayton     return WindowSP();
107544d93782SGreg Clayton   }
107644d93782SGreg Clayton 
1077b9c1b51eSKate Stone   void RemoveSubWindows() {
107844d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
107944d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
108044d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1081b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
108244d93782SGreg Clayton       (*pos)->Erase();
108344d93782SGreg Clayton     }
108444d93782SGreg Clayton     if (m_parent)
108544d93782SGreg Clayton       m_parent->Touch();
108644d93782SGreg Clayton     else
108744d93782SGreg Clayton       ::touchwin(stdscr);
108844d93782SGreg Clayton   }
108944d93782SGreg Clayton 
1090b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
109144d93782SGreg Clayton 
1092b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
109344d93782SGreg Clayton 
109444d93782SGreg Clayton   // Window drawing utilities
1095b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
109644d93782SGreg Clayton     attr_t attr = 0;
109744d93782SGreg Clayton     if (IsActive())
109844d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
109944d93782SGreg Clayton     else
110044d93782SGreg Clayton       attr = 0;
110144d93782SGreg Clayton     if (attr)
110244d93782SGreg Clayton       AttributeOn(attr);
110344d93782SGreg Clayton 
110444d93782SGreg Clayton     Box();
110544d93782SGreg Clayton     MoveCursor(3, 0);
110644d93782SGreg Clayton 
1107b9c1b51eSKate Stone     if (title && title[0]) {
110844d93782SGreg Clayton       PutChar('<');
110944d93782SGreg Clayton       PutCString(title);
111044d93782SGreg Clayton       PutChar('>');
111144d93782SGreg Clayton     }
111244d93782SGreg Clayton 
1113b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
111444d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
111544d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
111644d93782SGreg Clayton 
1117b9c1b51eSKate Stone       if (x > 0) {
111844d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
111944d93782SGreg Clayton         PutChar('[');
112044d93782SGreg Clayton         PutCString(bottom_message);
112144d93782SGreg Clayton         PutChar(']');
1122b9c1b51eSKate Stone       } else {
112344d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
112444d93782SGreg Clayton         PutChar('[');
112544d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
112644d93782SGreg Clayton       }
112744d93782SGreg Clayton     }
112844d93782SGreg Clayton     if (attr)
112944d93782SGreg Clayton       AttributeOff(attr);
113044d93782SGreg Clayton   }
113144d93782SGreg Clayton 
1132b9c1b51eSKate Stone   virtual void Draw(bool force) {
113344d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
113444d93782SGreg Clayton       return;
113544d93782SGreg Clayton 
113644d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
113744d93782SGreg Clayton       subwindow_sp->Draw(force);
113844d93782SGreg Clayton   }
113944d93782SGreg Clayton 
1140b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1141b9c1b51eSKate Stone     if (m_delegate_sp) {
114244d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
114344d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1144b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1145d5b44036SJonas Devlieghere         std::unique_ptr<HelpDialogDelegate> help_delegate_up(
1146b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
1147d5b44036SJonas Devlieghere         const size_t num_lines = help_delegate_up->GetNumLines();
1148d5b44036SJonas Devlieghere         const size_t max_length = help_delegate_up->GetMaxLineLength();
114944d93782SGreg Clayton         Rect bounds = GetBounds();
115044d93782SGreg Clayton         bounds.Inset(1, 1);
1151b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
115244d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
115344d93782SGreg Clayton           bounds.size.width = max_length + 4;
1154b9c1b51eSKate Stone         } else {
1155b9c1b51eSKate Stone           if (bounds.size.width > 100) {
115644d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
115744d93782SGreg Clayton             bounds.origin.x += inset_w;
115844d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
115944d93782SGreg Clayton           }
116044d93782SGreg Clayton         }
116144d93782SGreg Clayton 
1162b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
116344d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
116444d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1165b9c1b51eSKate Stone         } else {
1166b9c1b51eSKate Stone           if (bounds.size.height > 100) {
116744d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
116844d93782SGreg Clayton             bounds.origin.y += inset_h;
116944d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
117044d93782SGreg Clayton           }
117144d93782SGreg Clayton         }
11725fdb09bbSGreg Clayton         WindowSP help_window_sp;
11735fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11745fdb09bbSGreg Clayton         if (parent_window)
11755fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11765fdb09bbSGreg Clayton         else
11775fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1178b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1179d5b44036SJonas Devlieghere             WindowDelegateSP(help_delegate_up.release()));
118044d93782SGreg Clayton         return true;
118144d93782SGreg Clayton       }
118244d93782SGreg Clayton     }
118344d93782SGreg Clayton     return false;
118444d93782SGreg Clayton   }
118544d93782SGreg Clayton 
1186b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
118744d93782SGreg Clayton     // Always check the active window first
118844d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
118944d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1190b9c1b51eSKate Stone     if (active_window_sp) {
119144d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
119244d93782SGreg Clayton       if (result != eKeyNotHandled)
119344d93782SGreg Clayton         return result;
119444d93782SGreg Clayton     }
119544d93782SGreg Clayton 
1196b9c1b51eSKate Stone     if (m_delegate_sp) {
119744d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
119844d93782SGreg Clayton       if (result != eKeyNotHandled)
119944d93782SGreg Clayton         return result;
120044d93782SGreg Clayton     }
120144d93782SGreg Clayton 
120205097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
120305097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
120405097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
120505097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
120644d93782SGreg Clayton     Windows subwindows(m_subwindows);
1207b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1208b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
120944d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
121044d93782SGreg Clayton         if (result != eKeyNotHandled)
121144d93782SGreg Clayton           return result;
121244d93782SGreg Clayton       }
121344d93782SGreg Clayton     }
121444d93782SGreg Clayton 
121544d93782SGreg Clayton     return eKeyNotHandled;
121644d93782SGreg Clayton   }
121744d93782SGreg Clayton 
1218b9c1b51eSKate Stone   bool SetActiveWindow(Window *window) {
121944d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1220b9c1b51eSKate Stone     for (size_t i = 0; i < num_subwindows; ++i) {
1221b9c1b51eSKate Stone       if (m_subwindows[i].get() == window) {
122244d93782SGreg Clayton         m_prev_active_window_idx = m_curr_active_window_idx;
122344d93782SGreg Clayton         ::top_panel(window->m_panel);
122444d93782SGreg Clayton         m_curr_active_window_idx = i;
122544d93782SGreg Clayton         return true;
122644d93782SGreg Clayton       }
122744d93782SGreg Clayton     }
122844d93782SGreg Clayton     return false;
122944d93782SGreg Clayton   }
123044d93782SGreg Clayton 
1231b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1232b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1233b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1234b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
123544d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
123644d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1237b9c1b51eSKate Stone         } else if (IsActive()) {
123844d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
123944d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
124044d93782SGreg Clayton 
124144d93782SGreg Clayton           // Find first window that wants to be active if this window is active
124244d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1243b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1244b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
124544d93782SGreg Clayton               m_curr_active_window_idx = i;
124644d93782SGreg Clayton               break;
124744d93782SGreg Clayton             }
124844d93782SGreg Clayton           }
124944d93782SGreg Clayton         }
125044d93782SGreg Clayton       }
125144d93782SGreg Clayton 
125244d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
125344d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
125444d93782SGreg Clayton     }
125544d93782SGreg Clayton     return WindowSP();
125644d93782SGreg Clayton   }
125744d93782SGreg Clayton 
1258b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
125944d93782SGreg Clayton 
1260b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
126144d93782SGreg Clayton 
1262b9c1b51eSKate Stone   const WindowDelegateSP &GetDelegate() const { return m_delegate_sp; }
126344d93782SGreg Clayton 
1264b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
126544d93782SGreg Clayton     m_delegate_sp = delegate_sp;
126644d93782SGreg Clayton   }
126744d93782SGreg Clayton 
1268b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
126944d93782SGreg Clayton 
1270b9c1b51eSKate Stone   bool IsActive() const {
127144d93782SGreg Clayton     if (m_parent)
127244d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
127344d93782SGreg Clayton     else
127444d93782SGreg Clayton       return true; // Top level window is always active
127544d93782SGreg Clayton   }
127644d93782SGreg Clayton 
1277b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
127844d93782SGreg Clayton     // Move active focus to next window
127944d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1280b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
128144d93782SGreg Clayton       uint32_t idx = 0;
1282b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1283b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
128444d93782SGreg Clayton           m_curr_active_window_idx = idx;
128544d93782SGreg Clayton           break;
128644d93782SGreg Clayton         }
128744d93782SGreg Clayton         ++idx;
128844d93782SGreg Clayton       }
1289b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
129044d93782SGreg Clayton       bool handled = false;
129144d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1292b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1293b9c1b51eSKate Stone            ++idx) {
1294b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
129544d93782SGreg Clayton           m_curr_active_window_idx = idx;
129644d93782SGreg Clayton           handled = true;
129744d93782SGreg Clayton           break;
129844d93782SGreg Clayton         }
129944d93782SGreg Clayton       }
1300b9c1b51eSKate Stone       if (!handled) {
1301b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1302b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
130344d93782SGreg Clayton             m_curr_active_window_idx = idx;
130444d93782SGreg Clayton             break;
130544d93782SGreg Clayton           }
130644d93782SGreg Clayton         }
130744d93782SGreg Clayton       }
1308b9c1b51eSKate Stone     } else {
130944d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1310b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1311b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
131244d93782SGreg Clayton           m_curr_active_window_idx = idx;
131344d93782SGreg Clayton           break;
131444d93782SGreg Clayton         }
131544d93782SGreg Clayton       }
131644d93782SGreg Clayton     }
131744d93782SGreg Clayton   }
131844d93782SGreg Clayton 
1319b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1320315b6884SEugene Zelenko 
132144d93782SGreg Clayton protected:
132244d93782SGreg Clayton   std::string m_name;
132344d93782SGreg Clayton   WINDOW *m_window;
132444d93782SGreg Clayton   PANEL *m_panel;
132544d93782SGreg Clayton   Window *m_parent;
132644d93782SGreg Clayton   Windows m_subwindows;
132744d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
132844d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
132944d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
133044d93782SGreg Clayton   bool m_delete;
133144d93782SGreg Clayton   bool m_needs_update;
133244d93782SGreg Clayton   bool m_can_activate;
133344d93782SGreg Clayton   bool m_is_subwin;
133444d93782SGreg Clayton 
133544d93782SGreg Clayton private:
133644d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
133744d93782SGreg Clayton };
133844d93782SGreg Clayton 
1339b9c1b51eSKate Stone class MenuDelegate {
134044d93782SGreg Clayton public:
1341315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
134244d93782SGreg Clayton 
1343b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
134444d93782SGreg Clayton };
134544d93782SGreg Clayton 
1346b9c1b51eSKate Stone class Menu : public WindowDelegate {
134744d93782SGreg Clayton public:
1348b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
134944d93782SGreg Clayton 
135044d93782SGreg Clayton   // Menubar or separator constructor
135144d93782SGreg Clayton   Menu(Type type);
135244d93782SGreg Clayton 
135344d93782SGreg Clayton   // Menuitem constructor
1354b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
135544d93782SGreg Clayton        uint64_t identifier);
135644d93782SGreg Clayton 
1357315b6884SEugene Zelenko   ~Menu() override = default;
135844d93782SGreg Clayton 
1359b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
136044d93782SGreg Clayton 
1361b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
136244d93782SGreg Clayton     m_delegate_sp = delegate_sp;
136344d93782SGreg Clayton   }
136444d93782SGreg Clayton 
1365b9c1b51eSKate Stone   void RecalculateNameLengths();
136644d93782SGreg Clayton 
1367b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
136844d93782SGreg Clayton 
1369b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
137044d93782SGreg Clayton 
1371b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
137244d93782SGreg Clayton 
1373b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
137444d93782SGreg Clayton 
1375b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
137644d93782SGreg Clayton 
1377b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
137844d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1379b9c1b51eSKate Stone     if (m_delegate_sp) {
138044d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
138144d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
138244d93782SGreg Clayton         return result;
1383b9c1b51eSKate Stone     } else if (m_parent) {
138444d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
138544d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
138644d93782SGreg Clayton         return result;
138744d93782SGreg Clayton     }
138844d93782SGreg Clayton     return m_canned_result;
138944d93782SGreg Clayton   }
139044d93782SGreg Clayton 
1391b9c1b51eSKate Stone   MenuActionResult Action() {
139205097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
139305097246SAdrian Prantl     // delegate, and if not, try our parent menu
139444d93782SGreg Clayton     return ActionPrivate(*this);
139544d93782SGreg Clayton   }
139644d93782SGreg Clayton 
1397b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
139844d93782SGreg Clayton 
1399b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
140044d93782SGreg Clayton 
1401b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
140244d93782SGreg Clayton 
1403b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
140444d93782SGreg Clayton 
1405b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
140644d93782SGreg Clayton 
1407b9c1b51eSKate Stone   Type GetType() const { return m_type; }
140844d93782SGreg Clayton 
1409b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
141044d93782SGreg Clayton 
1411b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
141244d93782SGreg Clayton 
1413b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
141444d93782SGreg Clayton 
1415b9c1b51eSKate Stone   void SetKeyValue(int key_value) { m_key_value = key_value; }
141644d93782SGreg Clayton 
1417b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
141844d93782SGreg Clayton 
1419b9c1b51eSKate Stone   std::string &GetKeyName() { return m_key_name; }
142044d93782SGreg Clayton 
1421b9c1b51eSKate Stone   int GetDrawWidth() const {
142244d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
142344d93782SGreg Clayton   }
142444d93782SGreg Clayton 
1425b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
142644d93782SGreg Clayton 
1427b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
142844d93782SGreg Clayton 
142944d93782SGreg Clayton protected:
143044d93782SGreg Clayton   std::string m_name;
143144d93782SGreg Clayton   std::string m_key_name;
143244d93782SGreg Clayton   uint64_t m_identifier;
143344d93782SGreg Clayton   Type m_type;
143444d93782SGreg Clayton   int m_key_value;
143544d93782SGreg Clayton   int m_start_col;
143644d93782SGreg Clayton   int m_max_submenu_name_length;
143744d93782SGreg Clayton   int m_max_submenu_key_name_length;
143844d93782SGreg Clayton   int m_selected;
143944d93782SGreg Clayton   Menu *m_parent;
144044d93782SGreg Clayton   Menus m_submenus;
144144d93782SGreg Clayton   WindowSP m_menu_window_sp;
144244d93782SGreg Clayton   MenuActionResult m_canned_result;
144344d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
144444d93782SGreg Clayton };
144544d93782SGreg Clayton 
144644d93782SGreg Clayton // Menubar or separator constructor
1447b9c1b51eSKate Stone Menu::Menu(Type type)
1448b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1449b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1450b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1451b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1452b9c1b51eSKate Stone       m_delegate_sp() {}
145344d93782SGreg Clayton 
145444d93782SGreg Clayton // Menuitem constructor
1455b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1456b9c1b51eSKate Stone            uint64_t identifier)
1457b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1458b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1459b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1460b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1461b9c1b51eSKate Stone       m_delegate_sp() {
1462b9c1b51eSKate Stone   if (name && name[0]) {
146344d93782SGreg Clayton     m_name = name;
146444d93782SGreg Clayton     m_type = Type::Item;
146544d93782SGreg Clayton     if (key_name && key_name[0])
146644d93782SGreg Clayton       m_key_name = key_name;
1467b9c1b51eSKate Stone   } else {
146844d93782SGreg Clayton     m_type = Type::Separator;
146944d93782SGreg Clayton   }
147044d93782SGreg Clayton }
147144d93782SGreg Clayton 
1472b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
147344d93782SGreg Clayton   m_max_submenu_name_length = 0;
147444d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
147544d93782SGreg Clayton   Menus &submenus = GetSubmenus();
147644d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1477b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
147844d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14793985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
148044d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1481b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1482b9c1b51eSKate Stone         submenu->m_key_name.size())
148344d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
148444d93782SGreg Clayton   }
148544d93782SGreg Clayton }
148644d93782SGreg Clayton 
1487b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
148844d93782SGreg Clayton   menu_sp->m_parent = this;
14893985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
149044d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1491b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1492b9c1b51eSKate Stone       menu_sp->m_key_name.size())
149344d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
149444d93782SGreg Clayton   m_submenus.push_back(menu_sp);
149544d93782SGreg Clayton }
149644d93782SGreg Clayton 
1497b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1498b9c1b51eSKate Stone   if (m_type == Type::Separator) {
149944d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
150044d93782SGreg Clayton     window.PutChar(ACS_LTEE);
150144d93782SGreg Clayton     int width = window.GetWidth();
1502b9c1b51eSKate Stone     if (width > 2) {
150344d93782SGreg Clayton       width -= 2;
15043985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
150544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
150644d93782SGreg Clayton     }
150744d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1508b9c1b51eSKate Stone   } else {
150944d93782SGreg Clayton     const int shortcut_key = m_key_value;
151044d93782SGreg Clayton     bool underlined_shortcut = false;
151144d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
151244d93782SGreg Clayton     if (highlight)
151344d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1514b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
151544d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
151644d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
151744d93782SGreg Clayton       const char *name = m_name.c_str();
151844d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1519b9c1b51eSKate Stone       if (pos != std::string::npos) {
152044d93782SGreg Clayton         underlined_shortcut = true;
1521b9c1b51eSKate Stone         if (pos > 0) {
152244d93782SGreg Clayton           window.PutCString(name, pos);
152344d93782SGreg Clayton           name += pos;
152444d93782SGreg Clayton         }
152544d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
152644d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
152744d93782SGreg Clayton         window.PutChar(name[0]);
152844d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
152944d93782SGreg Clayton         name++;
153044d93782SGreg Clayton         if (name[0])
153144d93782SGreg Clayton           window.PutCString(name);
153244d93782SGreg Clayton       }
153344d93782SGreg Clayton     }
153444d93782SGreg Clayton 
1535b9c1b51eSKate Stone     if (!underlined_shortcut) {
153644d93782SGreg Clayton       window.PutCString(m_name.c_str());
153744d93782SGreg Clayton     }
153844d93782SGreg Clayton 
153944d93782SGreg Clayton     if (highlight)
154044d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
154144d93782SGreg Clayton 
1542b9c1b51eSKate Stone     if (m_key_name.empty()) {
1543b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
154444d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
154544d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
154644d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
154744d93782SGreg Clayton       }
1548b9c1b51eSKate Stone     } else {
154944d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
155044d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
155144d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
155244d93782SGreg Clayton     }
155344d93782SGreg Clayton   }
155444d93782SGreg Clayton }
155544d93782SGreg Clayton 
1556b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
155744d93782SGreg Clayton   Menus &submenus = GetSubmenus();
155844d93782SGreg Clayton   const size_t num_submenus = submenus.size();
155944d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
156044d93782SGreg Clayton   Menu::Type menu_type = GetType();
1561b9c1b51eSKate Stone   switch (menu_type) {
1562b9c1b51eSKate Stone   case Menu::Type::Bar: {
156344d93782SGreg Clayton     window.SetBackground(2);
156444d93782SGreg Clayton     window.MoveCursor(0, 0);
1565b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
156644d93782SGreg Clayton       Menu *menu = submenus[i].get();
156744d93782SGreg Clayton       if (i > 0)
156844d93782SGreg Clayton         window.PutChar(' ');
156944d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
157044d93782SGreg Clayton       window.PutCString("| ");
157144d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
157244d93782SGreg Clayton     }
157344d93782SGreg Clayton     window.PutCString(" |");
157444d93782SGreg Clayton     window.DeferredRefresh();
1575b9c1b51eSKate Stone   } break;
157644d93782SGreg Clayton 
1577b9c1b51eSKate Stone   case Menu::Type::Item: {
157844d93782SGreg Clayton     int y = 1;
157944d93782SGreg Clayton     int x = 3;
158044d93782SGreg Clayton     // Draw the menu
158144d93782SGreg Clayton     int cursor_x = 0;
158244d93782SGreg Clayton     int cursor_y = 0;
158344d93782SGreg Clayton     window.Erase();
158444d93782SGreg Clayton     window.SetBackground(2);
158544d93782SGreg Clayton     window.Box();
1586b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1587b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
158844d93782SGreg Clayton       window.MoveCursor(x, y + i);
1589b9c1b51eSKate Stone       if (is_selected) {
159044d93782SGreg Clayton         // Remember where we want the cursor to be
159144d93782SGreg Clayton         cursor_x = x - 1;
159244d93782SGreg Clayton         cursor_y = y + i;
159344d93782SGreg Clayton       }
159444d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
159544d93782SGreg Clayton     }
159644d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
159744d93782SGreg Clayton     window.DeferredRefresh();
1598b9c1b51eSKate Stone   } break;
159944d93782SGreg Clayton 
160044d93782SGreg Clayton   default:
160144d93782SGreg Clayton   case Menu::Type::Separator:
160244d93782SGreg Clayton     break;
160344d93782SGreg Clayton   }
160444d93782SGreg Clayton   return true; // Drawing handled...
160544d93782SGreg Clayton }
160644d93782SGreg Clayton 
1607b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
160844d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
160944d93782SGreg Clayton 
161044d93782SGreg Clayton   Menus &submenus = GetSubmenus();
161144d93782SGreg Clayton   const size_t num_submenus = submenus.size();
161244d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
161344d93782SGreg Clayton   Menu::Type menu_type = GetType();
1614b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
161544d93782SGreg Clayton     MenuSP run_menu_sp;
1616b9c1b51eSKate Stone     switch (key) {
161744d93782SGreg Clayton     case KEY_DOWN:
161844d93782SGreg Clayton     case KEY_UP:
161944d93782SGreg Clayton       // Show last menu or first menu
16203985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
162144d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
162244d93782SGreg Clayton       else if (!submenus.empty())
162344d93782SGreg Clayton         run_menu_sp = submenus.front();
162444d93782SGreg Clayton       result = eKeyHandled;
162544d93782SGreg Clayton       break;
162644d93782SGreg Clayton 
162744d93782SGreg Clayton     case KEY_RIGHT:
162844d93782SGreg Clayton       ++m_selected;
16293985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
163044d93782SGreg Clayton         m_selected = 0;
16313985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
163244d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
163344d93782SGreg Clayton       else if (!submenus.empty())
163444d93782SGreg Clayton         run_menu_sp = submenus.front();
163544d93782SGreg Clayton       result = eKeyHandled;
163644d93782SGreg Clayton       break;
163744d93782SGreg Clayton 
163844d93782SGreg Clayton     case KEY_LEFT:
163944d93782SGreg Clayton       --m_selected;
164044d93782SGreg Clayton       if (m_selected < 0)
164144d93782SGreg Clayton         m_selected = num_submenus - 1;
16423985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
164344d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
164444d93782SGreg Clayton       else if (!submenus.empty())
164544d93782SGreg Clayton         run_menu_sp = submenus.front();
164644d93782SGreg Clayton       result = eKeyHandled;
164744d93782SGreg Clayton       break;
164844d93782SGreg Clayton 
164944d93782SGreg Clayton     default:
1650b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1651b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
165244d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
165344d93782SGreg Clayton           run_menu_sp = submenus[i];
165444d93782SGreg Clayton           result = eKeyHandled;
165544d93782SGreg Clayton           break;
165644d93782SGreg Clayton         }
165744d93782SGreg Clayton       }
165844d93782SGreg Clayton       break;
165944d93782SGreg Clayton     }
166044d93782SGreg Clayton 
1661b9c1b51eSKate Stone     if (run_menu_sp) {
166205097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
166305097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
166405097246SAdrian Prantl       // decorations need to be calculated
166544d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
166644d93782SGreg Clayton         return eQuitApplication;
166744d93782SGreg Clayton 
166844d93782SGreg Clayton       Rect menu_bounds;
166944d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
167044d93782SGreg Clayton       menu_bounds.origin.y = 1;
167144d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
167244d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
167344d93782SGreg Clayton       if (m_menu_window_sp)
167444d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
167544d93782SGreg Clayton 
1676b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1677b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
167844d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
167944d93782SGreg Clayton     }
1680b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1681b9c1b51eSKate Stone     switch (key) {
168244d93782SGreg Clayton     case KEY_DOWN:
1683b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
168444d93782SGreg Clayton         const int start_select = m_selected;
1685b9c1b51eSKate Stone         while (++m_selected != start_select) {
16863985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
168744d93782SGreg Clayton             m_selected = 0;
168844d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
168944d93782SGreg Clayton             continue;
169044d93782SGreg Clayton           else
169144d93782SGreg Clayton             break;
169244d93782SGreg Clayton         }
169344d93782SGreg Clayton         return eKeyHandled;
169444d93782SGreg Clayton       }
169544d93782SGreg Clayton       break;
169644d93782SGreg Clayton 
169744d93782SGreg Clayton     case KEY_UP:
1698b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
169944d93782SGreg Clayton         const int start_select = m_selected;
1700b9c1b51eSKate Stone         while (--m_selected != start_select) {
17013985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
170244d93782SGreg Clayton             m_selected = num_submenus - 1;
170344d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
170444d93782SGreg Clayton             continue;
170544d93782SGreg Clayton           else
170644d93782SGreg Clayton             break;
170744d93782SGreg Clayton         }
170844d93782SGreg Clayton         return eKeyHandled;
170944d93782SGreg Clayton       }
171044d93782SGreg Clayton       break;
171144d93782SGreg Clayton 
171244d93782SGreg Clayton     case KEY_RETURN:
1713b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
171444d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
171544d93782SGreg Clayton           return eQuitApplication;
171644d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
171744d93782SGreg Clayton         return eKeyHandled;
171844d93782SGreg Clayton       }
171944d93782SGreg Clayton       break;
172044d93782SGreg Clayton 
1721b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1722b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
172344d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
172444d93782SGreg Clayton       return eKeyHandled;
172544d93782SGreg Clayton 
172644d93782SGreg Clayton     default:
1727b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
172844d93782SGreg Clayton         Menu *menu = submenus[i].get();
1729b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
173044d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
173144d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
173244d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
173344d93782SGreg Clayton             return eQuitApplication;
173444d93782SGreg Clayton           return eKeyHandled;
173544d93782SGreg Clayton         }
173644d93782SGreg Clayton       }
173744d93782SGreg Clayton       break;
173844d93782SGreg Clayton     }
1739b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
174044d93782SGreg Clayton   }
174144d93782SGreg Clayton   return result;
174244d93782SGreg Clayton }
174344d93782SGreg Clayton 
1744b9c1b51eSKate Stone class Application {
174544d93782SGreg Clayton public:
1746b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1747b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
174844d93782SGreg Clayton 
1749b9c1b51eSKate Stone   ~Application() {
175044d93782SGreg Clayton     m_window_delegates.clear();
175144d93782SGreg Clayton     m_window_sp.reset();
1752b9c1b51eSKate Stone     if (m_screen) {
175344d93782SGreg Clayton       ::delscreen(m_screen);
1754c5dac77aSEugene Zelenko       m_screen = nullptr;
175544d93782SGreg Clayton     }
175644d93782SGreg Clayton   }
175744d93782SGreg Clayton 
1758b9c1b51eSKate Stone   void Initialize() {
175944d93782SGreg Clayton     ::setlocale(LC_ALL, "");
176044d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
1761c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
176244d93782SGreg Clayton     ::start_color();
176344d93782SGreg Clayton     ::curs_set(0);
176444d93782SGreg Clayton     ::noecho();
176544d93782SGreg Clayton     ::keypad(stdscr, TRUE);
176644d93782SGreg Clayton   }
176744d93782SGreg Clayton 
1768b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
176944d93782SGreg Clayton 
1770b9c1b51eSKate Stone   void Run(Debugger &debugger) {
177144d93782SGreg Clayton     bool done = false;
177244d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
177344d93782SGreg Clayton 
177405097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
177505097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
177605097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
177705097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
177805097246SAdrian Prantl     // to receive async process events while in this loop.
177944d93782SGreg Clayton 
1780b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1781b9c1b51eSKate Stone                                             // of seconds seconds when calling
1782b9c1b51eSKate Stone                                             // Window::GetChar()
178344d93782SGreg Clayton 
1784b9c1b51eSKate Stone     ListenerSP listener_sp(
1785b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
178644d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
178744d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
178844d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
178944d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
179044d93782SGreg Clayton 
179144d93782SGreg Clayton     bool update = true;
179244d93782SGreg Clayton #if defined(__APPLE__)
179344d93782SGreg Clayton     std::deque<int> escape_chars;
179444d93782SGreg Clayton #endif
179544d93782SGreg Clayton 
1796b9c1b51eSKate Stone     while (!done) {
1797b9c1b51eSKate Stone       if (update) {
179844d93782SGreg Clayton         m_window_sp->Draw(false);
179905097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
180005097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
180105097246SAdrian Prantl         // blinking
180244d93782SGreg Clayton         update_panels();
180344d93782SGreg Clayton 
1804b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1805b9c1b51eSKate Stone         // corner
180644d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
180744d93782SGreg Clayton 
180844d93782SGreg Clayton         doupdate();
180944d93782SGreg Clayton         update = false;
181044d93782SGreg Clayton       }
181144d93782SGreg Clayton 
181244d93782SGreg Clayton #if defined(__APPLE__)
181305097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
181405097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1815b9c1b51eSKate Stone       // possible
181644d93782SGreg Clayton       int ch;
181744d93782SGreg Clayton       if (escape_chars.empty())
181844d93782SGreg Clayton         ch = m_window_sp->GetChar();
1819b9c1b51eSKate Stone       else {
182044d93782SGreg Clayton         ch = escape_chars.front();
182144d93782SGreg Clayton         escape_chars.pop_front();
182244d93782SGreg Clayton       }
1823b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
182444d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1825b9c1b51eSKate Stone         if (ch2 == 'O') {
182644d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1827b9c1b51eSKate Stone           switch (ch3) {
1828b9c1b51eSKate Stone           case 'P':
1829b9c1b51eSKate Stone             ch = KEY_F(1);
1830b9c1b51eSKate Stone             break;
1831b9c1b51eSKate Stone           case 'Q':
1832b9c1b51eSKate Stone             ch = KEY_F(2);
1833b9c1b51eSKate Stone             break;
1834b9c1b51eSKate Stone           case 'R':
1835b9c1b51eSKate Stone             ch = KEY_F(3);
1836b9c1b51eSKate Stone             break;
1837b9c1b51eSKate Stone           case 'S':
1838b9c1b51eSKate Stone             ch = KEY_F(4);
1839b9c1b51eSKate Stone             break;
184044d93782SGreg Clayton           default:
184144d93782SGreg Clayton             escape_chars.push_back(ch2);
184244d93782SGreg Clayton             if (ch3 != -1)
184344d93782SGreg Clayton               escape_chars.push_back(ch3);
184444d93782SGreg Clayton             break;
184544d93782SGreg Clayton           }
1846b9c1b51eSKate Stone         } else if (ch2 != -1)
184744d93782SGreg Clayton           escape_chars.push_back(ch2);
184844d93782SGreg Clayton       }
184944d93782SGreg Clayton #else
185044d93782SGreg Clayton       int ch = m_window_sp->GetChar();
185144d93782SGreg Clayton 
185244d93782SGreg Clayton #endif
1853b9c1b51eSKate Stone       if (ch == -1) {
1854b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
185544d93782SGreg Clayton           done = true;
1856b9c1b51eSKate Stone         } else {
185744d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
185844d93782SGreg Clayton           EventSP event_sp;
1859b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1860d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
186144d93782SGreg Clayton 
1862b9c1b51eSKate Stone             if (event_sp) {
186344d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1864b9c1b51eSKate Stone               if (broadcaster) {
186544d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1866b9c1b51eSKate Stone                 ConstString broadcaster_class(
1867b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1868b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1869b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1870b9c1b51eSKate Stone                       nullptr);
187144d93782SGreg Clayton                   update = true;
187244d93782SGreg Clayton                   continue; // Don't get any key, just update our view
187344d93782SGreg Clayton                 }
187444d93782SGreg Clayton               }
187544d93782SGreg Clayton             }
187644d93782SGreg Clayton           }
187744d93782SGreg Clayton         }
1878b9c1b51eSKate Stone       } else {
187944d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1880b9c1b51eSKate Stone         switch (key_result) {
188144d93782SGreg Clayton         case eKeyHandled:
1882c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
188344d93782SGreg Clayton           update = true;
188444d93782SGreg Clayton           break;
188544d93782SGreg Clayton         case eKeyNotHandled:
188644d93782SGreg Clayton           break;
188744d93782SGreg Clayton         case eQuitApplication:
188844d93782SGreg Clayton           done = true;
188944d93782SGreg Clayton           break;
189044d93782SGreg Clayton         }
189144d93782SGreg Clayton       }
189244d93782SGreg Clayton     }
189344d93782SGreg Clayton 
189444d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
189544d93782SGreg Clayton   }
189644d93782SGreg Clayton 
1897b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
189844d93782SGreg Clayton     if (!m_window_sp)
1899796ac80bSJonas Devlieghere       m_window_sp = std::make_shared<Window>("main", stdscr, false);
190044d93782SGreg Clayton     return m_window_sp;
190144d93782SGreg Clayton   }
190244d93782SGreg Clayton 
1903b9c1b51eSKate Stone   WindowDelegates &GetWindowDelegates() { return m_window_delegates; }
190444d93782SGreg Clayton 
190544d93782SGreg Clayton protected:
190644d93782SGreg Clayton   WindowSP m_window_sp;
190744d93782SGreg Clayton   WindowDelegates m_window_delegates;
190844d93782SGreg Clayton   SCREEN *m_screen;
190944d93782SGreg Clayton   FILE *m_in;
191044d93782SGreg Clayton   FILE *m_out;
191144d93782SGreg Clayton };
191244d93782SGreg Clayton 
191344d93782SGreg Clayton } // namespace curses
191444d93782SGreg Clayton 
191544d93782SGreg Clayton using namespace curses;
191644d93782SGreg Clayton 
1917b9c1b51eSKate Stone struct Row {
19188369b28dSGreg Clayton   ValueObjectManager value;
191944d93782SGreg Clayton   Row *parent;
19208369b28dSGreg Clayton   // The process stop ID when the children were calculated.
19218369b28dSGreg Clayton   uint32_t children_stop_id;
192244d93782SGreg Clayton   int row_idx;
192344d93782SGreg Clayton   int x;
192444d93782SGreg Clayton   int y;
192544d93782SGreg Clayton   bool might_have_children;
192644d93782SGreg Clayton   bool expanded;
192744d93782SGreg Clayton   bool calculated_children;
192844d93782SGreg Clayton   std::vector<Row> children;
192944d93782SGreg Clayton 
1930b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
19318369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
19328369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1933b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
193444d93782SGreg Clayton 
1935b9c1b51eSKate Stone   size_t GetDepth() const {
193644d93782SGreg Clayton     if (parent)
193744d93782SGreg Clayton       return 1 + parent->GetDepth();
193844d93782SGreg Clayton     return 0;
193944d93782SGreg Clayton   }
194044d93782SGreg Clayton 
1941b9c1b51eSKate Stone   void Expand() {
194244d93782SGreg Clayton     expanded = true;
19438369b28dSGreg Clayton   }
19448369b28dSGreg Clayton 
19458369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
19468369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
19478369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
19488369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
19498369b28dSGreg Clayton       children_stop_id = stop_id;
19508369b28dSGreg Clayton       calculated_children = false;
19518369b28dSGreg Clayton     }
1952b9c1b51eSKate Stone     if (!calculated_children) {
19538369b28dSGreg Clayton       children.clear();
195444d93782SGreg Clayton       calculated_children = true;
19558369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1956b9c1b51eSKate Stone       if (valobj) {
195744d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1958b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
195944d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
196044d93782SGreg Clayton         }
196144d93782SGreg Clayton       }
196244d93782SGreg Clayton     }
19638369b28dSGreg Clayton     return children;
196444d93782SGreg Clayton   }
196544d93782SGreg Clayton 
19668369b28dSGreg Clayton   void Unexpand() {
19678369b28dSGreg Clayton     expanded = false;
19688369b28dSGreg Clayton     calculated_children = false;
19698369b28dSGreg Clayton     children.clear();
19708369b28dSGreg Clayton   }
197144d93782SGreg Clayton 
1972b9c1b51eSKate Stone   void DrawTree(Window &window) {
197344d93782SGreg Clayton     if (parent)
197444d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
197544d93782SGreg Clayton 
1976b9c1b51eSKate Stone     if (might_have_children) {
197705097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
197805097246SAdrian Prantl       // "symbol" UTF8 string below
197944d93782SGreg Clayton       //            const char *symbol = "";
198044d93782SGreg Clayton       //            if (row.expanded)
198144d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
198244d93782SGreg Clayton       //            else
198344d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
198444d93782SGreg Clayton       //            window.PutCString (symbol);
198544d93782SGreg Clayton 
198605097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
198705097246SAdrian Prantl       // or '>' character...
198844d93782SGreg Clayton       //            if (expanded)
198944d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
199044d93782SGreg Clayton       //            else
199144d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
199205097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
199305097246SAdrian Prantl       // just use a diamond...
199444d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
199544d93782SGreg Clayton       window.PutChar(ACS_HLINE);
199644d93782SGreg Clayton     }
199744d93782SGreg Clayton   }
199844d93782SGreg Clayton 
1999b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
200044d93782SGreg Clayton     if (parent)
200144d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
200244d93782SGreg Clayton 
20038369b28dSGreg Clayton     if (&GetChildren().back() == child) {
200444d93782SGreg Clayton       // Last child
2005b9c1b51eSKate Stone       if (reverse_depth == 0) {
200644d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
200744d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2008b9c1b51eSKate Stone       } else {
200944d93782SGreg Clayton         window.PutChar(' ');
201044d93782SGreg Clayton         window.PutChar(' ');
201144d93782SGreg Clayton       }
2012b9c1b51eSKate Stone     } else {
2013b9c1b51eSKate Stone       if (reverse_depth == 0) {
201444d93782SGreg Clayton         window.PutChar(ACS_LTEE);
201544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2016b9c1b51eSKate Stone       } else {
201744d93782SGreg Clayton         window.PutChar(ACS_VLINE);
201844d93782SGreg Clayton         window.PutChar(' ');
201944d93782SGreg Clayton       }
202044d93782SGreg Clayton     }
202144d93782SGreg Clayton   }
202244d93782SGreg Clayton };
202344d93782SGreg Clayton 
2024b9c1b51eSKate Stone struct DisplayOptions {
202544d93782SGreg Clayton   bool show_types;
202644d93782SGreg Clayton };
202744d93782SGreg Clayton 
202844d93782SGreg Clayton class TreeItem;
202944d93782SGreg Clayton 
2030b9c1b51eSKate Stone class TreeDelegate {
203144d93782SGreg Clayton public:
2032c5dac77aSEugene Zelenko   TreeDelegate() = default;
2033315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
2034315b6884SEugene Zelenko 
203544d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
203644d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2037b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2038b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
203944d93782SGreg Clayton };
2040315b6884SEugene Zelenko 
204144d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
204244d93782SGreg Clayton 
2043b9c1b51eSKate Stone class TreeItem {
204444d93782SGreg Clayton public:
2045b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2046b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2047b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2048b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
204944d93782SGreg Clayton 
2050b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2051b9c1b51eSKate Stone     if (this != &rhs) {
205244d93782SGreg Clayton       m_parent = rhs.m_parent;
205344d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2054ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
205544d93782SGreg Clayton       m_identifier = rhs.m_identifier;
205644d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
205744d93782SGreg Clayton       m_children = rhs.m_children;
205844d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
205944d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
206044d93782SGreg Clayton     }
206144d93782SGreg Clayton     return *this;
206244d93782SGreg Clayton   }
206344d93782SGreg Clayton 
2064b9c1b51eSKate Stone   size_t GetDepth() const {
206544d93782SGreg Clayton     if (m_parent)
206644d93782SGreg Clayton       return 1 + m_parent->GetDepth();
206744d93782SGreg Clayton     return 0;
206844d93782SGreg Clayton   }
206944d93782SGreg Clayton 
2070b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
207144d93782SGreg Clayton 
2072b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
207344d93782SGreg Clayton 
2074b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
207544d93782SGreg Clayton 
2076b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
207744d93782SGreg Clayton 
2078b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
207944d93782SGreg Clayton 
2080b9c1b51eSKate Stone   size_t GetNumChildren() {
208144d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
208244d93782SGreg Clayton     return m_children.size();
208344d93782SGreg Clayton   }
208444d93782SGreg Clayton 
2085b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2086315b6884SEugene Zelenko 
2087b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
208844d93782SGreg Clayton     SetRowIndex(row_idx);
208944d93782SGreg Clayton     ++row_idx;
209044d93782SGreg Clayton 
2091ec990867SGreg Clayton     const bool expanded = IsExpanded();
2092ec990867SGreg Clayton 
209305097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
209405097246SAdrian Prantl     // number of children if the item is expanded
2095c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
209644d93782SGreg Clayton       GetNumChildren();
209744d93782SGreg Clayton 
2098b9c1b51eSKate Stone     for (auto &item : m_children) {
209944d93782SGreg Clayton       if (expanded)
210044d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
210144d93782SGreg Clayton       else
210244d93782SGreg Clayton         item.SetRowIndex(-1);
210344d93782SGreg Clayton     }
210444d93782SGreg Clayton   }
210544d93782SGreg Clayton 
2106b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
210744d93782SGreg Clayton 
2108b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
210944d93782SGreg Clayton 
2110b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
211144d93782SGreg Clayton 
2112b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
211344d93782SGreg Clayton 
2114b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2115b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
211644d93782SGreg Clayton     if (num_rows_left <= 0)
211744d93782SGreg Clayton       return false;
211844d93782SGreg Clayton 
2119b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
212044d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
212144d93782SGreg Clayton 
212244d93782SGreg Clayton       if (m_parent)
212344d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
212444d93782SGreg Clayton 
2125b9c1b51eSKate Stone       if (m_might_have_children) {
2126b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
212705097246SAdrian Prantl         // "symbol" UTF8 string below
212844d93782SGreg Clayton         //            const char *symbol = "";
212944d93782SGreg Clayton         //            if (row.expanded)
213044d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
213144d93782SGreg Clayton         //            else
213244d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
213344d93782SGreg Clayton         //            window.PutCString (symbol);
213444d93782SGreg Clayton 
213544d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
213644d93782SGreg Clayton         // 'v' or '>' character...
213744d93782SGreg Clayton         //            if (expanded)
213844d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
213944d93782SGreg Clayton         //            else
214044d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
214105097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
214205097246SAdrian Prantl         // just use a diamond...
214344d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
214444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
214544d93782SGreg Clayton       }
2146b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2147b9c1b51eSKate Stone                        window.IsActive();
214844d93782SGreg Clayton 
214944d93782SGreg Clayton       if (highlight)
215044d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
215144d93782SGreg Clayton 
215244d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
215344d93782SGreg Clayton 
215444d93782SGreg Clayton       if (highlight)
215544d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
215644d93782SGreg Clayton       ++row_idx;
215744d93782SGreg Clayton       --num_rows_left;
215844d93782SGreg Clayton     }
215944d93782SGreg Clayton 
216044d93782SGreg Clayton     if (num_rows_left <= 0)
216144d93782SGreg Clayton       return false; // We are done drawing...
216244d93782SGreg Clayton 
2163b9c1b51eSKate Stone     if (IsExpanded()) {
2164b9c1b51eSKate Stone       for (auto &item : m_children) {
216505097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
216605097246SAdrian Prantl         // done drawing and can exit this for loop
2167b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2168b9c1b51eSKate Stone                        num_rows_left))
216944d93782SGreg Clayton           break;
217044d93782SGreg Clayton       }
217144d93782SGreg Clayton     }
217244d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
217344d93782SGreg Clayton   }
217444d93782SGreg Clayton 
2175b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2176b9c1b51eSKate Stone                         uint32_t reverse_depth) {
217744d93782SGreg Clayton     if (m_parent)
217844d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
217944d93782SGreg Clayton 
2180b9c1b51eSKate Stone     if (&m_children.back() == child) {
218144d93782SGreg Clayton       // Last child
2182b9c1b51eSKate Stone       if (reverse_depth == 0) {
218344d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
218444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2185b9c1b51eSKate Stone       } else {
218644d93782SGreg Clayton         window.PutChar(' ');
218744d93782SGreg Clayton         window.PutChar(' ');
218844d93782SGreg Clayton       }
2189b9c1b51eSKate Stone     } else {
2190b9c1b51eSKate Stone       if (reverse_depth == 0) {
219144d93782SGreg Clayton         window.PutChar(ACS_LTEE);
219244d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2193b9c1b51eSKate Stone       } else {
219444d93782SGreg Clayton         window.PutChar(ACS_VLINE);
219544d93782SGreg Clayton         window.PutChar(' ');
219644d93782SGreg Clayton       }
219744d93782SGreg Clayton     }
219844d93782SGreg Clayton   }
219944d93782SGreg Clayton 
2200b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
22013985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
220244d93782SGreg Clayton       return this;
220344d93782SGreg Clayton     if (m_children.empty())
2204c5dac77aSEugene Zelenko       return nullptr;
2205b9c1b51eSKate Stone     if (IsExpanded()) {
2206b9c1b51eSKate Stone       for (auto &item : m_children) {
220744d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
220844d93782SGreg Clayton         if (selected_item_ptr)
220944d93782SGreg Clayton           return selected_item_ptr;
221044d93782SGreg Clayton       }
221144d93782SGreg Clayton     }
2212c5dac77aSEugene Zelenko     return nullptr;
221344d93782SGreg Clayton   }
221444d93782SGreg Clayton 
2215b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2216ec990867SGreg Clayton 
2217b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2218ec990867SGreg Clayton 
2219b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
222044d93782SGreg Clayton 
2221b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
222244d93782SGreg Clayton 
2223b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2224ec990867SGreg Clayton 
222544d93782SGreg Clayton protected:
222644d93782SGreg Clayton   TreeItem *m_parent;
222744d93782SGreg Clayton   TreeDelegate &m_delegate;
2228ec990867SGreg Clayton   void *m_user_data;
222944d93782SGreg Clayton   uint64_t m_identifier;
2230b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2231b9c1b51eSKate Stone                  // root item
223244d93782SGreg Clayton   std::vector<TreeItem> m_children;
223344d93782SGreg Clayton   bool m_might_have_children;
223444d93782SGreg Clayton   bool m_is_expanded;
223544d93782SGreg Clayton };
223644d93782SGreg Clayton 
2237b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
223844d93782SGreg Clayton public:
2239b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2240b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2241b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2242b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2243b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
224444d93782SGreg Clayton 
2245b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
224644d93782SGreg Clayton 
2247b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2248b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2249b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
225044d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
225144d93782SGreg Clayton 
225244d93782SGreg Clayton     bool display_content = false;
2253b9c1b51eSKate Stone     if (process) {
225444d93782SGreg Clayton       StateType state = process->GetState();
2255b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
225644d93782SGreg Clayton         // We are stopped, so it is ok to
225744d93782SGreg Clayton         display_content = true;
2258b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
225944d93782SGreg Clayton         return true; // Don't do any updating when we are running
226044d93782SGreg Clayton       }
226144d93782SGreg Clayton     }
226244d93782SGreg Clayton 
226344d93782SGreg Clayton     m_min_x = 2;
226444d93782SGreg Clayton     m_min_y = 1;
226544d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
226644d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
226744d93782SGreg Clayton 
226844d93782SGreg Clayton     window.Erase();
226944d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
227044d93782SGreg Clayton 
2271b9c1b51eSKate Stone     if (display_content) {
227244d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
227344d93782SGreg Clayton       m_num_rows = 0;
227444d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
227544d93782SGreg Clayton 
227605097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
227705097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
227805097246SAdrian Prantl       // rows by setting the first visible row accordingly.
227944d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
228044d93782SGreg Clayton         m_first_visible_row = 0;
228144d93782SGreg Clayton 
228244d93782SGreg Clayton       // Make sure the selected row is always visible
228344d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
228444d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
228544d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
228644d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
228744d93782SGreg Clayton 
228844d93782SGreg Clayton       int row_idx = 0;
228944d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2290b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2291b9c1b51eSKate Stone                   num_rows_left);
229244d93782SGreg Clayton       // Get the selected row
229344d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2294b9c1b51eSKate Stone     } else {
2295c5dac77aSEugene Zelenko       m_selected_item = nullptr;
229644d93782SGreg Clayton     }
229744d93782SGreg Clayton 
229844d93782SGreg Clayton     window.DeferredRefresh();
229944d93782SGreg Clayton 
230044d93782SGreg Clayton     return true; // Drawing handled
230144d93782SGreg Clayton   }
230244d93782SGreg Clayton 
2303b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
230444d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
230544d93782SGreg Clayton   }
230644d93782SGreg Clayton 
2307b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
230844d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
230944d93782SGreg Clayton         {KEY_UP, "Select previous item"},
231044d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
231144d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2312b9c1b51eSKate Stone         {KEY_LEFT,
2313b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
231444d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
231544d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
231644d93782SGreg Clayton         {'h', "Show help dialog"},
231744d93782SGreg Clayton         {' ', "Toggle item expansion"},
231844d93782SGreg Clayton         {',', "Page up"},
231944d93782SGreg Clayton         {'.', "Page down"},
2320b9c1b51eSKate Stone         {'\0', nullptr}};
232144d93782SGreg Clayton     return g_source_view_key_help;
232244d93782SGreg Clayton   }
232344d93782SGreg Clayton 
2324b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2325b9c1b51eSKate Stone     switch (c) {
232644d93782SGreg Clayton     case ',':
232744d93782SGreg Clayton     case KEY_PPAGE:
232844d93782SGreg Clayton       // Page up key
2329b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
233044d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
233144d93782SGreg Clayton           m_first_visible_row -= m_max_y;
233244d93782SGreg Clayton         else
233344d93782SGreg Clayton           m_first_visible_row = 0;
233444d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
233544d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
233644d93782SGreg Clayton         if (m_selected_item)
233744d93782SGreg Clayton           m_selected_item->ItemWasSelected();
233844d93782SGreg Clayton       }
233944d93782SGreg Clayton       return eKeyHandled;
234044d93782SGreg Clayton 
234144d93782SGreg Clayton     case '.':
234244d93782SGreg Clayton     case KEY_NPAGE:
234344d93782SGreg Clayton       // Page down key
2344b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2345b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
234644d93782SGreg Clayton           m_first_visible_row += m_max_y;
234744d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
234844d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
234944d93782SGreg Clayton           if (m_selected_item)
235044d93782SGreg Clayton             m_selected_item->ItemWasSelected();
235144d93782SGreg Clayton         }
235244d93782SGreg Clayton       }
235344d93782SGreg Clayton       return eKeyHandled;
235444d93782SGreg Clayton 
235544d93782SGreg Clayton     case KEY_UP:
2356b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
235744d93782SGreg Clayton         --m_selected_row_idx;
235844d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
235944d93782SGreg Clayton         if (m_selected_item)
236044d93782SGreg Clayton           m_selected_item->ItemWasSelected();
236144d93782SGreg Clayton       }
236244d93782SGreg Clayton       return eKeyHandled;
2363315b6884SEugene Zelenko 
236444d93782SGreg Clayton     case KEY_DOWN:
2365b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
236644d93782SGreg Clayton         ++m_selected_row_idx;
236744d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
236844d93782SGreg Clayton         if (m_selected_item)
236944d93782SGreg Clayton           m_selected_item->ItemWasSelected();
237044d93782SGreg Clayton       }
237144d93782SGreg Clayton       return eKeyHandled;
237244d93782SGreg Clayton 
237344d93782SGreg Clayton     case KEY_RIGHT:
2374b9c1b51eSKate Stone       if (m_selected_item) {
237544d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
237644d93782SGreg Clayton           m_selected_item->Expand();
237744d93782SGreg Clayton       }
237844d93782SGreg Clayton       return eKeyHandled;
237944d93782SGreg Clayton 
238044d93782SGreg Clayton     case KEY_LEFT:
2381b9c1b51eSKate Stone       if (m_selected_item) {
238244d93782SGreg Clayton         if (m_selected_item->IsExpanded())
238344d93782SGreg Clayton           m_selected_item->Unexpand();
2384b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
238544d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
238644d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
238744d93782SGreg Clayton           if (m_selected_item)
238844d93782SGreg Clayton             m_selected_item->ItemWasSelected();
238944d93782SGreg Clayton         }
239044d93782SGreg Clayton       }
239144d93782SGreg Clayton       return eKeyHandled;
239244d93782SGreg Clayton 
239344d93782SGreg Clayton     case ' ':
239444d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2395b9c1b51eSKate Stone       if (m_selected_item) {
239644d93782SGreg Clayton         if (m_selected_item->IsExpanded())
239744d93782SGreg Clayton           m_selected_item->Unexpand();
239844d93782SGreg Clayton         else
239944d93782SGreg Clayton           m_selected_item->Expand();
240044d93782SGreg Clayton       }
240144d93782SGreg Clayton       return eKeyHandled;
240244d93782SGreg Clayton 
240344d93782SGreg Clayton     case 'h':
240444d93782SGreg Clayton       window.CreateHelpSubwindow();
240544d93782SGreg Clayton       return eKeyHandled;
240644d93782SGreg Clayton 
240744d93782SGreg Clayton     default:
240844d93782SGreg Clayton       break;
240944d93782SGreg Clayton     }
241044d93782SGreg Clayton     return eKeyNotHandled;
241144d93782SGreg Clayton   }
241244d93782SGreg Clayton 
241344d93782SGreg Clayton protected:
241444d93782SGreg Clayton   Debugger &m_debugger;
241544d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
241644d93782SGreg Clayton   TreeItem m_root;
241744d93782SGreg Clayton   TreeItem *m_selected_item;
241844d93782SGreg Clayton   int m_num_rows;
241944d93782SGreg Clayton   int m_selected_row_idx;
242044d93782SGreg Clayton   int m_first_visible_row;
242144d93782SGreg Clayton   int m_min_x;
242244d93782SGreg Clayton   int m_min_y;
242344d93782SGreg Clayton   int m_max_x;
242444d93782SGreg Clayton   int m_max_y;
242544d93782SGreg Clayton };
242644d93782SGreg Clayton 
2427b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
242844d93782SGreg Clayton public:
2429b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2430b9c1b51eSKate Stone     FormatEntity::Parse(
2431b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2432554f68d3SGreg Clayton         m_format);
243344d93782SGreg Clayton   }
243444d93782SGreg Clayton 
2435315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
243644d93782SGreg Clayton 
2437b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2438ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2439b9c1b51eSKate Stone     if (thread) {
244044d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2441ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2442b9c1b51eSKate Stone       if (frame_sp) {
244344d93782SGreg Clayton         StreamString strm;
2444b9c1b51eSKate Stone         const SymbolContext &sc =
2445b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
244644d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2447b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2448b9c1b51eSKate Stone                                  nullptr, false, false)) {
244944d93782SGreg Clayton           int right_pad = 1;
2450c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
245144d93782SGreg Clayton         }
245244d93782SGreg Clayton       }
245344d93782SGreg Clayton     }
245444d93782SGreg Clayton   }
2455315b6884SEugene Zelenko 
2456b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
245744d93782SGreg Clayton     // No children for frames yet...
245844d93782SGreg Clayton   }
245944d93782SGreg Clayton 
2460b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2461ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2462b9c1b51eSKate Stone     if (thread) {
2463b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2464b9c1b51eSKate Stone           thread->GetID());
246544d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2466ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
246744d93782SGreg Clayton       return true;
246844d93782SGreg Clayton     }
246944d93782SGreg Clayton     return false;
247044d93782SGreg Clayton   }
2471315b6884SEugene Zelenko 
2472554f68d3SGreg Clayton protected:
2473554f68d3SGreg Clayton   FormatEntity::Entry m_format;
247444d93782SGreg Clayton };
247544d93782SGreg Clayton 
2476b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
247744d93782SGreg Clayton public:
2478b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2479b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2480b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2481b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2482b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2483554f68d3SGreg Clayton                         m_format);
248444d93782SGreg Clayton   }
248544d93782SGreg Clayton 
2486315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
248744d93782SGreg Clayton 
2488b9c1b51eSKate Stone   ProcessSP GetProcess() {
2489b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2490b9c1b51eSKate Stone         .GetExecutionContext()
2491b9c1b51eSKate Stone         .GetProcessSP();
2492ec990867SGreg Clayton   }
2493ec990867SGreg Clayton 
2494b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2495ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2496ec990867SGreg Clayton     if (process_sp)
2497ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2498ec990867SGreg Clayton     return ThreadSP();
2499ec990867SGreg Clayton   }
2500ec990867SGreg Clayton 
2501b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2502ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2503b9c1b51eSKate Stone     if (thread_sp) {
250444d93782SGreg Clayton       StreamString strm;
250544d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2506b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2507b9c1b51eSKate Stone                                nullptr, false, false)) {
250844d93782SGreg Clayton         int right_pad = 1;
2509c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
251044d93782SGreg Clayton       }
251144d93782SGreg Clayton     }
251244d93782SGreg Clayton   }
2513315b6884SEugene Zelenko 
2514b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2515ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2516b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
251744d93782SGreg Clayton       StateType state = process_sp->GetState();
2518b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2519ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2520b9c1b51eSKate Stone         if (thread_sp) {
2521b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2522b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
252344d93782SGreg Clayton             return; // Children are already up to date
2524b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
252544d93782SGreg Clayton             // Always expand the thread item the first time we show it
2526796ac80bSJonas Devlieghere             m_frame_delegate_sp = std::make_shared<FrameTreeDelegate>();
252744d93782SGreg Clayton           }
252844d93782SGreg Clayton 
252944d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
253044d93782SGreg Clayton           m_tid = thread_sp->GetID();
253144d93782SGreg Clayton 
253244d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
253344d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
253444d93782SGreg Clayton           item.Resize(num_frames, t);
2535b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2536ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
253744d93782SGreg Clayton             item[i].SetIdentifier(i);
253844d93782SGreg Clayton           }
253944d93782SGreg Clayton         }
254044d93782SGreg Clayton         return;
254144d93782SGreg Clayton       }
254244d93782SGreg Clayton     }
254344d93782SGreg Clayton     item.ClearChildren();
254444d93782SGreg Clayton   }
254544d93782SGreg Clayton 
2546b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2547ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2548b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2549ec990867SGreg Clayton       StateType state = process_sp->GetState();
2550b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2551ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2552b9c1b51eSKate Stone         if (thread_sp) {
255344d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2554bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
255544d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2556b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
255744d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
255844d93782SGreg Clayton             return true;
255944d93782SGreg Clayton           }
256044d93782SGreg Clayton         }
2561ec990867SGreg Clayton       }
2562ec990867SGreg Clayton     }
256344d93782SGreg Clayton     return false;
256444d93782SGreg Clayton   }
256544d93782SGreg Clayton 
256644d93782SGreg Clayton protected:
256744d93782SGreg Clayton   Debugger &m_debugger;
256844d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
256944d93782SGreg Clayton   lldb::user_id_t m_tid;
257044d93782SGreg Clayton   uint32_t m_stop_id;
2571554f68d3SGreg Clayton   FormatEntity::Entry m_format;
257244d93782SGreg Clayton };
257344d93782SGreg Clayton 
2574b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2575ec990867SGreg Clayton public:
2576b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2577b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2578b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2579554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2580554f68d3SGreg Clayton                         m_format);
2581ec990867SGreg Clayton   }
2582ec990867SGreg Clayton 
2583315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2584ec990867SGreg Clayton 
2585b9c1b51eSKate Stone   ProcessSP GetProcess() {
2586b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2587b9c1b51eSKate Stone         .GetExecutionContext()
2588b9c1b51eSKate Stone         .GetProcessSP();
2589ec990867SGreg Clayton   }
2590ec990867SGreg Clayton 
2591b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2592ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2593b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2594ec990867SGreg Clayton       StreamString strm;
2595ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2596b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2597b9c1b51eSKate Stone                                nullptr, false, false)) {
2598ec990867SGreg Clayton         int right_pad = 1;
2599c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2600ec990867SGreg Clayton       }
2601ec990867SGreg Clayton     }
2602ec990867SGreg Clayton   }
2603ec990867SGreg Clayton 
2604b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2605ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2606b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2607ec990867SGreg Clayton       StateType state = process_sp->GetState();
2608b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2609ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2610ec990867SGreg Clayton         if (m_stop_id == stop_id)
2611ec990867SGreg Clayton           return; // Children are already up to date
2612ec990867SGreg Clayton 
2613ec990867SGreg Clayton         m_stop_id = stop_id;
2614ec990867SGreg Clayton 
2615b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2616ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2617ec990867SGreg Clayton           // item.Expand();
2618796ac80bSJonas Devlieghere           m_thread_delegate_sp =
2619796ac80bSJonas Devlieghere               std::make_shared<ThreadTreeDelegate>(m_debugger);
2620ec990867SGreg Clayton         }
2621ec990867SGreg Clayton 
2622ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2623ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2624bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2625ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2626ec990867SGreg Clayton         item.Resize(num_threads, t);
2627b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2628ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2629ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2630ec990867SGreg Clayton         }
2631ec990867SGreg Clayton         return;
2632ec990867SGreg Clayton       }
2633ec990867SGreg Clayton     }
2634ec990867SGreg Clayton     item.ClearChildren();
2635ec990867SGreg Clayton   }
2636ec990867SGreg Clayton 
2637b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2638ec990867SGreg Clayton 
2639ec990867SGreg Clayton protected:
2640ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2641ec990867SGreg Clayton   Debugger &m_debugger;
2642ec990867SGreg Clayton   uint32_t m_stop_id;
2643554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2644ec990867SGreg Clayton };
2645ec990867SGreg Clayton 
2646b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
264744d93782SGreg Clayton public:
2648b9c1b51eSKate Stone   ValueObjectListDelegate()
26498369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2650b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2651b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {}
265244d93782SGreg Clayton 
2653b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
26548369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2655b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2656b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {
265744d93782SGreg Clayton     SetValues(valobj_list);
265844d93782SGreg Clayton   }
265944d93782SGreg Clayton 
2660315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
266144d93782SGreg Clayton 
2662b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2663c5dac77aSEugene Zelenko     m_selected_row = nullptr;
266444d93782SGreg Clayton     m_selected_row_idx = 0;
266544d93782SGreg Clayton     m_first_visible_row = 0;
266644d93782SGreg Clayton     m_num_rows = 0;
266744d93782SGreg Clayton     m_rows.clear();
26688369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
26698369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
267044d93782SGreg Clayton   }
267144d93782SGreg Clayton 
2672b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
267344d93782SGreg Clayton     m_num_rows = 0;
267444d93782SGreg Clayton     m_min_x = 2;
267544d93782SGreg Clayton     m_min_y = 1;
267644d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
267744d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
267844d93782SGreg Clayton 
267944d93782SGreg Clayton     window.Erase();
268044d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
268144d93782SGreg Clayton 
268244d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
268344d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
268444d93782SGreg Clayton 
268505097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
268605097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
268705097246SAdrian Prantl     // rows by setting the first visible row accordingly.
268844d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
268944d93782SGreg Clayton       m_first_visible_row = 0;
269044d93782SGreg Clayton 
269144d93782SGreg Clayton     // Make sure the selected row is always visible
269244d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
269344d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
269444d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
269544d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
269644d93782SGreg Clayton 
269744d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
269844d93782SGreg Clayton 
269944d93782SGreg Clayton     window.DeferredRefresh();
270044d93782SGreg Clayton 
270144d93782SGreg Clayton     // Get the selected row
270244d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
270305097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
270405097246SAdrian Prantl     // always on the same line
270544d93782SGreg Clayton     if (m_selected_row)
2706b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
270744d93782SGreg Clayton 
270844d93782SGreg Clayton     return true; // Drawing handled
270944d93782SGreg Clayton   }
271044d93782SGreg Clayton 
2711b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
271244d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
271344d93782SGreg Clayton         {KEY_UP, "Select previous item"},
271444d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
271544d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
271644d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
271744d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
271844d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
271944d93782SGreg Clayton         {'A', "Format as annotated address"},
272044d93782SGreg Clayton         {'b', "Format as binary"},
272144d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
272244d93782SGreg Clayton         {'c', "Format as character"},
272344d93782SGreg Clayton         {'d', "Format as a signed integer"},
272444d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
272544d93782SGreg Clayton         {'f', "Format as float"},
272644d93782SGreg Clayton         {'h', "Show help dialog"},
272744d93782SGreg Clayton         {'i', "Format as instructions"},
272844d93782SGreg Clayton         {'o', "Format as octal"},
272944d93782SGreg Clayton         {'p', "Format as pointer"},
273044d93782SGreg Clayton         {'s', "Format as C string"},
273144d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
273244d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
273344d93782SGreg Clayton         {'x', "Format as hex"},
273444d93782SGreg Clayton         {'X', "Format as uppercase hex"},
273544d93782SGreg Clayton         {' ', "Toggle item expansion"},
273644d93782SGreg Clayton         {',', "Page up"},
273744d93782SGreg Clayton         {'.', "Page down"},
2738b9c1b51eSKate Stone         {'\0', nullptr}};
273944d93782SGreg Clayton     return g_source_view_key_help;
274044d93782SGreg Clayton   }
274144d93782SGreg Clayton 
2742b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2743b9c1b51eSKate Stone     switch (c) {
274444d93782SGreg Clayton     case 'x':
274544d93782SGreg Clayton     case 'X':
274644d93782SGreg Clayton     case 'o':
274744d93782SGreg Clayton     case 's':
274844d93782SGreg Clayton     case 'u':
274944d93782SGreg Clayton     case 'd':
275044d93782SGreg Clayton     case 'D':
275144d93782SGreg Clayton     case 'i':
275244d93782SGreg Clayton     case 'A':
275344d93782SGreg Clayton     case 'p':
275444d93782SGreg Clayton     case 'c':
275544d93782SGreg Clayton     case 'b':
275644d93782SGreg Clayton     case 'B':
275744d93782SGreg Clayton     case 'f':
275844d93782SGreg Clayton       // Change the format for the currently selected item
27598369b28dSGreg Clayton       if (m_selected_row) {
27608369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
27618369b28dSGreg Clayton         if (valobj_sp)
27628369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
27638369b28dSGreg Clayton       }
276444d93782SGreg Clayton       return eKeyHandled;
276544d93782SGreg Clayton 
276644d93782SGreg Clayton     case 't':
276744d93782SGreg Clayton       // Toggle showing type names
276844d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
276944d93782SGreg Clayton       return eKeyHandled;
277044d93782SGreg Clayton 
277144d93782SGreg Clayton     case ',':
277244d93782SGreg Clayton     case KEY_PPAGE:
277344d93782SGreg Clayton       // Page up key
2774b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27753985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
277644d93782SGreg Clayton           m_first_visible_row -= m_max_y;
277744d93782SGreg Clayton         else
277844d93782SGreg Clayton           m_first_visible_row = 0;
277944d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
278044d93782SGreg Clayton       }
278144d93782SGreg Clayton       return eKeyHandled;
278244d93782SGreg Clayton 
278344d93782SGreg Clayton     case '.':
278444d93782SGreg Clayton     case KEY_NPAGE:
278544d93782SGreg Clayton       // Page down key
2786b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2787b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
278844d93782SGreg Clayton           m_first_visible_row += m_max_y;
278944d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
279044d93782SGreg Clayton         }
279144d93782SGreg Clayton       }
279244d93782SGreg Clayton       return eKeyHandled;
279344d93782SGreg Clayton 
279444d93782SGreg Clayton     case KEY_UP:
279544d93782SGreg Clayton       if (m_selected_row_idx > 0)
279644d93782SGreg Clayton         --m_selected_row_idx;
279744d93782SGreg Clayton       return eKeyHandled;
2798315b6884SEugene Zelenko 
279944d93782SGreg Clayton     case KEY_DOWN:
280044d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
280144d93782SGreg Clayton         ++m_selected_row_idx;
280244d93782SGreg Clayton       return eKeyHandled;
280344d93782SGreg Clayton 
280444d93782SGreg Clayton     case KEY_RIGHT:
2805b9c1b51eSKate Stone       if (m_selected_row) {
280644d93782SGreg Clayton         if (!m_selected_row->expanded)
280744d93782SGreg Clayton           m_selected_row->Expand();
280844d93782SGreg Clayton       }
280944d93782SGreg Clayton       return eKeyHandled;
281044d93782SGreg Clayton 
281144d93782SGreg Clayton     case KEY_LEFT:
2812b9c1b51eSKate Stone       if (m_selected_row) {
281344d93782SGreg Clayton         if (m_selected_row->expanded)
281444d93782SGreg Clayton           m_selected_row->Unexpand();
281544d93782SGreg Clayton         else if (m_selected_row->parent)
281644d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
281744d93782SGreg Clayton       }
281844d93782SGreg Clayton       return eKeyHandled;
281944d93782SGreg Clayton 
282044d93782SGreg Clayton     case ' ':
282144d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2822b9c1b51eSKate Stone       if (m_selected_row) {
282344d93782SGreg Clayton         if (m_selected_row->expanded)
282444d93782SGreg Clayton           m_selected_row->Unexpand();
282544d93782SGreg Clayton         else
282644d93782SGreg Clayton           m_selected_row->Expand();
282744d93782SGreg Clayton       }
282844d93782SGreg Clayton       return eKeyHandled;
282944d93782SGreg Clayton 
283044d93782SGreg Clayton     case 'h':
283144d93782SGreg Clayton       window.CreateHelpSubwindow();
283244d93782SGreg Clayton       return eKeyHandled;
283344d93782SGreg Clayton 
283444d93782SGreg Clayton     default:
283544d93782SGreg Clayton       break;
283644d93782SGreg Clayton     }
283744d93782SGreg Clayton     return eKeyNotHandled;
283844d93782SGreg Clayton   }
283944d93782SGreg Clayton 
284044d93782SGreg Clayton protected:
284144d93782SGreg Clayton   std::vector<Row> m_rows;
284244d93782SGreg Clayton   Row *m_selected_row;
284344d93782SGreg Clayton   uint32_t m_selected_row_idx;
284444d93782SGreg Clayton   uint32_t m_first_visible_row;
284544d93782SGreg Clayton   uint32_t m_num_rows;
284644d93782SGreg Clayton   int m_min_x;
284744d93782SGreg Clayton   int m_min_y;
284844d93782SGreg Clayton   int m_max_x;
284944d93782SGreg Clayton   int m_max_y;
285044d93782SGreg Clayton 
2851b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2852b9c1b51eSKate Stone     switch (c) {
2853b9c1b51eSKate Stone     case 'x':
2854b9c1b51eSKate Stone       return eFormatHex;
2855b9c1b51eSKate Stone     case 'X':
2856b9c1b51eSKate Stone       return eFormatHexUppercase;
2857b9c1b51eSKate Stone     case 'o':
2858b9c1b51eSKate Stone       return eFormatOctal;
2859b9c1b51eSKate Stone     case 's':
2860b9c1b51eSKate Stone       return eFormatCString;
2861b9c1b51eSKate Stone     case 'u':
2862b9c1b51eSKate Stone       return eFormatUnsigned;
2863b9c1b51eSKate Stone     case 'd':
2864b9c1b51eSKate Stone       return eFormatDecimal;
2865b9c1b51eSKate Stone     case 'D':
2866b9c1b51eSKate Stone       return eFormatDefault;
2867b9c1b51eSKate Stone     case 'i':
2868b9c1b51eSKate Stone       return eFormatInstruction;
2869b9c1b51eSKate Stone     case 'A':
2870b9c1b51eSKate Stone       return eFormatAddressInfo;
2871b9c1b51eSKate Stone     case 'p':
2872b9c1b51eSKate Stone       return eFormatPointer;
2873b9c1b51eSKate Stone     case 'c':
2874b9c1b51eSKate Stone       return eFormatChar;
2875b9c1b51eSKate Stone     case 'b':
2876b9c1b51eSKate Stone       return eFormatBinary;
2877b9c1b51eSKate Stone     case 'B':
2878b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2879b9c1b51eSKate Stone     case 'f':
2880b9c1b51eSKate Stone       return eFormatFloat;
288144d93782SGreg Clayton     }
288244d93782SGreg Clayton     return eFormatDefault;
288344d93782SGreg Clayton   }
288444d93782SGreg Clayton 
2885b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2886b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28878369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
288844d93782SGreg Clayton 
2889c5dac77aSEugene Zelenko     if (valobj == nullptr)
289044d93782SGreg Clayton       return false;
289144d93782SGreg Clayton 
2892b9c1b51eSKate Stone     const char *type_name =
2893b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
289444d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
289544d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
289644d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
289744d93782SGreg Clayton 
289844d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
289944d93782SGreg Clayton 
290044d93782SGreg Clayton     row.DrawTree(window);
290144d93782SGreg Clayton 
290244d93782SGreg Clayton     if (highlight)
290344d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
290444d93782SGreg Clayton 
290544d93782SGreg Clayton     if (type_name && type_name[0])
290644d93782SGreg Clayton       window.Printf("(%s) ", type_name);
290744d93782SGreg Clayton 
290844d93782SGreg Clayton     if (name && name[0])
290944d93782SGreg Clayton       window.PutCString(name);
291044d93782SGreg Clayton 
291144d93782SGreg Clayton     attr_t changd_attr = 0;
291244d93782SGreg Clayton     if (valobj->GetValueDidChange())
291344d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
291444d93782SGreg Clayton 
2915b9c1b51eSKate Stone     if (value && value[0]) {
291644d93782SGreg Clayton       window.PutCString(" = ");
291744d93782SGreg Clayton       if (changd_attr)
291844d93782SGreg Clayton         window.AttributeOn(changd_attr);
291944d93782SGreg Clayton       window.PutCString(value);
292044d93782SGreg Clayton       if (changd_attr)
292144d93782SGreg Clayton         window.AttributeOff(changd_attr);
292244d93782SGreg Clayton     }
292344d93782SGreg Clayton 
2924b9c1b51eSKate Stone     if (summary && summary[0]) {
292544d93782SGreg Clayton       window.PutChar(' ');
292644d93782SGreg Clayton       if (changd_attr)
292744d93782SGreg Clayton         window.AttributeOn(changd_attr);
292844d93782SGreg Clayton       window.PutCString(summary);
292944d93782SGreg Clayton       if (changd_attr)
293044d93782SGreg Clayton         window.AttributeOff(changd_attr);
293144d93782SGreg Clayton     }
293244d93782SGreg Clayton 
293344d93782SGreg Clayton     if (highlight)
293444d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
293544d93782SGreg Clayton 
293644d93782SGreg Clayton     return true;
293744d93782SGreg Clayton   }
2938315b6884SEugene Zelenko 
2939b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2940b9c1b51eSKate Stone                    DisplayOptions &options) {
294144d93782SGreg Clayton     // >   0x25B7
294244d93782SGreg Clayton     // \/  0x25BD
294344d93782SGreg Clayton 
294444d93782SGreg Clayton     bool window_is_active = window.IsActive();
2945b9c1b51eSKate Stone     for (auto &row : rows) {
294644d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
294744d93782SGreg Clayton       // Save the row index in each Row structure
294844d93782SGreg Clayton       row.row_idx = m_num_rows;
294944d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2950b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2951b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
295244d93782SGreg Clayton         row.x = m_min_x;
295344d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2954b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2955b9c1b51eSKate Stone                              window_is_active &&
2956b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2957b9c1b51eSKate Stone                              last_child)) {
295844d93782SGreg Clayton           ++m_num_rows;
2959b9c1b51eSKate Stone         } else {
296044d93782SGreg Clayton           row.x = 0;
296144d93782SGreg Clayton           row.y = 0;
296244d93782SGreg Clayton         }
2963b9c1b51eSKate Stone       } else {
296444d93782SGreg Clayton         row.x = 0;
296544d93782SGreg Clayton         row.y = 0;
296644d93782SGreg Clayton         ++m_num_rows;
296744d93782SGreg Clayton       }
296844d93782SGreg Clayton 
29698369b28dSGreg Clayton       auto &children = row.GetChildren();
29708369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
29718369b28dSGreg Clayton         DisplayRows(window, children, options);
297244d93782SGreg Clayton       }
297344d93782SGreg Clayton     }
297444d93782SGreg Clayton   }
297544d93782SGreg Clayton 
29768369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
297744d93782SGreg Clayton     int row_count = 0;
29788369b28dSGreg Clayton     for (auto &row : rows) {
297944d93782SGreg Clayton       ++row_count;
298044d93782SGreg Clayton       if (row.expanded)
29818369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
298244d93782SGreg Clayton     }
298344d93782SGreg Clayton     return row_count;
298444d93782SGreg Clayton   }
2985315b6884SEugene Zelenko 
2986b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2987b9c1b51eSKate Stone     for (auto &row : rows) {
298844d93782SGreg Clayton       if (row_index == 0)
298944d93782SGreg Clayton         return &row;
2990b9c1b51eSKate Stone       else {
299144d93782SGreg Clayton         --row_index;
29928369b28dSGreg Clayton         auto &children = row.GetChildren();
29938369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29948369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
299544d93782SGreg Clayton           if (result)
299644d93782SGreg Clayton             return result;
299744d93782SGreg Clayton         }
299844d93782SGreg Clayton       }
299944d93782SGreg Clayton     }
3000c5dac77aSEugene Zelenko     return nullptr;
300144d93782SGreg Clayton   }
300244d93782SGreg Clayton 
3003b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
300444d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
300544d93782SGreg Clayton   }
300644d93782SGreg Clayton 
3007b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
300844d93782SGreg Clayton 
300944d93782SGreg Clayton   static DisplayOptions g_options;
301044d93782SGreg Clayton };
301144d93782SGreg Clayton 
3012b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
301344d93782SGreg Clayton public:
3014b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
3015b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
3016b9c1b51eSKate Stone         m_frame_block(nullptr) {}
301744d93782SGreg Clayton 
3018315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
301944d93782SGreg Clayton 
3020b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
302144d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
302244d93782SGreg Clayton   }
302344d93782SGreg Clayton 
3024b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3025b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3026b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
302744d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3028c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
3029c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
303044d93782SGreg Clayton 
3031b9c1b51eSKate Stone     if (process) {
303244d93782SGreg Clayton       StateType state = process->GetState();
3033b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
303444d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
303544d93782SGreg Clayton         if (frame)
303644d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
3037b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
303844d93782SGreg Clayton         return true; // Don't do any updating when we are running
303944d93782SGreg Clayton       }
304044d93782SGreg Clayton     }
304144d93782SGreg Clayton 
304244d93782SGreg Clayton     ValueObjectList local_values;
3043b9c1b51eSKate Stone     if (frame_block) {
304444d93782SGreg Clayton       // Only update the variables if they have changed
3045b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
304644d93782SGreg Clayton         m_frame_block = frame_block;
304744d93782SGreg Clayton 
304844d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3049b9c1b51eSKate Stone         if (locals) {
305044d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
305144d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3052b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3053b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3054b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3055b9c1b51eSKate Stone             if (value_sp) {
3056eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3057eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3058eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3059eb72dc7dSGreg Clayton               else
3060eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3061eb72dc7dSGreg Clayton             }
3062eb72dc7dSGreg Clayton           }
306344d93782SGreg Clayton           // Update the values
306444d93782SGreg Clayton           SetValues(local_values);
306544d93782SGreg Clayton         }
306644d93782SGreg Clayton       }
3067b9c1b51eSKate Stone     } else {
3068c5dac77aSEugene Zelenko       m_frame_block = nullptr;
306944d93782SGreg Clayton       // Update the values with an empty list if there is no frame
307044d93782SGreg Clayton       SetValues(local_values);
307144d93782SGreg Clayton     }
307244d93782SGreg Clayton 
307344d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
307444d93782SGreg Clayton   }
307544d93782SGreg Clayton 
307644d93782SGreg Clayton protected:
307744d93782SGreg Clayton   Debugger &m_debugger;
307844d93782SGreg Clayton   Block *m_frame_block;
307944d93782SGreg Clayton };
308044d93782SGreg Clayton 
3081b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
308244d93782SGreg Clayton public:
3083b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3084b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
308544d93782SGreg Clayton 
3086315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
308744d93782SGreg Clayton 
3088b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
308944d93782SGreg Clayton     return "Register window keyboard shortcuts:";
309044d93782SGreg Clayton   }
309144d93782SGreg Clayton 
3092b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3093b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3094b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
309544d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
309644d93782SGreg Clayton 
309744d93782SGreg Clayton     ValueObjectList value_list;
3098b9c1b51eSKate Stone     if (frame) {
3099b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
310044d93782SGreg Clayton         m_stack_id = frame->GetStackID();
310144d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3102b9c1b51eSKate Stone         if (reg_ctx) {
310344d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3104b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3105b9c1b51eSKate Stone             value_list.Append(
3106b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
310744d93782SGreg Clayton           }
310844d93782SGreg Clayton         }
310944d93782SGreg Clayton         SetValues(value_list);
311044d93782SGreg Clayton       }
3111b9c1b51eSKate Stone     } else {
311244d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
311344d93782SGreg Clayton       if (process && process->IsAlive())
311444d93782SGreg Clayton         return true; // Don't do any updating if we are running
3115b9c1b51eSKate Stone       else {
311605097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
311705097246SAdrian Prantl         // process isn't alive anymore
311844d93782SGreg Clayton         SetValues(value_list);
311944d93782SGreg Clayton       }
312044d93782SGreg Clayton     }
312144d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
312244d93782SGreg Clayton   }
312344d93782SGreg Clayton 
312444d93782SGreg Clayton protected:
312544d93782SGreg Clayton   Debugger &m_debugger;
312644d93782SGreg Clayton   StackID m_stack_id;
312744d93782SGreg Clayton };
312844d93782SGreg Clayton 
3129b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
313044d93782SGreg Clayton   static char g_desc[32];
3131b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
313244d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
313344d93782SGreg Clayton     return g_desc;
313444d93782SGreg Clayton   }
3135b9c1b51eSKate Stone   switch (ch) {
3136b9c1b51eSKate Stone   case KEY_DOWN:
3137b9c1b51eSKate Stone     return "down";
3138b9c1b51eSKate Stone   case KEY_UP:
3139b9c1b51eSKate Stone     return "up";
3140b9c1b51eSKate Stone   case KEY_LEFT:
3141b9c1b51eSKate Stone     return "left";
3142b9c1b51eSKate Stone   case KEY_RIGHT:
3143b9c1b51eSKate Stone     return "right";
3144b9c1b51eSKate Stone   case KEY_HOME:
3145b9c1b51eSKate Stone     return "home";
3146b9c1b51eSKate Stone   case KEY_BACKSPACE:
3147b9c1b51eSKate Stone     return "backspace";
3148b9c1b51eSKate Stone   case KEY_DL:
3149b9c1b51eSKate Stone     return "delete-line";
3150b9c1b51eSKate Stone   case KEY_IL:
3151b9c1b51eSKate Stone     return "insert-line";
3152b9c1b51eSKate Stone   case KEY_DC:
3153b9c1b51eSKate Stone     return "delete-char";
3154b9c1b51eSKate Stone   case KEY_IC:
3155b9c1b51eSKate Stone     return "insert-char";
3156b9c1b51eSKate Stone   case KEY_CLEAR:
3157b9c1b51eSKate Stone     return "clear";
3158b9c1b51eSKate Stone   case KEY_EOS:
3159b9c1b51eSKate Stone     return "clear-to-eos";
3160b9c1b51eSKate Stone   case KEY_EOL:
3161b9c1b51eSKate Stone     return "clear-to-eol";
3162b9c1b51eSKate Stone   case KEY_SF:
3163b9c1b51eSKate Stone     return "scroll-forward";
3164b9c1b51eSKate Stone   case KEY_SR:
3165b9c1b51eSKate Stone     return "scroll-backward";
3166b9c1b51eSKate Stone   case KEY_NPAGE:
3167b9c1b51eSKate Stone     return "page-down";
3168b9c1b51eSKate Stone   case KEY_PPAGE:
3169b9c1b51eSKate Stone     return "page-up";
3170b9c1b51eSKate Stone   case KEY_STAB:
3171b9c1b51eSKate Stone     return "set-tab";
3172b9c1b51eSKate Stone   case KEY_CTAB:
3173b9c1b51eSKate Stone     return "clear-tab";
3174b9c1b51eSKate Stone   case KEY_CATAB:
3175b9c1b51eSKate Stone     return "clear-all-tabs";
3176b9c1b51eSKate Stone   case KEY_ENTER:
3177b9c1b51eSKate Stone     return "enter";
3178b9c1b51eSKate Stone   case KEY_PRINT:
3179b9c1b51eSKate Stone     return "print";
3180b9c1b51eSKate Stone   case KEY_LL:
3181b9c1b51eSKate Stone     return "lower-left key";
3182b9c1b51eSKate Stone   case KEY_A1:
3183b9c1b51eSKate Stone     return "upper left of keypad";
3184b9c1b51eSKate Stone   case KEY_A3:
3185b9c1b51eSKate Stone     return "upper right of keypad";
3186b9c1b51eSKate Stone   case KEY_B2:
3187b9c1b51eSKate Stone     return "center of keypad";
3188b9c1b51eSKate Stone   case KEY_C1:
3189b9c1b51eSKate Stone     return "lower left of keypad";
3190b9c1b51eSKate Stone   case KEY_C3:
3191b9c1b51eSKate Stone     return "lower right of keypad";
3192b9c1b51eSKate Stone   case KEY_BTAB:
3193b9c1b51eSKate Stone     return "back-tab key";
3194b9c1b51eSKate Stone   case KEY_BEG:
3195b9c1b51eSKate Stone     return "begin key";
3196b9c1b51eSKate Stone   case KEY_CANCEL:
3197b9c1b51eSKate Stone     return "cancel key";
3198b9c1b51eSKate Stone   case KEY_CLOSE:
3199b9c1b51eSKate Stone     return "close key";
3200b9c1b51eSKate Stone   case KEY_COMMAND:
3201b9c1b51eSKate Stone     return "command key";
3202b9c1b51eSKate Stone   case KEY_COPY:
3203b9c1b51eSKate Stone     return "copy key";
3204b9c1b51eSKate Stone   case KEY_CREATE:
3205b9c1b51eSKate Stone     return "create key";
3206b9c1b51eSKate Stone   case KEY_END:
3207b9c1b51eSKate Stone     return "end key";
3208b9c1b51eSKate Stone   case KEY_EXIT:
3209b9c1b51eSKate Stone     return "exit key";
3210b9c1b51eSKate Stone   case KEY_FIND:
3211b9c1b51eSKate Stone     return "find key";
3212b9c1b51eSKate Stone   case KEY_HELP:
3213b9c1b51eSKate Stone     return "help key";
3214b9c1b51eSKate Stone   case KEY_MARK:
3215b9c1b51eSKate Stone     return "mark key";
3216b9c1b51eSKate Stone   case KEY_MESSAGE:
3217b9c1b51eSKate Stone     return "message key";
3218b9c1b51eSKate Stone   case KEY_MOVE:
3219b9c1b51eSKate Stone     return "move key";
3220b9c1b51eSKate Stone   case KEY_NEXT:
3221b9c1b51eSKate Stone     return "next key";
3222b9c1b51eSKate Stone   case KEY_OPEN:
3223b9c1b51eSKate Stone     return "open key";
3224b9c1b51eSKate Stone   case KEY_OPTIONS:
3225b9c1b51eSKate Stone     return "options key";
3226b9c1b51eSKate Stone   case KEY_PREVIOUS:
3227b9c1b51eSKate Stone     return "previous key";
3228b9c1b51eSKate Stone   case KEY_REDO:
3229b9c1b51eSKate Stone     return "redo key";
3230b9c1b51eSKate Stone   case KEY_REFERENCE:
3231b9c1b51eSKate Stone     return "reference key";
3232b9c1b51eSKate Stone   case KEY_REFRESH:
3233b9c1b51eSKate Stone     return "refresh key";
3234b9c1b51eSKate Stone   case KEY_REPLACE:
3235b9c1b51eSKate Stone     return "replace key";
3236b9c1b51eSKate Stone   case KEY_RESTART:
3237b9c1b51eSKate Stone     return "restart key";
3238b9c1b51eSKate Stone   case KEY_RESUME:
3239b9c1b51eSKate Stone     return "resume key";
3240b9c1b51eSKate Stone   case KEY_SAVE:
3241b9c1b51eSKate Stone     return "save key";
3242b9c1b51eSKate Stone   case KEY_SBEG:
3243b9c1b51eSKate Stone     return "shifted begin key";
3244b9c1b51eSKate Stone   case KEY_SCANCEL:
3245b9c1b51eSKate Stone     return "shifted cancel key";
3246b9c1b51eSKate Stone   case KEY_SCOMMAND:
3247b9c1b51eSKate Stone     return "shifted command key";
3248b9c1b51eSKate Stone   case KEY_SCOPY:
3249b9c1b51eSKate Stone     return "shifted copy key";
3250b9c1b51eSKate Stone   case KEY_SCREATE:
3251b9c1b51eSKate Stone     return "shifted create key";
3252b9c1b51eSKate Stone   case KEY_SDC:
3253b9c1b51eSKate Stone     return "shifted delete-character key";
3254b9c1b51eSKate Stone   case KEY_SDL:
3255b9c1b51eSKate Stone     return "shifted delete-line key";
3256b9c1b51eSKate Stone   case KEY_SELECT:
3257b9c1b51eSKate Stone     return "select key";
3258b9c1b51eSKate Stone   case KEY_SEND:
3259b9c1b51eSKate Stone     return "shifted end key";
3260b9c1b51eSKate Stone   case KEY_SEOL:
3261b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3262b9c1b51eSKate Stone   case KEY_SEXIT:
3263b9c1b51eSKate Stone     return "shifted exit key";
3264b9c1b51eSKate Stone   case KEY_SFIND:
3265b9c1b51eSKate Stone     return "shifted find key";
3266b9c1b51eSKate Stone   case KEY_SHELP:
3267b9c1b51eSKate Stone     return "shifted help key";
3268b9c1b51eSKate Stone   case KEY_SHOME:
3269b9c1b51eSKate Stone     return "shifted home key";
3270b9c1b51eSKate Stone   case KEY_SIC:
3271b9c1b51eSKate Stone     return "shifted insert-character key";
3272b9c1b51eSKate Stone   case KEY_SLEFT:
3273b9c1b51eSKate Stone     return "shifted left-arrow key";
3274b9c1b51eSKate Stone   case KEY_SMESSAGE:
3275b9c1b51eSKate Stone     return "shifted message key";
3276b9c1b51eSKate Stone   case KEY_SMOVE:
3277b9c1b51eSKate Stone     return "shifted move key";
3278b9c1b51eSKate Stone   case KEY_SNEXT:
3279b9c1b51eSKate Stone     return "shifted next key";
3280b9c1b51eSKate Stone   case KEY_SOPTIONS:
3281b9c1b51eSKate Stone     return "shifted options key";
3282b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3283b9c1b51eSKate Stone     return "shifted previous key";
3284b9c1b51eSKate Stone   case KEY_SPRINT:
3285b9c1b51eSKate Stone     return "shifted print key";
3286b9c1b51eSKate Stone   case KEY_SREDO:
3287b9c1b51eSKate Stone     return "shifted redo key";
3288b9c1b51eSKate Stone   case KEY_SREPLACE:
3289b9c1b51eSKate Stone     return "shifted replace key";
3290b9c1b51eSKate Stone   case KEY_SRIGHT:
3291b9c1b51eSKate Stone     return "shifted right-arrow key";
3292b9c1b51eSKate Stone   case KEY_SRSUME:
3293b9c1b51eSKate Stone     return "shifted resume key";
3294b9c1b51eSKate Stone   case KEY_SSAVE:
3295b9c1b51eSKate Stone     return "shifted save key";
3296b9c1b51eSKate Stone   case KEY_SSUSPEND:
3297b9c1b51eSKate Stone     return "shifted suspend key";
3298b9c1b51eSKate Stone   case KEY_SUNDO:
3299b9c1b51eSKate Stone     return "shifted undo key";
3300b9c1b51eSKate Stone   case KEY_SUSPEND:
3301b9c1b51eSKate Stone     return "suspend key";
3302b9c1b51eSKate Stone   case KEY_UNDO:
3303b9c1b51eSKate Stone     return "undo key";
3304b9c1b51eSKate Stone   case KEY_MOUSE:
3305b9c1b51eSKate Stone     return "Mouse event has occurred";
3306b9c1b51eSKate Stone   case KEY_RESIZE:
3307b9c1b51eSKate Stone     return "Terminal resize event";
330827801f4fSBruce Mitchener #ifdef KEY_EVENT
3309b9c1b51eSKate Stone   case KEY_EVENT:
3310b9c1b51eSKate Stone     return "We were interrupted by an event";
331127801f4fSBruce Mitchener #endif
3312b9c1b51eSKate Stone   case KEY_RETURN:
3313b9c1b51eSKate Stone     return "return";
3314b9c1b51eSKate Stone   case ' ':
3315b9c1b51eSKate Stone     return "space";
3316b9c1b51eSKate Stone   case '\t':
3317b9c1b51eSKate Stone     return "tab";
3318b9c1b51eSKate Stone   case KEY_ESCAPE:
3319b9c1b51eSKate Stone     return "escape";
332044d93782SGreg Clayton   default:
332144d93782SGreg Clayton     if (isprint(ch))
332244d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
332344d93782SGreg Clayton     else
332444d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
332544d93782SGreg Clayton     return g_desc;
332644d93782SGreg Clayton   }
3327c5dac77aSEugene Zelenko   return nullptr;
332844d93782SGreg Clayton }
332944d93782SGreg Clayton 
3330b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3331b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3332b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3333b9c1b51eSKate Stone   if (text && text[0]) {
333444d93782SGreg Clayton     m_text.SplitIntoLines(text);
333544d93782SGreg Clayton     m_text.AppendString("");
333644d93782SGreg Clayton   }
3337b9c1b51eSKate Stone   if (key_help_array) {
3338b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
333944d93782SGreg Clayton       StreamString key_description;
3340b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3341b9c1b51eSKate Stone                              key->description);
3342c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
334344d93782SGreg Clayton     }
334444d93782SGreg Clayton   }
334544d93782SGreg Clayton }
334644d93782SGreg Clayton 
3347315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
334844d93782SGreg Clayton 
3349b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
335044d93782SGreg Clayton   window.Erase();
335144d93782SGreg Clayton   const int window_height = window.GetHeight();
335244d93782SGreg Clayton   int x = 2;
335344d93782SGreg Clayton   int y = 1;
335444d93782SGreg Clayton   const int min_y = y;
335544d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33563985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
335744d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
335844d93782SGreg Clayton   const char *bottom_message;
335944d93782SGreg Clayton   if (num_lines <= num_visible_lines)
336044d93782SGreg Clayton     bottom_message = "Press any key to exit";
336144d93782SGreg Clayton   else
336244d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
336344d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3364b9c1b51eSKate Stone   while (y <= max_y) {
336544d93782SGreg Clayton     window.MoveCursor(x, y);
3366b9c1b51eSKate Stone     window.PutCStringTruncated(
3367b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
336844d93782SGreg Clayton     ++y;
336944d93782SGreg Clayton   }
337044d93782SGreg Clayton   return true;
337144d93782SGreg Clayton }
337244d93782SGreg Clayton 
3373b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3374b9c1b51eSKate Stone                                                               int key) {
337544d93782SGreg Clayton   bool done = false;
337644d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
337744d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
337844d93782SGreg Clayton 
3379b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
338044d93782SGreg Clayton     done = true;
338105097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
338205097246SAdrian Prantl     // press will cause us to exit
3383b9c1b51eSKate Stone   } else {
3384b9c1b51eSKate Stone     switch (key) {
338544d93782SGreg Clayton     case KEY_UP:
338644d93782SGreg Clayton       if (m_first_visible_line > 0)
338744d93782SGreg Clayton         --m_first_visible_line;
338844d93782SGreg Clayton       break;
338944d93782SGreg Clayton 
339044d93782SGreg Clayton     case KEY_DOWN:
339144d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
339244d93782SGreg Clayton         ++m_first_visible_line;
339344d93782SGreg Clayton       break;
339444d93782SGreg Clayton 
339544d93782SGreg Clayton     case KEY_PPAGE:
339644d93782SGreg Clayton     case ',':
3397b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33983985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
339944d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
340044d93782SGreg Clayton         else
340144d93782SGreg Clayton           m_first_visible_line = 0;
340244d93782SGreg Clayton       }
340344d93782SGreg Clayton       break;
3404315b6884SEugene Zelenko 
340544d93782SGreg Clayton     case KEY_NPAGE:
340644d93782SGreg Clayton     case '.':
3407b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
340844d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
34093985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
341044d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
341144d93782SGreg Clayton       }
341244d93782SGreg Clayton       break;
3413315b6884SEugene Zelenko 
341444d93782SGreg Clayton     default:
341544d93782SGreg Clayton       done = true;
341644d93782SGreg Clayton       break;
341744d93782SGreg Clayton     }
341844d93782SGreg Clayton   }
341944d93782SGreg Clayton   if (done)
342044d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
342144d93782SGreg Clayton   return eKeyHandled;
342244d93782SGreg Clayton }
342344d93782SGreg Clayton 
3424b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
342544d93782SGreg Clayton public:
342644d93782SGreg Clayton   enum {
342744d93782SGreg Clayton     eMenuID_LLDB = 1,
342844d93782SGreg Clayton     eMenuID_LLDBAbout,
342944d93782SGreg Clayton     eMenuID_LLDBExit,
343044d93782SGreg Clayton 
343144d93782SGreg Clayton     eMenuID_Target,
343244d93782SGreg Clayton     eMenuID_TargetCreate,
343344d93782SGreg Clayton     eMenuID_TargetDelete,
343444d93782SGreg Clayton 
343544d93782SGreg Clayton     eMenuID_Process,
343644d93782SGreg Clayton     eMenuID_ProcessAttach,
343744d93782SGreg Clayton     eMenuID_ProcessDetach,
343844d93782SGreg Clayton     eMenuID_ProcessLaunch,
343944d93782SGreg Clayton     eMenuID_ProcessContinue,
344044d93782SGreg Clayton     eMenuID_ProcessHalt,
344144d93782SGreg Clayton     eMenuID_ProcessKill,
344244d93782SGreg Clayton 
344344d93782SGreg Clayton     eMenuID_Thread,
344444d93782SGreg Clayton     eMenuID_ThreadStepIn,
344544d93782SGreg Clayton     eMenuID_ThreadStepOver,
344644d93782SGreg Clayton     eMenuID_ThreadStepOut,
344744d93782SGreg Clayton 
344844d93782SGreg Clayton     eMenuID_View,
344944d93782SGreg Clayton     eMenuID_ViewBacktrace,
345044d93782SGreg Clayton     eMenuID_ViewRegisters,
345144d93782SGreg Clayton     eMenuID_ViewSource,
345244d93782SGreg Clayton     eMenuID_ViewVariables,
345344d93782SGreg Clayton 
345444d93782SGreg Clayton     eMenuID_Help,
345544d93782SGreg Clayton     eMenuID_HelpGUIHelp
345644d93782SGreg Clayton   };
345744d93782SGreg Clayton 
3458b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3459b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
346044d93782SGreg Clayton 
3461315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3462bd5ae6b4SGreg Clayton 
3463b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
346444d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
346544d93782SGreg Clayton   }
346644d93782SGreg Clayton 
3467b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3468b9c1b51eSKate Stone     switch (key) {
34695fdb09bbSGreg Clayton     case '\t':
347044d93782SGreg Clayton       window.SelectNextWindowAsActive();
347144d93782SGreg Clayton       return eKeyHandled;
34725fdb09bbSGreg Clayton 
34735fdb09bbSGreg Clayton     case 'h':
34745fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34755fdb09bbSGreg Clayton       return eKeyHandled;
34765fdb09bbSGreg Clayton 
34775fdb09bbSGreg Clayton     case KEY_ESCAPE:
34785fdb09bbSGreg Clayton       return eQuitApplication;
34795fdb09bbSGreg Clayton 
34805fdb09bbSGreg Clayton     default:
34815fdb09bbSGreg Clayton       break;
348244d93782SGreg Clayton     }
348344d93782SGreg Clayton     return eKeyNotHandled;
348444d93782SGreg Clayton   }
348544d93782SGreg Clayton 
3486b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34875fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34885fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3489b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3490b9c1b51eSKate Stone            "dialog to display them.\n\n"
34915fdb09bbSGreg Clayton            "Common key bindings for all views:";
34925fdb09bbSGreg Clayton   }
34935fdb09bbSGreg Clayton 
3494b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34955fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34965fdb09bbSGreg Clayton         {'\t', "Select next view"},
34975fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34985fdb09bbSGreg Clayton         {',', "Page up"},
34995fdb09bbSGreg Clayton         {'.', "Page down"},
35005fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
35015fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
35025fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
35035fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
35045fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
35055fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3506b9c1b51eSKate Stone         {'\0', nullptr}};
35075fdb09bbSGreg Clayton     return g_source_view_key_help;
35085fdb09bbSGreg Clayton   }
35095fdb09bbSGreg Clayton 
3510b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3511b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3512b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3513b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3514b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3515b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
351644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3517b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3518b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
35194b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
352044d93782SGreg Clayton       }
352144d93782SGreg Clayton     }
352244d93782SGreg Clayton       return MenuActionResult::Handled;
352344d93782SGreg Clayton 
3524b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3525b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3526b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3527b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
352844d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3529b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3530b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
353144d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
353244d93782SGreg Clayton       }
353344d93782SGreg Clayton     }
353444d93782SGreg Clayton       return MenuActionResult::Handled;
353544d93782SGreg Clayton 
3536b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3537b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3538b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3539b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
354044d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3541b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3542b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
354344d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
354444d93782SGreg Clayton       }
354544d93782SGreg Clayton     }
354644d93782SGreg Clayton       return MenuActionResult::Handled;
354744d93782SGreg Clayton 
3548b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3549b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3550b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3551b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
355244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3553b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3554b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
355544d93782SGreg Clayton           process->Resume();
355644d93782SGreg Clayton       }
355744d93782SGreg Clayton     }
355844d93782SGreg Clayton       return MenuActionResult::Handled;
355944d93782SGreg Clayton 
3560b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3561b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3562b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3563b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
356444d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
356544d93782SGreg Clayton         if (process && process->IsAlive())
3566ede3193bSJason Molenda           process->Destroy(false);
356744d93782SGreg Clayton       }
356844d93782SGreg Clayton     }
356944d93782SGreg Clayton       return MenuActionResult::Handled;
357044d93782SGreg Clayton 
3571b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3572b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3573b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3574b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
357544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
357644d93782SGreg Clayton         if (process && process->IsAlive())
357744d93782SGreg Clayton           process->Halt();
357844d93782SGreg Clayton       }
357944d93782SGreg Clayton     }
358044d93782SGreg Clayton       return MenuActionResult::Handled;
358144d93782SGreg Clayton 
3582b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3583b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3584b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3585b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
358644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
358744d93782SGreg Clayton         if (process && process->IsAlive())
358844d93782SGreg Clayton           process->Detach(false);
358944d93782SGreg Clayton       }
359044d93782SGreg Clayton     }
359144d93782SGreg Clayton       return MenuActionResult::Handled;
359244d93782SGreg Clayton 
3593b9c1b51eSKate Stone     case eMenuID_Process: {
3594b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
359505097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
359605097246SAdrian Prantl       // submenu.
359744d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3598b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3599b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
360044d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3601b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3602b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
360344d93782SGreg Clayton         if (submenus.size() == 7)
360444d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
360544d93782SGreg Clayton         else if (submenus.size() > 8)
360644d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
360744d93782SGreg Clayton 
360844d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3609bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
361044d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3611b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
361244d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
361344d93782SGreg Clayton           char menu_char = '\0';
361444d93782SGreg Clayton           if (i < 9)
361544d93782SGreg Clayton             menu_char = '1' + i;
361644d93782SGreg Clayton           StreamString thread_menu_title;
361744d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
361844d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
361944d93782SGreg Clayton           if (thread_name && thread_name[0])
362044d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3621b9c1b51eSKate Stone           else {
362244d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
362344d93782SGreg Clayton             if (queue_name && queue_name[0])
362444d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
362544d93782SGreg Clayton           }
3626b9c1b51eSKate Stone           menu.AddSubmenu(
3627c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3628c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
362944d93782SGreg Clayton         }
3630b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
363105097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
363205097246SAdrian Prantl         // previously added
363344d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
363444d93782SGreg Clayton       }
3635b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3636b9c1b51eSKate Stone       // lengths
363744d93782SGreg Clayton       menu.RecalculateNameLengths();
363844d93782SGreg Clayton     }
363944d93782SGreg Clayton       return MenuActionResult::Handled;
364044d93782SGreg Clayton 
3641b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
364244d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
364344d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
364444d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
364544d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
364644d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
364744d93782SGreg Clayton 
3648b9c1b51eSKate Stone       if (variables_window_sp) {
364944d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
365044d93782SGreg Clayton 
365144d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
365244d93782SGreg Clayton 
3653b9c1b51eSKate Stone         if (registers_window_sp) {
3654b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3655b9c1b51eSKate Stone           // registers window
365644d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
365744d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
365844d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3659b9c1b51eSKate Stone         } else {
366005097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
366105097246SAdrian Prantl           // to the source view
366244d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3663b9c1b51eSKate Stone                                    source_bounds.size.height +
3664b9c1b51eSKate Stone                                        variables_bounds.size.height);
366544d93782SGreg Clayton         }
3666b9c1b51eSKate Stone       } else {
366744d93782SGreg Clayton         Rect new_variables_rect;
3668b9c1b51eSKate Stone         if (registers_window_sp) {
366944d93782SGreg Clayton           // We have a registers window so split the area of the registers
367044d93782SGreg Clayton           // window into two columns where the left hand side will be the
367144d93782SGreg Clayton           // variables and the right hand side will be the registers
367244d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
367344d93782SGreg Clayton           Rect new_registers_rect;
3674b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3675b9c1b51eSKate Stone                                                    new_registers_rect);
367644d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3677b9c1b51eSKate Stone         } else {
367844d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
367944d93782SGreg Clayton           Rect new_source_rect;
3680b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3681b9c1b51eSKate Stone                                                   new_variables_rect);
368244d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
368344d93782SGreg Clayton         }
3684b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3685b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3686b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3687b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
368844d93782SGreg Clayton       }
368944d93782SGreg Clayton       touchwin(stdscr);
369044d93782SGreg Clayton     }
369144d93782SGreg Clayton       return MenuActionResult::Handled;
369244d93782SGreg Clayton 
3693b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
369444d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
369544d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
369644d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
369744d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
369844d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
369944d93782SGreg Clayton 
3700b9c1b51eSKate Stone       if (registers_window_sp) {
3701b9c1b51eSKate Stone         if (variables_window_sp) {
370244d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
370344d93782SGreg Clayton 
3704b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3705b9c1b51eSKate Stone           // variables window
3706b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3707b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
370844d93782SGreg Clayton                                       variables_bounds.size.height);
3709b9c1b51eSKate Stone         } else {
371005097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
371105097246SAdrian Prantl           // to the source view
371244d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3713b9c1b51eSKate Stone                                    source_bounds.size.height +
3714b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
371544d93782SGreg Clayton         }
371644d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3717b9c1b51eSKate Stone       } else {
371844d93782SGreg Clayton         Rect new_regs_rect;
3719b9c1b51eSKate Stone         if (variables_window_sp) {
372005097246SAdrian Prantl           // We have a variables window, split it into two columns where the
372105097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
372205097246SAdrian Prantl           // be the registers
372344d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
372444d93782SGreg Clayton           Rect new_vars_rect;
3725b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3726b9c1b51eSKate Stone                                                    new_regs_rect);
372744d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3728b9c1b51eSKate Stone         } else {
372944d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
373044d93782SGreg Clayton           Rect new_source_rect;
3731b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3732b9c1b51eSKate Stone                                                   new_regs_rect);
373344d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
373444d93782SGreg Clayton         }
3735b9c1b51eSKate Stone         WindowSP new_window_sp =
3736b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3737b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3738b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
373944d93782SGreg Clayton       }
374044d93782SGreg Clayton       touchwin(stdscr);
374144d93782SGreg Clayton     }
374244d93782SGreg Clayton       return MenuActionResult::Handled;
374344d93782SGreg Clayton 
374444d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37455fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
374644d93782SGreg Clayton       return MenuActionResult::Handled;
374744d93782SGreg Clayton 
374844d93782SGreg Clayton     default:
374944d93782SGreg Clayton       break;
375044d93782SGreg Clayton     }
375144d93782SGreg Clayton 
375244d93782SGreg Clayton     return MenuActionResult::NotHandled;
375344d93782SGreg Clayton   }
3754b9c1b51eSKate Stone 
375544d93782SGreg Clayton protected:
375644d93782SGreg Clayton   Application &m_app;
375744d93782SGreg Clayton   Debugger &m_debugger;
375844d93782SGreg Clayton };
375944d93782SGreg Clayton 
3760b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
376144d93782SGreg Clayton public:
3762b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3763b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
376444d93782SGreg Clayton   }
376544d93782SGreg Clayton 
3766315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3767bd5ae6b4SGreg Clayton 
3768b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3769b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3770b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
377144d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
377244d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
377344d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
377444d93782SGreg Clayton     window.Erase();
377544d93782SGreg Clayton     window.SetBackground(2);
377644d93782SGreg Clayton     window.MoveCursor(0, 0);
3777b9c1b51eSKate Stone     if (process) {
377844d93782SGreg Clayton       const StateType state = process->GetState();
3779b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3780b9c1b51eSKate Stone                     StateAsCString(state));
378144d93782SGreg Clayton 
3782b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37835b031ebcSEd Maste         StreamString strm;
3784b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3785b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
378644d93782SGreg Clayton           window.MoveCursor(40, 0);
3787c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37885b031ebcSEd Maste         }
378944d93782SGreg Clayton 
379044d93782SGreg Clayton         window.MoveCursor(60, 0);
379144d93782SGreg Clayton         if (frame)
3792b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3793b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3794b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3795b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3796b9c1b51eSKate Stone       } else if (state == eStateExited) {
379744d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
379844d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
379944d93782SGreg Clayton         if (exit_desc && exit_desc[0])
380044d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
380144d93782SGreg Clayton         else
380244d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
380344d93782SGreg Clayton       }
380444d93782SGreg Clayton     }
380544d93782SGreg Clayton     window.DeferredRefresh();
380644d93782SGreg Clayton     return true;
380744d93782SGreg Clayton   }
380844d93782SGreg Clayton 
380944d93782SGreg Clayton protected:
381044d93782SGreg Clayton   Debugger &m_debugger;
3811554f68d3SGreg Clayton   FormatEntity::Entry m_format;
381244d93782SGreg Clayton };
381344d93782SGreg Clayton 
3814b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
381544d93782SGreg Clayton public:
3816b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3817b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3818b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3819b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3820b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3821b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
382244d93782SGreg Clayton 
3823315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
382444d93782SGreg Clayton 
3825b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
382644d93782SGreg Clayton 
3827b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
382844d93782SGreg Clayton 
3829b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
383044d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
383144d93782SGreg Clayton   }
383244d93782SGreg Clayton 
3833b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
383444d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
383544d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
383644d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
383744d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
383844d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
383944d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
384044d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
384144d93782SGreg Clayton         {'c', "Continue process"},
384244d93782SGreg Clayton         {'d', "Detach and resume process"},
384344d93782SGreg Clayton         {'D', "Detach with process suspended"},
384444d93782SGreg Clayton         {'h', "Show help dialog"},
384544d93782SGreg Clayton         {'k', "Kill process"},
384644d93782SGreg Clayton         {'n', "Step over (source line)"},
384744d93782SGreg Clayton         {'N', "Step over (single instruction)"},
384844d93782SGreg Clayton         {'o', "Step out"},
384944d93782SGreg Clayton         {'s', "Step in (source line)"},
385044d93782SGreg Clayton         {'S', "Step in (single instruction)"},
385144d93782SGreg Clayton         {',', "Page up"},
385244d93782SGreg Clayton         {'.', "Page down"},
3853b9c1b51eSKate Stone         {'\0', nullptr}};
385444d93782SGreg Clayton     return g_source_view_key_help;
385544d93782SGreg Clayton   }
385644d93782SGreg Clayton 
3857b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3858b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3859b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
386044d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3861c5dac77aSEugene Zelenko     Thread *thread = nullptr;
386244d93782SGreg Clayton 
386344d93782SGreg Clayton     bool update_location = false;
3864b9c1b51eSKate Stone     if (process) {
386544d93782SGreg Clayton       StateType state = process->GetState();
3866b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
386744d93782SGreg Clayton         // We are stopped, so it is ok to
386844d93782SGreg Clayton         update_location = true;
386944d93782SGreg Clayton       }
387044d93782SGreg Clayton     }
387144d93782SGreg Clayton 
387244d93782SGreg Clayton     m_min_x = 1;
3873ec990867SGreg Clayton     m_min_y = 2;
387444d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
387544d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
387644d93782SGreg Clayton 
387744d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
387844d93782SGreg Clayton     StackFrameSP frame_sp;
387944d93782SGreg Clayton     bool set_selected_line_to_pc = false;
388044d93782SGreg Clayton 
3881b9c1b51eSKate Stone     if (update_location) {
388244d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
388344d93782SGreg Clayton       bool thread_changed = false;
3884b9c1b51eSKate Stone       if (process_alive) {
388544d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3886b9c1b51eSKate Stone         if (thread) {
388744d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
388844d93782SGreg Clayton           auto tid = thread->GetID();
388944d93782SGreg Clayton           thread_changed = tid != m_tid;
389044d93782SGreg Clayton           m_tid = tid;
3891b9c1b51eSKate Stone         } else {
3892b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
389344d93782SGreg Clayton             thread_changed = true;
389444d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
389544d93782SGreg Clayton           }
389644d93782SGreg Clayton         }
389744d93782SGreg Clayton       }
389844d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
389944d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
390044d93782SGreg Clayton       bool frame_changed = false;
390144d93782SGreg Clayton       m_stop_id = stop_id;
3902ec990867SGreg Clayton       m_title.Clear();
3903b9c1b51eSKate Stone       if (frame_sp) {
390444d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3905b9c1b51eSKate Stone         if (m_sc.module_sp) {
3906b9c1b51eSKate Stone           m_title.Printf(
3907b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3908ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3909ec990867SGreg Clayton           if (func_name)
3910ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3911ec990867SGreg Clayton         }
391244d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
391344d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
391444d93782SGreg Clayton         m_frame_idx = frame_idx;
3915b9c1b51eSKate Stone       } else {
391644d93782SGreg Clayton         m_sc.Clear(true);
391744d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
391844d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
391944d93782SGreg Clayton       }
392044d93782SGreg Clayton 
3921b9c1b51eSKate Stone       const bool context_changed =
3922b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
392344d93782SGreg Clayton 
3924b9c1b51eSKate Stone       if (process_alive) {
3925b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
392644d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
392744d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
392844d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
392944d93782SGreg Clayton           // Update the selected line if the stop ID changed...
393044d93782SGreg Clayton           if (context_changed)
393144d93782SGreg Clayton             m_selected_line = m_pc_line;
393244d93782SGreg Clayton 
3933b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
393405097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
393505097246SAdrian Prantl             // (source file missing)
3936b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
393744d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
393844d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3939b9c1b51eSKate Stone             } else {
394044d93782SGreg Clayton               if (m_selected_line > 10)
394144d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
394244d93782SGreg Clayton               else
394344d93782SGreg Clayton                 m_first_visible_line = 0;
394444d93782SGreg Clayton             }
3945b9c1b51eSKate Stone           } else {
394644d93782SGreg Clayton             // File changed, set selected line to the line with the PC
394744d93782SGreg Clayton             m_selected_line = m_pc_line;
3948b9c1b51eSKate Stone             m_file_sp =
3949b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3950b9c1b51eSKate Stone             if (m_file_sp) {
395144d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
3952eebf32faSPavel Labath               m_line_width = 1;
395344d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
395444d93782SGreg Clayton                 ++m_line_width;
395544d93782SGreg Clayton 
3956b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3957b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
395844d93782SGreg Clayton                 m_first_visible_line = 0;
395944d93782SGreg Clayton               else
396044d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
396144d93782SGreg Clayton             }
396244d93782SGreg Clayton           }
3963b9c1b51eSKate Stone         } else {
396444d93782SGreg Clayton           m_file_sp.reset();
396544d93782SGreg Clayton         }
396644d93782SGreg Clayton 
3967b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
396844d93782SGreg Clayton           // Show disassembly
396944d93782SGreg Clayton           bool prefer_file_cache = false;
3970b9c1b51eSKate Stone           if (m_sc.function) {
3971b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
397244d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3973b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3974b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3975b9c1b51eSKate Stone               if (m_disassembly_sp) {
397644d93782SGreg Clayton                 set_selected_line_to_pc = true;
397744d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3978b9c1b51eSKate Stone               } else {
397944d93782SGreg Clayton                 m_disassembly_range.Clear();
398044d93782SGreg Clayton               }
3981b9c1b51eSKate Stone             } else {
398244d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
398344d93782SGreg Clayton             }
3984b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3985b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
398644d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3987b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3988b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3989b9c1b51eSKate Stone               if (m_disassembly_sp) {
399044d93782SGreg Clayton                 set_selected_line_to_pc = true;
3991b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3992b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
399344d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3994b9c1b51eSKate Stone               } else {
399544d93782SGreg Clayton                 m_disassembly_range.Clear();
399644d93782SGreg Clayton               }
3997b9c1b51eSKate Stone             } else {
399844d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
399944d93782SGreg Clayton             }
400044d93782SGreg Clayton           }
400144d93782SGreg Clayton         }
4002b9c1b51eSKate Stone       } else {
400344d93782SGreg Clayton         m_pc_line = UINT32_MAX;
400444d93782SGreg Clayton       }
400544d93782SGreg Clayton     }
400644d93782SGreg Clayton 
4007ec990867SGreg Clayton     const int window_width = window.GetWidth();
400844d93782SGreg Clayton     window.Erase();
400944d93782SGreg Clayton     window.DrawTitleBox("Sources");
4010b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
4011ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
4012ec990867SGreg Clayton       window.MoveCursor(1, 1);
4013ec990867SGreg Clayton       window.PutChar(' ');
4014c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
4015ec990867SGreg Clayton       int x = window.GetCursorX();
4016b9c1b51eSKate Stone       if (x < window_width - 1) {
4017ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
4018ec990867SGreg Clayton       }
4019ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
4020ec990867SGreg Clayton     }
402144d93782SGreg Clayton 
402244d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
402344d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
4024b9c1b51eSKate Stone     if (num_source_lines > 0) {
402544d93782SGreg Clayton       // Display source
402644d93782SGreg Clayton       BreakpointLines bp_lines;
4027b9c1b51eSKate Stone       if (target) {
402844d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
402944d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
4030b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
403144d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
403244d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
4033b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
4034b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
4035b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
403644d93782SGreg Clayton             LineEntry bp_loc_line_entry;
4037b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
4038b9c1b51eSKate Stone                     bp_loc_line_entry)) {
4039b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
404044d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
404144d93782SGreg Clayton               }
404244d93782SGreg Clayton             }
404344d93782SGreg Clayton           }
404444d93782SGreg Clayton         }
404544d93782SGreg Clayton       }
404644d93782SGreg Clayton 
404744d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
404844d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
404944d93782SGreg Clayton 
4050b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
405144d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4052b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4053ec990867SGreg Clayton           const int line_y = m_min_y + i;
405444d93782SGreg Clayton           window.MoveCursor(1, line_y);
405544d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
405644d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
405744d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
405844d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
405944d93782SGreg Clayton           attr_t highlight_attr = 0;
406044d93782SGreg Clayton           attr_t bp_attr = 0;
406144d93782SGreg Clayton           if (is_pc_line)
406244d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
406344d93782SGreg Clayton           else if (line_is_selected)
406444d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
406544d93782SGreg Clayton 
406644d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
406744d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
406844d93782SGreg Clayton 
406944d93782SGreg Clayton           if (bp_attr)
407044d93782SGreg Clayton             window.AttributeOn(bp_attr);
407144d93782SGreg Clayton 
4072eebf32faSPavel Labath           window.Printf(" %*u ", m_line_width, curr_line + 1);
407344d93782SGreg Clayton 
407444d93782SGreg Clayton           if (bp_attr)
407544d93782SGreg Clayton             window.AttributeOff(bp_attr);
407644d93782SGreg Clayton 
407744d93782SGreg Clayton           window.PutChar(ACS_VLINE);
407844d93782SGreg Clayton           // Mark the line with the PC with a diamond
407944d93782SGreg Clayton           if (is_pc_line)
408044d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
408144d93782SGreg Clayton           else
408244d93782SGreg Clayton             window.PutChar(' ');
408344d93782SGreg Clayton 
408444d93782SGreg Clayton           if (highlight_attr)
408544d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4086b9c1b51eSKate Stone           const uint32_t line_len =
4087b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
408844d93782SGreg Clayton           if (line_len > 0)
408944d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
409044d93782SGreg Clayton 
4091b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4092b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
409344d93782SGreg Clayton             StopInfoSP stop_info_sp;
409444d93782SGreg Clayton             if (thread)
409544d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4096b9c1b51eSKate Stone             if (stop_info_sp) {
409744d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4098b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
409944d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4100ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
410144d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4102b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4103b9c1b51eSKate Stone                 // line_y);
4104b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4105b9c1b51eSKate Stone                               stop_description);
410644d93782SGreg Clayton               }
4107b9c1b51eSKate Stone             } else {
4108ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
410944d93782SGreg Clayton             }
411044d93782SGreg Clayton           }
411144d93782SGreg Clayton           if (highlight_attr)
411244d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4113b9c1b51eSKate Stone         } else {
411444d93782SGreg Clayton           break;
411544d93782SGreg Clayton         }
411644d93782SGreg Clayton       }
4117b9c1b51eSKate Stone     } else {
411844d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4119b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
412044d93782SGreg Clayton         // Display disassembly
412144d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
412244d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4123b9c1b51eSKate Stone         if (target) {
412444d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
412544d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4126b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
412744d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
412844d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4129b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4130b9c1b51eSKate Stone                  ++bp_loc_idx) {
4131b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4132b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
413344d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4134b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4135b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4136b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
413744d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
413844d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
413944d93782SGreg Clayton               }
414044d93782SGreg Clayton             }
414144d93782SGreg Clayton           }
414244d93782SGreg Clayton         }
414344d93782SGreg Clayton 
414444d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
414544d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
414644d93782SGreg Clayton 
414744d93782SGreg Clayton         StreamString strm;
414844d93782SGreg Clayton 
414944d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
415044d93782SGreg Clayton         Address pc_address;
415144d93782SGreg Clayton 
415244d93782SGreg Clayton         if (frame_sp)
415344d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4154b9c1b51eSKate Stone         const uint32_t pc_idx =
4155b9c1b51eSKate Stone             pc_address.IsValid()
4156b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4157b9c1b51eSKate Stone                 : UINT32_MAX;
4158b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
415944d93782SGreg Clayton           m_selected_line = pc_idx;
416044d93782SGreg Clayton         }
416144d93782SGreg Clayton 
416244d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41633985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
416444d93782SGreg Clayton           m_first_visible_line = 0;
416544d93782SGreg Clayton 
4166b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41673985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
416844d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
416944d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
417044d93782SGreg Clayton         }
417144d93782SGreg Clayton 
4172b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
417344d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
417444d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
417544d93782SGreg Clayton           if (!inst)
417644d93782SGreg Clayton             break;
417744d93782SGreg Clayton 
4178ec990867SGreg Clayton           const int line_y = m_min_y + i;
4179ec990867SGreg Clayton           window.MoveCursor(1, line_y);
418044d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
418144d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
418244d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
418344d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
418444d93782SGreg Clayton           attr_t highlight_attr = 0;
418544d93782SGreg Clayton           attr_t bp_attr = 0;
418644d93782SGreg Clayton           if (is_pc_line)
418744d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
418844d93782SGreg Clayton           else if (line_is_selected)
418944d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
419044d93782SGreg Clayton 
4191b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4192b9c1b51eSKate Stone               bp_file_addrs.end())
419344d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
419444d93782SGreg Clayton 
419544d93782SGreg Clayton           if (bp_attr)
419644d93782SGreg Clayton             window.AttributeOn(bp_attr);
419744d93782SGreg Clayton 
4198324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4199b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4200b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
420144d93782SGreg Clayton 
420244d93782SGreg Clayton           if (bp_attr)
420344d93782SGreg Clayton             window.AttributeOff(bp_attr);
420444d93782SGreg Clayton 
420544d93782SGreg Clayton           window.PutChar(ACS_VLINE);
420644d93782SGreg Clayton           // Mark the line with the PC with a diamond
420744d93782SGreg Clayton           if (is_pc_line)
420844d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
420944d93782SGreg Clayton           else
421044d93782SGreg Clayton             window.PutChar(' ');
421144d93782SGreg Clayton 
421244d93782SGreg Clayton           if (highlight_attr)
421344d93782SGreg Clayton             window.AttributeOn(highlight_attr);
421444d93782SGreg Clayton 
421544d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
421644d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
421744d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
421844d93782SGreg Clayton 
4219c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4220c5dac77aSEugene Zelenko             mnemonic = nullptr;
4221c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4222c5dac77aSEugene Zelenko             operands = nullptr;
4223c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4224c5dac77aSEugene Zelenko             comment = nullptr;
422544d93782SGreg Clayton 
422644d93782SGreg Clayton           strm.Clear();
422744d93782SGreg Clayton 
4228c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
422944d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4230c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
423144d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4232c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
423344d93782SGreg Clayton             strm.Printf("%s", mnemonic);
423444d93782SGreg Clayton 
423544d93782SGreg Clayton           int right_pad = 1;
4236c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
423744d93782SGreg Clayton 
4238b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4239b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
424044d93782SGreg Clayton             StopInfoSP stop_info_sp;
424144d93782SGreg Clayton             if (thread)
424244d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4243b9c1b51eSKate Stone             if (stop_info_sp) {
424444d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4245b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
424644d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4247ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
424844d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4249b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4250b9c1b51eSKate Stone                 // line_y);
4251b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4252b9c1b51eSKate Stone                               stop_description);
425344d93782SGreg Clayton               }
4254b9c1b51eSKate Stone             } else {
4255ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
425644d93782SGreg Clayton             }
425744d93782SGreg Clayton           }
425844d93782SGreg Clayton           if (highlight_attr)
425944d93782SGreg Clayton             window.AttributeOff(highlight_attr);
426044d93782SGreg Clayton         }
426144d93782SGreg Clayton       }
426244d93782SGreg Clayton     }
426344d93782SGreg Clayton     window.DeferredRefresh();
426444d93782SGreg Clayton     return true; // Drawing handled
426544d93782SGreg Clayton   }
426644d93782SGreg Clayton 
4267b9c1b51eSKate Stone   size_t GetNumLines() {
426844d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
426944d93782SGreg Clayton     if (num_lines == 0)
427044d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
427144d93782SGreg Clayton     return num_lines;
427244d93782SGreg Clayton   }
427344d93782SGreg Clayton 
4274b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
427544d93782SGreg Clayton     if (m_file_sp)
427644d93782SGreg Clayton       return m_file_sp->GetNumLines();
427744d93782SGreg Clayton     return 0;
427844d93782SGreg Clayton   }
4279315b6884SEugene Zelenko 
4280b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
428144d93782SGreg Clayton     if (m_disassembly_sp)
428244d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
428344d93782SGreg Clayton     return 0;
428444d93782SGreg Clayton   }
428544d93782SGreg Clayton 
4286b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
428744d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
428844d93782SGreg Clayton     const size_t num_lines = GetNumLines();
428944d93782SGreg Clayton 
4290b9c1b51eSKate Stone     switch (c) {
429144d93782SGreg Clayton     case ',':
429244d93782SGreg Clayton     case KEY_PPAGE:
429344d93782SGreg Clayton       // Page up key
42943985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
429544d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
429644d93782SGreg Clayton       else
429744d93782SGreg Clayton         m_first_visible_line = 0;
429844d93782SGreg Clayton       m_selected_line = m_first_visible_line;
429944d93782SGreg Clayton       return eKeyHandled;
430044d93782SGreg Clayton 
430144d93782SGreg Clayton     case '.':
430244d93782SGreg Clayton     case KEY_NPAGE:
430344d93782SGreg Clayton       // Page down key
430444d93782SGreg Clayton       {
430544d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
430644d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
430744d93782SGreg Clayton         else if (num_lines < num_visible_lines)
430844d93782SGreg Clayton           m_first_visible_line = 0;
430944d93782SGreg Clayton         else
431044d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
431144d93782SGreg Clayton         m_selected_line = m_first_visible_line;
431244d93782SGreg Clayton       }
431344d93782SGreg Clayton       return eKeyHandled;
431444d93782SGreg Clayton 
431544d93782SGreg Clayton     case KEY_UP:
4316b9c1b51eSKate Stone       if (m_selected_line > 0) {
431744d93782SGreg Clayton         m_selected_line--;
43183985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
431944d93782SGreg Clayton           m_first_visible_line = m_selected_line;
432044d93782SGreg Clayton       }
432144d93782SGreg Clayton       return eKeyHandled;
432244d93782SGreg Clayton 
432344d93782SGreg Clayton     case KEY_DOWN:
4324b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
432544d93782SGreg Clayton         m_selected_line++;
432644d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
432744d93782SGreg Clayton           m_first_visible_line++;
432844d93782SGreg Clayton       }
432944d93782SGreg Clayton       return eKeyHandled;
433044d93782SGreg Clayton 
433144d93782SGreg Clayton     case '\r':
433244d93782SGreg Clayton     case '\n':
433344d93782SGreg Clayton     case KEY_ENTER:
433444d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4335b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4336b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4337b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4338b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4339b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4340b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
434144d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4342b9c1b51eSKate Stone               m_selected_line +
4343b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4344431b1584SAdrian Prantl               0,     // Unspecified column.
43452411167fSJim Ingham               0,     // No offset
434644d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
434744d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
434844d93782SGreg Clayton               false,               // internal
4349055ad9beSIlia K               false,               // request_hardware
4350055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
435144d93782SGreg Clayton           // Make breakpoint one shot
435244d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
435344d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
435444d93782SGreg Clayton         }
4355b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4356b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4357b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4358b9c1b51eSKate Stone                                       .get();
4359b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4360b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4361b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
436244d93782SGreg Clayton           Address addr = inst->GetAddress();
4363b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4364b9c1b51eSKate Stone               addr,   // lldb_private::Address
436544d93782SGreg Clayton               false,  // internal
436644d93782SGreg Clayton               false); // request_hardware
436744d93782SGreg Clayton           // Make breakpoint one shot
436844d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
436944d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
437044d93782SGreg Clayton         }
437144d93782SGreg Clayton       }
437244d93782SGreg Clayton       return eKeyHandled;
437344d93782SGreg Clayton 
437444d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4375b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4376b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4377b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4378b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4379b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4380b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
438144d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4382b9c1b51eSKate Stone               m_selected_line +
4383b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4384431b1584SAdrian Prantl               0,     // No column specified.
43852411167fSJim Ingham               0,     // No offset
438644d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
438744d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
438844d93782SGreg Clayton               false,               // internal
4389055ad9beSIlia K               false,               // request_hardware
4390055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
439144d93782SGreg Clayton         }
4392b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4393b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4394b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4395b9c1b51eSKate Stone                                       .get();
4396b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4397b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4398b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
439944d93782SGreg Clayton           Address addr = inst->GetAddress();
4400b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4401b9c1b51eSKate Stone               addr,   // lldb_private::Address
440244d93782SGreg Clayton               false,  // internal
440344d93782SGreg Clayton               false); // request_hardware
440444d93782SGreg Clayton         }
440544d93782SGreg Clayton       }
440644d93782SGreg Clayton       return eKeyHandled;
440744d93782SGreg Clayton 
440844d93782SGreg Clayton     case 'd': // 'd' == detach and let run
440944d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
441044d93782SGreg Clayton     {
4411b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4412b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
441344d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
441444d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
441544d93782SGreg Clayton     }
441644d93782SGreg Clayton       return eKeyHandled;
441744d93782SGreg Clayton 
441844d93782SGreg Clayton     case 'k':
441944d93782SGreg Clayton       // 'k' == kill
442044d93782SGreg Clayton       {
4421b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4422b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
442344d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4424ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
442544d93782SGreg Clayton       }
442644d93782SGreg Clayton       return eKeyHandled;
442744d93782SGreg Clayton 
442844d93782SGreg Clayton     case 'c':
442944d93782SGreg Clayton       // 'c' == continue
443044d93782SGreg Clayton       {
4431b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4432b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
443344d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
443444d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
443544d93782SGreg Clayton       }
443644d93782SGreg Clayton       return eKeyHandled;
443744d93782SGreg Clayton 
443844d93782SGreg Clayton     case 'o':
443944d93782SGreg Clayton       // 'o' == step out
444044d93782SGreg Clayton       {
4441b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4442b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4443b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4444b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
444544d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
444644d93782SGreg Clayton         }
444744d93782SGreg Clayton       }
444844d93782SGreg Clayton       return eKeyHandled;
4449315b6884SEugene Zelenko 
445044d93782SGreg Clayton     case 'n': // 'n' == step over
445144d93782SGreg Clayton     case 'N': // 'N' == step over instruction
445244d93782SGreg Clayton     {
4453b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4454b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4455b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4456b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
445744d93782SGreg Clayton         bool source_step = (c == 'n');
445844d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
445944d93782SGreg Clayton       }
446044d93782SGreg Clayton     }
446144d93782SGreg Clayton       return eKeyHandled;
4462315b6884SEugene Zelenko 
446344d93782SGreg Clayton     case 's': // 's' == step into
446444d93782SGreg Clayton     case 'S': // 'S' == step into instruction
446544d93782SGreg Clayton     {
4466b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4467b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4468b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4469b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
447044d93782SGreg Clayton         bool source_step = (c == 's');
44714b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
447244d93782SGreg Clayton       }
447344d93782SGreg Clayton     }
447444d93782SGreg Clayton       return eKeyHandled;
447544d93782SGreg Clayton 
447644d93782SGreg Clayton     case 'h':
447744d93782SGreg Clayton       window.CreateHelpSubwindow();
447844d93782SGreg Clayton       return eKeyHandled;
447944d93782SGreg Clayton 
448044d93782SGreg Clayton     default:
448144d93782SGreg Clayton       break;
448244d93782SGreg Clayton     }
448344d93782SGreg Clayton     return eKeyNotHandled;
448444d93782SGreg Clayton   }
448544d93782SGreg Clayton 
448644d93782SGreg Clayton protected:
448744d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
448844d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
448944d93782SGreg Clayton 
449044d93782SGreg Clayton   Debugger &m_debugger;
449144d93782SGreg Clayton   SymbolContext m_sc;
449244d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
449344d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
449444d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
449544d93782SGreg Clayton   AddressRange m_disassembly_range;
4496ec990867SGreg Clayton   StreamString m_title;
449744d93782SGreg Clayton   lldb::user_id_t m_tid;
449844d93782SGreg Clayton   int m_line_width;
449944d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
450044d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
450144d93782SGreg Clayton   uint32_t m_stop_id;
450244d93782SGreg Clayton   uint32_t m_frame_idx;
450344d93782SGreg Clayton   int m_first_visible_line;
450444d93782SGreg Clayton   int m_min_x;
450544d93782SGreg Clayton   int m_min_y;
450644d93782SGreg Clayton   int m_max_x;
450744d93782SGreg Clayton   int m_max_y;
450844d93782SGreg Clayton };
450944d93782SGreg Clayton 
451044d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
451144d93782SGreg Clayton 
4512b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4513b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
451444d93782SGreg Clayton 
4515b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
451644d93782SGreg Clayton   IOHandler::Activate();
4517b9c1b51eSKate Stone   if (!m_app_ap) {
451844d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
451944d93782SGreg Clayton 
452044d93782SGreg Clayton     // This is both a window and a menu delegate
4521b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4522b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
452344d93782SGreg Clayton 
4524b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4525b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4526b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4527b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4528b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4529b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
453044d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4531b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4532b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
453344d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
453444d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
453544d93782SGreg Clayton 
4536b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4537b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4538b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4539b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4540b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4541b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
454244d93782SGreg Clayton 
4543b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4544b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4545b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4546b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4547b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4548b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4549b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4550b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
455144d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4552b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4553b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4554b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4555b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4556b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4557b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4558b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
455944d93782SGreg Clayton 
4560b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4561b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4562b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4563b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4564b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4565b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4566b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4567b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4568b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
456944d93782SGreg Clayton 
4570b9c1b51eSKate Stone     MenuSP view_menu_sp(
4571b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4572b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4573b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4574b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4575b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4576b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4577b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4578b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4579b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4580b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4581b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4582b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
458344d93782SGreg Clayton 
4584b9c1b51eSKate Stone     MenuSP help_menu_sp(
4585b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4586b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4587b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
458844d93782SGreg Clayton 
458944d93782SGreg Clayton     m_app_ap->Initialize();
459044d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
459144d93782SGreg Clayton 
459244d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
459344d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
459444d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
459544d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
459644d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
459744d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
459844d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
459944d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
460044d93782SGreg Clayton 
460144d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
460244d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
460344d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
460444d93782SGreg Clayton     Rect source_bounds;
460544d93782SGreg Clayton     Rect variables_bounds;
460644d93782SGreg Clayton     Rect threads_bounds;
460744d93782SGreg Clayton     Rect source_variables_bounds;
4608b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4609b9c1b51eSKate Stone                                            threads_bounds);
4610b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4611b9c1b51eSKate Stone                                                       variables_bounds);
461244d93782SGreg Clayton 
4613b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4614b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
461505097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
461605097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4617b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4618b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
461944d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
462044d93782SGreg Clayton 
4621b9c1b51eSKate Stone     WindowSP source_window_sp(
4622b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4623b9c1b51eSKate Stone     WindowSP variables_window_sp(
4624b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4625b9c1b51eSKate Stone     WindowSP threads_window_sp(
4626b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4627b9c1b51eSKate Stone     WindowSP status_window_sp(
46286bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4629b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4630b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4631b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4632b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4633b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4634b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4635b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4636b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4637ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4638b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4639b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4640b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4641b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
464244d93782SGreg Clayton 
46435fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
46445fdb09bbSGreg Clayton     static bool g_showed_help = false;
4645b9c1b51eSKate Stone     if (!g_showed_help) {
46465fdb09bbSGreg Clayton       g_showed_help = true;
46475fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46485fdb09bbSGreg Clayton     }
46495fdb09bbSGreg Clayton 
465044d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
465144d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
465244d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
465344d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
465444d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
465544d93782SGreg Clayton   }
465644d93782SGreg Clayton }
465744d93782SGreg Clayton 
4658b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
465944d93782SGreg Clayton 
4660b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
466144d93782SGreg Clayton   m_app_ap->Run(m_debugger);
466244d93782SGreg Clayton   SetIsDone(true);
466344d93782SGreg Clayton }
466444d93782SGreg Clayton 
4665315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
466644d93782SGreg Clayton 
4667b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
466844d93782SGreg Clayton 
4669b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
467044d93782SGreg Clayton 
4671b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
467244d93782SGreg Clayton 
4673315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4674