144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
344d93782SGreg Clayton //                     The LLVM Compiler Infrastructure
444d93782SGreg Clayton //
544d93782SGreg Clayton // This file is distributed under the University of Illinois Open Source
644d93782SGreg Clayton // License. See LICENSE.TXT for details.
744d93782SGreg Clayton //
844d93782SGreg Clayton //===----------------------------------------------------------------------===//
944d93782SGreg Clayton 
10315b6884SEugene Zelenko // C Includes
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
16315b6884SEugene Zelenko // C++ Includes
177c9aa073STodd Fiala #if defined(__APPLE__)
187c9aa073STodd Fiala #include <deque>
197c9aa073STodd Fiala #endif
2044d93782SGreg Clayton #include <string>
2144d93782SGreg Clayton 
22315b6884SEugene Zelenko // Other libraries and framework includes
23315b6884SEugene Zelenko // Project includes
2444d93782SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h"
2544d93782SGreg Clayton #include "lldb/Core/Debugger.h"
26b9c1b51eSKate Stone #include "lldb/Core/IOHandler.h"
27ec990867SGreg Clayton #include "lldb/Core/Module.h"
2844d93782SGreg Clayton #include "lldb/Core/State.h"
2944d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
3044d93782SGreg Clayton #include "lldb/Core/ValueObjectRegister.h"
31cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3244d93782SGreg Clayton #include "lldb/Host/Editline.h"
33cacde7dfSTodd Fiala #endif
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3544d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
3644d93782SGreg Clayton #include "lldb/Symbol/Block.h"
3744d93782SGreg Clayton #include "lldb/Symbol/Function.h"
3844d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
3944d93782SGreg Clayton #include "lldb/Target/RegisterContext.h"
4044d93782SGreg Clayton #include "lldb/Target/ThreadPlan.h"
41c5dac77aSEugene Zelenko #ifndef LLDB_DISABLE_CURSES
42c5dac77aSEugene Zelenko #include "lldb/Core/ValueObject.h"
43c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
44c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
45c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
46b9c1b51eSKate Stone #include "lldb/Target/Target.h"
47b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
48c5dac77aSEugene Zelenko #endif
497c9aa073STodd Fiala 
50fab31220STed Woodward #ifdef _MSC_VER
51fab31220STed Woodward #include <Windows.h>
52fab31220STed Woodward #endif
53fab31220STed Woodward 
5444d93782SGreg Clayton using namespace lldb;
5544d93782SGreg Clayton using namespace lldb_private;
5644d93782SGreg Clayton 
57b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type)
58b9c1b51eSKate Stone     : IOHandler(debugger, type,
5944d93782SGreg Clayton                 StreamFileSP(), // Adopt STDIN from top input reader
6044d93782SGreg Clayton                 StreamFileSP(), // Adopt STDOUT from top input reader
61340b0309SGreg Clayton                 StreamFileSP(), // Adopt STDERR from top input reader
62340b0309SGreg Clayton                 0)              // Flags
63b9c1b51eSKate Stone {}
6444d93782SGreg Clayton 
65b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
6644d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
6744d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
68b9c1b51eSKate Stone                      const lldb::StreamFileSP &error_sp, uint32_t flags)
69b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
70b9c1b51eSKate Stone       m_error_sp(error_sp), m_popped(false), m_flags(flags), m_type(type),
71b9c1b51eSKate Stone       m_user_data(nullptr), m_done(false), m_active(false) {
7244d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
7344d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
74b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
7544d93782SGreg Clayton                                              m_error_sp);
7644d93782SGreg Clayton }
7744d93782SGreg Clayton 
78315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
7944d93782SGreg Clayton 
80b9c1b51eSKate Stone int IOHandler::GetInputFD() {
81c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
8244d93782SGreg Clayton }
8344d93782SGreg Clayton 
84b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
85c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
8644d93782SGreg Clayton }
8744d93782SGreg Clayton 
88b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
89c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
9044d93782SGreg Clayton }
9144d93782SGreg Clayton 
92b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
93c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
9444d93782SGreg Clayton }
9544d93782SGreg Clayton 
96b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
97c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
9844d93782SGreg Clayton }
9944d93782SGreg Clayton 
100b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
101c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
10244d93782SGreg Clayton }
10344d93782SGreg Clayton 
104b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
10544d93782SGreg Clayton 
106b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
10744d93782SGreg Clayton 
108b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
10944d93782SGreg Clayton 
110b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
111340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
112340b0309SGreg Clayton }
113340b0309SGreg Clayton 
114b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
115340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
116340b0309SGreg Clayton }
11744d93782SGreg Clayton 
118b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
119e30f11d9SKate Stone 
120b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
121e30f11d9SKate Stone 
122b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
123b9c1b51eSKate Stone   if (stream) {
12416ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1254446487dSPavel Labath     if (m_top)
1264446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1274446487dSPavel Labath   }
1284446487dSPavel Labath }
1294446487dSPavel Labath 
130b9c1b51eSKate Stone IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, const char *prompt,
131b9c1b51eSKate Stone                                    bool default_response)
132b9c1b51eSKate Stone     : IOHandlerEditline(
133b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
134c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
135*514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
136*514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
13744d93782SGreg Clayton           false,             // Multi-line
138e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
139b9c1b51eSKate Stone           0, *this),
140b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
14144d93782SGreg Clayton   StreamString prompt_stream;
14244d93782SGreg Clayton   prompt_stream.PutCString(prompt);
14344d93782SGreg Clayton   if (m_default_response)
14444d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
14544d93782SGreg Clayton   else
14644d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
14744d93782SGreg Clayton 
148*514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
14944d93782SGreg Clayton }
15044d93782SGreg Clayton 
151315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
15244d93782SGreg Clayton 
153b9c1b51eSKate Stone int IOHandlerConfirm::IOHandlerComplete(IOHandler &io_handler,
15444d93782SGreg Clayton                                         const char *current_line,
15544d93782SGreg Clayton                                         const char *cursor,
15644d93782SGreg Clayton                                         const char *last_char,
15744d93782SGreg Clayton                                         int skip_first_n_matches,
158b9c1b51eSKate Stone                                         int max_matches, StringList &matches) {
159b9c1b51eSKate Stone   if (current_line == cursor) {
160b9c1b51eSKate Stone     if (m_default_response) {
16144d93782SGreg Clayton       matches.AppendString("y");
162b9c1b51eSKate Stone     } else {
16344d93782SGreg Clayton       matches.AppendString("n");
16444d93782SGreg Clayton     }
16544d93782SGreg Clayton   }
16644d93782SGreg Clayton   return matches.GetSize();
16744d93782SGreg Clayton }
16844d93782SGreg Clayton 
169b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
170b9c1b51eSKate Stone                                               std::string &line) {
171b9c1b51eSKate Stone   if (line.empty()) {
17244d93782SGreg Clayton     // User just hit enter, set the response to the default
17344d93782SGreg Clayton     m_user_response = m_default_response;
17444d93782SGreg Clayton     io_handler.SetIsDone(true);
17544d93782SGreg Clayton     return;
17644d93782SGreg Clayton   }
17744d93782SGreg Clayton 
178b9c1b51eSKate Stone   if (line.size() == 1) {
179b9c1b51eSKate Stone     switch (line[0]) {
18044d93782SGreg Clayton     case 'y':
18144d93782SGreg Clayton     case 'Y':
18244d93782SGreg Clayton       m_user_response = true;
18344d93782SGreg Clayton       io_handler.SetIsDone(true);
18444d93782SGreg Clayton       return;
18544d93782SGreg Clayton     case 'n':
18644d93782SGreg Clayton     case 'N':
18744d93782SGreg Clayton       m_user_response = false;
18844d93782SGreg Clayton       io_handler.SetIsDone(true);
18944d93782SGreg Clayton       return;
19044d93782SGreg Clayton     default:
19144d93782SGreg Clayton       break;
19244d93782SGreg Clayton     }
19344d93782SGreg Clayton   }
19444d93782SGreg Clayton 
195b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
19644d93782SGreg Clayton     m_user_response = true;
19744d93782SGreg Clayton     io_handler.SetIsDone(true);
198b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
19944d93782SGreg Clayton     m_user_response = false;
20044d93782SGreg Clayton     io_handler.SetIsDone(true);
20144d93782SGreg Clayton   }
20244d93782SGreg Clayton }
20344d93782SGreg Clayton 
204b9c1b51eSKate Stone int IOHandlerDelegate::IOHandlerComplete(IOHandler &io_handler,
20544d93782SGreg Clayton                                          const char *current_line,
20644d93782SGreg Clayton                                          const char *cursor,
20744d93782SGreg Clayton                                          const char *last_char,
20844d93782SGreg Clayton                                          int skip_first_n_matches,
209b9c1b51eSKate Stone                                          int max_matches, StringList &matches) {
210b9c1b51eSKate Stone   switch (m_completion) {
21144d93782SGreg Clayton   case Completion::None:
21244d93782SGreg Clayton     break;
21344d93782SGreg Clayton 
21444d93782SGreg Clayton   case Completion::LLDBCommand:
215b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
216b9c1b51eSKate Stone         current_line, cursor, last_char, skip_first_n_matches, max_matches,
21744d93782SGreg Clayton         matches);
21844d93782SGreg Clayton 
219b9c1b51eSKate Stone   case Completion::Expression: {
22044d93782SGreg Clayton     bool word_complete = false;
22144d93782SGreg Clayton     const char *word_start = cursor;
22244d93782SGreg Clayton     if (cursor > current_line)
22344d93782SGreg Clayton       --word_start;
22444d93782SGreg Clayton     while (word_start > current_line && !isspace(*word_start))
22544d93782SGreg Clayton       --word_start;
226b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
227b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
228b9c1b51eSKate Stone         CommandCompletions::eVariablePathCompletion, word_start,
229b9c1b51eSKate Stone         skip_first_n_matches, max_matches, nullptr, word_complete, matches);
23044d93782SGreg Clayton 
23144d93782SGreg Clayton     size_t num_matches = matches.GetSize();
232b9c1b51eSKate Stone     if (num_matches > 0) {
23344d93782SGreg Clayton       std::string common_prefix;
23444d93782SGreg Clayton       matches.LongestCommonPrefix(common_prefix);
23544d93782SGreg Clayton       const size_t partial_name_len = strlen(word_start);
23644d93782SGreg Clayton 
23744d93782SGreg Clayton       // If we matched a unique single command, add a space...
238b9c1b51eSKate Stone       // Only do this if the completer told us this was a complete word,
239b9c1b51eSKate Stone       // however...
240b9c1b51eSKate Stone       if (num_matches == 1 && word_complete) {
24144d93782SGreg Clayton         common_prefix.push_back(' ');
24244d93782SGreg Clayton       }
24344d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
24444d93782SGreg Clayton       matches.InsertStringAtIndex(0, std::move(common_prefix));
24544d93782SGreg Clayton     }
24644d93782SGreg Clayton     return num_matches;
247b9c1b51eSKate Stone   } break;
24844d93782SGreg Clayton   }
24944d93782SGreg Clayton 
25044d93782SGreg Clayton   return 0;
25144d93782SGreg Clayton }
25244d93782SGreg Clayton 
253b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
254b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
25544d93782SGreg Clayton     const char *editline_name, // Used for saving history files
256*514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
257*514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
258*514d8cd8SZachary Turner     IOHandlerDelegate &delegate)
259b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
26044d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
26144d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
26244d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
263340b0309SGreg Clayton                         0,              // Flags
26444d93782SGreg Clayton                         editline_name,  // Used for saving history files
265b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
266b9c1b51eSKate Stone                         line_number_start, delegate) {}
26744d93782SGreg Clayton 
268b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
269b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
270b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
271b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
27244d93782SGreg Clayton     const char *editline_name, // Used for saving history files
273*514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
274*514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
275*514d8cd8SZachary Turner     IOHandlerDelegate &delegate)
276b9c1b51eSKate Stone     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags),
277cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
27844d93782SGreg Clayton       m_editline_ap(),
279cacde7dfSTodd Fiala #endif
280b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
281b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
282b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
283b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
284b9c1b51eSKate Stone       m_editing(false) {
28544d93782SGreg Clayton   SetPrompt(prompt);
28644d93782SGreg Clayton 
287cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
288914b8d98SDeepak Panickal   bool use_editline = false;
289340b0309SGreg Clayton 
290340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
29144d93782SGreg Clayton 
292b9c1b51eSKate Stone   if (use_editline) {
293b9c1b51eSKate Stone     m_editline_ap.reset(new Editline(editline_name, GetInputFILE(),
294b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
295e30f11d9SKate Stone                                      m_color_prompts));
296e30f11d9SKate Stone     m_editline_ap->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
29744d93782SGreg Clayton     m_editline_ap->SetAutoCompleteCallback(AutoCompleteCallback, this);
298e30f11d9SKate Stone     // See if the delegate supports fixing indentation
299e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
300b9c1b51eSKate Stone     if (indent_chars) {
301b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
302b9c1b51eSKate Stone       // indentation
303e30f11d9SKate Stone       // character is typed, the delegate gets a chance to fix it
304b9c1b51eSKate Stone       m_editline_ap->SetFixIndentationCallback(FixIndentationCallback, this,
305b9c1b51eSKate Stone                                                indent_chars);
306e30f11d9SKate Stone     }
30744d93782SGreg Clayton   }
308cacde7dfSTodd Fiala #endif
309e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
310*514d8cd8SZachary Turner   SetPrompt(prompt);
311e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
31244d93782SGreg Clayton }
31344d93782SGreg Clayton 
314b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
315cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
31644d93782SGreg Clayton   m_editline_ap.reset();
317cacde7dfSTodd Fiala #endif
31844d93782SGreg Clayton }
31944d93782SGreg Clayton 
320b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
321e30f11d9SKate Stone   IOHandler::Activate();
322e30f11d9SKate Stone   m_delegate.IOHandlerActivated(*this);
323e30f11d9SKate Stone }
324e30f11d9SKate Stone 
325b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
326e30f11d9SKate Stone   IOHandler::Deactivate();
327e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
328e30f11d9SKate Stone }
329e30f11d9SKate Stone 
330b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
331cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
332b9c1b51eSKate Stone   if (m_editline_ap) {
333e30f11d9SKate Stone     return m_editline_ap->GetLine(line, interrupted);
334b9c1b51eSKate Stone   } else {
335cacde7dfSTodd Fiala #endif
33644d93782SGreg Clayton     line.clear();
33744d93782SGreg Clayton 
33844d93782SGreg Clayton     FILE *in = GetInputFILE();
339b9c1b51eSKate Stone     if (in) {
340b9c1b51eSKate Stone       if (GetIsInteractive()) {
341c5dac77aSEugene Zelenko         const char *prompt = nullptr;
342e30f11d9SKate Stone 
343e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
344e30f11d9SKate Stone           prompt = GetContinuationPrompt();
345e30f11d9SKate Stone 
346c5dac77aSEugene Zelenko         if (prompt == nullptr)
347e30f11d9SKate Stone           prompt = GetPrompt();
348e30f11d9SKate Stone 
349b9c1b51eSKate Stone         if (prompt && prompt[0]) {
35044d93782SGreg Clayton           FILE *out = GetOutputFILE();
351b9c1b51eSKate Stone           if (out) {
35244d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
35344d93782SGreg Clayton             ::fflush(out);
35444d93782SGreg Clayton           }
35544d93782SGreg Clayton         }
35644d93782SGreg Clayton       }
35744d93782SGreg Clayton       char buffer[256];
35844d93782SGreg Clayton       bool done = false;
3590f86e6e7SGreg Clayton       bool got_line = false;
360e034a04eSGreg Clayton       m_editing = true;
361b9c1b51eSKate Stone       while (!done) {
362b9c1b51eSKate Stone         if (fgets(buffer, sizeof(buffer), in) == nullptr) {
363c7797accSGreg Clayton           const int saved_errno = errno;
364c9cf5798SGreg Clayton           if (feof(in))
36544d93782SGreg Clayton             done = true;
366b9c1b51eSKate Stone           else if (ferror(in)) {
367c7797accSGreg Clayton             if (saved_errno != EINTR)
368c7797accSGreg Clayton               done = true;
369c7797accSGreg Clayton           }
370b9c1b51eSKate Stone         } else {
3710f86e6e7SGreg Clayton           got_line = true;
37244d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
37344d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
37444d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
375b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
37644d93782SGreg Clayton             done = true;
37744d93782SGreg Clayton             // Strip trailing newlines
378b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
37944d93782SGreg Clayton               --buffer_len;
38044d93782SGreg Clayton               if (buffer_len == 0)
38144d93782SGreg Clayton                 break;
38244d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
38344d93782SGreg Clayton             }
38444d93782SGreg Clayton           }
38544d93782SGreg Clayton           line.append(buffer, buffer_len);
38644d93782SGreg Clayton         }
38744d93782SGreg Clayton       }
388e034a04eSGreg Clayton       m_editing = false;
3890f86e6e7SGreg Clayton       // We might have gotten a newline on a line by itself
3900f86e6e7SGreg Clayton       // make sure to return true in this case.
3910f86e6e7SGreg Clayton       return got_line;
392b9c1b51eSKate Stone     } else {
39344d93782SGreg Clayton       // No more input file, we are done...
39444d93782SGreg Clayton       SetIsDone(true);
39544d93782SGreg Clayton     }
396340b0309SGreg Clayton     return false;
397cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
39844d93782SGreg Clayton   }
399cacde7dfSTodd Fiala #endif
40044d93782SGreg Clayton }
40144d93782SGreg Clayton 
402cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
403b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
40444d93782SGreg Clayton                                                 StringList &lines,
405b9c1b51eSKate Stone                                                 void *baton) {
40644d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
407b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
408b9c1b51eSKate Stone                                                               lines);
409e30f11d9SKate Stone }
410e30f11d9SKate Stone 
411b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
412e30f11d9SKate Stone                                               const StringList &lines,
413e30f11d9SKate Stone                                               int cursor_position,
414b9c1b51eSKate Stone                                               void *baton) {
415e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
416b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
417b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
41844d93782SGreg Clayton }
41944d93782SGreg Clayton 
420b9c1b51eSKate Stone int IOHandlerEditline::AutoCompleteCallback(const char *current_line,
42144d93782SGreg Clayton                                             const char *cursor,
42244d93782SGreg Clayton                                             const char *last_char,
42344d93782SGreg Clayton                                             int skip_first_n_matches,
42444d93782SGreg Clayton                                             int max_matches,
425b9c1b51eSKate Stone                                             StringList &matches, void *baton) {
42644d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
42744d93782SGreg Clayton   if (editline_reader)
428b9c1b51eSKate Stone     return editline_reader->m_delegate.IOHandlerComplete(
429b9c1b51eSKate Stone         *editline_reader, current_line, cursor, last_char, skip_first_n_matches,
430b9c1b51eSKate Stone         max_matches, matches);
43144d93782SGreg Clayton   return 0;
43244d93782SGreg Clayton }
433cacde7dfSTodd Fiala #endif
43444d93782SGreg Clayton 
435b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
436cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
437b9c1b51eSKate Stone   if (m_editline_ap) {
43844d93782SGreg Clayton     return m_editline_ap->GetPrompt();
439b9c1b51eSKate Stone   } else {
440cacde7dfSTodd Fiala #endif
441cacde7dfSTodd Fiala     if (m_prompt.empty())
442c5dac77aSEugene Zelenko       return nullptr;
443cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
444cacde7dfSTodd Fiala   }
445cacde7dfSTodd Fiala #endif
44644d93782SGreg Clayton   return m_prompt.c_str();
44744d93782SGreg Clayton }
44844d93782SGreg Clayton 
449*514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
450*514d8cd8SZachary Turner   m_prompt = prompt;
451*514d8cd8SZachary Turner 
452cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
45344d93782SGreg Clayton   if (m_editline_ap)
454c5dac77aSEugene Zelenko     m_editline_ap->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
455cacde7dfSTodd Fiala #endif
45644d93782SGreg Clayton   return true;
45744d93782SGreg Clayton }
45844d93782SGreg Clayton 
459b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
460b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
461b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
462e30f11d9SKate Stone }
463e30f11d9SKate Stone 
464*514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
465*514d8cd8SZachary Turner   m_continuation_prompt = prompt;
466e30f11d9SKate Stone 
467d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
468e30f11d9SKate Stone   if (m_editline_ap)
469b9c1b51eSKate Stone     m_editline_ap->SetContinuationPrompt(m_continuation_prompt.empty()
470b9c1b51eSKate Stone                                              ? nullptr
471b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
472d553d00cSZachary Turner #endif
473e30f11d9SKate Stone }
474e30f11d9SKate Stone 
475b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
476f6913cd7SGreg Clayton   m_base_line_number = line;
477f6913cd7SGreg Clayton }
478e30f11d9SKate Stone 
479b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
480d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
481e30f11d9SKate Stone   if (m_editline_ap)
482e30f11d9SKate Stone     return m_editline_ap->GetCurrentLine();
483e30f11d9SKate Stone #endif
484e30f11d9SKate Stone   return m_curr_line_idx;
485e30f11d9SKate Stone }
486e30f11d9SKate Stone 
487b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
488e30f11d9SKate Stone   m_current_lines_ptr = &lines;
489e30f11d9SKate Stone 
49044d93782SGreg Clayton   bool success = false;
491cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
492b9c1b51eSKate Stone   if (m_editline_ap) {
493e30f11d9SKate Stone     return m_editline_ap->GetLines(m_base_line_number, lines, interrupted);
494b9c1b51eSKate Stone   } else {
495cacde7dfSTodd Fiala #endif
496e30f11d9SKate Stone     bool done = false;
497c3d874a5SGreg Clayton     Error error;
49844d93782SGreg Clayton 
499b9c1b51eSKate Stone     while (!done) {
500f6913cd7SGreg Clayton       // Show line numbers if we are asked to
50144d93782SGreg Clayton       std::string line;
502b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
503f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
504f6913cd7SGreg Clayton         if (out)
505b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
506b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
507f6913cd7SGreg Clayton       }
508f6913cd7SGreg Clayton 
509e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
510e30f11d9SKate Stone 
511f0066ad0SGreg Clayton       bool interrupted = false;
512b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
51344d93782SGreg Clayton         lines.AppendString(line);
514e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
515b9c1b51eSKate Stone       } else {
516e30f11d9SKate Stone         done = true;
51744d93782SGreg Clayton       }
51844d93782SGreg Clayton     }
51944d93782SGreg Clayton     success = lines.GetSize() > 0;
520cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
52144d93782SGreg Clayton   }
522cacde7dfSTodd Fiala #endif
52344d93782SGreg Clayton   return success;
52444d93782SGreg Clayton }
52544d93782SGreg Clayton 
52644d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data
52744d93782SGreg Clayton // from the "in" and place output into "out" and "err and return
52844d93782SGreg Clayton // when done.
529b9c1b51eSKate Stone void IOHandlerEditline::Run() {
53044d93782SGreg Clayton   std::string line;
531b9c1b51eSKate Stone   while (IsActive()) {
532f0066ad0SGreg Clayton     bool interrupted = false;
533b9c1b51eSKate Stone     if (m_multi_line) {
53444d93782SGreg Clayton       StringList lines;
535b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
536b9c1b51eSKate Stone         if (interrupted) {
537e30f11d9SKate Stone           m_done = m_interrupt_exits;
538e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
539e30f11d9SKate Stone 
540b9c1b51eSKate Stone         } else {
54144d93782SGreg Clayton           line = lines.CopyList();
54244d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
54344d93782SGreg Clayton         }
544b9c1b51eSKate Stone       } else {
54544d93782SGreg Clayton         m_done = true;
54644d93782SGreg Clayton       }
547b9c1b51eSKate Stone     } else {
548b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
549e30f11d9SKate Stone         if (interrupted)
550e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
551e30f11d9SKate Stone         else
55244d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
553b9c1b51eSKate Stone       } else {
55444d93782SGreg Clayton         m_done = true;
55544d93782SGreg Clayton       }
55644d93782SGreg Clayton     }
55744d93782SGreg Clayton   }
55844d93782SGreg Clayton }
55944d93782SGreg Clayton 
560b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
561cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
562e68f5d6bSGreg Clayton   if (m_editline_ap)
5634446487dSPavel Labath     m_editline_ap->Cancel();
564cacde7dfSTodd Fiala #endif
565e68f5d6bSGreg Clayton }
566e68f5d6bSGreg Clayton 
567b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
568f0066ad0SGreg Clayton   // Let the delgate handle it first
569f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
570f0066ad0SGreg Clayton     return true;
571f0066ad0SGreg Clayton 
572cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
57344d93782SGreg Clayton   if (m_editline_ap)
574f0066ad0SGreg Clayton     return m_editline_ap->Interrupt();
575cacde7dfSTodd Fiala #endif
576f0066ad0SGreg Clayton   return false;
57744d93782SGreg Clayton }
57844d93782SGreg Clayton 
579b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
580cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
58144d93782SGreg Clayton   if (m_editline_ap)
58244d93782SGreg Clayton     m_editline_ap->Interrupt();
583cacde7dfSTodd Fiala #endif
58444d93782SGreg Clayton }
58544d93782SGreg Clayton 
586b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
5874446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
5884446487dSPavel Labath   if (m_editline_ap)
5894446487dSPavel Labath     m_editline_ap->PrintAsync(stream, s, len);
5904446487dSPavel Labath   else
5914446487dSPavel Labath #endif
592fab31220STed Woodward   {
593fab31220STed Woodward     const char *prompt = GetPrompt();
594fab31220STed Woodward #ifdef _MSC_VER
595b9c1b51eSKate Stone     if (prompt) {
596fab31220STed Woodward       // Back up over previous prompt using Windows API
597fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
598fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
599fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
600fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
601fab31220STed Woodward       coord.X -= strlen(prompt);
602fab31220STed Woodward       if (coord.X < 0)
603fab31220STed Woodward         coord.X = 0;
604fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
605fab31220STed Woodward     }
606fab31220STed Woodward #endif
6074446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
608fab31220STed Woodward     if (prompt)
609b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
610b9c1b51eSKate Stone                             strlen(prompt));
611fab31220STed Woodward   }
6124446487dSPavel Labath }
6134446487dSPavel Labath 
614914b8d98SDeepak Panickal // we may want curses to be disabled for some builds
615914b8d98SDeepak Panickal // for instance, windows
616914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
617914b8d98SDeepak Panickal 
61844d93782SGreg Clayton #define KEY_RETURN 10
61944d93782SGreg Clayton #define KEY_ESCAPE 27
62044d93782SGreg Clayton 
621b9c1b51eSKate Stone namespace curses {
62244d93782SGreg Clayton class Menu;
62344d93782SGreg Clayton class MenuDelegate;
62444d93782SGreg Clayton class Window;
62544d93782SGreg Clayton class WindowDelegate;
62644d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
62744d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
62844d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
62944d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
63044d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
63144d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
63244d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
63344d93782SGreg Clayton 
63444d93782SGreg Clayton #if 0
63544d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
63644d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
63744d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
63844d93782SGreg Clayton #endif
639315b6884SEugene Zelenko 
640b9c1b51eSKate Stone struct Point {
64144d93782SGreg Clayton   int x;
64244d93782SGreg Clayton   int y;
64344d93782SGreg Clayton 
644b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
64544d93782SGreg Clayton 
646b9c1b51eSKate Stone   void Clear() {
64744d93782SGreg Clayton     x = 0;
64844d93782SGreg Clayton     y = 0;
64944d93782SGreg Clayton   }
65044d93782SGreg Clayton 
651b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
65244d93782SGreg Clayton     x += rhs.x;
65344d93782SGreg Clayton     y += rhs.y;
65444d93782SGreg Clayton     return *this;
65544d93782SGreg Clayton   }
65644d93782SGreg Clayton 
657b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
65844d93782SGreg Clayton };
65944d93782SGreg Clayton 
660b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
66144d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
66244d93782SGreg Clayton }
663315b6884SEugene Zelenko 
664b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
66544d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
66644d93782SGreg Clayton }
66744d93782SGreg Clayton 
668b9c1b51eSKate Stone struct Size {
66944d93782SGreg Clayton   int width;
67044d93782SGreg Clayton   int height;
671b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
67244d93782SGreg Clayton 
673b9c1b51eSKate Stone   void Clear() {
67444d93782SGreg Clayton     width = 0;
67544d93782SGreg Clayton     height = 0;
67644d93782SGreg Clayton   }
67744d93782SGreg Clayton 
678b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
67944d93782SGreg Clayton };
68044d93782SGreg Clayton 
681b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
68244d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
68344d93782SGreg Clayton }
684315b6884SEugene Zelenko 
685b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
68644d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
68744d93782SGreg Clayton }
68844d93782SGreg Clayton 
689b9c1b51eSKate Stone struct Rect {
69044d93782SGreg Clayton   Point origin;
69144d93782SGreg Clayton   Size size;
69244d93782SGreg Clayton 
693b9c1b51eSKate Stone   Rect() : origin(), size() {}
69444d93782SGreg Clayton 
695b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
69644d93782SGreg Clayton 
697b9c1b51eSKate Stone   void Clear() {
69844d93782SGreg Clayton     origin.Clear();
69944d93782SGreg Clayton     size.Clear();
70044d93782SGreg Clayton   }
70144d93782SGreg Clayton 
702b9c1b51eSKate Stone   void Dump() {
703b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
704b9c1b51eSKate Stone            size.height);
70544d93782SGreg Clayton   }
70644d93782SGreg Clayton 
707b9c1b51eSKate Stone   void Inset(int w, int h) {
70844d93782SGreg Clayton     if (size.width > w * 2)
70944d93782SGreg Clayton       size.width -= w * 2;
71044d93782SGreg Clayton     origin.x += w;
71144d93782SGreg Clayton 
71244d93782SGreg Clayton     if (size.height > h * 2)
71344d93782SGreg Clayton       size.height -= h * 2;
71444d93782SGreg Clayton     origin.y += h;
71544d93782SGreg Clayton   }
716315b6884SEugene Zelenko 
71744d93782SGreg Clayton   // Return a status bar rectangle which is the last line of
71844d93782SGreg Clayton   // this rectangle. This rectangle will be modified to not
71944d93782SGreg Clayton   // include the status bar area.
720b9c1b51eSKate Stone   Rect MakeStatusBar() {
72144d93782SGreg Clayton     Rect status_bar;
722b9c1b51eSKate Stone     if (size.height > 1) {
72344d93782SGreg Clayton       status_bar.origin.x = origin.x;
72444d93782SGreg Clayton       status_bar.origin.y = size.height;
72544d93782SGreg Clayton       status_bar.size.width = size.width;
72644d93782SGreg Clayton       status_bar.size.height = 1;
72744d93782SGreg Clayton       --size.height;
72844d93782SGreg Clayton     }
72944d93782SGreg Clayton     return status_bar;
73044d93782SGreg Clayton   }
73144d93782SGreg Clayton 
73244d93782SGreg Clayton   // Return a menubar rectangle which is the first line of
73344d93782SGreg Clayton   // this rectangle. This rectangle will be modified to not
73444d93782SGreg Clayton   // include the menubar area.
735b9c1b51eSKate Stone   Rect MakeMenuBar() {
73644d93782SGreg Clayton     Rect menubar;
737b9c1b51eSKate Stone     if (size.height > 1) {
73844d93782SGreg Clayton       menubar.origin.x = origin.x;
73944d93782SGreg Clayton       menubar.origin.y = origin.y;
74044d93782SGreg Clayton       menubar.size.width = size.width;
74144d93782SGreg Clayton       menubar.size.height = 1;
74244d93782SGreg Clayton       ++origin.y;
74344d93782SGreg Clayton       --size.height;
74444d93782SGreg Clayton     }
74544d93782SGreg Clayton     return menubar;
74644d93782SGreg Clayton   }
74744d93782SGreg Clayton 
748b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
749b9c1b51eSKate Stone                                  Rect &bottom) const {
75044d93782SGreg Clayton     float top_height = top_percentage * size.height;
75144d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
75244d93782SGreg Clayton   }
75344d93782SGreg Clayton 
754b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
75544d93782SGreg Clayton     top = *this;
756b9c1b51eSKate Stone     if (top_height < size.height) {
75744d93782SGreg Clayton       top.size.height = top_height;
75844d93782SGreg Clayton       bottom.origin.x = origin.x;
75944d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
76044d93782SGreg Clayton       bottom.size.width = size.width;
76144d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
762b9c1b51eSKate Stone     } else {
76344d93782SGreg Clayton       bottom.Clear();
76444d93782SGreg Clayton     }
76544d93782SGreg Clayton   }
76644d93782SGreg Clayton 
767b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
768b9c1b51eSKate Stone                                Rect &right) const {
76944d93782SGreg Clayton     float left_width = left_percentage * size.width;
77044d93782SGreg Clayton     VerticalSplit(left_width, left, right);
77144d93782SGreg Clayton   }
77244d93782SGreg Clayton 
773b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
77444d93782SGreg Clayton     left = *this;
775b9c1b51eSKate Stone     if (left_width < size.width) {
77644d93782SGreg Clayton       left.size.width = left_width;
77744d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
77844d93782SGreg Clayton       right.origin.y = origin.y;
77944d93782SGreg Clayton       right.size.width = size.width - left.size.width;
78044d93782SGreg Clayton       right.size.height = size.height;
781b9c1b51eSKate Stone     } else {
78244d93782SGreg Clayton       right.Clear();
78344d93782SGreg Clayton     }
78444d93782SGreg Clayton   }
78544d93782SGreg Clayton };
78644d93782SGreg Clayton 
787b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
78844d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
78944d93782SGreg Clayton }
790315b6884SEugene Zelenko 
791b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
79244d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
79344d93782SGreg Clayton }
79444d93782SGreg Clayton 
795b9c1b51eSKate Stone enum HandleCharResult {
79644d93782SGreg Clayton   eKeyNotHandled = 0,
79744d93782SGreg Clayton   eKeyHandled = 1,
79844d93782SGreg Clayton   eQuitApplication = 2
79944d93782SGreg Clayton };
80044d93782SGreg Clayton 
801b9c1b51eSKate Stone enum class MenuActionResult {
80244d93782SGreg Clayton   Handled,
80344d93782SGreg Clayton   NotHandled,
80444d93782SGreg Clayton   Quit // Exit all menus and quit
80544d93782SGreg Clayton };
80644d93782SGreg Clayton 
807b9c1b51eSKate Stone struct KeyHelp {
80844d93782SGreg Clayton   int ch;
80944d93782SGreg Clayton   const char *description;
81044d93782SGreg Clayton };
81144d93782SGreg Clayton 
812b9c1b51eSKate Stone class WindowDelegate {
81344d93782SGreg Clayton public:
814b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
81544d93782SGreg Clayton 
816b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
81744d93782SGreg Clayton     return false; // Drawing not handled
81844d93782SGreg Clayton   }
81944d93782SGreg Clayton 
820b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
82144d93782SGreg Clayton     return eKeyNotHandled;
82244d93782SGreg Clayton   }
82344d93782SGreg Clayton 
824b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
82544d93782SGreg Clayton 
826b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
82744d93782SGreg Clayton };
82844d93782SGreg Clayton 
829b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
83044d93782SGreg Clayton public:
83144d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
83244d93782SGreg Clayton 
833bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
83444d93782SGreg Clayton 
835b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
83644d93782SGreg Clayton 
837b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
83844d93782SGreg Clayton 
839b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
84044d93782SGreg Clayton 
841b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
84244d93782SGreg Clayton 
84344d93782SGreg Clayton protected:
84444d93782SGreg Clayton   StringList m_text;
84544d93782SGreg Clayton   int m_first_visible_line;
84644d93782SGreg Clayton };
84744d93782SGreg Clayton 
848b9c1b51eSKate Stone class Window {
84944d93782SGreg Clayton public:
850b9c1b51eSKate Stone   Window(const char *name)
851b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
852b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
853b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
854b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
85544d93782SGreg Clayton 
856b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
857b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
858b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
859b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
860b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
86144d93782SGreg Clayton     if (w)
86244d93782SGreg Clayton       Reset(w);
86344d93782SGreg Clayton   }
86444d93782SGreg Clayton 
865b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
866b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
867b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
868b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
869b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
870b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
871b9c1b51eSKate Stone                    bounds.origin.y));
87244d93782SGreg Clayton   }
87344d93782SGreg Clayton 
874b9c1b51eSKate Stone   virtual ~Window() {
87544d93782SGreg Clayton     RemoveSubWindows();
87644d93782SGreg Clayton     Reset();
87744d93782SGreg Clayton   }
87844d93782SGreg Clayton 
879b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
88044d93782SGreg Clayton     if (m_window == w)
88144d93782SGreg Clayton       return;
88244d93782SGreg Clayton 
883b9c1b51eSKate Stone     if (m_panel) {
88444d93782SGreg Clayton       ::del_panel(m_panel);
885c5dac77aSEugene Zelenko       m_panel = nullptr;
88644d93782SGreg Clayton     }
887b9c1b51eSKate Stone     if (m_window && m_delete) {
88844d93782SGreg Clayton       ::delwin(m_window);
889c5dac77aSEugene Zelenko       m_window = nullptr;
89044d93782SGreg Clayton       m_delete = false;
89144d93782SGreg Clayton     }
892b9c1b51eSKate Stone     if (w) {
89344d93782SGreg Clayton       m_window = w;
89444d93782SGreg Clayton       m_panel = ::new_panel(m_window);
89544d93782SGreg Clayton       m_delete = del;
89644d93782SGreg Clayton     }
89744d93782SGreg Clayton   }
89844d93782SGreg Clayton 
89944d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
90044d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
901b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
902b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
903b9c1b51eSKate Stone   }
90444d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
90544d93782SGreg Clayton   void Erase() { ::werase(m_window); }
906b9c1b51eSKate Stone   Rect GetBounds() {
907b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
908b9c1b51eSKate Stone   } // Get the rectangle in our parent window
90944d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
91044d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
91144d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
912b9c1b51eSKate Stone   Rect GetFrame() {
913b9c1b51eSKate Stone     return Rect(Point(), GetSize());
914b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
91544d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
91644d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
91744d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
91844d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
91944d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
92044d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
92144d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
92244d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
92344d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
92444d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
92544d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
926b9c1b51eSKate Stone   void Resize(const Size &size) {
927b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
928b9c1b51eSKate Stone   }
92944d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
93044d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
93144d93782SGreg Clayton   void Refresh() { ::wrefresh(m_window); }
932b9c1b51eSKate Stone   void DeferredRefresh() {
93344d93782SGreg Clayton     // We are using panels, so we don't need to call this...
93444d93782SGreg Clayton     //::wnoutrefresh(m_window);
93544d93782SGreg Clayton   }
936b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
937b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
938b9c1b51eSKate Stone   }
93944d93782SGreg Clayton   void UnderlineOn() { AttributeOn(A_UNDERLINE); }
94044d93782SGreg Clayton   void UnderlineOff() { AttributeOff(A_UNDERLINE); }
94144d93782SGreg Clayton 
942b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
94344d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
944b9c1b51eSKate Stone     if (bytes_left > right_pad) {
94544d93782SGreg Clayton       bytes_left -= right_pad;
94644d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
94744d93782SGreg Clayton     }
94844d93782SGreg Clayton   }
94944d93782SGreg Clayton 
950b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
95144d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
952b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
95344d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
95444d93782SGreg Clayton       Size size = GetSize();
955b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
956b9c1b51eSKate Stone                      origin.x),
957b9c1b51eSKate Stone             true);
958b9c1b51eSKate Stone     } else {
95944d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
96044d93782SGreg Clayton     }
96144d93782SGreg Clayton   }
96244d93782SGreg Clayton 
963b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
96444d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
965b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
96644d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
967b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
968b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
969b9c1b51eSKate Stone             true);
970b9c1b51eSKate Stone     } else {
97144d93782SGreg Clayton       if (moving_window)
97244d93782SGreg Clayton         MoveWindow(bounds.origin);
97344d93782SGreg Clayton       Resize(bounds.size);
97444d93782SGreg Clayton     }
97544d93782SGreg Clayton   }
97644d93782SGreg Clayton 
977b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
97844d93782SGreg Clayton     va_list args;
97944d93782SGreg Clayton     va_start(args, format);
98044d93782SGreg Clayton     vwprintw(m_window, format, args);
98144d93782SGreg Clayton     va_end(args);
98244d93782SGreg Clayton   }
98344d93782SGreg Clayton 
984b9c1b51eSKate Stone   void Touch() {
98544d93782SGreg Clayton     ::touchwin(m_window);
98644d93782SGreg Clayton     if (m_parent)
98744d93782SGreg Clayton       m_parent->Touch();
98844d93782SGreg Clayton   }
98944d93782SGreg Clayton 
990b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
991b9c1b51eSKate Stone                            bool make_active) {
99244d93782SGreg Clayton     WindowSP subwindow_sp;
993b9c1b51eSKate Stone     if (m_window) {
994b9c1b51eSKate Stone       subwindow_sp.reset(new Window(
995b9c1b51eSKate Stone           name, ::subwin(m_window, bounds.size.height, bounds.size.width,
996b9c1b51eSKate Stone                          bounds.origin.y, bounds.origin.x),
997b9c1b51eSKate Stone           true));
99844d93782SGreg Clayton       subwindow_sp->m_is_subwin = true;
999b9c1b51eSKate Stone     } else {
1000b9c1b51eSKate Stone       subwindow_sp.reset(
1001b9c1b51eSKate Stone           new Window(name, ::newwin(bounds.size.height, bounds.size.width,
1002b9c1b51eSKate Stone                                     bounds.origin.y, bounds.origin.x),
1003b9c1b51eSKate Stone                      true));
100444d93782SGreg Clayton       subwindow_sp->m_is_subwin = false;
100544d93782SGreg Clayton     }
100644d93782SGreg Clayton     subwindow_sp->m_parent = this;
1007b9c1b51eSKate Stone     if (make_active) {
100844d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
100944d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
101044d93782SGreg Clayton     }
101144d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
101244d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
101344d93782SGreg Clayton     m_needs_update = true;
101444d93782SGreg Clayton     return subwindow_sp;
101544d93782SGreg Clayton   }
101644d93782SGreg Clayton 
1017b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
101844d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
101944d93782SGreg Clayton     size_t i = 0;
1020b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1021b9c1b51eSKate Stone       if ((*pos).get() == window) {
102244d93782SGreg Clayton         if (m_prev_active_window_idx == i)
102344d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1024b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1025b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
102644d93782SGreg Clayton           --m_prev_active_window_idx;
102744d93782SGreg Clayton 
102844d93782SGreg Clayton         if (m_curr_active_window_idx == i)
102944d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1030b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1031b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
103244d93782SGreg Clayton           --m_curr_active_window_idx;
103344d93782SGreg Clayton         window->Erase();
103444d93782SGreg Clayton         m_subwindows.erase(pos);
103544d93782SGreg Clayton         m_needs_update = true;
103644d93782SGreg Clayton         if (m_parent)
103744d93782SGreg Clayton           m_parent->Touch();
103844d93782SGreg Clayton         else
103944d93782SGreg Clayton           ::touchwin(stdscr);
104044d93782SGreg Clayton         return true;
104144d93782SGreg Clayton       }
104244d93782SGreg Clayton     }
104344d93782SGreg Clayton     return false;
104444d93782SGreg Clayton   }
104544d93782SGreg Clayton 
1046b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
104744d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
104844d93782SGreg Clayton     size_t i = 0;
1049b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
105044d93782SGreg Clayton       if ((*pos)->m_name.compare(name) == 0)
105144d93782SGreg Clayton         return *pos;
105244d93782SGreg Clayton     }
105344d93782SGreg Clayton     return WindowSP();
105444d93782SGreg Clayton   }
105544d93782SGreg Clayton 
1056b9c1b51eSKate Stone   void RemoveSubWindows() {
105744d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
105844d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
105944d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1060b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
106144d93782SGreg Clayton       (*pos)->Erase();
106244d93782SGreg Clayton     }
106344d93782SGreg Clayton     if (m_parent)
106444d93782SGreg Clayton       m_parent->Touch();
106544d93782SGreg Clayton     else
106644d93782SGreg Clayton       ::touchwin(stdscr);
106744d93782SGreg Clayton   }
106844d93782SGreg Clayton 
1069b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
107044d93782SGreg Clayton 
1071b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
107244d93782SGreg Clayton 
107344d93782SGreg Clayton   //----------------------------------------------------------------------
107444d93782SGreg Clayton   // Window drawing utilities
107544d93782SGreg Clayton   //----------------------------------------------------------------------
1076b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
107744d93782SGreg Clayton     attr_t attr = 0;
107844d93782SGreg Clayton     if (IsActive())
107944d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
108044d93782SGreg Clayton     else
108144d93782SGreg Clayton       attr = 0;
108244d93782SGreg Clayton     if (attr)
108344d93782SGreg Clayton       AttributeOn(attr);
108444d93782SGreg Clayton 
108544d93782SGreg Clayton     Box();
108644d93782SGreg Clayton     MoveCursor(3, 0);
108744d93782SGreg Clayton 
1088b9c1b51eSKate Stone     if (title && title[0]) {
108944d93782SGreg Clayton       PutChar('<');
109044d93782SGreg Clayton       PutCString(title);
109144d93782SGreg Clayton       PutChar('>');
109244d93782SGreg Clayton     }
109344d93782SGreg Clayton 
1094b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
109544d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
109644d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
109744d93782SGreg Clayton 
1098b9c1b51eSKate Stone       if (x > 0) {
109944d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
110044d93782SGreg Clayton         PutChar('[');
110144d93782SGreg Clayton         PutCString(bottom_message);
110244d93782SGreg Clayton         PutChar(']');
1103b9c1b51eSKate Stone       } else {
110444d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
110544d93782SGreg Clayton         PutChar('[');
110644d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
110744d93782SGreg Clayton       }
110844d93782SGreg Clayton     }
110944d93782SGreg Clayton     if (attr)
111044d93782SGreg Clayton       AttributeOff(attr);
111144d93782SGreg Clayton   }
111244d93782SGreg Clayton 
1113b9c1b51eSKate Stone   virtual void Draw(bool force) {
111444d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
111544d93782SGreg Clayton       return;
111644d93782SGreg Clayton 
111744d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
111844d93782SGreg Clayton       subwindow_sp->Draw(force);
111944d93782SGreg Clayton   }
112044d93782SGreg Clayton 
1121b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1122b9c1b51eSKate Stone     if (m_delegate_sp) {
112344d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
112444d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1125b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1126b9c1b51eSKate Stone         std::auto_ptr<HelpDialogDelegate> help_delegate_ap(
1127b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
112844d93782SGreg Clayton         const size_t num_lines = help_delegate_ap->GetNumLines();
112944d93782SGreg Clayton         const size_t max_length = help_delegate_ap->GetMaxLineLength();
113044d93782SGreg Clayton         Rect bounds = GetBounds();
113144d93782SGreg Clayton         bounds.Inset(1, 1);
1132b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
113344d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
113444d93782SGreg Clayton           bounds.size.width = max_length + 4;
1135b9c1b51eSKate Stone         } else {
1136b9c1b51eSKate Stone           if (bounds.size.width > 100) {
113744d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
113844d93782SGreg Clayton             bounds.origin.x += inset_w;
113944d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
114044d93782SGreg Clayton           }
114144d93782SGreg Clayton         }
114244d93782SGreg Clayton 
1143b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
114444d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
114544d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1146b9c1b51eSKate Stone         } else {
1147b9c1b51eSKate Stone           if (bounds.size.height > 100) {
114844d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
114944d93782SGreg Clayton             bounds.origin.y += inset_h;
115044d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
115144d93782SGreg Clayton           }
115244d93782SGreg Clayton         }
11535fdb09bbSGreg Clayton         WindowSP help_window_sp;
11545fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11555fdb09bbSGreg Clayton         if (parent_window)
11565fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11575fdb09bbSGreg Clayton         else
11585fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1159b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1160b9c1b51eSKate Stone             WindowDelegateSP(help_delegate_ap.release()));
116144d93782SGreg Clayton         return true;
116244d93782SGreg Clayton       }
116344d93782SGreg Clayton     }
116444d93782SGreg Clayton     return false;
116544d93782SGreg Clayton   }
116644d93782SGreg Clayton 
1167b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
116844d93782SGreg Clayton     // Always check the active window first
116944d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
117044d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1171b9c1b51eSKate Stone     if (active_window_sp) {
117244d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
117344d93782SGreg Clayton       if (result != eKeyNotHandled)
117444d93782SGreg Clayton         return result;
117544d93782SGreg Clayton     }
117644d93782SGreg Clayton 
1177b9c1b51eSKate Stone     if (m_delegate_sp) {
117844d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
117944d93782SGreg Clayton       if (result != eKeyNotHandled)
118044d93782SGreg Clayton         return result;
118144d93782SGreg Clayton     }
118244d93782SGreg Clayton 
118344d93782SGreg Clayton     // Then check for any windows that want any keys
118444d93782SGreg Clayton     // that weren't handled. This is typically only
118544d93782SGreg Clayton     // for a menubar.
118644d93782SGreg Clayton     // Make a copy of the subwindows in case any HandleChar()
118744d93782SGreg Clayton     // functions muck with the subwindows. If we don't do this,
118844d93782SGreg Clayton     // we can crash when iterating over the subwindows.
118944d93782SGreg Clayton     Windows subwindows(m_subwindows);
1190b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1191b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
119244d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
119344d93782SGreg Clayton         if (result != eKeyNotHandled)
119444d93782SGreg Clayton           return result;
119544d93782SGreg Clayton       }
119644d93782SGreg Clayton     }
119744d93782SGreg Clayton 
119844d93782SGreg Clayton     return eKeyNotHandled;
119944d93782SGreg Clayton   }
120044d93782SGreg Clayton 
1201b9c1b51eSKate Stone   bool SetActiveWindow(Window *window) {
120244d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1203b9c1b51eSKate Stone     for (size_t i = 0; i < num_subwindows; ++i) {
1204b9c1b51eSKate Stone       if (m_subwindows[i].get() == window) {
120544d93782SGreg Clayton         m_prev_active_window_idx = m_curr_active_window_idx;
120644d93782SGreg Clayton         ::top_panel(window->m_panel);
120744d93782SGreg Clayton         m_curr_active_window_idx = i;
120844d93782SGreg Clayton         return true;
120944d93782SGreg Clayton       }
121044d93782SGreg Clayton     }
121144d93782SGreg Clayton     return false;
121244d93782SGreg Clayton   }
121344d93782SGreg Clayton 
1214b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1215b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1216b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1217b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
121844d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
121944d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1220b9c1b51eSKate Stone         } else if (IsActive()) {
122144d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
122244d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
122344d93782SGreg Clayton 
122444d93782SGreg Clayton           // Find first window that wants to be active if this window is active
122544d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1226b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1227b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
122844d93782SGreg Clayton               m_curr_active_window_idx = i;
122944d93782SGreg Clayton               break;
123044d93782SGreg Clayton             }
123144d93782SGreg Clayton           }
123244d93782SGreg Clayton         }
123344d93782SGreg Clayton       }
123444d93782SGreg Clayton 
123544d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
123644d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
123744d93782SGreg Clayton     }
123844d93782SGreg Clayton     return WindowSP();
123944d93782SGreg Clayton   }
124044d93782SGreg Clayton 
1241b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
124244d93782SGreg Clayton 
1243b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
124444d93782SGreg Clayton 
1245b9c1b51eSKate Stone   const WindowDelegateSP &GetDelegate() const { return m_delegate_sp; }
124644d93782SGreg Clayton 
1247b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
124844d93782SGreg Clayton     m_delegate_sp = delegate_sp;
124944d93782SGreg Clayton   }
125044d93782SGreg Clayton 
1251b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
125244d93782SGreg Clayton 
1253b9c1b51eSKate Stone   bool IsActive() const {
125444d93782SGreg Clayton     if (m_parent)
125544d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
125644d93782SGreg Clayton     else
125744d93782SGreg Clayton       return true; // Top level window is always active
125844d93782SGreg Clayton   }
125944d93782SGreg Clayton 
1260b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
126144d93782SGreg Clayton     // Move active focus to next window
126244d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1263b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
126444d93782SGreg Clayton       uint32_t idx = 0;
1265b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1266b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
126744d93782SGreg Clayton           m_curr_active_window_idx = idx;
126844d93782SGreg Clayton           break;
126944d93782SGreg Clayton         }
127044d93782SGreg Clayton         ++idx;
127144d93782SGreg Clayton       }
1272b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
127344d93782SGreg Clayton       bool handled = false;
127444d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1275b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1276b9c1b51eSKate Stone            ++idx) {
1277b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
127844d93782SGreg Clayton           m_curr_active_window_idx = idx;
127944d93782SGreg Clayton           handled = true;
128044d93782SGreg Clayton           break;
128144d93782SGreg Clayton         }
128244d93782SGreg Clayton       }
1283b9c1b51eSKate Stone       if (!handled) {
1284b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1285b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
128644d93782SGreg Clayton             m_curr_active_window_idx = idx;
128744d93782SGreg Clayton             break;
128844d93782SGreg Clayton           }
128944d93782SGreg Clayton         }
129044d93782SGreg Clayton       }
1291b9c1b51eSKate Stone     } else {
129244d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1293b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1294b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
129544d93782SGreg Clayton           m_curr_active_window_idx = idx;
129644d93782SGreg Clayton           break;
129744d93782SGreg Clayton         }
129844d93782SGreg Clayton       }
129944d93782SGreg Clayton     }
130044d93782SGreg Clayton   }
130144d93782SGreg Clayton 
1302b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1303315b6884SEugene Zelenko 
130444d93782SGreg Clayton protected:
130544d93782SGreg Clayton   std::string m_name;
130644d93782SGreg Clayton   WINDOW *m_window;
130744d93782SGreg Clayton   PANEL *m_panel;
130844d93782SGreg Clayton   Window *m_parent;
130944d93782SGreg Clayton   Windows m_subwindows;
131044d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
131144d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
131244d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
131344d93782SGreg Clayton   bool m_delete;
131444d93782SGreg Clayton   bool m_needs_update;
131544d93782SGreg Clayton   bool m_can_activate;
131644d93782SGreg Clayton   bool m_is_subwin;
131744d93782SGreg Clayton 
131844d93782SGreg Clayton private:
131944d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
132044d93782SGreg Clayton };
132144d93782SGreg Clayton 
1322b9c1b51eSKate Stone class MenuDelegate {
132344d93782SGreg Clayton public:
1324315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
132544d93782SGreg Clayton 
1326b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
132744d93782SGreg Clayton };
132844d93782SGreg Clayton 
1329b9c1b51eSKate Stone class Menu : public WindowDelegate {
133044d93782SGreg Clayton public:
1331b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
133244d93782SGreg Clayton 
133344d93782SGreg Clayton   // Menubar or separator constructor
133444d93782SGreg Clayton   Menu(Type type);
133544d93782SGreg Clayton 
133644d93782SGreg Clayton   // Menuitem constructor
1337b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
133844d93782SGreg Clayton        uint64_t identifier);
133944d93782SGreg Clayton 
1340315b6884SEugene Zelenko   ~Menu() override = default;
134144d93782SGreg Clayton 
1342b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
134344d93782SGreg Clayton 
1344b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
134544d93782SGreg Clayton     m_delegate_sp = delegate_sp;
134644d93782SGreg Clayton   }
134744d93782SGreg Clayton 
1348b9c1b51eSKate Stone   void RecalculateNameLengths();
134944d93782SGreg Clayton 
1350b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
135144d93782SGreg Clayton 
1352b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
135344d93782SGreg Clayton 
1354b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
135544d93782SGreg Clayton 
1356b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
135744d93782SGreg Clayton 
1358b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
135944d93782SGreg Clayton 
1360b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
136144d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1362b9c1b51eSKate Stone     if (m_delegate_sp) {
136344d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
136444d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
136544d93782SGreg Clayton         return result;
1366b9c1b51eSKate Stone     } else if (m_parent) {
136744d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
136844d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
136944d93782SGreg Clayton         return result;
137044d93782SGreg Clayton     }
137144d93782SGreg Clayton     return m_canned_result;
137244d93782SGreg Clayton   }
137344d93782SGreg Clayton 
1374b9c1b51eSKate Stone   MenuActionResult Action() {
137544d93782SGreg Clayton     // Call the recursive action so it can try to handle it
137644d93782SGreg Clayton     // with the menu delegate, and if not, try our parent menu
137744d93782SGreg Clayton     return ActionPrivate(*this);
137844d93782SGreg Clayton   }
137944d93782SGreg Clayton 
1380b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
138144d93782SGreg Clayton 
1382b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
138344d93782SGreg Clayton 
1384b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
138544d93782SGreg Clayton 
1386b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
138744d93782SGreg Clayton 
1388b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
138944d93782SGreg Clayton 
1390b9c1b51eSKate Stone   Type GetType() const { return m_type; }
139144d93782SGreg Clayton 
1392b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
139344d93782SGreg Clayton 
1394b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
139544d93782SGreg Clayton 
1396b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
139744d93782SGreg Clayton 
1398b9c1b51eSKate Stone   void SetKeyValue(int key_value) { m_key_value = key_value; }
139944d93782SGreg Clayton 
1400b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
140144d93782SGreg Clayton 
1402b9c1b51eSKate Stone   std::string &GetKeyName() { return m_key_name; }
140344d93782SGreg Clayton 
1404b9c1b51eSKate Stone   int GetDrawWidth() const {
140544d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
140644d93782SGreg Clayton   }
140744d93782SGreg Clayton 
1408b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
140944d93782SGreg Clayton 
1410b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
141144d93782SGreg Clayton 
141244d93782SGreg Clayton protected:
141344d93782SGreg Clayton   std::string m_name;
141444d93782SGreg Clayton   std::string m_key_name;
141544d93782SGreg Clayton   uint64_t m_identifier;
141644d93782SGreg Clayton   Type m_type;
141744d93782SGreg Clayton   int m_key_value;
141844d93782SGreg Clayton   int m_start_col;
141944d93782SGreg Clayton   int m_max_submenu_name_length;
142044d93782SGreg Clayton   int m_max_submenu_key_name_length;
142144d93782SGreg Clayton   int m_selected;
142244d93782SGreg Clayton   Menu *m_parent;
142344d93782SGreg Clayton   Menus m_submenus;
142444d93782SGreg Clayton   WindowSP m_menu_window_sp;
142544d93782SGreg Clayton   MenuActionResult m_canned_result;
142644d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
142744d93782SGreg Clayton };
142844d93782SGreg Clayton 
142944d93782SGreg Clayton // Menubar or separator constructor
1430b9c1b51eSKate Stone Menu::Menu(Type type)
1431b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1432b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1433b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1434b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1435b9c1b51eSKate Stone       m_delegate_sp() {}
143644d93782SGreg Clayton 
143744d93782SGreg Clayton // Menuitem constructor
1438b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1439b9c1b51eSKate Stone            uint64_t identifier)
1440b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1441b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1442b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1443b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1444b9c1b51eSKate Stone       m_delegate_sp() {
1445b9c1b51eSKate Stone   if (name && name[0]) {
144644d93782SGreg Clayton     m_name = name;
144744d93782SGreg Clayton     m_type = Type::Item;
144844d93782SGreg Clayton     if (key_name && key_name[0])
144944d93782SGreg Clayton       m_key_name = key_name;
1450b9c1b51eSKate Stone   } else {
145144d93782SGreg Clayton     m_type = Type::Separator;
145244d93782SGreg Clayton   }
145344d93782SGreg Clayton }
145444d93782SGreg Clayton 
1455b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
145644d93782SGreg Clayton   m_max_submenu_name_length = 0;
145744d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
145844d93782SGreg Clayton   Menus &submenus = GetSubmenus();
145944d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1460b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
146144d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14623985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
146344d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1464b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1465b9c1b51eSKate Stone         submenu->m_key_name.size())
146644d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
146744d93782SGreg Clayton   }
146844d93782SGreg Clayton }
146944d93782SGreg Clayton 
1470b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
147144d93782SGreg Clayton   menu_sp->m_parent = this;
14723985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
147344d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1474b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1475b9c1b51eSKate Stone       menu_sp->m_key_name.size())
147644d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
147744d93782SGreg Clayton   m_submenus.push_back(menu_sp);
147844d93782SGreg Clayton }
147944d93782SGreg Clayton 
1480b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1481b9c1b51eSKate Stone   if (m_type == Type::Separator) {
148244d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
148344d93782SGreg Clayton     window.PutChar(ACS_LTEE);
148444d93782SGreg Clayton     int width = window.GetWidth();
1485b9c1b51eSKate Stone     if (width > 2) {
148644d93782SGreg Clayton       width -= 2;
14873985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
148844d93782SGreg Clayton         window.PutChar(ACS_HLINE);
148944d93782SGreg Clayton     }
149044d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1491b9c1b51eSKate Stone   } else {
149244d93782SGreg Clayton     const int shortcut_key = m_key_value;
149344d93782SGreg Clayton     bool underlined_shortcut = false;
149444d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
149544d93782SGreg Clayton     if (highlight)
149644d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1497b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
149844d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
149944d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
150044d93782SGreg Clayton       const char *name = m_name.c_str();
150144d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1502b9c1b51eSKate Stone       if (pos != std::string::npos) {
150344d93782SGreg Clayton         underlined_shortcut = true;
1504b9c1b51eSKate Stone         if (pos > 0) {
150544d93782SGreg Clayton           window.PutCString(name, pos);
150644d93782SGreg Clayton           name += pos;
150744d93782SGreg Clayton         }
150844d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
150944d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
151044d93782SGreg Clayton         window.PutChar(name[0]);
151144d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
151244d93782SGreg Clayton         name++;
151344d93782SGreg Clayton         if (name[0])
151444d93782SGreg Clayton           window.PutCString(name);
151544d93782SGreg Clayton       }
151644d93782SGreg Clayton     }
151744d93782SGreg Clayton 
1518b9c1b51eSKate Stone     if (!underlined_shortcut) {
151944d93782SGreg Clayton       window.PutCString(m_name.c_str());
152044d93782SGreg Clayton     }
152144d93782SGreg Clayton 
152244d93782SGreg Clayton     if (highlight)
152344d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
152444d93782SGreg Clayton 
1525b9c1b51eSKate Stone     if (m_key_name.empty()) {
1526b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
152744d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
152844d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
152944d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
153044d93782SGreg Clayton       }
1531b9c1b51eSKate Stone     } else {
153244d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
153344d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
153444d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
153544d93782SGreg Clayton     }
153644d93782SGreg Clayton   }
153744d93782SGreg Clayton }
153844d93782SGreg Clayton 
1539b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
154044d93782SGreg Clayton   Menus &submenus = GetSubmenus();
154144d93782SGreg Clayton   const size_t num_submenus = submenus.size();
154244d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
154344d93782SGreg Clayton   Menu::Type menu_type = GetType();
1544b9c1b51eSKate Stone   switch (menu_type) {
1545b9c1b51eSKate Stone   case Menu::Type::Bar: {
154644d93782SGreg Clayton     window.SetBackground(2);
154744d93782SGreg Clayton     window.MoveCursor(0, 0);
1548b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
154944d93782SGreg Clayton       Menu *menu = submenus[i].get();
155044d93782SGreg Clayton       if (i > 0)
155144d93782SGreg Clayton         window.PutChar(' ');
155244d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
155344d93782SGreg Clayton       window.PutCString("| ");
155444d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
155544d93782SGreg Clayton     }
155644d93782SGreg Clayton     window.PutCString(" |");
155744d93782SGreg Clayton     window.DeferredRefresh();
1558b9c1b51eSKate Stone   } break;
155944d93782SGreg Clayton 
1560b9c1b51eSKate Stone   case Menu::Type::Item: {
156144d93782SGreg Clayton     int y = 1;
156244d93782SGreg Clayton     int x = 3;
156344d93782SGreg Clayton     // Draw the menu
156444d93782SGreg Clayton     int cursor_x = 0;
156544d93782SGreg Clayton     int cursor_y = 0;
156644d93782SGreg Clayton     window.Erase();
156744d93782SGreg Clayton     window.SetBackground(2);
156844d93782SGreg Clayton     window.Box();
1569b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1570b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
157144d93782SGreg Clayton       window.MoveCursor(x, y + i);
1572b9c1b51eSKate Stone       if (is_selected) {
157344d93782SGreg Clayton         // Remember where we want the cursor to be
157444d93782SGreg Clayton         cursor_x = x - 1;
157544d93782SGreg Clayton         cursor_y = y + i;
157644d93782SGreg Clayton       }
157744d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
157844d93782SGreg Clayton     }
157944d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
158044d93782SGreg Clayton     window.DeferredRefresh();
1581b9c1b51eSKate Stone   } break;
158244d93782SGreg Clayton 
158344d93782SGreg Clayton   default:
158444d93782SGreg Clayton   case Menu::Type::Separator:
158544d93782SGreg Clayton     break;
158644d93782SGreg Clayton   }
158744d93782SGreg Clayton   return true; // Drawing handled...
158844d93782SGreg Clayton }
158944d93782SGreg Clayton 
1590b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
159144d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
159244d93782SGreg Clayton 
159344d93782SGreg Clayton   Menus &submenus = GetSubmenus();
159444d93782SGreg Clayton   const size_t num_submenus = submenus.size();
159544d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
159644d93782SGreg Clayton   Menu::Type menu_type = GetType();
1597b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
159844d93782SGreg Clayton     MenuSP run_menu_sp;
1599b9c1b51eSKate Stone     switch (key) {
160044d93782SGreg Clayton     case KEY_DOWN:
160144d93782SGreg Clayton     case KEY_UP:
160244d93782SGreg Clayton       // Show last menu or first menu
16033985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
160444d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
160544d93782SGreg Clayton       else if (!submenus.empty())
160644d93782SGreg Clayton         run_menu_sp = submenus.front();
160744d93782SGreg Clayton       result = eKeyHandled;
160844d93782SGreg Clayton       break;
160944d93782SGreg Clayton 
161044d93782SGreg Clayton     case KEY_RIGHT:
161144d93782SGreg Clayton       ++m_selected;
16123985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
161344d93782SGreg Clayton         m_selected = 0;
16143985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
161544d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
161644d93782SGreg Clayton       else if (!submenus.empty())
161744d93782SGreg Clayton         run_menu_sp = submenus.front();
161844d93782SGreg Clayton       result = eKeyHandled;
161944d93782SGreg Clayton       break;
162044d93782SGreg Clayton 
162144d93782SGreg Clayton     case KEY_LEFT:
162244d93782SGreg Clayton       --m_selected;
162344d93782SGreg Clayton       if (m_selected < 0)
162444d93782SGreg Clayton         m_selected = num_submenus - 1;
16253985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
162644d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
162744d93782SGreg Clayton       else if (!submenus.empty())
162844d93782SGreg Clayton         run_menu_sp = submenus.front();
162944d93782SGreg Clayton       result = eKeyHandled;
163044d93782SGreg Clayton       break;
163144d93782SGreg Clayton 
163244d93782SGreg Clayton     default:
1633b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1634b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
163544d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
163644d93782SGreg Clayton           run_menu_sp = submenus[i];
163744d93782SGreg Clayton           result = eKeyHandled;
163844d93782SGreg Clayton           break;
163944d93782SGreg Clayton         }
164044d93782SGreg Clayton       }
164144d93782SGreg Clayton       break;
164244d93782SGreg Clayton     }
164344d93782SGreg Clayton 
1644b9c1b51eSKate Stone     if (run_menu_sp) {
164544d93782SGreg Clayton       // Run the action on this menu in case we need to populate the
164644d93782SGreg Clayton       // menu with dynamic content and also in case check marks, and
164785025451SKamil Rytarowski       // any other menu decorations need to be calculated
164844d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
164944d93782SGreg Clayton         return eQuitApplication;
165044d93782SGreg Clayton 
165144d93782SGreg Clayton       Rect menu_bounds;
165244d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
165344d93782SGreg Clayton       menu_bounds.origin.y = 1;
165444d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
165544d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
165644d93782SGreg Clayton       if (m_menu_window_sp)
165744d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
165844d93782SGreg Clayton 
1659b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1660b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
166144d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
166244d93782SGreg Clayton     }
1663b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1664b9c1b51eSKate Stone     switch (key) {
166544d93782SGreg Clayton     case KEY_DOWN:
1666b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
166744d93782SGreg Clayton         const int start_select = m_selected;
1668b9c1b51eSKate Stone         while (++m_selected != start_select) {
16693985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
167044d93782SGreg Clayton             m_selected = 0;
167144d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
167244d93782SGreg Clayton             continue;
167344d93782SGreg Clayton           else
167444d93782SGreg Clayton             break;
167544d93782SGreg Clayton         }
167644d93782SGreg Clayton         return eKeyHandled;
167744d93782SGreg Clayton       }
167844d93782SGreg Clayton       break;
167944d93782SGreg Clayton 
168044d93782SGreg Clayton     case KEY_UP:
1681b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
168244d93782SGreg Clayton         const int start_select = m_selected;
1683b9c1b51eSKate Stone         while (--m_selected != start_select) {
16843985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
168544d93782SGreg Clayton             m_selected = num_submenus - 1;
168644d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
168744d93782SGreg Clayton             continue;
168844d93782SGreg Clayton           else
168944d93782SGreg Clayton             break;
169044d93782SGreg Clayton         }
169144d93782SGreg Clayton         return eKeyHandled;
169244d93782SGreg Clayton       }
169344d93782SGreg Clayton       break;
169444d93782SGreg Clayton 
169544d93782SGreg Clayton     case KEY_RETURN:
1696b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
169744d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
169844d93782SGreg Clayton           return eQuitApplication;
169944d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
170044d93782SGreg Clayton         return eKeyHandled;
170144d93782SGreg Clayton       }
170244d93782SGreg Clayton       break;
170344d93782SGreg Clayton 
1704b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1705b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
170644d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
170744d93782SGreg Clayton       return eKeyHandled;
170844d93782SGreg Clayton 
170944d93782SGreg Clayton     default:
1710b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
171144d93782SGreg Clayton         Menu *menu = submenus[i].get();
1712b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
171344d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
171444d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
171544d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
171644d93782SGreg Clayton             return eQuitApplication;
171744d93782SGreg Clayton           return eKeyHandled;
171844d93782SGreg Clayton         }
171944d93782SGreg Clayton       }
172044d93782SGreg Clayton       break;
172144d93782SGreg Clayton     }
1722b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
172344d93782SGreg Clayton   }
172444d93782SGreg Clayton   return result;
172544d93782SGreg Clayton }
172644d93782SGreg Clayton 
1727b9c1b51eSKate Stone class Application {
172844d93782SGreg Clayton public:
1729b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1730b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
173144d93782SGreg Clayton 
1732b9c1b51eSKate Stone   ~Application() {
173344d93782SGreg Clayton     m_window_delegates.clear();
173444d93782SGreg Clayton     m_window_sp.reset();
1735b9c1b51eSKate Stone     if (m_screen) {
173644d93782SGreg Clayton       ::delscreen(m_screen);
1737c5dac77aSEugene Zelenko       m_screen = nullptr;
173844d93782SGreg Clayton     }
173944d93782SGreg Clayton   }
174044d93782SGreg Clayton 
1741b9c1b51eSKate Stone   void Initialize() {
174244d93782SGreg Clayton     ::setlocale(LC_ALL, "");
174344d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
174444d93782SGreg Clayton #if 0
174544d93782SGreg Clayton             ::initscr();
174644d93782SGreg Clayton #else
1747c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
174844d93782SGreg Clayton #endif
174944d93782SGreg Clayton     ::start_color();
175044d93782SGreg Clayton     ::curs_set(0);
175144d93782SGreg Clayton     ::noecho();
175244d93782SGreg Clayton     ::keypad(stdscr, TRUE);
175344d93782SGreg Clayton   }
175444d93782SGreg Clayton 
1755b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
175644d93782SGreg Clayton 
1757b9c1b51eSKate Stone   void Run(Debugger &debugger) {
175844d93782SGreg Clayton     bool done = false;
175944d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
176044d93782SGreg Clayton 
176144d93782SGreg Clayton     // Alas the threading model in curses is a bit lame so we need to
176244d93782SGreg Clayton     // resort to polling every 0.5 seconds. We could poll for stdin
176344d93782SGreg Clayton     // ourselves and then pass the keys down but then we need to
176444d93782SGreg Clayton     // translate all of the escape sequences ourselves. So we resort to
176544d93782SGreg Clayton     // polling for input because we need to receive async process events
176644d93782SGreg Clayton     // while in this loop.
176744d93782SGreg Clayton 
1768b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1769b9c1b51eSKate Stone                                             // of seconds seconds when calling
1770b9c1b51eSKate Stone                                             // Window::GetChar()
177144d93782SGreg Clayton 
1772b9c1b51eSKate Stone     ListenerSP listener_sp(
1773b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
177444d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
177544d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
177644d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
177744d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
177844d93782SGreg Clayton 
177944d93782SGreg Clayton     bool update = true;
178044d93782SGreg Clayton #if defined(__APPLE__)
178144d93782SGreg Clayton     std::deque<int> escape_chars;
178244d93782SGreg Clayton #endif
178344d93782SGreg Clayton 
1784b9c1b51eSKate Stone     while (!done) {
1785b9c1b51eSKate Stone       if (update) {
178644d93782SGreg Clayton         m_window_sp->Draw(false);
178744d93782SGreg Clayton         // All windows should be calling Window::DeferredRefresh() instead
178844d93782SGreg Clayton         // of Window::Refresh() so we can do a single update and avoid
178944d93782SGreg Clayton         // any screen blinking
179044d93782SGreg Clayton         update_panels();
179144d93782SGreg Clayton 
1792b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1793b9c1b51eSKate Stone         // corner
179444d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
179544d93782SGreg Clayton 
179644d93782SGreg Clayton         doupdate();
179744d93782SGreg Clayton         update = false;
179844d93782SGreg Clayton       }
179944d93782SGreg Clayton 
180044d93782SGreg Clayton #if defined(__APPLE__)
180144d93782SGreg Clayton       // Terminal.app doesn't map its function keys correctly, F1-F4 default to:
1802b9c1b51eSKate Stone       // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1803b9c1b51eSKate Stone       // possible
180444d93782SGreg Clayton       int ch;
180544d93782SGreg Clayton       if (escape_chars.empty())
180644d93782SGreg Clayton         ch = m_window_sp->GetChar();
1807b9c1b51eSKate Stone       else {
180844d93782SGreg Clayton         ch = escape_chars.front();
180944d93782SGreg Clayton         escape_chars.pop_front();
181044d93782SGreg Clayton       }
1811b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
181244d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1813b9c1b51eSKate Stone         if (ch2 == 'O') {
181444d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1815b9c1b51eSKate Stone           switch (ch3) {
1816b9c1b51eSKate Stone           case 'P':
1817b9c1b51eSKate Stone             ch = KEY_F(1);
1818b9c1b51eSKate Stone             break;
1819b9c1b51eSKate Stone           case 'Q':
1820b9c1b51eSKate Stone             ch = KEY_F(2);
1821b9c1b51eSKate Stone             break;
1822b9c1b51eSKate Stone           case 'R':
1823b9c1b51eSKate Stone             ch = KEY_F(3);
1824b9c1b51eSKate Stone             break;
1825b9c1b51eSKate Stone           case 'S':
1826b9c1b51eSKate Stone             ch = KEY_F(4);
1827b9c1b51eSKate Stone             break;
182844d93782SGreg Clayton           default:
182944d93782SGreg Clayton             escape_chars.push_back(ch2);
183044d93782SGreg Clayton             if (ch3 != -1)
183144d93782SGreg Clayton               escape_chars.push_back(ch3);
183244d93782SGreg Clayton             break;
183344d93782SGreg Clayton           }
1834b9c1b51eSKate Stone         } else if (ch2 != -1)
183544d93782SGreg Clayton           escape_chars.push_back(ch2);
183644d93782SGreg Clayton       }
183744d93782SGreg Clayton #else
183844d93782SGreg Clayton       int ch = m_window_sp->GetChar();
183944d93782SGreg Clayton 
184044d93782SGreg Clayton #endif
1841b9c1b51eSKate Stone       if (ch == -1) {
1842b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
184344d93782SGreg Clayton           done = true;
1844b9c1b51eSKate Stone         } else {
184544d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
184644d93782SGreg Clayton           EventSP event_sp;
1847b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
184844d93782SGreg Clayton             listener_sp->GetNextEvent(event_sp);
184944d93782SGreg Clayton 
1850b9c1b51eSKate Stone             if (event_sp) {
185144d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1852b9c1b51eSKate Stone               if (broadcaster) {
185344d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1854b9c1b51eSKate Stone                 ConstString broadcaster_class(
1855b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1856b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1857b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1858b9c1b51eSKate Stone                       nullptr);
185944d93782SGreg Clayton                   update = true;
186044d93782SGreg Clayton                   continue; // Don't get any key, just update our view
186144d93782SGreg Clayton                 }
186244d93782SGreg Clayton               }
186344d93782SGreg Clayton             }
186444d93782SGreg Clayton           }
186544d93782SGreg Clayton         }
1866b9c1b51eSKate Stone       } else {
186744d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1868b9c1b51eSKate Stone         switch (key_result) {
186944d93782SGreg Clayton         case eKeyHandled:
1870c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
187144d93782SGreg Clayton           update = true;
187244d93782SGreg Clayton           break;
187344d93782SGreg Clayton         case eKeyNotHandled:
187444d93782SGreg Clayton           break;
187544d93782SGreg Clayton         case eQuitApplication:
187644d93782SGreg Clayton           done = true;
187744d93782SGreg Clayton           break;
187844d93782SGreg Clayton         }
187944d93782SGreg Clayton       }
188044d93782SGreg Clayton     }
188144d93782SGreg Clayton 
188244d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
188344d93782SGreg Clayton   }
188444d93782SGreg Clayton 
1885b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
188644d93782SGreg Clayton     if (!m_window_sp)
188744d93782SGreg Clayton       m_window_sp.reset(new Window("main", stdscr, false));
188844d93782SGreg Clayton     return m_window_sp;
188944d93782SGreg Clayton   }
189044d93782SGreg Clayton 
1891b9c1b51eSKate Stone   WindowDelegates &GetWindowDelegates() { return m_window_delegates; }
189244d93782SGreg Clayton 
189344d93782SGreg Clayton protected:
189444d93782SGreg Clayton   WindowSP m_window_sp;
189544d93782SGreg Clayton   WindowDelegates m_window_delegates;
189644d93782SGreg Clayton   SCREEN *m_screen;
189744d93782SGreg Clayton   FILE *m_in;
189844d93782SGreg Clayton   FILE *m_out;
189944d93782SGreg Clayton };
190044d93782SGreg Clayton 
190144d93782SGreg Clayton } // namespace curses
190244d93782SGreg Clayton 
190344d93782SGreg Clayton using namespace curses;
190444d93782SGreg Clayton 
1905b9c1b51eSKate Stone struct Row {
190644d93782SGreg Clayton   ValueObjectSP valobj;
190744d93782SGreg Clayton   Row *parent;
190844d93782SGreg Clayton   int row_idx;
190944d93782SGreg Clayton   int x;
191044d93782SGreg Clayton   int y;
191144d93782SGreg Clayton   bool might_have_children;
191244d93782SGreg Clayton   bool expanded;
191344d93782SGreg Clayton   bool calculated_children;
191444d93782SGreg Clayton   std::vector<Row> children;
191544d93782SGreg Clayton 
1916b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
1917b9c1b51eSKate Stone       : valobj(v), parent(p), row_idx(0), x(1), y(1),
191844d93782SGreg Clayton         might_have_children(v ? v->MightHaveChildren() : false),
1919b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
192044d93782SGreg Clayton 
1921b9c1b51eSKate Stone   size_t GetDepth() const {
192244d93782SGreg Clayton     if (parent)
192344d93782SGreg Clayton       return 1 + parent->GetDepth();
192444d93782SGreg Clayton     return 0;
192544d93782SGreg Clayton   }
192644d93782SGreg Clayton 
1927b9c1b51eSKate Stone   void Expand() {
192844d93782SGreg Clayton     expanded = true;
1929b9c1b51eSKate Stone     if (!calculated_children) {
193044d93782SGreg Clayton       calculated_children = true;
1931b9c1b51eSKate Stone       if (valobj) {
193244d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1933b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
193444d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
193544d93782SGreg Clayton         }
193644d93782SGreg Clayton       }
193744d93782SGreg Clayton     }
193844d93782SGreg Clayton   }
193944d93782SGreg Clayton 
1940b9c1b51eSKate Stone   void Unexpand() { expanded = false; }
194144d93782SGreg Clayton 
1942b9c1b51eSKate Stone   void DrawTree(Window &window) {
194344d93782SGreg Clayton     if (parent)
194444d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
194544d93782SGreg Clayton 
1946b9c1b51eSKate Stone     if (might_have_children) {
194744d93782SGreg Clayton       // It we can get UTF8 characters to work we should try to use the "symbol"
194844d93782SGreg Clayton       // UTF8 string below
194944d93782SGreg Clayton       //            const char *symbol = "";
195044d93782SGreg Clayton       //            if (row.expanded)
195144d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
195244d93782SGreg Clayton       //            else
195344d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
195444d93782SGreg Clayton       //            window.PutCString (symbol);
195544d93782SGreg Clayton 
195644d93782SGreg Clayton       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
195744d93782SGreg Clayton       // 'v' or '>' character...
195844d93782SGreg Clayton       //            if (expanded)
195944d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
196044d93782SGreg Clayton       //            else
196144d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
196244d93782SGreg Clayton       // Since we can't find any good looking right arrow/down arrow
196344d93782SGreg Clayton       // symbols, just use a diamond...
196444d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
196544d93782SGreg Clayton       window.PutChar(ACS_HLINE);
196644d93782SGreg Clayton     }
196744d93782SGreg Clayton   }
196844d93782SGreg Clayton 
1969b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
197044d93782SGreg Clayton     if (parent)
197144d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
197244d93782SGreg Clayton 
1973b9c1b51eSKate Stone     if (&children.back() == child) {
197444d93782SGreg Clayton       // Last child
1975b9c1b51eSKate Stone       if (reverse_depth == 0) {
197644d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
197744d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1978b9c1b51eSKate Stone       } else {
197944d93782SGreg Clayton         window.PutChar(' ');
198044d93782SGreg Clayton         window.PutChar(' ');
198144d93782SGreg Clayton       }
1982b9c1b51eSKate Stone     } else {
1983b9c1b51eSKate Stone       if (reverse_depth == 0) {
198444d93782SGreg Clayton         window.PutChar(ACS_LTEE);
198544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1986b9c1b51eSKate Stone       } else {
198744d93782SGreg Clayton         window.PutChar(ACS_VLINE);
198844d93782SGreg Clayton         window.PutChar(' ');
198944d93782SGreg Clayton       }
199044d93782SGreg Clayton     }
199144d93782SGreg Clayton   }
199244d93782SGreg Clayton };
199344d93782SGreg Clayton 
1994b9c1b51eSKate Stone struct DisplayOptions {
199544d93782SGreg Clayton   bool show_types;
199644d93782SGreg Clayton };
199744d93782SGreg Clayton 
199844d93782SGreg Clayton class TreeItem;
199944d93782SGreg Clayton 
2000b9c1b51eSKate Stone class TreeDelegate {
200144d93782SGreg Clayton public:
2002c5dac77aSEugene Zelenko   TreeDelegate() = default;
2003315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
2004315b6884SEugene Zelenko 
200544d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
200644d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2007b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2008b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
200944d93782SGreg Clayton };
2010315b6884SEugene Zelenko 
201144d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
201244d93782SGreg Clayton 
2013b9c1b51eSKate Stone class TreeItem {
201444d93782SGreg Clayton public:
2015b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2016b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2017b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2018b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
201944d93782SGreg Clayton 
2020b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2021b9c1b51eSKate Stone     if (this != &rhs) {
202244d93782SGreg Clayton       m_parent = rhs.m_parent;
202344d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2024ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
202544d93782SGreg Clayton       m_identifier = rhs.m_identifier;
202644d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
202744d93782SGreg Clayton       m_children = rhs.m_children;
202844d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
202944d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
203044d93782SGreg Clayton     }
203144d93782SGreg Clayton     return *this;
203244d93782SGreg Clayton   }
203344d93782SGreg Clayton 
2034b9c1b51eSKate Stone   size_t GetDepth() const {
203544d93782SGreg Clayton     if (m_parent)
203644d93782SGreg Clayton       return 1 + m_parent->GetDepth();
203744d93782SGreg Clayton     return 0;
203844d93782SGreg Clayton   }
203944d93782SGreg Clayton 
2040b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
204144d93782SGreg Clayton 
2042b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
204344d93782SGreg Clayton 
2044b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
204544d93782SGreg Clayton 
2046b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
204744d93782SGreg Clayton 
2048b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
204944d93782SGreg Clayton 
2050b9c1b51eSKate Stone   size_t GetNumChildren() {
205144d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
205244d93782SGreg Clayton     return m_children.size();
205344d93782SGreg Clayton   }
205444d93782SGreg Clayton 
2055b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2056315b6884SEugene Zelenko 
2057b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
205844d93782SGreg Clayton     SetRowIndex(row_idx);
205944d93782SGreg Clayton     ++row_idx;
206044d93782SGreg Clayton 
2061ec990867SGreg Clayton     const bool expanded = IsExpanded();
2062ec990867SGreg Clayton 
2063ec990867SGreg Clayton     // The root item must calculate its children,
2064ec990867SGreg Clayton     // or we must calculate the number of children
2065ec990867SGreg Clayton     // if the item is expanded
2066c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
206744d93782SGreg Clayton       GetNumChildren();
206844d93782SGreg Clayton 
2069b9c1b51eSKate Stone     for (auto &item : m_children) {
207044d93782SGreg Clayton       if (expanded)
207144d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
207244d93782SGreg Clayton       else
207344d93782SGreg Clayton         item.SetRowIndex(-1);
207444d93782SGreg Clayton     }
207544d93782SGreg Clayton   }
207644d93782SGreg Clayton 
2077b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
207844d93782SGreg Clayton 
2079b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
208044d93782SGreg Clayton 
2081b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
208244d93782SGreg Clayton 
2083b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
208444d93782SGreg Clayton 
2085b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2086b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
208744d93782SGreg Clayton     if (num_rows_left <= 0)
208844d93782SGreg Clayton       return false;
208944d93782SGreg Clayton 
2090b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
209144d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
209244d93782SGreg Clayton 
209344d93782SGreg Clayton       if (m_parent)
209444d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
209544d93782SGreg Clayton 
2096b9c1b51eSKate Stone       if (m_might_have_children) {
2097b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
2098b9c1b51eSKate Stone         // "symbol"
209944d93782SGreg Clayton         // UTF8 string below
210044d93782SGreg Clayton         //            const char *symbol = "";
210144d93782SGreg Clayton         //            if (row.expanded)
210244d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
210344d93782SGreg Clayton         //            else
210444d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
210544d93782SGreg Clayton         //            window.PutCString (symbol);
210644d93782SGreg Clayton 
210744d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
210844d93782SGreg Clayton         // 'v' or '>' character...
210944d93782SGreg Clayton         //            if (expanded)
211044d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
211144d93782SGreg Clayton         //            else
211244d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
211344d93782SGreg Clayton         // Since we can't find any good looking right arrow/down arrow
211444d93782SGreg Clayton         // symbols, just use a diamond...
211544d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
211644d93782SGreg Clayton         window.PutChar(ACS_HLINE);
211744d93782SGreg Clayton       }
2118b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2119b9c1b51eSKate Stone                        window.IsActive();
212044d93782SGreg Clayton 
212144d93782SGreg Clayton       if (highlight)
212244d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
212344d93782SGreg Clayton 
212444d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
212544d93782SGreg Clayton 
212644d93782SGreg Clayton       if (highlight)
212744d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
212844d93782SGreg Clayton       ++row_idx;
212944d93782SGreg Clayton       --num_rows_left;
213044d93782SGreg Clayton     }
213144d93782SGreg Clayton 
213244d93782SGreg Clayton     if (num_rows_left <= 0)
213344d93782SGreg Clayton       return false; // We are done drawing...
213444d93782SGreg Clayton 
2135b9c1b51eSKate Stone     if (IsExpanded()) {
2136b9c1b51eSKate Stone       for (auto &item : m_children) {
213744d93782SGreg Clayton         // If we displayed all the rows and item.Draw() returns
213844d93782SGreg Clayton         // false we are done drawing and can exit this for loop
2139b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2140b9c1b51eSKate Stone                        num_rows_left))
214144d93782SGreg Clayton           break;
214244d93782SGreg Clayton       }
214344d93782SGreg Clayton     }
214444d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
214544d93782SGreg Clayton   }
214644d93782SGreg Clayton 
2147b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2148b9c1b51eSKate Stone                         uint32_t reverse_depth) {
214944d93782SGreg Clayton     if (m_parent)
215044d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
215144d93782SGreg Clayton 
2152b9c1b51eSKate Stone     if (&m_children.back() == child) {
215344d93782SGreg Clayton       // Last child
2154b9c1b51eSKate Stone       if (reverse_depth == 0) {
215544d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
215644d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2157b9c1b51eSKate Stone       } else {
215844d93782SGreg Clayton         window.PutChar(' ');
215944d93782SGreg Clayton         window.PutChar(' ');
216044d93782SGreg Clayton       }
2161b9c1b51eSKate Stone     } else {
2162b9c1b51eSKate Stone       if (reverse_depth == 0) {
216344d93782SGreg Clayton         window.PutChar(ACS_LTEE);
216444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2165b9c1b51eSKate Stone       } else {
216644d93782SGreg Clayton         window.PutChar(ACS_VLINE);
216744d93782SGreg Clayton         window.PutChar(' ');
216844d93782SGreg Clayton       }
216944d93782SGreg Clayton     }
217044d93782SGreg Clayton   }
217144d93782SGreg Clayton 
2172b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21733985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
217444d93782SGreg Clayton       return this;
217544d93782SGreg Clayton     if (m_children.empty())
2176c5dac77aSEugene Zelenko       return nullptr;
2177b9c1b51eSKate Stone     if (IsExpanded()) {
2178b9c1b51eSKate Stone       for (auto &item : m_children) {
217944d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
218044d93782SGreg Clayton         if (selected_item_ptr)
218144d93782SGreg Clayton           return selected_item_ptr;
218244d93782SGreg Clayton       }
218344d93782SGreg Clayton     }
2184c5dac77aSEugene Zelenko     return nullptr;
218544d93782SGreg Clayton   }
218644d93782SGreg Clayton 
2187b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2188ec990867SGreg Clayton 
2189b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2190ec990867SGreg Clayton 
2191b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
219244d93782SGreg Clayton 
2193b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
219444d93782SGreg Clayton 
2195b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2196ec990867SGreg Clayton 
219744d93782SGreg Clayton protected:
219844d93782SGreg Clayton   TreeItem *m_parent;
219944d93782SGreg Clayton   TreeDelegate &m_delegate;
2200ec990867SGreg Clayton   void *m_user_data;
220144d93782SGreg Clayton   uint64_t m_identifier;
2202b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2203b9c1b51eSKate Stone                  // root item
220444d93782SGreg Clayton   std::vector<TreeItem> m_children;
220544d93782SGreg Clayton   bool m_might_have_children;
220644d93782SGreg Clayton   bool m_is_expanded;
220744d93782SGreg Clayton };
220844d93782SGreg Clayton 
2209b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
221044d93782SGreg Clayton public:
2211b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2212b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2213b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2214b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2215b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
221644d93782SGreg Clayton 
2217b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
221844d93782SGreg Clayton 
2219b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2220b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2221b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
222244d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
222344d93782SGreg Clayton 
222444d93782SGreg Clayton     bool display_content = false;
2225b9c1b51eSKate Stone     if (process) {
222644d93782SGreg Clayton       StateType state = process->GetState();
2227b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
222844d93782SGreg Clayton         // We are stopped, so it is ok to
222944d93782SGreg Clayton         display_content = true;
2230b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
223144d93782SGreg Clayton         return true; // Don't do any updating when we are running
223244d93782SGreg Clayton       }
223344d93782SGreg Clayton     }
223444d93782SGreg Clayton 
223544d93782SGreg Clayton     m_min_x = 2;
223644d93782SGreg Clayton     m_min_y = 1;
223744d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
223844d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
223944d93782SGreg Clayton 
224044d93782SGreg Clayton     window.Erase();
224144d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
224244d93782SGreg Clayton 
2243b9c1b51eSKate Stone     if (display_content) {
224444d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
224544d93782SGreg Clayton       m_num_rows = 0;
224644d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
224744d93782SGreg Clayton 
224844d93782SGreg Clayton       // If we unexpanded while having something selected our
224944d93782SGreg Clayton       // total number of rows is less than the num visible rows,
225044d93782SGreg Clayton       // then make sure we show all the rows by setting the first
225144d93782SGreg Clayton       // visible row accordingly.
225244d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
225344d93782SGreg Clayton         m_first_visible_row = 0;
225444d93782SGreg Clayton 
225544d93782SGreg Clayton       // Make sure the selected row is always visible
225644d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
225744d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
225844d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
225944d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
226044d93782SGreg Clayton 
226144d93782SGreg Clayton       int row_idx = 0;
226244d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2263b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2264b9c1b51eSKate Stone                   num_rows_left);
226544d93782SGreg Clayton       // Get the selected row
226644d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2267b9c1b51eSKate Stone     } else {
2268c5dac77aSEugene Zelenko       m_selected_item = nullptr;
226944d93782SGreg Clayton     }
227044d93782SGreg Clayton 
227144d93782SGreg Clayton     window.DeferredRefresh();
227244d93782SGreg Clayton 
227344d93782SGreg Clayton     return true; // Drawing handled
227444d93782SGreg Clayton   }
227544d93782SGreg Clayton 
2276b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
227744d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
227844d93782SGreg Clayton   }
227944d93782SGreg Clayton 
2280b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
228144d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
228244d93782SGreg Clayton         {KEY_UP, "Select previous item"},
228344d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
228444d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2285b9c1b51eSKate Stone         {KEY_LEFT,
2286b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
228744d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
228844d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
228944d93782SGreg Clayton         {'h', "Show help dialog"},
229044d93782SGreg Clayton         {' ', "Toggle item expansion"},
229144d93782SGreg Clayton         {',', "Page up"},
229244d93782SGreg Clayton         {'.', "Page down"},
2293b9c1b51eSKate Stone         {'\0', nullptr}};
229444d93782SGreg Clayton     return g_source_view_key_help;
229544d93782SGreg Clayton   }
229644d93782SGreg Clayton 
2297b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2298b9c1b51eSKate Stone     switch (c) {
229944d93782SGreg Clayton     case ',':
230044d93782SGreg Clayton     case KEY_PPAGE:
230144d93782SGreg Clayton       // Page up key
2302b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
230344d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
230444d93782SGreg Clayton           m_first_visible_row -= m_max_y;
230544d93782SGreg Clayton         else
230644d93782SGreg Clayton           m_first_visible_row = 0;
230744d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
230844d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
230944d93782SGreg Clayton         if (m_selected_item)
231044d93782SGreg Clayton           m_selected_item->ItemWasSelected();
231144d93782SGreg Clayton       }
231244d93782SGreg Clayton       return eKeyHandled;
231344d93782SGreg Clayton 
231444d93782SGreg Clayton     case '.':
231544d93782SGreg Clayton     case KEY_NPAGE:
231644d93782SGreg Clayton       // Page down key
2317b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2318b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
231944d93782SGreg Clayton           m_first_visible_row += m_max_y;
232044d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
232144d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
232244d93782SGreg Clayton           if (m_selected_item)
232344d93782SGreg Clayton             m_selected_item->ItemWasSelected();
232444d93782SGreg Clayton         }
232544d93782SGreg Clayton       }
232644d93782SGreg Clayton       return eKeyHandled;
232744d93782SGreg Clayton 
232844d93782SGreg Clayton     case KEY_UP:
2329b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
233044d93782SGreg Clayton         --m_selected_row_idx;
233144d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
233244d93782SGreg Clayton         if (m_selected_item)
233344d93782SGreg Clayton           m_selected_item->ItemWasSelected();
233444d93782SGreg Clayton       }
233544d93782SGreg Clayton       return eKeyHandled;
2336315b6884SEugene Zelenko 
233744d93782SGreg Clayton     case KEY_DOWN:
2338b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
233944d93782SGreg Clayton         ++m_selected_row_idx;
234044d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
234144d93782SGreg Clayton         if (m_selected_item)
234244d93782SGreg Clayton           m_selected_item->ItemWasSelected();
234344d93782SGreg Clayton       }
234444d93782SGreg Clayton       return eKeyHandled;
234544d93782SGreg Clayton 
234644d93782SGreg Clayton     case KEY_RIGHT:
2347b9c1b51eSKate Stone       if (m_selected_item) {
234844d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
234944d93782SGreg Clayton           m_selected_item->Expand();
235044d93782SGreg Clayton       }
235144d93782SGreg Clayton       return eKeyHandled;
235244d93782SGreg Clayton 
235344d93782SGreg Clayton     case KEY_LEFT:
2354b9c1b51eSKate Stone       if (m_selected_item) {
235544d93782SGreg Clayton         if (m_selected_item->IsExpanded())
235644d93782SGreg Clayton           m_selected_item->Unexpand();
2357b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
235844d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
235944d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
236044d93782SGreg Clayton           if (m_selected_item)
236144d93782SGreg Clayton             m_selected_item->ItemWasSelected();
236244d93782SGreg Clayton         }
236344d93782SGreg Clayton       }
236444d93782SGreg Clayton       return eKeyHandled;
236544d93782SGreg Clayton 
236644d93782SGreg Clayton     case ' ':
236744d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2368b9c1b51eSKate Stone       if (m_selected_item) {
236944d93782SGreg Clayton         if (m_selected_item->IsExpanded())
237044d93782SGreg Clayton           m_selected_item->Unexpand();
237144d93782SGreg Clayton         else
237244d93782SGreg Clayton           m_selected_item->Expand();
237344d93782SGreg Clayton       }
237444d93782SGreg Clayton       return eKeyHandled;
237544d93782SGreg Clayton 
237644d93782SGreg Clayton     case 'h':
237744d93782SGreg Clayton       window.CreateHelpSubwindow();
237844d93782SGreg Clayton       return eKeyHandled;
237944d93782SGreg Clayton 
238044d93782SGreg Clayton     default:
238144d93782SGreg Clayton       break;
238244d93782SGreg Clayton     }
238344d93782SGreg Clayton     return eKeyNotHandled;
238444d93782SGreg Clayton   }
238544d93782SGreg Clayton 
238644d93782SGreg Clayton protected:
238744d93782SGreg Clayton   Debugger &m_debugger;
238844d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
238944d93782SGreg Clayton   TreeItem m_root;
239044d93782SGreg Clayton   TreeItem *m_selected_item;
239144d93782SGreg Clayton   int m_num_rows;
239244d93782SGreg Clayton   int m_selected_row_idx;
239344d93782SGreg Clayton   int m_first_visible_row;
239444d93782SGreg Clayton   int m_min_x;
239544d93782SGreg Clayton   int m_min_y;
239644d93782SGreg Clayton   int m_max_x;
239744d93782SGreg Clayton   int m_max_y;
239844d93782SGreg Clayton };
239944d93782SGreg Clayton 
2400b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
240144d93782SGreg Clayton public:
2402b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2403b9c1b51eSKate Stone     FormatEntity::Parse(
2404b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2405554f68d3SGreg Clayton         m_format);
240644d93782SGreg Clayton   }
240744d93782SGreg Clayton 
2408315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
240944d93782SGreg Clayton 
2410b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2411ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2412b9c1b51eSKate Stone     if (thread) {
241344d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2414ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2415b9c1b51eSKate Stone       if (frame_sp) {
241644d93782SGreg Clayton         StreamString strm;
2417b9c1b51eSKate Stone         const SymbolContext &sc =
2418b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
241944d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2420b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2421b9c1b51eSKate Stone                                  nullptr, false, false)) {
242244d93782SGreg Clayton           int right_pad = 1;
242344d93782SGreg Clayton           window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
242444d93782SGreg Clayton         }
242544d93782SGreg Clayton       }
242644d93782SGreg Clayton     }
242744d93782SGreg Clayton   }
2428315b6884SEugene Zelenko 
2429b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
243044d93782SGreg Clayton     // No children for frames yet...
243144d93782SGreg Clayton   }
243244d93782SGreg Clayton 
2433b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2434ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2435b9c1b51eSKate Stone     if (thread) {
2436b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2437b9c1b51eSKate Stone           thread->GetID());
243844d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2439ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
244044d93782SGreg Clayton       return true;
244144d93782SGreg Clayton     }
244244d93782SGreg Clayton     return false;
244344d93782SGreg Clayton   }
2444315b6884SEugene Zelenko 
2445554f68d3SGreg Clayton protected:
2446554f68d3SGreg Clayton   FormatEntity::Entry m_format;
244744d93782SGreg Clayton };
244844d93782SGreg Clayton 
2449b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
245044d93782SGreg Clayton public:
2451b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2452b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2453b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2454b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2455b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2456554f68d3SGreg Clayton                         m_format);
245744d93782SGreg Clayton   }
245844d93782SGreg Clayton 
2459315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
246044d93782SGreg Clayton 
2461b9c1b51eSKate Stone   ProcessSP GetProcess() {
2462b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2463b9c1b51eSKate Stone         .GetExecutionContext()
2464b9c1b51eSKate Stone         .GetProcessSP();
2465ec990867SGreg Clayton   }
2466ec990867SGreg Clayton 
2467b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2468ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2469ec990867SGreg Clayton     if (process_sp)
2470ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2471ec990867SGreg Clayton     return ThreadSP();
2472ec990867SGreg Clayton   }
2473ec990867SGreg Clayton 
2474b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2475ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2476b9c1b51eSKate Stone     if (thread_sp) {
247744d93782SGreg Clayton       StreamString strm;
247844d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2479b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2480b9c1b51eSKate Stone                                nullptr, false, false)) {
248144d93782SGreg Clayton         int right_pad = 1;
248244d93782SGreg Clayton         window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
248344d93782SGreg Clayton       }
248444d93782SGreg Clayton     }
248544d93782SGreg Clayton   }
2486315b6884SEugene Zelenko 
2487b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2488ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2489b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
249044d93782SGreg Clayton       StateType state = process_sp->GetState();
2491b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2492ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2493b9c1b51eSKate Stone         if (thread_sp) {
2494b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2495b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
249644d93782SGreg Clayton             return; // Children are already up to date
2497b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
249844d93782SGreg Clayton             // Always expand the thread item the first time we show it
2499ec990867SGreg Clayton             m_frame_delegate_sp.reset(new FrameTreeDelegate());
250044d93782SGreg Clayton           }
250144d93782SGreg Clayton 
250244d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
250344d93782SGreg Clayton           m_tid = thread_sp->GetID();
250444d93782SGreg Clayton 
250544d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
250644d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
250744d93782SGreg Clayton           item.Resize(num_frames, t);
2508b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2509ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
251044d93782SGreg Clayton             item[i].SetIdentifier(i);
251144d93782SGreg Clayton           }
251244d93782SGreg Clayton         }
251344d93782SGreg Clayton         return;
251444d93782SGreg Clayton       }
251544d93782SGreg Clayton     }
251644d93782SGreg Clayton     item.ClearChildren();
251744d93782SGreg Clayton   }
251844d93782SGreg Clayton 
2519b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2520ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2521b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2522ec990867SGreg Clayton       StateType state = process_sp->GetState();
2523b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2524ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2525b9c1b51eSKate Stone         if (thread_sp) {
252644d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2527bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
252844d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2529b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
253044d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
253144d93782SGreg Clayton             return true;
253244d93782SGreg Clayton           }
253344d93782SGreg Clayton         }
2534ec990867SGreg Clayton       }
2535ec990867SGreg Clayton     }
253644d93782SGreg Clayton     return false;
253744d93782SGreg Clayton   }
253844d93782SGreg Clayton 
253944d93782SGreg Clayton protected:
254044d93782SGreg Clayton   Debugger &m_debugger;
254144d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
254244d93782SGreg Clayton   lldb::user_id_t m_tid;
254344d93782SGreg Clayton   uint32_t m_stop_id;
2544554f68d3SGreg Clayton   FormatEntity::Entry m_format;
254544d93782SGreg Clayton };
254644d93782SGreg Clayton 
2547b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2548ec990867SGreg Clayton public:
2549b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2550b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2551b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2552554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2553554f68d3SGreg Clayton                         m_format);
2554ec990867SGreg Clayton   }
2555ec990867SGreg Clayton 
2556315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2557ec990867SGreg Clayton 
2558b9c1b51eSKate Stone   ProcessSP GetProcess() {
2559b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2560b9c1b51eSKate Stone         .GetExecutionContext()
2561b9c1b51eSKate Stone         .GetProcessSP();
2562ec990867SGreg Clayton   }
2563ec990867SGreg Clayton 
2564b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2565ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2566b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2567ec990867SGreg Clayton       StreamString strm;
2568ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2569b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2570b9c1b51eSKate Stone                                nullptr, false, false)) {
2571ec990867SGreg Clayton         int right_pad = 1;
2572ec990867SGreg Clayton         window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
2573ec990867SGreg Clayton       }
2574ec990867SGreg Clayton     }
2575ec990867SGreg Clayton   }
2576ec990867SGreg Clayton 
2577b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2578ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2579b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2580ec990867SGreg Clayton       StateType state = process_sp->GetState();
2581b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2582ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2583ec990867SGreg Clayton         if (m_stop_id == stop_id)
2584ec990867SGreg Clayton           return; // Children are already up to date
2585ec990867SGreg Clayton 
2586ec990867SGreg Clayton         m_stop_id = stop_id;
2587ec990867SGreg Clayton 
2588b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2589ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2590ec990867SGreg Clayton           // item.Expand();
2591ec990867SGreg Clayton           m_thread_delegate_sp.reset(new ThreadTreeDelegate(m_debugger));
2592ec990867SGreg Clayton         }
2593ec990867SGreg Clayton 
2594ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2595ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2596bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2597ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2598ec990867SGreg Clayton         item.Resize(num_threads, t);
2599b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2600ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2601ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2602ec990867SGreg Clayton         }
2603ec990867SGreg Clayton         return;
2604ec990867SGreg Clayton       }
2605ec990867SGreg Clayton     }
2606ec990867SGreg Clayton     item.ClearChildren();
2607ec990867SGreg Clayton   }
2608ec990867SGreg Clayton 
2609b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2610ec990867SGreg Clayton 
2611ec990867SGreg Clayton protected:
2612ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2613ec990867SGreg Clayton   Debugger &m_debugger;
2614ec990867SGreg Clayton   uint32_t m_stop_id;
2615554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2616ec990867SGreg Clayton };
2617ec990867SGreg Clayton 
2618b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
261944d93782SGreg Clayton public:
2620b9c1b51eSKate Stone   ValueObjectListDelegate()
2621b9c1b51eSKate Stone       : m_valobj_list(), m_rows(), m_selected_row(nullptr),
2622b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2623b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {}
262444d93782SGreg Clayton 
2625b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
2626b9c1b51eSKate Stone       : m_valobj_list(valobj_list), m_rows(), m_selected_row(nullptr),
2627b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2628b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {
262944d93782SGreg Clayton     SetValues(valobj_list);
263044d93782SGreg Clayton   }
263144d93782SGreg Clayton 
2632315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
263344d93782SGreg Clayton 
2634b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2635c5dac77aSEugene Zelenko     m_selected_row = nullptr;
263644d93782SGreg Clayton     m_selected_row_idx = 0;
263744d93782SGreg Clayton     m_first_visible_row = 0;
263844d93782SGreg Clayton     m_num_rows = 0;
263944d93782SGreg Clayton     m_rows.clear();
264044d93782SGreg Clayton     m_valobj_list = valobj_list;
264144d93782SGreg Clayton     const size_t num_values = m_valobj_list.GetSize();
264244d93782SGreg Clayton     for (size_t i = 0; i < num_values; ++i)
2643c5dac77aSEugene Zelenko       m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), nullptr));
264444d93782SGreg Clayton   }
264544d93782SGreg Clayton 
2646b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
264744d93782SGreg Clayton     m_num_rows = 0;
264844d93782SGreg Clayton     m_min_x = 2;
264944d93782SGreg Clayton     m_min_y = 1;
265044d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
265144d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
265244d93782SGreg Clayton 
265344d93782SGreg Clayton     window.Erase();
265444d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
265544d93782SGreg Clayton 
265644d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
265744d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
265844d93782SGreg Clayton 
265944d93782SGreg Clayton     // If we unexpanded while having something selected our
266044d93782SGreg Clayton     // total number of rows is less than the num visible rows,
266144d93782SGreg Clayton     // then make sure we show all the rows by setting the first
266244d93782SGreg Clayton     // visible row accordingly.
266344d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
266444d93782SGreg Clayton       m_first_visible_row = 0;
266544d93782SGreg Clayton 
266644d93782SGreg Clayton     // Make sure the selected row is always visible
266744d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
266844d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
266944d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
267044d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
267144d93782SGreg Clayton 
267244d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
267344d93782SGreg Clayton 
267444d93782SGreg Clayton     window.DeferredRefresh();
267544d93782SGreg Clayton 
267644d93782SGreg Clayton     // Get the selected row
267744d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
267844d93782SGreg Clayton     // Keep the cursor on the selected row so the highlight and the cursor
267944d93782SGreg Clayton     // are always on the same line
268044d93782SGreg Clayton     if (m_selected_row)
2681b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
268244d93782SGreg Clayton 
268344d93782SGreg Clayton     return true; // Drawing handled
268444d93782SGreg Clayton   }
268544d93782SGreg Clayton 
2686b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
268744d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
268844d93782SGreg Clayton         {KEY_UP, "Select previous item"},
268944d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
269044d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
269144d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
269244d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
269344d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
269444d93782SGreg Clayton         {'A', "Format as annotated address"},
269544d93782SGreg Clayton         {'b', "Format as binary"},
269644d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
269744d93782SGreg Clayton         {'c', "Format as character"},
269844d93782SGreg Clayton         {'d', "Format as a signed integer"},
269944d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
270044d93782SGreg Clayton         {'f', "Format as float"},
270144d93782SGreg Clayton         {'h', "Show help dialog"},
270244d93782SGreg Clayton         {'i', "Format as instructions"},
270344d93782SGreg Clayton         {'o', "Format as octal"},
270444d93782SGreg Clayton         {'p', "Format as pointer"},
270544d93782SGreg Clayton         {'s', "Format as C string"},
270644d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
270744d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
270844d93782SGreg Clayton         {'x', "Format as hex"},
270944d93782SGreg Clayton         {'X', "Format as uppercase hex"},
271044d93782SGreg Clayton         {' ', "Toggle item expansion"},
271144d93782SGreg Clayton         {',', "Page up"},
271244d93782SGreg Clayton         {'.', "Page down"},
2713b9c1b51eSKate Stone         {'\0', nullptr}};
271444d93782SGreg Clayton     return g_source_view_key_help;
271544d93782SGreg Clayton   }
271644d93782SGreg Clayton 
2717b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2718b9c1b51eSKate Stone     switch (c) {
271944d93782SGreg Clayton     case 'x':
272044d93782SGreg Clayton     case 'X':
272144d93782SGreg Clayton     case 'o':
272244d93782SGreg Clayton     case 's':
272344d93782SGreg Clayton     case 'u':
272444d93782SGreg Clayton     case 'd':
272544d93782SGreg Clayton     case 'D':
272644d93782SGreg Clayton     case 'i':
272744d93782SGreg Clayton     case 'A':
272844d93782SGreg Clayton     case 'p':
272944d93782SGreg Clayton     case 'c':
273044d93782SGreg Clayton     case 'b':
273144d93782SGreg Clayton     case 'B':
273244d93782SGreg Clayton     case 'f':
273344d93782SGreg Clayton       // Change the format for the currently selected item
273444d93782SGreg Clayton       if (m_selected_row)
273544d93782SGreg Clayton         m_selected_row->valobj->SetFormat(FormatForChar(c));
273644d93782SGreg Clayton       return eKeyHandled;
273744d93782SGreg Clayton 
273844d93782SGreg Clayton     case 't':
273944d93782SGreg Clayton       // Toggle showing type names
274044d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
274144d93782SGreg Clayton       return eKeyHandled;
274244d93782SGreg Clayton 
274344d93782SGreg Clayton     case ',':
274444d93782SGreg Clayton     case KEY_PPAGE:
274544d93782SGreg Clayton       // Page up key
2746b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27473985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
274844d93782SGreg Clayton           m_first_visible_row -= m_max_y;
274944d93782SGreg Clayton         else
275044d93782SGreg Clayton           m_first_visible_row = 0;
275144d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
275244d93782SGreg Clayton       }
275344d93782SGreg Clayton       return eKeyHandled;
275444d93782SGreg Clayton 
275544d93782SGreg Clayton     case '.':
275644d93782SGreg Clayton     case KEY_NPAGE:
275744d93782SGreg Clayton       // Page down key
2758b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2759b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
276044d93782SGreg Clayton           m_first_visible_row += m_max_y;
276144d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
276244d93782SGreg Clayton         }
276344d93782SGreg Clayton       }
276444d93782SGreg Clayton       return eKeyHandled;
276544d93782SGreg Clayton 
276644d93782SGreg Clayton     case KEY_UP:
276744d93782SGreg Clayton       if (m_selected_row_idx > 0)
276844d93782SGreg Clayton         --m_selected_row_idx;
276944d93782SGreg Clayton       return eKeyHandled;
2770315b6884SEugene Zelenko 
277144d93782SGreg Clayton     case KEY_DOWN:
277244d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
277344d93782SGreg Clayton         ++m_selected_row_idx;
277444d93782SGreg Clayton       return eKeyHandled;
277544d93782SGreg Clayton 
277644d93782SGreg Clayton     case KEY_RIGHT:
2777b9c1b51eSKate Stone       if (m_selected_row) {
277844d93782SGreg Clayton         if (!m_selected_row->expanded)
277944d93782SGreg Clayton           m_selected_row->Expand();
278044d93782SGreg Clayton       }
278144d93782SGreg Clayton       return eKeyHandled;
278244d93782SGreg Clayton 
278344d93782SGreg Clayton     case KEY_LEFT:
2784b9c1b51eSKate Stone       if (m_selected_row) {
278544d93782SGreg Clayton         if (m_selected_row->expanded)
278644d93782SGreg Clayton           m_selected_row->Unexpand();
278744d93782SGreg Clayton         else if (m_selected_row->parent)
278844d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
278944d93782SGreg Clayton       }
279044d93782SGreg Clayton       return eKeyHandled;
279144d93782SGreg Clayton 
279244d93782SGreg Clayton     case ' ':
279344d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2794b9c1b51eSKate Stone       if (m_selected_row) {
279544d93782SGreg Clayton         if (m_selected_row->expanded)
279644d93782SGreg Clayton           m_selected_row->Unexpand();
279744d93782SGreg Clayton         else
279844d93782SGreg Clayton           m_selected_row->Expand();
279944d93782SGreg Clayton       }
280044d93782SGreg Clayton       return eKeyHandled;
280144d93782SGreg Clayton 
280244d93782SGreg Clayton     case 'h':
280344d93782SGreg Clayton       window.CreateHelpSubwindow();
280444d93782SGreg Clayton       return eKeyHandled;
280544d93782SGreg Clayton 
280644d93782SGreg Clayton     default:
280744d93782SGreg Clayton       break;
280844d93782SGreg Clayton     }
280944d93782SGreg Clayton     return eKeyNotHandled;
281044d93782SGreg Clayton   }
281144d93782SGreg Clayton 
281244d93782SGreg Clayton protected:
281344d93782SGreg Clayton   ValueObjectList m_valobj_list;
281444d93782SGreg Clayton   std::vector<Row> m_rows;
281544d93782SGreg Clayton   Row *m_selected_row;
281644d93782SGreg Clayton   uint32_t m_selected_row_idx;
281744d93782SGreg Clayton   uint32_t m_first_visible_row;
281844d93782SGreg Clayton   uint32_t m_num_rows;
281944d93782SGreg Clayton   int m_min_x;
282044d93782SGreg Clayton   int m_min_y;
282144d93782SGreg Clayton   int m_max_x;
282244d93782SGreg Clayton   int m_max_y;
282344d93782SGreg Clayton 
2824b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2825b9c1b51eSKate Stone     switch (c) {
2826b9c1b51eSKate Stone     case 'x':
2827b9c1b51eSKate Stone       return eFormatHex;
2828b9c1b51eSKate Stone     case 'X':
2829b9c1b51eSKate Stone       return eFormatHexUppercase;
2830b9c1b51eSKate Stone     case 'o':
2831b9c1b51eSKate Stone       return eFormatOctal;
2832b9c1b51eSKate Stone     case 's':
2833b9c1b51eSKate Stone       return eFormatCString;
2834b9c1b51eSKate Stone     case 'u':
2835b9c1b51eSKate Stone       return eFormatUnsigned;
2836b9c1b51eSKate Stone     case 'd':
2837b9c1b51eSKate Stone       return eFormatDecimal;
2838b9c1b51eSKate Stone     case 'D':
2839b9c1b51eSKate Stone       return eFormatDefault;
2840b9c1b51eSKate Stone     case 'i':
2841b9c1b51eSKate Stone       return eFormatInstruction;
2842b9c1b51eSKate Stone     case 'A':
2843b9c1b51eSKate Stone       return eFormatAddressInfo;
2844b9c1b51eSKate Stone     case 'p':
2845b9c1b51eSKate Stone       return eFormatPointer;
2846b9c1b51eSKate Stone     case 'c':
2847b9c1b51eSKate Stone       return eFormatChar;
2848b9c1b51eSKate Stone     case 'b':
2849b9c1b51eSKate Stone       return eFormatBinary;
2850b9c1b51eSKate Stone     case 'B':
2851b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2852b9c1b51eSKate Stone     case 'f':
2853b9c1b51eSKate Stone       return eFormatFloat;
285444d93782SGreg Clayton     }
285544d93782SGreg Clayton     return eFormatDefault;
285644d93782SGreg Clayton   }
285744d93782SGreg Clayton 
2858b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2859b9c1b51eSKate Stone                         bool highlight, bool last_child) {
286044d93782SGreg Clayton     ValueObject *valobj = row.valobj.get();
286144d93782SGreg Clayton 
2862c5dac77aSEugene Zelenko     if (valobj == nullptr)
286344d93782SGreg Clayton       return false;
286444d93782SGreg Clayton 
2865b9c1b51eSKate Stone     const char *type_name =
2866b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
286744d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
286844d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
286944d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
287044d93782SGreg Clayton 
287144d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
287244d93782SGreg Clayton 
287344d93782SGreg Clayton     row.DrawTree(window);
287444d93782SGreg Clayton 
287544d93782SGreg Clayton     if (highlight)
287644d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
287744d93782SGreg Clayton 
287844d93782SGreg Clayton     if (type_name && type_name[0])
287944d93782SGreg Clayton       window.Printf("(%s) ", type_name);
288044d93782SGreg Clayton 
288144d93782SGreg Clayton     if (name && name[0])
288244d93782SGreg Clayton       window.PutCString(name);
288344d93782SGreg Clayton 
288444d93782SGreg Clayton     attr_t changd_attr = 0;
288544d93782SGreg Clayton     if (valobj->GetValueDidChange())
288644d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
288744d93782SGreg Clayton 
2888b9c1b51eSKate Stone     if (value && value[0]) {
288944d93782SGreg Clayton       window.PutCString(" = ");
289044d93782SGreg Clayton       if (changd_attr)
289144d93782SGreg Clayton         window.AttributeOn(changd_attr);
289244d93782SGreg Clayton       window.PutCString(value);
289344d93782SGreg Clayton       if (changd_attr)
289444d93782SGreg Clayton         window.AttributeOff(changd_attr);
289544d93782SGreg Clayton     }
289644d93782SGreg Clayton 
2897b9c1b51eSKate Stone     if (summary && summary[0]) {
289844d93782SGreg Clayton       window.PutChar(' ');
289944d93782SGreg Clayton       if (changd_attr)
290044d93782SGreg Clayton         window.AttributeOn(changd_attr);
290144d93782SGreg Clayton       window.PutCString(summary);
290244d93782SGreg Clayton       if (changd_attr)
290344d93782SGreg Clayton         window.AttributeOff(changd_attr);
290444d93782SGreg Clayton     }
290544d93782SGreg Clayton 
290644d93782SGreg Clayton     if (highlight)
290744d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
290844d93782SGreg Clayton 
290944d93782SGreg Clayton     return true;
291044d93782SGreg Clayton   }
2911315b6884SEugene Zelenko 
2912b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2913b9c1b51eSKate Stone                    DisplayOptions &options) {
291444d93782SGreg Clayton     // >   0x25B7
291544d93782SGreg Clayton     // \/  0x25BD
291644d93782SGreg Clayton 
291744d93782SGreg Clayton     bool window_is_active = window.IsActive();
2918b9c1b51eSKate Stone     for (auto &row : rows) {
291944d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
292044d93782SGreg Clayton       // Save the row index in each Row structure
292144d93782SGreg Clayton       row.row_idx = m_num_rows;
292244d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2923b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2924b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
292544d93782SGreg Clayton         row.x = m_min_x;
292644d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2927b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2928b9c1b51eSKate Stone                              window_is_active &&
2929b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2930b9c1b51eSKate Stone                              last_child)) {
293144d93782SGreg Clayton           ++m_num_rows;
2932b9c1b51eSKate Stone         } else {
293344d93782SGreg Clayton           row.x = 0;
293444d93782SGreg Clayton           row.y = 0;
293544d93782SGreg Clayton         }
2936b9c1b51eSKate Stone       } else {
293744d93782SGreg Clayton         row.x = 0;
293844d93782SGreg Clayton         row.y = 0;
293944d93782SGreg Clayton         ++m_num_rows;
294044d93782SGreg Clayton       }
294144d93782SGreg Clayton 
2942b9c1b51eSKate Stone       if (row.expanded && !row.children.empty()) {
2943b9c1b51eSKate Stone         DisplayRows(window, row.children, options);
294444d93782SGreg Clayton       }
294544d93782SGreg Clayton     }
294644d93782SGreg Clayton   }
294744d93782SGreg Clayton 
2948b9c1b51eSKate Stone   int CalculateTotalNumberRows(const std::vector<Row> &rows) {
294944d93782SGreg Clayton     int row_count = 0;
2950b9c1b51eSKate Stone     for (const auto &row : rows) {
295144d93782SGreg Clayton       ++row_count;
295244d93782SGreg Clayton       if (row.expanded)
295344d93782SGreg Clayton         row_count += CalculateTotalNumberRows(row.children);
295444d93782SGreg Clayton     }
295544d93782SGreg Clayton     return row_count;
295644d93782SGreg Clayton   }
2957315b6884SEugene Zelenko 
2958b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2959b9c1b51eSKate Stone     for (auto &row : rows) {
296044d93782SGreg Clayton       if (row_index == 0)
296144d93782SGreg Clayton         return &row;
2962b9c1b51eSKate Stone       else {
296344d93782SGreg Clayton         --row_index;
2964b9c1b51eSKate Stone         if (row.expanded && !row.children.empty()) {
296544d93782SGreg Clayton           Row *result = GetRowForRowIndexImpl(row.children, row_index);
296644d93782SGreg Clayton           if (result)
296744d93782SGreg Clayton             return result;
296844d93782SGreg Clayton         }
296944d93782SGreg Clayton       }
297044d93782SGreg Clayton     }
2971c5dac77aSEugene Zelenko     return nullptr;
297244d93782SGreg Clayton   }
297344d93782SGreg Clayton 
2974b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
297544d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
297644d93782SGreg Clayton   }
297744d93782SGreg Clayton 
2978b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
297944d93782SGreg Clayton 
298044d93782SGreg Clayton   static DisplayOptions g_options;
298144d93782SGreg Clayton };
298244d93782SGreg Clayton 
2983b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
298444d93782SGreg Clayton public:
2985b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
2986b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
2987b9c1b51eSKate Stone         m_frame_block(nullptr) {}
298844d93782SGreg Clayton 
2989315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
299044d93782SGreg Clayton 
2991b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
299244d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
299344d93782SGreg Clayton   }
299444d93782SGreg Clayton 
2995b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2996b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2997b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
299844d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
2999c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
3000c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
300144d93782SGreg Clayton 
3002b9c1b51eSKate Stone     if (process) {
300344d93782SGreg Clayton       StateType state = process->GetState();
3004b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
300544d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
300644d93782SGreg Clayton         if (frame)
300744d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
3008b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
300944d93782SGreg Clayton         return true; // Don't do any updating when we are running
301044d93782SGreg Clayton       }
301144d93782SGreg Clayton     }
301244d93782SGreg Clayton 
301344d93782SGreg Clayton     ValueObjectList local_values;
3014b9c1b51eSKate Stone     if (frame_block) {
301544d93782SGreg Clayton       // Only update the variables if they have changed
3016b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
301744d93782SGreg Clayton         m_frame_block = frame_block;
301844d93782SGreg Clayton 
301944d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3020b9c1b51eSKate Stone         if (locals) {
302144d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
302244d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3023b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3024b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3025b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3026b9c1b51eSKate Stone             if (value_sp) {
3027eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3028eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3029eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3030eb72dc7dSGreg Clayton               else
3031eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3032eb72dc7dSGreg Clayton             }
3033eb72dc7dSGreg Clayton           }
303444d93782SGreg Clayton           // Update the values
303544d93782SGreg Clayton           SetValues(local_values);
303644d93782SGreg Clayton         }
303744d93782SGreg Clayton       }
3038b9c1b51eSKate Stone     } else {
3039c5dac77aSEugene Zelenko       m_frame_block = nullptr;
304044d93782SGreg Clayton       // Update the values with an empty list if there is no frame
304144d93782SGreg Clayton       SetValues(local_values);
304244d93782SGreg Clayton     }
304344d93782SGreg Clayton 
304444d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
304544d93782SGreg Clayton   }
304644d93782SGreg Clayton 
304744d93782SGreg Clayton protected:
304844d93782SGreg Clayton   Debugger &m_debugger;
304944d93782SGreg Clayton   Block *m_frame_block;
305044d93782SGreg Clayton };
305144d93782SGreg Clayton 
3052b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
305344d93782SGreg Clayton public:
3054b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3055b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
305644d93782SGreg Clayton 
3057315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
305844d93782SGreg Clayton 
3059b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
306044d93782SGreg Clayton     return "Register window keyboard shortcuts:";
306144d93782SGreg Clayton   }
306244d93782SGreg Clayton 
3063b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3064b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3065b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
306644d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
306744d93782SGreg Clayton 
306844d93782SGreg Clayton     ValueObjectList value_list;
3069b9c1b51eSKate Stone     if (frame) {
3070b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
307144d93782SGreg Clayton         m_stack_id = frame->GetStackID();
307244d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3073b9c1b51eSKate Stone         if (reg_ctx) {
307444d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3075b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3076b9c1b51eSKate Stone             value_list.Append(
3077b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
307844d93782SGreg Clayton           }
307944d93782SGreg Clayton         }
308044d93782SGreg Clayton         SetValues(value_list);
308144d93782SGreg Clayton       }
3082b9c1b51eSKate Stone     } else {
308344d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
308444d93782SGreg Clayton       if (process && process->IsAlive())
308544d93782SGreg Clayton         return true; // Don't do any updating if we are running
3086b9c1b51eSKate Stone       else {
308744d93782SGreg Clayton         // Update the values with an empty list if there
308844d93782SGreg Clayton         // is no process or the process isn't alive anymore
308944d93782SGreg Clayton         SetValues(value_list);
309044d93782SGreg Clayton       }
309144d93782SGreg Clayton     }
309244d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
309344d93782SGreg Clayton   }
309444d93782SGreg Clayton 
309544d93782SGreg Clayton protected:
309644d93782SGreg Clayton   Debugger &m_debugger;
309744d93782SGreg Clayton   StackID m_stack_id;
309844d93782SGreg Clayton };
309944d93782SGreg Clayton 
3100b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
310144d93782SGreg Clayton   static char g_desc[32];
3102b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
310344d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
310444d93782SGreg Clayton     return g_desc;
310544d93782SGreg Clayton   }
3106b9c1b51eSKate Stone   switch (ch) {
3107b9c1b51eSKate Stone   case KEY_DOWN:
3108b9c1b51eSKate Stone     return "down";
3109b9c1b51eSKate Stone   case KEY_UP:
3110b9c1b51eSKate Stone     return "up";
3111b9c1b51eSKate Stone   case KEY_LEFT:
3112b9c1b51eSKate Stone     return "left";
3113b9c1b51eSKate Stone   case KEY_RIGHT:
3114b9c1b51eSKate Stone     return "right";
3115b9c1b51eSKate Stone   case KEY_HOME:
3116b9c1b51eSKate Stone     return "home";
3117b9c1b51eSKate Stone   case KEY_BACKSPACE:
3118b9c1b51eSKate Stone     return "backspace";
3119b9c1b51eSKate Stone   case KEY_DL:
3120b9c1b51eSKate Stone     return "delete-line";
3121b9c1b51eSKate Stone   case KEY_IL:
3122b9c1b51eSKate Stone     return "insert-line";
3123b9c1b51eSKate Stone   case KEY_DC:
3124b9c1b51eSKate Stone     return "delete-char";
3125b9c1b51eSKate Stone   case KEY_IC:
3126b9c1b51eSKate Stone     return "insert-char";
3127b9c1b51eSKate Stone   case KEY_CLEAR:
3128b9c1b51eSKate Stone     return "clear";
3129b9c1b51eSKate Stone   case KEY_EOS:
3130b9c1b51eSKate Stone     return "clear-to-eos";
3131b9c1b51eSKate Stone   case KEY_EOL:
3132b9c1b51eSKate Stone     return "clear-to-eol";
3133b9c1b51eSKate Stone   case KEY_SF:
3134b9c1b51eSKate Stone     return "scroll-forward";
3135b9c1b51eSKate Stone   case KEY_SR:
3136b9c1b51eSKate Stone     return "scroll-backward";
3137b9c1b51eSKate Stone   case KEY_NPAGE:
3138b9c1b51eSKate Stone     return "page-down";
3139b9c1b51eSKate Stone   case KEY_PPAGE:
3140b9c1b51eSKate Stone     return "page-up";
3141b9c1b51eSKate Stone   case KEY_STAB:
3142b9c1b51eSKate Stone     return "set-tab";
3143b9c1b51eSKate Stone   case KEY_CTAB:
3144b9c1b51eSKate Stone     return "clear-tab";
3145b9c1b51eSKate Stone   case KEY_CATAB:
3146b9c1b51eSKate Stone     return "clear-all-tabs";
3147b9c1b51eSKate Stone   case KEY_ENTER:
3148b9c1b51eSKate Stone     return "enter";
3149b9c1b51eSKate Stone   case KEY_PRINT:
3150b9c1b51eSKate Stone     return "print";
3151b9c1b51eSKate Stone   case KEY_LL:
3152b9c1b51eSKate Stone     return "lower-left key";
3153b9c1b51eSKate Stone   case KEY_A1:
3154b9c1b51eSKate Stone     return "upper left of keypad";
3155b9c1b51eSKate Stone   case KEY_A3:
3156b9c1b51eSKate Stone     return "upper right of keypad";
3157b9c1b51eSKate Stone   case KEY_B2:
3158b9c1b51eSKate Stone     return "center of keypad";
3159b9c1b51eSKate Stone   case KEY_C1:
3160b9c1b51eSKate Stone     return "lower left of keypad";
3161b9c1b51eSKate Stone   case KEY_C3:
3162b9c1b51eSKate Stone     return "lower right of keypad";
3163b9c1b51eSKate Stone   case KEY_BTAB:
3164b9c1b51eSKate Stone     return "back-tab key";
3165b9c1b51eSKate Stone   case KEY_BEG:
3166b9c1b51eSKate Stone     return "begin key";
3167b9c1b51eSKate Stone   case KEY_CANCEL:
3168b9c1b51eSKate Stone     return "cancel key";
3169b9c1b51eSKate Stone   case KEY_CLOSE:
3170b9c1b51eSKate Stone     return "close key";
3171b9c1b51eSKate Stone   case KEY_COMMAND:
3172b9c1b51eSKate Stone     return "command key";
3173b9c1b51eSKate Stone   case KEY_COPY:
3174b9c1b51eSKate Stone     return "copy key";
3175b9c1b51eSKate Stone   case KEY_CREATE:
3176b9c1b51eSKate Stone     return "create key";
3177b9c1b51eSKate Stone   case KEY_END:
3178b9c1b51eSKate Stone     return "end key";
3179b9c1b51eSKate Stone   case KEY_EXIT:
3180b9c1b51eSKate Stone     return "exit key";
3181b9c1b51eSKate Stone   case KEY_FIND:
3182b9c1b51eSKate Stone     return "find key";
3183b9c1b51eSKate Stone   case KEY_HELP:
3184b9c1b51eSKate Stone     return "help key";
3185b9c1b51eSKate Stone   case KEY_MARK:
3186b9c1b51eSKate Stone     return "mark key";
3187b9c1b51eSKate Stone   case KEY_MESSAGE:
3188b9c1b51eSKate Stone     return "message key";
3189b9c1b51eSKate Stone   case KEY_MOVE:
3190b9c1b51eSKate Stone     return "move key";
3191b9c1b51eSKate Stone   case KEY_NEXT:
3192b9c1b51eSKate Stone     return "next key";
3193b9c1b51eSKate Stone   case KEY_OPEN:
3194b9c1b51eSKate Stone     return "open key";
3195b9c1b51eSKate Stone   case KEY_OPTIONS:
3196b9c1b51eSKate Stone     return "options key";
3197b9c1b51eSKate Stone   case KEY_PREVIOUS:
3198b9c1b51eSKate Stone     return "previous key";
3199b9c1b51eSKate Stone   case KEY_REDO:
3200b9c1b51eSKate Stone     return "redo key";
3201b9c1b51eSKate Stone   case KEY_REFERENCE:
3202b9c1b51eSKate Stone     return "reference key";
3203b9c1b51eSKate Stone   case KEY_REFRESH:
3204b9c1b51eSKate Stone     return "refresh key";
3205b9c1b51eSKate Stone   case KEY_REPLACE:
3206b9c1b51eSKate Stone     return "replace key";
3207b9c1b51eSKate Stone   case KEY_RESTART:
3208b9c1b51eSKate Stone     return "restart key";
3209b9c1b51eSKate Stone   case KEY_RESUME:
3210b9c1b51eSKate Stone     return "resume key";
3211b9c1b51eSKate Stone   case KEY_SAVE:
3212b9c1b51eSKate Stone     return "save key";
3213b9c1b51eSKate Stone   case KEY_SBEG:
3214b9c1b51eSKate Stone     return "shifted begin key";
3215b9c1b51eSKate Stone   case KEY_SCANCEL:
3216b9c1b51eSKate Stone     return "shifted cancel key";
3217b9c1b51eSKate Stone   case KEY_SCOMMAND:
3218b9c1b51eSKate Stone     return "shifted command key";
3219b9c1b51eSKate Stone   case KEY_SCOPY:
3220b9c1b51eSKate Stone     return "shifted copy key";
3221b9c1b51eSKate Stone   case KEY_SCREATE:
3222b9c1b51eSKate Stone     return "shifted create key";
3223b9c1b51eSKate Stone   case KEY_SDC:
3224b9c1b51eSKate Stone     return "shifted delete-character key";
3225b9c1b51eSKate Stone   case KEY_SDL:
3226b9c1b51eSKate Stone     return "shifted delete-line key";
3227b9c1b51eSKate Stone   case KEY_SELECT:
3228b9c1b51eSKate Stone     return "select key";
3229b9c1b51eSKate Stone   case KEY_SEND:
3230b9c1b51eSKate Stone     return "shifted end key";
3231b9c1b51eSKate Stone   case KEY_SEOL:
3232b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3233b9c1b51eSKate Stone   case KEY_SEXIT:
3234b9c1b51eSKate Stone     return "shifted exit key";
3235b9c1b51eSKate Stone   case KEY_SFIND:
3236b9c1b51eSKate Stone     return "shifted find key";
3237b9c1b51eSKate Stone   case KEY_SHELP:
3238b9c1b51eSKate Stone     return "shifted help key";
3239b9c1b51eSKate Stone   case KEY_SHOME:
3240b9c1b51eSKate Stone     return "shifted home key";
3241b9c1b51eSKate Stone   case KEY_SIC:
3242b9c1b51eSKate Stone     return "shifted insert-character key";
3243b9c1b51eSKate Stone   case KEY_SLEFT:
3244b9c1b51eSKate Stone     return "shifted left-arrow key";
3245b9c1b51eSKate Stone   case KEY_SMESSAGE:
3246b9c1b51eSKate Stone     return "shifted message key";
3247b9c1b51eSKate Stone   case KEY_SMOVE:
3248b9c1b51eSKate Stone     return "shifted move key";
3249b9c1b51eSKate Stone   case KEY_SNEXT:
3250b9c1b51eSKate Stone     return "shifted next key";
3251b9c1b51eSKate Stone   case KEY_SOPTIONS:
3252b9c1b51eSKate Stone     return "shifted options key";
3253b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3254b9c1b51eSKate Stone     return "shifted previous key";
3255b9c1b51eSKate Stone   case KEY_SPRINT:
3256b9c1b51eSKate Stone     return "shifted print key";
3257b9c1b51eSKate Stone   case KEY_SREDO:
3258b9c1b51eSKate Stone     return "shifted redo key";
3259b9c1b51eSKate Stone   case KEY_SREPLACE:
3260b9c1b51eSKate Stone     return "shifted replace key";
3261b9c1b51eSKate Stone   case KEY_SRIGHT:
3262b9c1b51eSKate Stone     return "shifted right-arrow key";
3263b9c1b51eSKate Stone   case KEY_SRSUME:
3264b9c1b51eSKate Stone     return "shifted resume key";
3265b9c1b51eSKate Stone   case KEY_SSAVE:
3266b9c1b51eSKate Stone     return "shifted save key";
3267b9c1b51eSKate Stone   case KEY_SSUSPEND:
3268b9c1b51eSKate Stone     return "shifted suspend key";
3269b9c1b51eSKate Stone   case KEY_SUNDO:
3270b9c1b51eSKate Stone     return "shifted undo key";
3271b9c1b51eSKate Stone   case KEY_SUSPEND:
3272b9c1b51eSKate Stone     return "suspend key";
3273b9c1b51eSKate Stone   case KEY_UNDO:
3274b9c1b51eSKate Stone     return "undo key";
3275b9c1b51eSKate Stone   case KEY_MOUSE:
3276b9c1b51eSKate Stone     return "Mouse event has occurred";
3277b9c1b51eSKate Stone   case KEY_RESIZE:
3278b9c1b51eSKate Stone     return "Terminal resize event";
327927801f4fSBruce Mitchener #ifdef KEY_EVENT
3280b9c1b51eSKate Stone   case KEY_EVENT:
3281b9c1b51eSKate Stone     return "We were interrupted by an event";
328227801f4fSBruce Mitchener #endif
3283b9c1b51eSKate Stone   case KEY_RETURN:
3284b9c1b51eSKate Stone     return "return";
3285b9c1b51eSKate Stone   case ' ':
3286b9c1b51eSKate Stone     return "space";
3287b9c1b51eSKate Stone   case '\t':
3288b9c1b51eSKate Stone     return "tab";
3289b9c1b51eSKate Stone   case KEY_ESCAPE:
3290b9c1b51eSKate Stone     return "escape";
329144d93782SGreg Clayton   default:
329244d93782SGreg Clayton     if (isprint(ch))
329344d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
329444d93782SGreg Clayton     else
329544d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
329644d93782SGreg Clayton     return g_desc;
329744d93782SGreg Clayton   }
3298c5dac77aSEugene Zelenko   return nullptr;
329944d93782SGreg Clayton }
330044d93782SGreg Clayton 
3301b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3302b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3303b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3304b9c1b51eSKate Stone   if (text && text[0]) {
330544d93782SGreg Clayton     m_text.SplitIntoLines(text);
330644d93782SGreg Clayton     m_text.AppendString("");
330744d93782SGreg Clayton   }
3308b9c1b51eSKate Stone   if (key_help_array) {
3309b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
331044d93782SGreg Clayton       StreamString key_description;
3311b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3312b9c1b51eSKate Stone                              key->description);
331344d93782SGreg Clayton       m_text.AppendString(std::move(key_description.GetString()));
331444d93782SGreg Clayton     }
331544d93782SGreg Clayton   }
331644d93782SGreg Clayton }
331744d93782SGreg Clayton 
3318315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
331944d93782SGreg Clayton 
3320b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
332144d93782SGreg Clayton   window.Erase();
332244d93782SGreg Clayton   const int window_height = window.GetHeight();
332344d93782SGreg Clayton   int x = 2;
332444d93782SGreg Clayton   int y = 1;
332544d93782SGreg Clayton   const int min_y = y;
332644d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33273985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
332844d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
332944d93782SGreg Clayton   const char *bottom_message;
333044d93782SGreg Clayton   if (num_lines <= num_visible_lines)
333144d93782SGreg Clayton     bottom_message = "Press any key to exit";
333244d93782SGreg Clayton   else
333344d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
333444d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3335b9c1b51eSKate Stone   while (y <= max_y) {
333644d93782SGreg Clayton     window.MoveCursor(x, y);
3337b9c1b51eSKate Stone     window.PutCStringTruncated(
3338b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
333944d93782SGreg Clayton     ++y;
334044d93782SGreg Clayton   }
334144d93782SGreg Clayton   return true;
334244d93782SGreg Clayton }
334344d93782SGreg Clayton 
3344b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3345b9c1b51eSKate Stone                                                               int key) {
334644d93782SGreg Clayton   bool done = false;
334744d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
334844d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
334944d93782SGreg Clayton 
3350b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
335144d93782SGreg Clayton     done = true;
335244d93782SGreg Clayton     // If we have all lines visible and don't need scrolling, then any
335344d93782SGreg Clayton     // key press will cause us to exit
3354b9c1b51eSKate Stone   } else {
3355b9c1b51eSKate Stone     switch (key) {
335644d93782SGreg Clayton     case KEY_UP:
335744d93782SGreg Clayton       if (m_first_visible_line > 0)
335844d93782SGreg Clayton         --m_first_visible_line;
335944d93782SGreg Clayton       break;
336044d93782SGreg Clayton 
336144d93782SGreg Clayton     case KEY_DOWN:
336244d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
336344d93782SGreg Clayton         ++m_first_visible_line;
336444d93782SGreg Clayton       break;
336544d93782SGreg Clayton 
336644d93782SGreg Clayton     case KEY_PPAGE:
336744d93782SGreg Clayton     case ',':
3368b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33693985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
337044d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
337144d93782SGreg Clayton         else
337244d93782SGreg Clayton           m_first_visible_line = 0;
337344d93782SGreg Clayton       }
337444d93782SGreg Clayton       break;
3375315b6884SEugene Zelenko 
337644d93782SGreg Clayton     case KEY_NPAGE:
337744d93782SGreg Clayton     case '.':
3378b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
337944d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
33803985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
338144d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
338244d93782SGreg Clayton       }
338344d93782SGreg Clayton       break;
3384315b6884SEugene Zelenko 
338544d93782SGreg Clayton     default:
338644d93782SGreg Clayton       done = true;
338744d93782SGreg Clayton       break;
338844d93782SGreg Clayton     }
338944d93782SGreg Clayton   }
339044d93782SGreg Clayton   if (done)
339144d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
339244d93782SGreg Clayton   return eKeyHandled;
339344d93782SGreg Clayton }
339444d93782SGreg Clayton 
3395b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
339644d93782SGreg Clayton public:
339744d93782SGreg Clayton   enum {
339844d93782SGreg Clayton     eMenuID_LLDB = 1,
339944d93782SGreg Clayton     eMenuID_LLDBAbout,
340044d93782SGreg Clayton     eMenuID_LLDBExit,
340144d93782SGreg Clayton 
340244d93782SGreg Clayton     eMenuID_Target,
340344d93782SGreg Clayton     eMenuID_TargetCreate,
340444d93782SGreg Clayton     eMenuID_TargetDelete,
340544d93782SGreg Clayton 
340644d93782SGreg Clayton     eMenuID_Process,
340744d93782SGreg Clayton     eMenuID_ProcessAttach,
340844d93782SGreg Clayton     eMenuID_ProcessDetach,
340944d93782SGreg Clayton     eMenuID_ProcessLaunch,
341044d93782SGreg Clayton     eMenuID_ProcessContinue,
341144d93782SGreg Clayton     eMenuID_ProcessHalt,
341244d93782SGreg Clayton     eMenuID_ProcessKill,
341344d93782SGreg Clayton 
341444d93782SGreg Clayton     eMenuID_Thread,
341544d93782SGreg Clayton     eMenuID_ThreadStepIn,
341644d93782SGreg Clayton     eMenuID_ThreadStepOver,
341744d93782SGreg Clayton     eMenuID_ThreadStepOut,
341844d93782SGreg Clayton 
341944d93782SGreg Clayton     eMenuID_View,
342044d93782SGreg Clayton     eMenuID_ViewBacktrace,
342144d93782SGreg Clayton     eMenuID_ViewRegisters,
342244d93782SGreg Clayton     eMenuID_ViewSource,
342344d93782SGreg Clayton     eMenuID_ViewVariables,
342444d93782SGreg Clayton 
342544d93782SGreg Clayton     eMenuID_Help,
342644d93782SGreg Clayton     eMenuID_HelpGUIHelp
342744d93782SGreg Clayton   };
342844d93782SGreg Clayton 
3429b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3430b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
343144d93782SGreg Clayton 
3432315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3433bd5ae6b4SGreg Clayton 
3434b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
343544d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
343644d93782SGreg Clayton   }
343744d93782SGreg Clayton 
3438b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3439b9c1b51eSKate Stone     switch (key) {
34405fdb09bbSGreg Clayton     case '\t':
344144d93782SGreg Clayton       window.SelectNextWindowAsActive();
344244d93782SGreg Clayton       return eKeyHandled;
34435fdb09bbSGreg Clayton 
34445fdb09bbSGreg Clayton     case 'h':
34455fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34465fdb09bbSGreg Clayton       return eKeyHandled;
34475fdb09bbSGreg Clayton 
34485fdb09bbSGreg Clayton     case KEY_ESCAPE:
34495fdb09bbSGreg Clayton       return eQuitApplication;
34505fdb09bbSGreg Clayton 
34515fdb09bbSGreg Clayton     default:
34525fdb09bbSGreg Clayton       break;
345344d93782SGreg Clayton     }
345444d93782SGreg Clayton     return eKeyNotHandled;
345544d93782SGreg Clayton   }
345644d93782SGreg Clayton 
3457b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34585fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34595fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3460b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3461b9c1b51eSKate Stone            "dialog to display them.\n\n"
34625fdb09bbSGreg Clayton            "Common key bindings for all views:";
34635fdb09bbSGreg Clayton   }
34645fdb09bbSGreg Clayton 
3465b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34665fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34675fdb09bbSGreg Clayton         {'\t', "Select next view"},
34685fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34695fdb09bbSGreg Clayton         {',', "Page up"},
34705fdb09bbSGreg Clayton         {'.', "Page down"},
34715fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34725fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34735fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
34745fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
34755fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
34765fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3477b9c1b51eSKate Stone         {'\0', nullptr}};
34785fdb09bbSGreg Clayton     return g_source_view_key_help;
34795fdb09bbSGreg Clayton   }
34805fdb09bbSGreg Clayton 
3481b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3482b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3483b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3484b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3485b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3486b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
348744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3488b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3489b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
34904b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
349144d93782SGreg Clayton       }
349244d93782SGreg Clayton     }
349344d93782SGreg Clayton       return MenuActionResult::Handled;
349444d93782SGreg Clayton 
3495b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3496b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3497b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3498b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
349944d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3500b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3501b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
350244d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
350344d93782SGreg Clayton       }
350444d93782SGreg Clayton     }
350544d93782SGreg Clayton       return MenuActionResult::Handled;
350644d93782SGreg Clayton 
3507b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3508b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3509b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3510b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
351144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3512b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3513b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
351444d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
351544d93782SGreg Clayton       }
351644d93782SGreg Clayton     }
351744d93782SGreg Clayton       return MenuActionResult::Handled;
351844d93782SGreg Clayton 
3519b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3520b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3521b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3522b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
352344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3524b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3525b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
352644d93782SGreg Clayton           process->Resume();
352744d93782SGreg Clayton       }
352844d93782SGreg Clayton     }
352944d93782SGreg Clayton       return MenuActionResult::Handled;
353044d93782SGreg Clayton 
3531b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3532b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3533b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3534b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
353544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
353644d93782SGreg Clayton         if (process && process->IsAlive())
3537ede3193bSJason Molenda           process->Destroy(false);
353844d93782SGreg Clayton       }
353944d93782SGreg Clayton     }
354044d93782SGreg Clayton       return MenuActionResult::Handled;
354144d93782SGreg Clayton 
3542b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3543b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3544b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3545b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
354644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
354744d93782SGreg Clayton         if (process && process->IsAlive())
354844d93782SGreg Clayton           process->Halt();
354944d93782SGreg Clayton       }
355044d93782SGreg Clayton     }
355144d93782SGreg Clayton       return MenuActionResult::Handled;
355244d93782SGreg Clayton 
3553b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3554b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3555b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3556b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
355744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
355844d93782SGreg Clayton         if (process && process->IsAlive())
355944d93782SGreg Clayton           process->Detach(false);
356044d93782SGreg Clayton       }
356144d93782SGreg Clayton     }
356244d93782SGreg Clayton       return MenuActionResult::Handled;
356344d93782SGreg Clayton 
3564b9c1b51eSKate Stone     case eMenuID_Process: {
3565b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
3566b9c1b51eSKate Stone       // when
356744d93782SGreg Clayton       // the Process menu gets selected and is about to display its submenu.
356844d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3569b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3570b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
357144d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3572b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3573b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
357444d93782SGreg Clayton         if (submenus.size() == 7)
357544d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
357644d93782SGreg Clayton         else if (submenus.size() > 8)
357744d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
357844d93782SGreg Clayton 
357944d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3580bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
358144d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3582b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
358344d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
358444d93782SGreg Clayton           char menu_char = '\0';
358544d93782SGreg Clayton           if (i < 9)
358644d93782SGreg Clayton             menu_char = '1' + i;
358744d93782SGreg Clayton           StreamString thread_menu_title;
358844d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
358944d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
359044d93782SGreg Clayton           if (thread_name && thread_name[0])
359144d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3592b9c1b51eSKate Stone           else {
359344d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
359444d93782SGreg Clayton             if (queue_name && queue_name[0])
359544d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
359644d93782SGreg Clayton           }
3597b9c1b51eSKate Stone           menu.AddSubmenu(
3598b9c1b51eSKate Stone               MenuSP(new Menu(thread_menu_title.GetString().c_str(), nullptr,
3599b9c1b51eSKate Stone                               menu_char, thread_sp->GetID())));
360044d93782SGreg Clayton         }
3601b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
360244d93782SGreg Clayton         // Remove the separator and any other thread submenu items
360344d93782SGreg Clayton         // that were previously added
360444d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
360544d93782SGreg Clayton       }
3606b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3607b9c1b51eSKate Stone       // lengths
360844d93782SGreg Clayton       menu.RecalculateNameLengths();
360944d93782SGreg Clayton     }
361044d93782SGreg Clayton       return MenuActionResult::Handled;
361144d93782SGreg Clayton 
3612b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
361344d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
361444d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
361544d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
361644d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
361744d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
361844d93782SGreg Clayton 
3619b9c1b51eSKate Stone       if (variables_window_sp) {
362044d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
362144d93782SGreg Clayton 
362244d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
362344d93782SGreg Clayton 
3624b9c1b51eSKate Stone         if (registers_window_sp) {
3625b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3626b9c1b51eSKate Stone           // registers window
362744d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
362844d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
362944d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3630b9c1b51eSKate Stone         } else {
363144d93782SGreg Clayton           // We have no registers window showing so give the bottom
363244d93782SGreg Clayton           // area back to the source view
363344d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3634b9c1b51eSKate Stone                                    source_bounds.size.height +
3635b9c1b51eSKate Stone                                        variables_bounds.size.height);
363644d93782SGreg Clayton         }
3637b9c1b51eSKate Stone       } else {
363844d93782SGreg Clayton         Rect new_variables_rect;
3639b9c1b51eSKate Stone         if (registers_window_sp) {
364044d93782SGreg Clayton           // We have a registers window so split the area of the registers
364144d93782SGreg Clayton           // window into two columns where the left hand side will be the
364244d93782SGreg Clayton           // variables and the right hand side will be the registers
364344d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
364444d93782SGreg Clayton           Rect new_registers_rect;
3645b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3646b9c1b51eSKate Stone                                                    new_registers_rect);
364744d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3648b9c1b51eSKate Stone         } else {
364944d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
365044d93782SGreg Clayton           Rect new_source_rect;
3651b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3652b9c1b51eSKate Stone                                                   new_variables_rect);
365344d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
365444d93782SGreg Clayton         }
3655b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3656b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3657b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3658b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
365944d93782SGreg Clayton       }
366044d93782SGreg Clayton       touchwin(stdscr);
366144d93782SGreg Clayton     }
366244d93782SGreg Clayton       return MenuActionResult::Handled;
366344d93782SGreg Clayton 
3664b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
366544d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
366644d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
366744d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
366844d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
366944d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
367044d93782SGreg Clayton 
3671b9c1b51eSKate Stone       if (registers_window_sp) {
3672b9c1b51eSKate Stone         if (variables_window_sp) {
367344d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
367444d93782SGreg Clayton 
3675b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3676b9c1b51eSKate Stone           // variables window
3677b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3678b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
367944d93782SGreg Clayton                                       variables_bounds.size.height);
3680b9c1b51eSKate Stone         } else {
368144d93782SGreg Clayton           // We have no variables window showing so give the bottom
368244d93782SGreg Clayton           // area back to the source view
368344d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3684b9c1b51eSKate Stone                                    source_bounds.size.height +
3685b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
368644d93782SGreg Clayton         }
368744d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3688b9c1b51eSKate Stone       } else {
368944d93782SGreg Clayton         Rect new_regs_rect;
3690b9c1b51eSKate Stone         if (variables_window_sp) {
369144d93782SGreg Clayton           // We have a variables window, split it into two columns
369244d93782SGreg Clayton           // where the left hand side will be the variables and the
369344d93782SGreg Clayton           // right hand side will be the registers
369444d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
369544d93782SGreg Clayton           Rect new_vars_rect;
3696b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3697b9c1b51eSKate Stone                                                    new_regs_rect);
369844d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3699b9c1b51eSKate Stone         } else {
370044d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
370144d93782SGreg Clayton           Rect new_source_rect;
3702b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3703b9c1b51eSKate Stone                                                   new_regs_rect);
370444d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
370544d93782SGreg Clayton         }
3706b9c1b51eSKate Stone         WindowSP new_window_sp =
3707b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3708b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3709b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
371044d93782SGreg Clayton       }
371144d93782SGreg Clayton       touchwin(stdscr);
371244d93782SGreg Clayton     }
371344d93782SGreg Clayton       return MenuActionResult::Handled;
371444d93782SGreg Clayton 
371544d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37165fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
371744d93782SGreg Clayton       return MenuActionResult::Handled;
371844d93782SGreg Clayton 
371944d93782SGreg Clayton     default:
372044d93782SGreg Clayton       break;
372144d93782SGreg Clayton     }
372244d93782SGreg Clayton 
372344d93782SGreg Clayton     return MenuActionResult::NotHandled;
372444d93782SGreg Clayton   }
3725b9c1b51eSKate Stone 
372644d93782SGreg Clayton protected:
372744d93782SGreg Clayton   Application &m_app;
372844d93782SGreg Clayton   Debugger &m_debugger;
372944d93782SGreg Clayton };
373044d93782SGreg Clayton 
3731b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
373244d93782SGreg Clayton public:
3733b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3734b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
373544d93782SGreg Clayton   }
373644d93782SGreg Clayton 
3737315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3738bd5ae6b4SGreg Clayton 
3739b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3740b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3741b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
374244d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
374344d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
374444d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
374544d93782SGreg Clayton     window.Erase();
374644d93782SGreg Clayton     window.SetBackground(2);
374744d93782SGreg Clayton     window.MoveCursor(0, 0);
3748b9c1b51eSKate Stone     if (process) {
374944d93782SGreg Clayton       const StateType state = process->GetState();
3750b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3751b9c1b51eSKate Stone                     StateAsCString(state));
375244d93782SGreg Clayton 
3753b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37545b031ebcSEd Maste         StreamString strm;
3755b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3756b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
375744d93782SGreg Clayton           window.MoveCursor(40, 0);
37585b031ebcSEd Maste           window.PutCStringTruncated(strm.GetString().c_str(), 1);
37595b031ebcSEd Maste         }
376044d93782SGreg Clayton 
376144d93782SGreg Clayton         window.MoveCursor(60, 0);
376244d93782SGreg Clayton         if (frame)
3763b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3764b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3765b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3766b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3767b9c1b51eSKate Stone       } else if (state == eStateExited) {
376844d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
376944d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
377044d93782SGreg Clayton         if (exit_desc && exit_desc[0])
377144d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
377244d93782SGreg Clayton         else
377344d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
377444d93782SGreg Clayton       }
377544d93782SGreg Clayton     }
377644d93782SGreg Clayton     window.DeferredRefresh();
377744d93782SGreg Clayton     return true;
377844d93782SGreg Clayton   }
377944d93782SGreg Clayton 
378044d93782SGreg Clayton protected:
378144d93782SGreg Clayton   Debugger &m_debugger;
3782554f68d3SGreg Clayton   FormatEntity::Entry m_format;
378344d93782SGreg Clayton };
378444d93782SGreg Clayton 
3785b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
378644d93782SGreg Clayton public:
3787b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3788b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3789b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3790b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3791b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3792b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
379344d93782SGreg Clayton 
3794315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
379544d93782SGreg Clayton 
3796b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
379744d93782SGreg Clayton 
3798b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
379944d93782SGreg Clayton 
3800b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
380144d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
380244d93782SGreg Clayton   }
380344d93782SGreg Clayton 
3804b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
380544d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
380644d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
380744d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
380844d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
380944d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
381044d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
381144d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
381244d93782SGreg Clayton         {'c', "Continue process"},
381344d93782SGreg Clayton         {'d', "Detach and resume process"},
381444d93782SGreg Clayton         {'D', "Detach with process suspended"},
381544d93782SGreg Clayton         {'h', "Show help dialog"},
381644d93782SGreg Clayton         {'k', "Kill process"},
381744d93782SGreg Clayton         {'n', "Step over (source line)"},
381844d93782SGreg Clayton         {'N', "Step over (single instruction)"},
381944d93782SGreg Clayton         {'o', "Step out"},
382044d93782SGreg Clayton         {'s', "Step in (source line)"},
382144d93782SGreg Clayton         {'S', "Step in (single instruction)"},
382244d93782SGreg Clayton         {',', "Page up"},
382344d93782SGreg Clayton         {'.', "Page down"},
3824b9c1b51eSKate Stone         {'\0', nullptr}};
382544d93782SGreg Clayton     return g_source_view_key_help;
382644d93782SGreg Clayton   }
382744d93782SGreg Clayton 
3828b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3829b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3830b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
383144d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3832c5dac77aSEugene Zelenko     Thread *thread = nullptr;
383344d93782SGreg Clayton 
383444d93782SGreg Clayton     bool update_location = false;
3835b9c1b51eSKate Stone     if (process) {
383644d93782SGreg Clayton       StateType state = process->GetState();
3837b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
383844d93782SGreg Clayton         // We are stopped, so it is ok to
383944d93782SGreg Clayton         update_location = true;
384044d93782SGreg Clayton       }
384144d93782SGreg Clayton     }
384244d93782SGreg Clayton 
384344d93782SGreg Clayton     m_min_x = 1;
3844ec990867SGreg Clayton     m_min_y = 2;
384544d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
384644d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
384744d93782SGreg Clayton 
384844d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
384944d93782SGreg Clayton     StackFrameSP frame_sp;
385044d93782SGreg Clayton     bool set_selected_line_to_pc = false;
385144d93782SGreg Clayton 
3852b9c1b51eSKate Stone     if (update_location) {
385344d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
385444d93782SGreg Clayton       bool thread_changed = false;
3855b9c1b51eSKate Stone       if (process_alive) {
385644d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3857b9c1b51eSKate Stone         if (thread) {
385844d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
385944d93782SGreg Clayton           auto tid = thread->GetID();
386044d93782SGreg Clayton           thread_changed = tid != m_tid;
386144d93782SGreg Clayton           m_tid = tid;
3862b9c1b51eSKate Stone         } else {
3863b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
386444d93782SGreg Clayton             thread_changed = true;
386544d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
386644d93782SGreg Clayton           }
386744d93782SGreg Clayton         }
386844d93782SGreg Clayton       }
386944d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
387044d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
387144d93782SGreg Clayton       bool frame_changed = false;
387244d93782SGreg Clayton       m_stop_id = stop_id;
3873ec990867SGreg Clayton       m_title.Clear();
3874b9c1b51eSKate Stone       if (frame_sp) {
387544d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3876b9c1b51eSKate Stone         if (m_sc.module_sp) {
3877b9c1b51eSKate Stone           m_title.Printf(
3878b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3879ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3880ec990867SGreg Clayton           if (func_name)
3881ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3882ec990867SGreg Clayton         }
388344d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
388444d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
388544d93782SGreg Clayton         m_frame_idx = frame_idx;
3886b9c1b51eSKate Stone       } else {
388744d93782SGreg Clayton         m_sc.Clear(true);
388844d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
388944d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
389044d93782SGreg Clayton       }
389144d93782SGreg Clayton 
3892b9c1b51eSKate Stone       const bool context_changed =
3893b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
389444d93782SGreg Clayton 
3895b9c1b51eSKate Stone       if (process_alive) {
3896b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
389744d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
389844d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
389944d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
390044d93782SGreg Clayton           // Update the selected line if the stop ID changed...
390144d93782SGreg Clayton           if (context_changed)
390244d93782SGreg Clayton             m_selected_line = m_pc_line;
390344d93782SGreg Clayton 
3904b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
390544d93782SGreg Clayton             // Same file, nothing to do, we should either have the
390644d93782SGreg Clayton             // lines or not (source file missing)
3907b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
390844d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
390944d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3910b9c1b51eSKate Stone             } else {
391144d93782SGreg Clayton               if (m_selected_line > 10)
391244d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
391344d93782SGreg Clayton               else
391444d93782SGreg Clayton                 m_first_visible_line = 0;
391544d93782SGreg Clayton             }
3916b9c1b51eSKate Stone           } else {
391744d93782SGreg Clayton             // File changed, set selected line to the line with the PC
391844d93782SGreg Clayton             m_selected_line = m_pc_line;
3919b9c1b51eSKate Stone             m_file_sp =
3920b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3921b9c1b51eSKate Stone             if (m_file_sp) {
392244d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
392344d93782SGreg Clayton               int m_line_width = 1;
392444d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
392544d93782SGreg Clayton                 ++m_line_width;
392644d93782SGreg Clayton 
3927b9c1b51eSKate Stone               snprintf(m_line_format, sizeof(m_line_format), " %%%iu ",
3928b9c1b51eSKate Stone                        m_line_width);
3929b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3930b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
393144d93782SGreg Clayton                 m_first_visible_line = 0;
393244d93782SGreg Clayton               else
393344d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
393444d93782SGreg Clayton             }
393544d93782SGreg Clayton           }
3936b9c1b51eSKate Stone         } else {
393744d93782SGreg Clayton           m_file_sp.reset();
393844d93782SGreg Clayton         }
393944d93782SGreg Clayton 
3940b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
394144d93782SGreg Clayton           // Show disassembly
394244d93782SGreg Clayton           bool prefer_file_cache = false;
3943b9c1b51eSKate Stone           if (m_sc.function) {
3944b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
394544d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3946b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3947b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3948b9c1b51eSKate Stone               if (m_disassembly_sp) {
394944d93782SGreg Clayton                 set_selected_line_to_pc = true;
395044d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3951b9c1b51eSKate Stone               } else {
395244d93782SGreg Clayton                 m_disassembly_range.Clear();
395344d93782SGreg Clayton               }
3954b9c1b51eSKate Stone             } else {
395544d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
395644d93782SGreg Clayton             }
3957b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3958b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
395944d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3960b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3961b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3962b9c1b51eSKate Stone               if (m_disassembly_sp) {
396344d93782SGreg Clayton                 set_selected_line_to_pc = true;
3964b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3965b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
396644d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3967b9c1b51eSKate Stone               } else {
396844d93782SGreg Clayton                 m_disassembly_range.Clear();
396944d93782SGreg Clayton               }
3970b9c1b51eSKate Stone             } else {
397144d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
397244d93782SGreg Clayton             }
397344d93782SGreg Clayton           }
397444d93782SGreg Clayton         }
3975b9c1b51eSKate Stone       } else {
397644d93782SGreg Clayton         m_pc_line = UINT32_MAX;
397744d93782SGreg Clayton       }
397844d93782SGreg Clayton     }
397944d93782SGreg Clayton 
3980ec990867SGreg Clayton     const int window_width = window.GetWidth();
398144d93782SGreg Clayton     window.Erase();
398244d93782SGreg Clayton     window.DrawTitleBox("Sources");
3983b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
3984ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
3985ec990867SGreg Clayton       window.MoveCursor(1, 1);
3986ec990867SGreg Clayton       window.PutChar(' ');
3987ec990867SGreg Clayton       window.PutCStringTruncated(m_title.GetString().c_str(), 1);
3988ec990867SGreg Clayton       int x = window.GetCursorX();
3989b9c1b51eSKate Stone       if (x < window_width - 1) {
3990ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
3991ec990867SGreg Clayton       }
3992ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
3993ec990867SGreg Clayton     }
399444d93782SGreg Clayton 
399544d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
399644d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
3997b9c1b51eSKate Stone     if (num_source_lines > 0) {
399844d93782SGreg Clayton       // Display source
399944d93782SGreg Clayton       BreakpointLines bp_lines;
4000b9c1b51eSKate Stone       if (target) {
400144d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
400244d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
4003b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
400444d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
400544d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
4006b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
4007b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
4008b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
400944d93782SGreg Clayton             LineEntry bp_loc_line_entry;
4010b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
4011b9c1b51eSKate Stone                     bp_loc_line_entry)) {
4012b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
401344d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
401444d93782SGreg Clayton               }
401544d93782SGreg Clayton             }
401644d93782SGreg Clayton           }
401744d93782SGreg Clayton         }
401844d93782SGreg Clayton       }
401944d93782SGreg Clayton 
402044d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
402144d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
402244d93782SGreg Clayton 
4023b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
402444d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4025b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4026ec990867SGreg Clayton           const int line_y = m_min_y + i;
402744d93782SGreg Clayton           window.MoveCursor(1, line_y);
402844d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
402944d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
403044d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
403144d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
403244d93782SGreg Clayton           attr_t highlight_attr = 0;
403344d93782SGreg Clayton           attr_t bp_attr = 0;
403444d93782SGreg Clayton           if (is_pc_line)
403544d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
403644d93782SGreg Clayton           else if (line_is_selected)
403744d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
403844d93782SGreg Clayton 
403944d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
404044d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
404144d93782SGreg Clayton 
404244d93782SGreg Clayton           if (bp_attr)
404344d93782SGreg Clayton             window.AttributeOn(bp_attr);
404444d93782SGreg Clayton 
404544d93782SGreg Clayton           window.Printf(m_line_format, curr_line + 1);
404644d93782SGreg Clayton 
404744d93782SGreg Clayton           if (bp_attr)
404844d93782SGreg Clayton             window.AttributeOff(bp_attr);
404944d93782SGreg Clayton 
405044d93782SGreg Clayton           window.PutChar(ACS_VLINE);
405144d93782SGreg Clayton           // Mark the line with the PC with a diamond
405244d93782SGreg Clayton           if (is_pc_line)
405344d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
405444d93782SGreg Clayton           else
405544d93782SGreg Clayton             window.PutChar(' ');
405644d93782SGreg Clayton 
405744d93782SGreg Clayton           if (highlight_attr)
405844d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4059b9c1b51eSKate Stone           const uint32_t line_len =
4060b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
406144d93782SGreg Clayton           if (line_len > 0)
406244d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
406344d93782SGreg Clayton 
4064b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4065b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
406644d93782SGreg Clayton             StopInfoSP stop_info_sp;
406744d93782SGreg Clayton             if (thread)
406844d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4069b9c1b51eSKate Stone             if (stop_info_sp) {
407044d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4071b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
407244d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4073ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
407444d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4075b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4076b9c1b51eSKate Stone                 // line_y);
4077b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4078b9c1b51eSKate Stone                               stop_description);
407944d93782SGreg Clayton               }
4080b9c1b51eSKate Stone             } else {
4081ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
408244d93782SGreg Clayton             }
408344d93782SGreg Clayton           }
408444d93782SGreg Clayton           if (highlight_attr)
408544d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4086b9c1b51eSKate Stone         } else {
408744d93782SGreg Clayton           break;
408844d93782SGreg Clayton         }
408944d93782SGreg Clayton       }
4090b9c1b51eSKate Stone     } else {
409144d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4092b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
409344d93782SGreg Clayton         // Display disassembly
409444d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
409544d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4096b9c1b51eSKate Stone         if (target) {
409744d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
409844d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4099b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
410044d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
410144d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4102b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4103b9c1b51eSKate Stone                  ++bp_loc_idx) {
4104b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4105b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
410644d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4107b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4108b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4109b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
411044d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
411144d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
411244d93782SGreg Clayton               }
411344d93782SGreg Clayton             }
411444d93782SGreg Clayton           }
411544d93782SGreg Clayton         }
411644d93782SGreg Clayton 
411744d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
411844d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
411944d93782SGreg Clayton 
412044d93782SGreg Clayton         StreamString strm;
412144d93782SGreg Clayton 
412244d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
412344d93782SGreg Clayton         Address pc_address;
412444d93782SGreg Clayton 
412544d93782SGreg Clayton         if (frame_sp)
412644d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4127b9c1b51eSKate Stone         const uint32_t pc_idx =
4128b9c1b51eSKate Stone             pc_address.IsValid()
4129b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4130b9c1b51eSKate Stone                 : UINT32_MAX;
4131b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
413244d93782SGreg Clayton           m_selected_line = pc_idx;
413344d93782SGreg Clayton         }
413444d93782SGreg Clayton 
413544d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41363985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
413744d93782SGreg Clayton           m_first_visible_line = 0;
413844d93782SGreg Clayton 
4139b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41403985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
414144d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
414244d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
414344d93782SGreg Clayton         }
414444d93782SGreg Clayton 
4145b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
414644d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
414744d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
414844d93782SGreg Clayton           if (!inst)
414944d93782SGreg Clayton             break;
415044d93782SGreg Clayton 
4151ec990867SGreg Clayton           const int line_y = m_min_y + i;
4152ec990867SGreg Clayton           window.MoveCursor(1, line_y);
415344d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
415444d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
415544d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
415644d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
415744d93782SGreg Clayton           attr_t highlight_attr = 0;
415844d93782SGreg Clayton           attr_t bp_attr = 0;
415944d93782SGreg Clayton           if (is_pc_line)
416044d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
416144d93782SGreg Clayton           else if (line_is_selected)
416244d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
416344d93782SGreg Clayton 
4164b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4165b9c1b51eSKate Stone               bp_file_addrs.end())
416644d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
416744d93782SGreg Clayton 
416844d93782SGreg Clayton           if (bp_attr)
416944d93782SGreg Clayton             window.AttributeOn(bp_attr);
417044d93782SGreg Clayton 
4171324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4172b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4173b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
417444d93782SGreg Clayton 
417544d93782SGreg Clayton           if (bp_attr)
417644d93782SGreg Clayton             window.AttributeOff(bp_attr);
417744d93782SGreg Clayton 
417844d93782SGreg Clayton           window.PutChar(ACS_VLINE);
417944d93782SGreg Clayton           // Mark the line with the PC with a diamond
418044d93782SGreg Clayton           if (is_pc_line)
418144d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
418244d93782SGreg Clayton           else
418344d93782SGreg Clayton             window.PutChar(' ');
418444d93782SGreg Clayton 
418544d93782SGreg Clayton           if (highlight_attr)
418644d93782SGreg Clayton             window.AttributeOn(highlight_attr);
418744d93782SGreg Clayton 
418844d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
418944d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
419044d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
419144d93782SGreg Clayton 
4192c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4193c5dac77aSEugene Zelenko             mnemonic = nullptr;
4194c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4195c5dac77aSEugene Zelenko             operands = nullptr;
4196c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4197c5dac77aSEugene Zelenko             comment = nullptr;
419844d93782SGreg Clayton 
419944d93782SGreg Clayton           strm.Clear();
420044d93782SGreg Clayton 
4201c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
420244d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4203c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
420444d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4205c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
420644d93782SGreg Clayton             strm.Printf("%s", mnemonic);
420744d93782SGreg Clayton 
420844d93782SGreg Clayton           int right_pad = 1;
420944d93782SGreg Clayton           window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
421044d93782SGreg Clayton 
4211b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4212b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
421344d93782SGreg Clayton             StopInfoSP stop_info_sp;
421444d93782SGreg Clayton             if (thread)
421544d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4216b9c1b51eSKate Stone             if (stop_info_sp) {
421744d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4218b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
421944d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4220ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
422144d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4222b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4223b9c1b51eSKate Stone                 // line_y);
4224b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4225b9c1b51eSKate Stone                               stop_description);
422644d93782SGreg Clayton               }
4227b9c1b51eSKate Stone             } else {
4228ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
422944d93782SGreg Clayton             }
423044d93782SGreg Clayton           }
423144d93782SGreg Clayton           if (highlight_attr)
423244d93782SGreg Clayton             window.AttributeOff(highlight_attr);
423344d93782SGreg Clayton         }
423444d93782SGreg Clayton       }
423544d93782SGreg Clayton     }
423644d93782SGreg Clayton     window.DeferredRefresh();
423744d93782SGreg Clayton     return true; // Drawing handled
423844d93782SGreg Clayton   }
423944d93782SGreg Clayton 
4240b9c1b51eSKate Stone   size_t GetNumLines() {
424144d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
424244d93782SGreg Clayton     if (num_lines == 0)
424344d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
424444d93782SGreg Clayton     return num_lines;
424544d93782SGreg Clayton   }
424644d93782SGreg Clayton 
4247b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
424844d93782SGreg Clayton     if (m_file_sp)
424944d93782SGreg Clayton       return m_file_sp->GetNumLines();
425044d93782SGreg Clayton     return 0;
425144d93782SGreg Clayton   }
4252315b6884SEugene Zelenko 
4253b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
425444d93782SGreg Clayton     if (m_disassembly_sp)
425544d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
425644d93782SGreg Clayton     return 0;
425744d93782SGreg Clayton   }
425844d93782SGreg Clayton 
4259b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
426044d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
426144d93782SGreg Clayton     const size_t num_lines = GetNumLines();
426244d93782SGreg Clayton 
4263b9c1b51eSKate Stone     switch (c) {
426444d93782SGreg Clayton     case ',':
426544d93782SGreg Clayton     case KEY_PPAGE:
426644d93782SGreg Clayton       // Page up key
42673985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
426844d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
426944d93782SGreg Clayton       else
427044d93782SGreg Clayton         m_first_visible_line = 0;
427144d93782SGreg Clayton       m_selected_line = m_first_visible_line;
427244d93782SGreg Clayton       return eKeyHandled;
427344d93782SGreg Clayton 
427444d93782SGreg Clayton     case '.':
427544d93782SGreg Clayton     case KEY_NPAGE:
427644d93782SGreg Clayton       // Page down key
427744d93782SGreg Clayton       {
427844d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
427944d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
428044d93782SGreg Clayton         else if (num_lines < num_visible_lines)
428144d93782SGreg Clayton           m_first_visible_line = 0;
428244d93782SGreg Clayton         else
428344d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
428444d93782SGreg Clayton         m_selected_line = m_first_visible_line;
428544d93782SGreg Clayton       }
428644d93782SGreg Clayton       return eKeyHandled;
428744d93782SGreg Clayton 
428844d93782SGreg Clayton     case KEY_UP:
4289b9c1b51eSKate Stone       if (m_selected_line > 0) {
429044d93782SGreg Clayton         m_selected_line--;
42913985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
429244d93782SGreg Clayton           m_first_visible_line = m_selected_line;
429344d93782SGreg Clayton       }
429444d93782SGreg Clayton       return eKeyHandled;
429544d93782SGreg Clayton 
429644d93782SGreg Clayton     case KEY_DOWN:
4297b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
429844d93782SGreg Clayton         m_selected_line++;
429944d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
430044d93782SGreg Clayton           m_first_visible_line++;
430144d93782SGreg Clayton       }
430244d93782SGreg Clayton       return eKeyHandled;
430344d93782SGreg Clayton 
430444d93782SGreg Clayton     case '\r':
430544d93782SGreg Clayton     case '\n':
430644d93782SGreg Clayton     case KEY_ENTER:
430744d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4308b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4309b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4310b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4311b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4312b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4313b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
431444d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4315b9c1b51eSKate Stone               m_selected_line +
4316b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
43172411167fSJim Ingham               0,     // No offset
431844d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
431944d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
432044d93782SGreg Clayton               false,               // internal
4321055ad9beSIlia K               false,               // request_hardware
4322055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
432344d93782SGreg Clayton           // Make breakpoint one shot
432444d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
432544d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
432644d93782SGreg Clayton         }
4327b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4328b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4329b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4330b9c1b51eSKate Stone                                       .get();
4331b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4332b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4333b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
433444d93782SGreg Clayton           Address addr = inst->GetAddress();
4335b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4336b9c1b51eSKate Stone               addr,   // lldb_private::Address
433744d93782SGreg Clayton               false,  // internal
433844d93782SGreg Clayton               false); // request_hardware
433944d93782SGreg Clayton           // Make breakpoint one shot
434044d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
434144d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
434244d93782SGreg Clayton         }
434344d93782SGreg Clayton       }
434444d93782SGreg Clayton       return eKeyHandled;
434544d93782SGreg Clayton 
434644d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4347b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4348b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4349b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4350b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4351b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4352b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
435344d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4354b9c1b51eSKate Stone               m_selected_line +
4355b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
43562411167fSJim Ingham               0,     // No offset
435744d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
435844d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
435944d93782SGreg Clayton               false,               // internal
4360055ad9beSIlia K               false,               // request_hardware
4361055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
436244d93782SGreg Clayton         }
4363b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4364b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4365b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4366b9c1b51eSKate Stone                                       .get();
4367b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4368b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4369b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
437044d93782SGreg Clayton           Address addr = inst->GetAddress();
4371b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4372b9c1b51eSKate Stone               addr,   // lldb_private::Address
437344d93782SGreg Clayton               false,  // internal
437444d93782SGreg Clayton               false); // request_hardware
437544d93782SGreg Clayton         }
437644d93782SGreg Clayton       }
437744d93782SGreg Clayton       return eKeyHandled;
437844d93782SGreg Clayton 
437944d93782SGreg Clayton     case 'd': // 'd' == detach and let run
438044d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
438144d93782SGreg Clayton     {
4382b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4383b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
438444d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
438544d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
438644d93782SGreg Clayton     }
438744d93782SGreg Clayton       return eKeyHandled;
438844d93782SGreg Clayton 
438944d93782SGreg Clayton     case 'k':
439044d93782SGreg Clayton       // 'k' == kill
439144d93782SGreg Clayton       {
4392b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4393b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
439444d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4395ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
439644d93782SGreg Clayton       }
439744d93782SGreg Clayton       return eKeyHandled;
439844d93782SGreg Clayton 
439944d93782SGreg Clayton     case 'c':
440044d93782SGreg Clayton       // 'c' == continue
440144d93782SGreg Clayton       {
4402b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4403b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
440444d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
440544d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
440644d93782SGreg Clayton       }
440744d93782SGreg Clayton       return eKeyHandled;
440844d93782SGreg Clayton 
440944d93782SGreg Clayton     case 'o':
441044d93782SGreg Clayton       // 'o' == step out
441144d93782SGreg Clayton       {
4412b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4413b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4414b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4415b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
441644d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
441744d93782SGreg Clayton         }
441844d93782SGreg Clayton       }
441944d93782SGreg Clayton       return eKeyHandled;
4420315b6884SEugene Zelenko 
442144d93782SGreg Clayton     case 'n': // 'n' == step over
442244d93782SGreg Clayton     case 'N': // 'N' == step over instruction
442344d93782SGreg Clayton     {
4424b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4425b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4426b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4427b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
442844d93782SGreg Clayton         bool source_step = (c == 'n');
442944d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
443044d93782SGreg Clayton       }
443144d93782SGreg Clayton     }
443244d93782SGreg Clayton       return eKeyHandled;
4433315b6884SEugene Zelenko 
443444d93782SGreg Clayton     case 's': // 's' == step into
443544d93782SGreg Clayton     case 'S': // 'S' == step into instruction
443644d93782SGreg Clayton     {
4437b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4438b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4439b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4440b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
444144d93782SGreg Clayton         bool source_step = (c == 's');
44424b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
444344d93782SGreg Clayton       }
444444d93782SGreg Clayton     }
444544d93782SGreg Clayton       return eKeyHandled;
444644d93782SGreg Clayton 
444744d93782SGreg Clayton     case 'h':
444844d93782SGreg Clayton       window.CreateHelpSubwindow();
444944d93782SGreg Clayton       return eKeyHandled;
445044d93782SGreg Clayton 
445144d93782SGreg Clayton     default:
445244d93782SGreg Clayton       break;
445344d93782SGreg Clayton     }
445444d93782SGreg Clayton     return eKeyNotHandled;
445544d93782SGreg Clayton   }
445644d93782SGreg Clayton 
445744d93782SGreg Clayton protected:
445844d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
445944d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
446044d93782SGreg Clayton 
446144d93782SGreg Clayton   Debugger &m_debugger;
446244d93782SGreg Clayton   SymbolContext m_sc;
446344d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
446444d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
446544d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
446644d93782SGreg Clayton   AddressRange m_disassembly_range;
4467ec990867SGreg Clayton   StreamString m_title;
446844d93782SGreg Clayton   lldb::user_id_t m_tid;
446944d93782SGreg Clayton   char m_line_format[8];
447044d93782SGreg Clayton   int m_line_width;
447144d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
447244d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
447344d93782SGreg Clayton   uint32_t m_stop_id;
447444d93782SGreg Clayton   uint32_t m_frame_idx;
447544d93782SGreg Clayton   int m_first_visible_line;
447644d93782SGreg Clayton   int m_min_x;
447744d93782SGreg Clayton   int m_min_y;
447844d93782SGreg Clayton   int m_max_x;
447944d93782SGreg Clayton   int m_max_y;
448044d93782SGreg Clayton };
448144d93782SGreg Clayton 
448244d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
448344d93782SGreg Clayton 
4484b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4485b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
448644d93782SGreg Clayton 
4487b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
448844d93782SGreg Clayton   IOHandler::Activate();
4489b9c1b51eSKate Stone   if (!m_app_ap) {
449044d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
449144d93782SGreg Clayton 
449244d93782SGreg Clayton     // This is both a window and a menu delegate
4493b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4494b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
449544d93782SGreg Clayton 
4496b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4497b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4498b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4499b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4500b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4501b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
450244d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4503b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4504b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
450544d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
450644d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
450744d93782SGreg Clayton 
4508b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4509b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4510b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4511b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4512b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4513b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
451444d93782SGreg Clayton 
4515b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4516b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4517b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4518b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4519b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4520b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4521b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4522b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
452344d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4524b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4525b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4526b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4527b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4528b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4529b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4530b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
453144d93782SGreg Clayton 
4532b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4533b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4534b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4535b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4536b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4537b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4538b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4539b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4540b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
454144d93782SGreg Clayton 
4542b9c1b51eSKate Stone     MenuSP view_menu_sp(
4543b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4544b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4545b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4546b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4547b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4548b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4549b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4550b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4551b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4552b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4553b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4554b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
455544d93782SGreg Clayton 
4556b9c1b51eSKate Stone     MenuSP help_menu_sp(
4557b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4558b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4559b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
456044d93782SGreg Clayton 
456144d93782SGreg Clayton     m_app_ap->Initialize();
456244d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
456344d93782SGreg Clayton 
456444d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
456544d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
456644d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
456744d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
456844d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
456944d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
457044d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
457144d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
457244d93782SGreg Clayton 
457344d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
457444d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
457544d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
457644d93782SGreg Clayton     Rect source_bounds;
457744d93782SGreg Clayton     Rect variables_bounds;
457844d93782SGreg Clayton     Rect threads_bounds;
457944d93782SGreg Clayton     Rect source_variables_bounds;
4580b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4581b9c1b51eSKate Stone                                            threads_bounds);
4582b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4583b9c1b51eSKate Stone                                                       variables_bounds);
458444d93782SGreg Clayton 
4585b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4586b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
458744d93782SGreg Clayton     // Let the menubar get keys if the active window doesn't handle the
458844d93782SGreg Clayton     // keys that are typed so it can respond to menubar key presses.
4589b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4590b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
459144d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
459244d93782SGreg Clayton 
4593b9c1b51eSKate Stone     WindowSP source_window_sp(
4594b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4595b9c1b51eSKate Stone     WindowSP variables_window_sp(
4596b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4597b9c1b51eSKate Stone     WindowSP threads_window_sp(
4598b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4599b9c1b51eSKate Stone     WindowSP status_window_sp(
4600b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4601b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4602b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4603b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4604b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4605b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4606b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4607b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4608b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4609ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4610b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4611b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4612b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4613b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
461444d93782SGreg Clayton 
46155fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
46165fdb09bbSGreg Clayton     static bool g_showed_help = false;
4617b9c1b51eSKate Stone     if (!g_showed_help) {
46185fdb09bbSGreg Clayton       g_showed_help = true;
46195fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46205fdb09bbSGreg Clayton     }
46215fdb09bbSGreg Clayton 
462244d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
462344d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
462444d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
462544d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
462644d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
462744d93782SGreg Clayton   }
462844d93782SGreg Clayton }
462944d93782SGreg Clayton 
4630b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
463144d93782SGreg Clayton 
4632b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
463344d93782SGreg Clayton   m_app_ap->Run(m_debugger);
463444d93782SGreg Clayton   SetIsDone(true);
463544d93782SGreg Clayton }
463644d93782SGreg Clayton 
4637315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
463844d93782SGreg Clayton 
4639b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
464044d93782SGreg Clayton 
4641b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
464244d93782SGreg Clayton 
4643b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
464444d93782SGreg Clayton 
4645315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4646