144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
644d93782SGreg Clayton //
744d93782SGreg Clayton //===----------------------------------------------------------------------===//
844d93782SGreg Clayton 
92f3df613SZachary Turner #include "lldb/Core/IOHandler.h"
102f3df613SZachary Turner 
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
167c9aa073STodd Fiala #if defined(__APPLE__)
177c9aa073STodd Fiala #include <deque>
187c9aa073STodd Fiala #endif
1944d93782SGreg Clayton #include <string>
2044d93782SGreg Clayton 
2144d93782SGreg Clayton #include "lldb/Core/Debugger.h"
2244d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
23672d2c12SJonas Devlieghere #include "lldb/Host/File.h"
24672d2c12SJonas Devlieghere #include "lldb/Utility/Predicate.h"
25672d2c12SJonas Devlieghere #include "lldb/Utility/Status.h"
26672d2c12SJonas Devlieghere #include "lldb/Utility/StreamString.h"
27672d2c12SJonas Devlieghere #include "lldb/Utility/StringList.h"
28672d2c12SJonas Devlieghere #include "lldb/lldb-forward.h"
292f3df613SZachary Turner 
30cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3144d93782SGreg Clayton #include "lldb/Host/Editline.h"
32cacde7dfSTodd Fiala #endif
3344d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
352f3df613SZachary Turner #ifndef LLDB_DISABLE_CURSES
362f3df613SZachary Turner #include "lldb/Breakpoint/BreakpointLocation.h"
372f3df613SZachary Turner #include "lldb/Core/Module.h"
382f3df613SZachary Turner #include "lldb/Core/ValueObject.h"
392f3df613SZachary Turner #include "lldb/Core/ValueObjectRegister.h"
4044d93782SGreg Clayton #include "lldb/Symbol/Block.h"
4144d93782SGreg Clayton #include "lldb/Symbol/Function.h"
4244d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
43c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
44c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
452f3df613SZachary Turner #include "lldb/Target/RegisterContext.h"
46c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
472f3df613SZachary Turner #include "lldb/Target/StopInfo.h"
48b9c1b51eSKate Stone #include "lldb/Target/Target.h"
49b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
50d821c997SPavel Labath #include "lldb/Utility/State.h"
51c5dac77aSEugene Zelenko #endif
527c9aa073STodd Fiala 
53672d2c12SJonas Devlieghere #include "llvm/ADT/StringRef.h"
542f3df613SZachary Turner 
55fab31220STed Woodward #ifdef _MSC_VER
56aaea8ee6SZachary Turner #include "lldb/Host/windows/windows.h"
57fab31220STed Woodward #endif
58fab31220STed Woodward 
59672d2c12SJonas Devlieghere #include <memory>
60672d2c12SJonas Devlieghere #include <mutex>
612f3df613SZachary Turner 
62672d2c12SJonas Devlieghere #include <assert.h>
63672d2c12SJonas Devlieghere #include <ctype.h>
64672d2c12SJonas Devlieghere #include <errno.h>
65672d2c12SJonas Devlieghere #include <locale.h>
66672d2c12SJonas Devlieghere #include <stdint.h>
67672d2c12SJonas Devlieghere #include <stdio.h>
68672d2c12SJonas Devlieghere #include <string.h>
69672d2c12SJonas Devlieghere #include <type_traits>
702f3df613SZachary Turner 
7144d93782SGreg Clayton using namespace lldb;
7244d93782SGreg Clayton using namespace lldb_private;
7344d93782SGreg Clayton 
74b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type)
75b9c1b51eSKate Stone     : IOHandler(debugger, type,
7644d93782SGreg Clayton                 StreamFileSP(), // Adopt STDIN from top input reader
7744d93782SGreg Clayton                 StreamFileSP(), // Adopt STDOUT from top input reader
78340b0309SGreg Clayton                 StreamFileSP(), // Adopt STDERR from top input reader
79340b0309SGreg Clayton                 0)              // Flags
80b9c1b51eSKate Stone {}
8144d93782SGreg Clayton 
82b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
8344d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
8444d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
85b9c1b51eSKate Stone                      const lldb::StreamFileSP &error_sp, uint32_t flags)
86b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
87b9c1b51eSKate Stone       m_error_sp(error_sp), m_popped(false), m_flags(flags), m_type(type),
88b9c1b51eSKate Stone       m_user_data(nullptr), m_done(false), m_active(false) {
8944d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
9044d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
91b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
9244d93782SGreg Clayton                                              m_error_sp);
9344d93782SGreg Clayton }
9444d93782SGreg Clayton 
95315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
9644d93782SGreg Clayton 
97b9c1b51eSKate Stone int IOHandler::GetInputFD() {
98c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
9944d93782SGreg Clayton }
10044d93782SGreg Clayton 
101b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
102c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10344d93782SGreg Clayton }
10444d93782SGreg Clayton 
105b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
106c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
10744d93782SGreg Clayton }
10844d93782SGreg Clayton 
109b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
110c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11144d93782SGreg Clayton }
11244d93782SGreg Clayton 
113b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
114c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
11544d93782SGreg Clayton }
11644d93782SGreg Clayton 
117b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
118c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
11944d93782SGreg Clayton }
12044d93782SGreg Clayton 
121b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
12244d93782SGreg Clayton 
123b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
12444d93782SGreg Clayton 
125b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
12644d93782SGreg Clayton 
127b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
128340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
129340b0309SGreg Clayton }
130340b0309SGreg Clayton 
131b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
132340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
133340b0309SGreg Clayton }
13444d93782SGreg Clayton 
135b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
136e30f11d9SKate Stone 
137b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
138e30f11d9SKate Stone 
139b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
140b9c1b51eSKate Stone   if (stream) {
14116ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1424446487dSPavel Labath     if (m_top)
1434446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1444446487dSPavel Labath   }
1454446487dSPavel Labath }
1464446487dSPavel Labath 
1477a120c8bSZachary Turner IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, llvm::StringRef prompt,
148b9c1b51eSKate Stone                                    bool default_response)
149b9c1b51eSKate Stone     : IOHandlerEditline(
150b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
151c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
152514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
153514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
15444d93782SGreg Clayton           false,             // Multi-line
155e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
156b9c1b51eSKate Stone           0, *this),
157b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
15844d93782SGreg Clayton   StreamString prompt_stream;
15944d93782SGreg Clayton   prompt_stream.PutCString(prompt);
16044d93782SGreg Clayton   if (m_default_response)
16144d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
16244d93782SGreg Clayton   else
16344d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
16444d93782SGreg Clayton 
165514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
16644d93782SGreg Clayton }
16744d93782SGreg Clayton 
168315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
16944d93782SGreg Clayton 
1707f88829cSRaphael Isemann int IOHandlerConfirm::IOHandlerComplete(
1717f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
1727f88829cSRaphael Isemann     const char *last_char, int skip_first_n_matches, int max_matches,
1737f88829cSRaphael Isemann     StringList &matches, StringList &descriptions) {
174b9c1b51eSKate Stone   if (current_line == cursor) {
175b9c1b51eSKate Stone     if (m_default_response) {
17644d93782SGreg Clayton       matches.AppendString("y");
177b9c1b51eSKate Stone     } else {
17844d93782SGreg Clayton       matches.AppendString("n");
17944d93782SGreg Clayton     }
18044d93782SGreg Clayton   }
18144d93782SGreg Clayton   return matches.GetSize();
18244d93782SGreg Clayton }
18344d93782SGreg Clayton 
184b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
185b9c1b51eSKate Stone                                               std::string &line) {
186b9c1b51eSKate Stone   if (line.empty()) {
18744d93782SGreg Clayton     // User just hit enter, set the response to the default
18844d93782SGreg Clayton     m_user_response = m_default_response;
18944d93782SGreg Clayton     io_handler.SetIsDone(true);
19044d93782SGreg Clayton     return;
19144d93782SGreg Clayton   }
19244d93782SGreg Clayton 
193b9c1b51eSKate Stone   if (line.size() == 1) {
194b9c1b51eSKate Stone     switch (line[0]) {
19544d93782SGreg Clayton     case 'y':
19644d93782SGreg Clayton     case 'Y':
19744d93782SGreg Clayton       m_user_response = true;
19844d93782SGreg Clayton       io_handler.SetIsDone(true);
19944d93782SGreg Clayton       return;
20044d93782SGreg Clayton     case 'n':
20144d93782SGreg Clayton     case 'N':
20244d93782SGreg Clayton       m_user_response = false;
20344d93782SGreg Clayton       io_handler.SetIsDone(true);
20444d93782SGreg Clayton       return;
20544d93782SGreg Clayton     default:
20644d93782SGreg Clayton       break;
20744d93782SGreg Clayton     }
20844d93782SGreg Clayton   }
20944d93782SGreg Clayton 
210b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
21144d93782SGreg Clayton     m_user_response = true;
21244d93782SGreg Clayton     io_handler.SetIsDone(true);
213b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
21444d93782SGreg Clayton     m_user_response = false;
21544d93782SGreg Clayton     io_handler.SetIsDone(true);
21644d93782SGreg Clayton   }
21744d93782SGreg Clayton }
21844d93782SGreg Clayton 
2197f88829cSRaphael Isemann int IOHandlerDelegate::IOHandlerComplete(
2207f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
2217f88829cSRaphael Isemann     const char *last_char, int skip_first_n_matches, int max_matches,
2227f88829cSRaphael Isemann     StringList &matches, StringList &descriptions) {
223b9c1b51eSKate Stone   switch (m_completion) {
22444d93782SGreg Clayton   case Completion::None:
22544d93782SGreg Clayton     break;
22644d93782SGreg Clayton 
22744d93782SGreg Clayton   case Completion::LLDBCommand:
228b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
229b9c1b51eSKate Stone         current_line, cursor, last_char, skip_first_n_matches, max_matches,
2307f88829cSRaphael Isemann         matches, descriptions);
231b9c1b51eSKate Stone   case Completion::Expression: {
2327f88829cSRaphael Isemann     CompletionResult result;
233a2e76c0bSRaphael Isemann     CompletionRequest request(current_line, current_line - cursor,
2347f88829cSRaphael Isemann                               skip_first_n_matches, max_matches, result);
235b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
236b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
237a2e76c0bSRaphael Isemann         CommandCompletions::eVariablePathCompletion, request, nullptr);
2387f88829cSRaphael Isemann     result.GetMatches(matches);
2397f88829cSRaphael Isemann     result.GetDescriptions(descriptions);
24044d93782SGreg Clayton 
2411a6d7ab5SRaphael Isemann     size_t num_matches = request.GetNumberOfMatches();
242b9c1b51eSKate Stone     if (num_matches > 0) {
24344d93782SGreg Clayton       std::string common_prefix;
2441a6d7ab5SRaphael Isemann       matches.LongestCommonPrefix(common_prefix);
245a2e76c0bSRaphael Isemann       const size_t partial_name_len = request.GetCursorArgumentPrefix().size();
24644d93782SGreg Clayton 
24705097246SAdrian Prantl       // If we matched a unique single command, add a space... Only do this if
24805097246SAdrian Prantl       // the completer told us this was a complete word, however...
249a2e76c0bSRaphael Isemann       if (num_matches == 1 && request.GetWordComplete()) {
25044d93782SGreg Clayton         common_prefix.push_back(' ');
25144d93782SGreg Clayton       }
25244d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
25344d93782SGreg Clayton       matches.InsertStringAtIndex(0, std::move(common_prefix));
25444d93782SGreg Clayton     }
25544d93782SGreg Clayton     return num_matches;
256b9c1b51eSKate Stone   } break;
25744d93782SGreg Clayton   }
25844d93782SGreg Clayton 
25944d93782SGreg Clayton   return 0;
26044d93782SGreg Clayton }
26144d93782SGreg Clayton 
262b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
263b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
26444d93782SGreg Clayton     const char *editline_name, // Used for saving history files
265514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
266514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
267514d8cd8SZachary Turner     IOHandlerDelegate &delegate)
268b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
26944d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
27044d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
27144d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
272340b0309SGreg Clayton                         0,              // Flags
27344d93782SGreg Clayton                         editline_name,  // Used for saving history files
274b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
275b9c1b51eSKate Stone                         line_number_start, delegate) {}
27644d93782SGreg Clayton 
277b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
278b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
279b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
280b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
28144d93782SGreg Clayton     const char *editline_name, // Used for saving history files
282514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
283514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
284514d8cd8SZachary Turner     IOHandlerDelegate &delegate)
285b9c1b51eSKate Stone     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags),
286cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
287d5b44036SJonas Devlieghere       m_editline_up(),
288cacde7dfSTodd Fiala #endif
289b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
290b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
291b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
292b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
293b9c1b51eSKate Stone       m_editing(false) {
29444d93782SGreg Clayton   SetPrompt(prompt);
29544d93782SGreg Clayton 
296cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
297914b8d98SDeepak Panickal   bool use_editline = false;
298340b0309SGreg Clayton 
299340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
30044d93782SGreg Clayton 
301b9c1b51eSKate Stone   if (use_editline) {
302d5b44036SJonas Devlieghere     m_editline_up.reset(new Editline(editline_name, GetInputFILE(),
303b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
304e30f11d9SKate Stone                                      m_color_prompts));
305d5b44036SJonas Devlieghere     m_editline_up->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
306d5b44036SJonas Devlieghere     m_editline_up->SetAutoCompleteCallback(AutoCompleteCallback, this);
307e30f11d9SKate Stone     // See if the delegate supports fixing indentation
308e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
309b9c1b51eSKate Stone     if (indent_chars) {
310b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
31105097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
312d5b44036SJonas Devlieghere       m_editline_up->SetFixIndentationCallback(FixIndentationCallback, this,
313b9c1b51eSKate Stone                                                indent_chars);
314e30f11d9SKate Stone     }
31544d93782SGreg Clayton   }
316cacde7dfSTodd Fiala #endif
317e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
318514d8cd8SZachary Turner   SetPrompt(prompt);
319e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
32044d93782SGreg Clayton }
32144d93782SGreg Clayton 
322b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
323cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
324d5b44036SJonas Devlieghere   m_editline_up.reset();
325cacde7dfSTodd Fiala #endif
32644d93782SGreg Clayton }
32744d93782SGreg Clayton 
328b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
329e30f11d9SKate Stone   IOHandler::Activate();
330e30f11d9SKate Stone   m_delegate.IOHandlerActivated(*this);
331e30f11d9SKate Stone }
332e30f11d9SKate Stone 
333b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
334e30f11d9SKate Stone   IOHandler::Deactivate();
335e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
336e30f11d9SKate Stone }
337e30f11d9SKate Stone 
338b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
339cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
340d5b44036SJonas Devlieghere   if (m_editline_up) {
341d5b44036SJonas Devlieghere     return m_editline_up->GetLine(line, interrupted);
342b9c1b51eSKate Stone   } else {
343cacde7dfSTodd Fiala #endif
34444d93782SGreg Clayton     line.clear();
34544d93782SGreg Clayton 
34644d93782SGreg Clayton     FILE *in = GetInputFILE();
347b9c1b51eSKate Stone     if (in) {
348b9c1b51eSKate Stone       if (GetIsInteractive()) {
349c5dac77aSEugene Zelenko         const char *prompt = nullptr;
350e30f11d9SKate Stone 
351e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
352e30f11d9SKate Stone           prompt = GetContinuationPrompt();
353e30f11d9SKate Stone 
354c5dac77aSEugene Zelenko         if (prompt == nullptr)
355e30f11d9SKate Stone           prompt = GetPrompt();
356e30f11d9SKate Stone 
357b9c1b51eSKate Stone         if (prompt && prompt[0]) {
35844d93782SGreg Clayton           FILE *out = GetOutputFILE();
359b9c1b51eSKate Stone           if (out) {
36044d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
36144d93782SGreg Clayton             ::fflush(out);
36244d93782SGreg Clayton           }
36344d93782SGreg Clayton         }
36444d93782SGreg Clayton       }
36544d93782SGreg Clayton       char buffer[256];
36644d93782SGreg Clayton       bool done = false;
3670f86e6e7SGreg Clayton       bool got_line = false;
368e034a04eSGreg Clayton       m_editing = true;
369b9c1b51eSKate Stone       while (!done) {
370b9c1b51eSKate Stone         if (fgets(buffer, sizeof(buffer), in) == nullptr) {
371c7797accSGreg Clayton           const int saved_errno = errno;
372c9cf5798SGreg Clayton           if (feof(in))
37344d93782SGreg Clayton             done = true;
374b9c1b51eSKate Stone           else if (ferror(in)) {
375c7797accSGreg Clayton             if (saved_errno != EINTR)
376c7797accSGreg Clayton               done = true;
377c7797accSGreg Clayton           }
378b9c1b51eSKate Stone         } else {
3790f86e6e7SGreg Clayton           got_line = true;
38044d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
38144d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
38244d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
383b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
38444d93782SGreg Clayton             done = true;
38544d93782SGreg Clayton             // Strip trailing newlines
386b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
38744d93782SGreg Clayton               --buffer_len;
38844d93782SGreg Clayton               if (buffer_len == 0)
38944d93782SGreg Clayton                 break;
39044d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
39144d93782SGreg Clayton             }
39244d93782SGreg Clayton           }
39344d93782SGreg Clayton           line.append(buffer, buffer_len);
39444d93782SGreg Clayton         }
39544d93782SGreg Clayton       }
396e034a04eSGreg Clayton       m_editing = false;
39705097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
39805097246SAdrian Prantl       // true in this case.
3990f86e6e7SGreg Clayton       return got_line;
400b9c1b51eSKate Stone     } else {
40144d93782SGreg Clayton       // No more input file, we are done...
40244d93782SGreg Clayton       SetIsDone(true);
40344d93782SGreg Clayton     }
404340b0309SGreg Clayton     return false;
405cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
40644d93782SGreg Clayton   }
407cacde7dfSTodd Fiala #endif
40844d93782SGreg Clayton }
40944d93782SGreg Clayton 
410cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
411b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
41244d93782SGreg Clayton                                                 StringList &lines,
413b9c1b51eSKate Stone                                                 void *baton) {
41444d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
415b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
416b9c1b51eSKate Stone                                                               lines);
417e30f11d9SKate Stone }
418e30f11d9SKate Stone 
419b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
420e30f11d9SKate Stone                                               const StringList &lines,
421e30f11d9SKate Stone                                               int cursor_position,
422b9c1b51eSKate Stone                                               void *baton) {
423e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
424b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
425b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
42644d93782SGreg Clayton }
42744d93782SGreg Clayton 
4287f88829cSRaphael Isemann int IOHandlerEditline::AutoCompleteCallback(
4297f88829cSRaphael Isemann     const char *current_line, const char *cursor, const char *last_char,
4307f88829cSRaphael Isemann     int skip_first_n_matches, int max_matches, StringList &matches,
4317f88829cSRaphael Isemann     StringList &descriptions, void *baton) {
43244d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
43344d93782SGreg Clayton   if (editline_reader)
434b9c1b51eSKate Stone     return editline_reader->m_delegate.IOHandlerComplete(
435b9c1b51eSKate Stone         *editline_reader, current_line, cursor, last_char, skip_first_n_matches,
4367f88829cSRaphael Isemann         max_matches, matches, descriptions);
43744d93782SGreg Clayton   return 0;
43844d93782SGreg Clayton }
439cacde7dfSTodd Fiala #endif
44044d93782SGreg Clayton 
441b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
442cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
443d5b44036SJonas Devlieghere   if (m_editline_up) {
444d5b44036SJonas Devlieghere     return m_editline_up->GetPrompt();
445b9c1b51eSKate Stone   } else {
446cacde7dfSTodd Fiala #endif
447cacde7dfSTodd Fiala     if (m_prompt.empty())
448c5dac77aSEugene Zelenko       return nullptr;
449cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
450cacde7dfSTodd Fiala   }
451cacde7dfSTodd Fiala #endif
45244d93782SGreg Clayton   return m_prompt.c_str();
45344d93782SGreg Clayton }
45444d93782SGreg Clayton 
455514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
456514d8cd8SZachary Turner   m_prompt = prompt;
457514d8cd8SZachary Turner 
458cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
459d5b44036SJonas Devlieghere   if (m_editline_up)
460d5b44036SJonas Devlieghere     m_editline_up->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
461cacde7dfSTodd Fiala #endif
46244d93782SGreg Clayton   return true;
46344d93782SGreg Clayton }
46444d93782SGreg Clayton 
465b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
466b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
467b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
468e30f11d9SKate Stone }
469e30f11d9SKate Stone 
470514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
471514d8cd8SZachary Turner   m_continuation_prompt = prompt;
472e30f11d9SKate Stone 
473d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
474d5b44036SJonas Devlieghere   if (m_editline_up)
475d5b44036SJonas Devlieghere     m_editline_up->SetContinuationPrompt(m_continuation_prompt.empty()
476b9c1b51eSKate Stone                                              ? nullptr
477b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
478d553d00cSZachary Turner #endif
479e30f11d9SKate Stone }
480e30f11d9SKate Stone 
481b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
482f6913cd7SGreg Clayton   m_base_line_number = line;
483f6913cd7SGreg Clayton }
484e30f11d9SKate Stone 
485b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
486d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
487d5b44036SJonas Devlieghere   if (m_editline_up)
488d5b44036SJonas Devlieghere     return m_editline_up->GetCurrentLine();
489e30f11d9SKate Stone #endif
490e30f11d9SKate Stone   return m_curr_line_idx;
491e30f11d9SKate Stone }
492e30f11d9SKate Stone 
493b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
494e30f11d9SKate Stone   m_current_lines_ptr = &lines;
495e30f11d9SKate Stone 
49644d93782SGreg Clayton   bool success = false;
497cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
498d5b44036SJonas Devlieghere   if (m_editline_up) {
499d5b44036SJonas Devlieghere     return m_editline_up->GetLines(m_base_line_number, lines, interrupted);
500b9c1b51eSKate Stone   } else {
501cacde7dfSTodd Fiala #endif
502e30f11d9SKate Stone     bool done = false;
50397206d57SZachary Turner     Status error;
50444d93782SGreg Clayton 
505b9c1b51eSKate Stone     while (!done) {
506f6913cd7SGreg Clayton       // Show line numbers if we are asked to
50744d93782SGreg Clayton       std::string line;
508b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
509f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
510f6913cd7SGreg Clayton         if (out)
511b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
512b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
513f6913cd7SGreg Clayton       }
514f6913cd7SGreg Clayton 
515e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
516e30f11d9SKate Stone 
517f0066ad0SGreg Clayton       bool interrupted = false;
518b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
51944d93782SGreg Clayton         lines.AppendString(line);
520e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
521b9c1b51eSKate Stone       } else {
522e30f11d9SKate Stone         done = true;
52344d93782SGreg Clayton       }
52444d93782SGreg Clayton     }
52544d93782SGreg Clayton     success = lines.GetSize() > 0;
526cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
52744d93782SGreg Clayton   }
528cacde7dfSTodd Fiala #endif
52944d93782SGreg Clayton   return success;
53044d93782SGreg Clayton }
53144d93782SGreg Clayton 
53205097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
53305097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
534b9c1b51eSKate Stone void IOHandlerEditline::Run() {
53544d93782SGreg Clayton   std::string line;
536b9c1b51eSKate Stone   while (IsActive()) {
537f0066ad0SGreg Clayton     bool interrupted = false;
538b9c1b51eSKate Stone     if (m_multi_line) {
53944d93782SGreg Clayton       StringList lines;
540b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
541b9c1b51eSKate Stone         if (interrupted) {
542e30f11d9SKate Stone           m_done = m_interrupt_exits;
543e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
544e30f11d9SKate Stone 
545b9c1b51eSKate Stone         } else {
54644d93782SGreg Clayton           line = lines.CopyList();
54744d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
54844d93782SGreg Clayton         }
549b9c1b51eSKate Stone       } else {
55044d93782SGreg Clayton         m_done = true;
55144d93782SGreg Clayton       }
552b9c1b51eSKate Stone     } else {
553b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
554e30f11d9SKate Stone         if (interrupted)
555e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
556e30f11d9SKate Stone         else
55744d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
558b9c1b51eSKate Stone       } else {
55944d93782SGreg Clayton         m_done = true;
56044d93782SGreg Clayton       }
56144d93782SGreg Clayton     }
56244d93782SGreg Clayton   }
56344d93782SGreg Clayton }
56444d93782SGreg Clayton 
565b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
566cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
567d5b44036SJonas Devlieghere   if (m_editline_up)
568d5b44036SJonas Devlieghere     m_editline_up->Cancel();
569cacde7dfSTodd Fiala #endif
570e68f5d6bSGreg Clayton }
571e68f5d6bSGreg Clayton 
572b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
573f0066ad0SGreg Clayton   // Let the delgate handle it first
574f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
575f0066ad0SGreg Clayton     return true;
576f0066ad0SGreg Clayton 
577cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
578d5b44036SJonas Devlieghere   if (m_editline_up)
579d5b44036SJonas Devlieghere     return m_editline_up->Interrupt();
580cacde7dfSTodd Fiala #endif
581f0066ad0SGreg Clayton   return false;
58244d93782SGreg Clayton }
58344d93782SGreg Clayton 
584b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
585cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
586d5b44036SJonas Devlieghere   if (m_editline_up)
587d5b44036SJonas Devlieghere     m_editline_up->Interrupt();
588cacde7dfSTodd Fiala #endif
58944d93782SGreg Clayton }
59044d93782SGreg Clayton 
591b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
5924446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
593d5b44036SJonas Devlieghere   if (m_editline_up)
594d5b44036SJonas Devlieghere     m_editline_up->PrintAsync(stream, s, len);
5954446487dSPavel Labath   else
5964446487dSPavel Labath #endif
597fab31220STed Woodward   {
598fab31220STed Woodward #ifdef _MSC_VER
599341e4789SDawn Perchik     const char *prompt = GetPrompt();
600b9c1b51eSKate Stone     if (prompt) {
601fab31220STed Woodward       // Back up over previous prompt using Windows API
602fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
603fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
604fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
605fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
606fab31220STed Woodward       coord.X -= strlen(prompt);
607fab31220STed Woodward       if (coord.X < 0)
608fab31220STed Woodward         coord.X = 0;
609fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
610fab31220STed Woodward     }
611fab31220STed Woodward #endif
6124446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
613341e4789SDawn Perchik #ifdef _MSC_VER
614fab31220STed Woodward     if (prompt)
615b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
616b9c1b51eSKate Stone                             strlen(prompt));
617341e4789SDawn Perchik #endif
618fab31220STed Woodward   }
6194446487dSPavel Labath }
6204446487dSPavel Labath 
62105097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
622914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
623914b8d98SDeepak Panickal 
62444d93782SGreg Clayton #define KEY_RETURN 10
62544d93782SGreg Clayton #define KEY_ESCAPE 27
62644d93782SGreg Clayton 
627b9c1b51eSKate Stone namespace curses {
62844d93782SGreg Clayton class Menu;
62944d93782SGreg Clayton class MenuDelegate;
63044d93782SGreg Clayton class Window;
63144d93782SGreg Clayton class WindowDelegate;
63244d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
63344d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
63444d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
63544d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
63644d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
63744d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
63844d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
63944d93782SGreg Clayton 
64044d93782SGreg Clayton #if 0
64144d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
64244d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
64344d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
64444d93782SGreg Clayton #endif
645315b6884SEugene Zelenko 
646b9c1b51eSKate Stone struct Point {
64744d93782SGreg Clayton   int x;
64844d93782SGreg Clayton   int y;
64944d93782SGreg Clayton 
650b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
65144d93782SGreg Clayton 
652b9c1b51eSKate Stone   void Clear() {
65344d93782SGreg Clayton     x = 0;
65444d93782SGreg Clayton     y = 0;
65544d93782SGreg Clayton   }
65644d93782SGreg Clayton 
657b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
65844d93782SGreg Clayton     x += rhs.x;
65944d93782SGreg Clayton     y += rhs.y;
66044d93782SGreg Clayton     return *this;
66144d93782SGreg Clayton   }
66244d93782SGreg Clayton 
663b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
66444d93782SGreg Clayton };
66544d93782SGreg Clayton 
666b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
66744d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
66844d93782SGreg Clayton }
669315b6884SEugene Zelenko 
670b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
67144d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
67244d93782SGreg Clayton }
67344d93782SGreg Clayton 
674b9c1b51eSKate Stone struct Size {
67544d93782SGreg Clayton   int width;
67644d93782SGreg Clayton   int height;
677b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
67844d93782SGreg Clayton 
679b9c1b51eSKate Stone   void Clear() {
68044d93782SGreg Clayton     width = 0;
68144d93782SGreg Clayton     height = 0;
68244d93782SGreg Clayton   }
68344d93782SGreg Clayton 
684b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
68544d93782SGreg Clayton };
68644d93782SGreg Clayton 
687b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
68844d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
68944d93782SGreg Clayton }
690315b6884SEugene Zelenko 
691b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
69244d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
69344d93782SGreg Clayton }
69444d93782SGreg Clayton 
695b9c1b51eSKate Stone struct Rect {
69644d93782SGreg Clayton   Point origin;
69744d93782SGreg Clayton   Size size;
69844d93782SGreg Clayton 
699b9c1b51eSKate Stone   Rect() : origin(), size() {}
70044d93782SGreg Clayton 
701b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
70244d93782SGreg Clayton 
703b9c1b51eSKate Stone   void Clear() {
70444d93782SGreg Clayton     origin.Clear();
70544d93782SGreg Clayton     size.Clear();
70644d93782SGreg Clayton   }
70744d93782SGreg Clayton 
708b9c1b51eSKate Stone   void Dump() {
709b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
710b9c1b51eSKate Stone            size.height);
71144d93782SGreg Clayton   }
71244d93782SGreg Clayton 
713b9c1b51eSKate Stone   void Inset(int w, int h) {
71444d93782SGreg Clayton     if (size.width > w * 2)
71544d93782SGreg Clayton       size.width -= w * 2;
71644d93782SGreg Clayton     origin.x += w;
71744d93782SGreg Clayton 
71844d93782SGreg Clayton     if (size.height > h * 2)
71944d93782SGreg Clayton       size.height -= h * 2;
72044d93782SGreg Clayton     origin.y += h;
72144d93782SGreg Clayton   }
722315b6884SEugene Zelenko 
72305097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
72405097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
725b9c1b51eSKate Stone   Rect MakeStatusBar() {
72644d93782SGreg Clayton     Rect status_bar;
727b9c1b51eSKate Stone     if (size.height > 1) {
72844d93782SGreg Clayton       status_bar.origin.x = origin.x;
72944d93782SGreg Clayton       status_bar.origin.y = size.height;
73044d93782SGreg Clayton       status_bar.size.width = size.width;
73144d93782SGreg Clayton       status_bar.size.height = 1;
73244d93782SGreg Clayton       --size.height;
73344d93782SGreg Clayton     }
73444d93782SGreg Clayton     return status_bar;
73544d93782SGreg Clayton   }
73644d93782SGreg Clayton 
73705097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
73805097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
739b9c1b51eSKate Stone   Rect MakeMenuBar() {
74044d93782SGreg Clayton     Rect menubar;
741b9c1b51eSKate Stone     if (size.height > 1) {
74244d93782SGreg Clayton       menubar.origin.x = origin.x;
74344d93782SGreg Clayton       menubar.origin.y = origin.y;
74444d93782SGreg Clayton       menubar.size.width = size.width;
74544d93782SGreg Clayton       menubar.size.height = 1;
74644d93782SGreg Clayton       ++origin.y;
74744d93782SGreg Clayton       --size.height;
74844d93782SGreg Clayton     }
74944d93782SGreg Clayton     return menubar;
75044d93782SGreg Clayton   }
75144d93782SGreg Clayton 
752b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
753b9c1b51eSKate Stone                                  Rect &bottom) const {
75444d93782SGreg Clayton     float top_height = top_percentage * size.height;
75544d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
75644d93782SGreg Clayton   }
75744d93782SGreg Clayton 
758b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
75944d93782SGreg Clayton     top = *this;
760b9c1b51eSKate Stone     if (top_height < size.height) {
76144d93782SGreg Clayton       top.size.height = top_height;
76244d93782SGreg Clayton       bottom.origin.x = origin.x;
76344d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
76444d93782SGreg Clayton       bottom.size.width = size.width;
76544d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
766b9c1b51eSKate Stone     } else {
76744d93782SGreg Clayton       bottom.Clear();
76844d93782SGreg Clayton     }
76944d93782SGreg Clayton   }
77044d93782SGreg Clayton 
771b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
772b9c1b51eSKate Stone                                Rect &right) const {
77344d93782SGreg Clayton     float left_width = left_percentage * size.width;
77444d93782SGreg Clayton     VerticalSplit(left_width, left, right);
77544d93782SGreg Clayton   }
77644d93782SGreg Clayton 
777b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
77844d93782SGreg Clayton     left = *this;
779b9c1b51eSKate Stone     if (left_width < size.width) {
78044d93782SGreg Clayton       left.size.width = left_width;
78144d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
78244d93782SGreg Clayton       right.origin.y = origin.y;
78344d93782SGreg Clayton       right.size.width = size.width - left.size.width;
78444d93782SGreg Clayton       right.size.height = size.height;
785b9c1b51eSKate Stone     } else {
78644d93782SGreg Clayton       right.Clear();
78744d93782SGreg Clayton     }
78844d93782SGreg Clayton   }
78944d93782SGreg Clayton };
79044d93782SGreg Clayton 
791b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
79244d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
79344d93782SGreg Clayton }
794315b6884SEugene Zelenko 
795b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
79644d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
79744d93782SGreg Clayton }
79844d93782SGreg Clayton 
799b9c1b51eSKate Stone enum HandleCharResult {
80044d93782SGreg Clayton   eKeyNotHandled = 0,
80144d93782SGreg Clayton   eKeyHandled = 1,
80244d93782SGreg Clayton   eQuitApplication = 2
80344d93782SGreg Clayton };
80444d93782SGreg Clayton 
805b9c1b51eSKate Stone enum class MenuActionResult {
80644d93782SGreg Clayton   Handled,
80744d93782SGreg Clayton   NotHandled,
80844d93782SGreg Clayton   Quit // Exit all menus and quit
80944d93782SGreg Clayton };
81044d93782SGreg Clayton 
811b9c1b51eSKate Stone struct KeyHelp {
81244d93782SGreg Clayton   int ch;
81344d93782SGreg Clayton   const char *description;
81444d93782SGreg Clayton };
81544d93782SGreg Clayton 
816b9c1b51eSKate Stone class WindowDelegate {
81744d93782SGreg Clayton public:
818b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
81944d93782SGreg Clayton 
820b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
82144d93782SGreg Clayton     return false; // Drawing not handled
82244d93782SGreg Clayton   }
82344d93782SGreg Clayton 
824b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
82544d93782SGreg Clayton     return eKeyNotHandled;
82644d93782SGreg Clayton   }
82744d93782SGreg Clayton 
828b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
82944d93782SGreg Clayton 
830b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
83144d93782SGreg Clayton };
83244d93782SGreg Clayton 
833b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
83444d93782SGreg Clayton public:
83544d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
83644d93782SGreg Clayton 
837bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
83844d93782SGreg Clayton 
839b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
84044d93782SGreg Clayton 
841b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
84244d93782SGreg Clayton 
843b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
84444d93782SGreg Clayton 
845b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
84644d93782SGreg Clayton 
84744d93782SGreg Clayton protected:
84844d93782SGreg Clayton   StringList m_text;
84944d93782SGreg Clayton   int m_first_visible_line;
85044d93782SGreg Clayton };
85144d93782SGreg Clayton 
852b9c1b51eSKate Stone class Window {
85344d93782SGreg Clayton public:
854b9c1b51eSKate Stone   Window(const char *name)
855b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
856b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
857b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
858b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
85944d93782SGreg Clayton 
860b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
861b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
862b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
863b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
864b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
86544d93782SGreg Clayton     if (w)
86644d93782SGreg Clayton       Reset(w);
86744d93782SGreg Clayton   }
86844d93782SGreg Clayton 
869b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
870b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
871b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
872b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
873b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
874b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
875b9c1b51eSKate Stone                    bounds.origin.y));
87644d93782SGreg Clayton   }
87744d93782SGreg Clayton 
878b9c1b51eSKate Stone   virtual ~Window() {
87944d93782SGreg Clayton     RemoveSubWindows();
88044d93782SGreg Clayton     Reset();
88144d93782SGreg Clayton   }
88244d93782SGreg Clayton 
883b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
88444d93782SGreg Clayton     if (m_window == w)
88544d93782SGreg Clayton       return;
88644d93782SGreg Clayton 
887b9c1b51eSKate Stone     if (m_panel) {
88844d93782SGreg Clayton       ::del_panel(m_panel);
889c5dac77aSEugene Zelenko       m_panel = nullptr;
89044d93782SGreg Clayton     }
891b9c1b51eSKate Stone     if (m_window && m_delete) {
89244d93782SGreg Clayton       ::delwin(m_window);
893c5dac77aSEugene Zelenko       m_window = nullptr;
89444d93782SGreg Clayton       m_delete = false;
89544d93782SGreg Clayton     }
896b9c1b51eSKate Stone     if (w) {
89744d93782SGreg Clayton       m_window = w;
89844d93782SGreg Clayton       m_panel = ::new_panel(m_window);
89944d93782SGreg Clayton       m_delete = del;
90044d93782SGreg Clayton     }
90144d93782SGreg Clayton   }
90244d93782SGreg Clayton 
90344d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
90444d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
905b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
906b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
907b9c1b51eSKate Stone   }
90844d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
90944d93782SGreg Clayton   void Erase() { ::werase(m_window); }
910b9c1b51eSKate Stone   Rect GetBounds() {
911b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
912b9c1b51eSKate Stone   } // Get the rectangle in our parent window
91344d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
91444d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
91544d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
916b9c1b51eSKate Stone   Rect GetFrame() {
917b9c1b51eSKate Stone     return Rect(Point(), GetSize());
918b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
91944d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
92044d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
92144d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
92244d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
92344d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
92444d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
92544d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
92644d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
92744d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
92844d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
92944d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
930b9c1b51eSKate Stone   void Resize(const Size &size) {
931b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
932b9c1b51eSKate Stone   }
93344d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
93444d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
93544d93782SGreg Clayton   void Refresh() { ::wrefresh(m_window); }
936b9c1b51eSKate Stone   void DeferredRefresh() {
93744d93782SGreg Clayton     // We are using panels, so we don't need to call this...
93844d93782SGreg Clayton     //::wnoutrefresh(m_window);
93944d93782SGreg Clayton   }
940b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
941b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
942b9c1b51eSKate Stone   }
94344d93782SGreg Clayton   void UnderlineOn() { AttributeOn(A_UNDERLINE); }
94444d93782SGreg Clayton   void UnderlineOff() { AttributeOff(A_UNDERLINE); }
94544d93782SGreg Clayton 
946b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
94744d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
948b9c1b51eSKate Stone     if (bytes_left > right_pad) {
94944d93782SGreg Clayton       bytes_left -= right_pad;
95044d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
95144d93782SGreg Clayton     }
95244d93782SGreg Clayton   }
95344d93782SGreg Clayton 
954b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
95544d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
956b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
95744d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
95844d93782SGreg Clayton       Size size = GetSize();
959b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
960b9c1b51eSKate Stone                      origin.x),
961b9c1b51eSKate Stone             true);
962b9c1b51eSKate Stone     } else {
96344d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
96444d93782SGreg Clayton     }
96544d93782SGreg Clayton   }
96644d93782SGreg Clayton 
967b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
96844d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
969b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
97044d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
971b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
972b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
973b9c1b51eSKate Stone             true);
974b9c1b51eSKate Stone     } else {
97544d93782SGreg Clayton       if (moving_window)
97644d93782SGreg Clayton         MoveWindow(bounds.origin);
97744d93782SGreg Clayton       Resize(bounds.size);
97844d93782SGreg Clayton     }
97944d93782SGreg Clayton   }
98044d93782SGreg Clayton 
981b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
98244d93782SGreg Clayton     va_list args;
98344d93782SGreg Clayton     va_start(args, format);
98444d93782SGreg Clayton     vwprintw(m_window, format, args);
98544d93782SGreg Clayton     va_end(args);
98644d93782SGreg Clayton   }
98744d93782SGreg Clayton 
988b9c1b51eSKate Stone   void Touch() {
98944d93782SGreg Clayton     ::touchwin(m_window);
99044d93782SGreg Clayton     if (m_parent)
99144d93782SGreg Clayton       m_parent->Touch();
99244d93782SGreg Clayton   }
99344d93782SGreg Clayton 
994b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
995b9c1b51eSKate Stone                            bool make_active) {
996c6091d2bSJonas Devlieghere     auto get_window = [this, &bounds]() {
997c6091d2bSJonas Devlieghere       return m_window
998c6091d2bSJonas Devlieghere                  ? ::subwin(m_window, bounds.size.height, bounds.size.width,
999c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x)
1000c6091d2bSJonas Devlieghere                  : ::newwin(bounds.size.height, bounds.size.width,
1001c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x);
1002c6091d2bSJonas Devlieghere     };
1003c6091d2bSJonas Devlieghere     WindowSP subwindow_sp = std::make_shared<Window>(name, get_window(), true);
1004c6091d2bSJonas Devlieghere     subwindow_sp->m_is_subwin = subwindow_sp.operator bool();
100544d93782SGreg Clayton     subwindow_sp->m_parent = this;
1006b9c1b51eSKate Stone     if (make_active) {
100744d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
100844d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
100944d93782SGreg Clayton     }
101044d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
101144d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
101244d93782SGreg Clayton     m_needs_update = true;
101344d93782SGreg Clayton     return subwindow_sp;
101444d93782SGreg Clayton   }
101544d93782SGreg Clayton 
1016b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
101744d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
101844d93782SGreg Clayton     size_t i = 0;
1019b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1020b9c1b51eSKate Stone       if ((*pos).get() == window) {
102144d93782SGreg Clayton         if (m_prev_active_window_idx == i)
102244d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1023b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1024b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
102544d93782SGreg Clayton           --m_prev_active_window_idx;
102644d93782SGreg Clayton 
102744d93782SGreg Clayton         if (m_curr_active_window_idx == i)
102844d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1029b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1030b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
103144d93782SGreg Clayton           --m_curr_active_window_idx;
103244d93782SGreg Clayton         window->Erase();
103344d93782SGreg Clayton         m_subwindows.erase(pos);
103444d93782SGreg Clayton         m_needs_update = true;
103544d93782SGreg Clayton         if (m_parent)
103644d93782SGreg Clayton           m_parent->Touch();
103744d93782SGreg Clayton         else
103844d93782SGreg Clayton           ::touchwin(stdscr);
103944d93782SGreg Clayton         return true;
104044d93782SGreg Clayton       }
104144d93782SGreg Clayton     }
104244d93782SGreg Clayton     return false;
104344d93782SGreg Clayton   }
104444d93782SGreg Clayton 
1045b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
104644d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
104744d93782SGreg Clayton     size_t i = 0;
1048b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
10498d20cfdfSJonas Devlieghere       if ((*pos)->m_name == name)
105044d93782SGreg Clayton         return *pos;
105144d93782SGreg Clayton     }
105244d93782SGreg Clayton     return WindowSP();
105344d93782SGreg Clayton   }
105444d93782SGreg Clayton 
1055b9c1b51eSKate Stone   void RemoveSubWindows() {
105644d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
105744d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
105844d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1059b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
106044d93782SGreg Clayton       (*pos)->Erase();
106144d93782SGreg Clayton     }
106244d93782SGreg Clayton     if (m_parent)
106344d93782SGreg Clayton       m_parent->Touch();
106444d93782SGreg Clayton     else
106544d93782SGreg Clayton       ::touchwin(stdscr);
106644d93782SGreg Clayton   }
106744d93782SGreg Clayton 
1068b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
106944d93782SGreg Clayton 
1070b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
107144d93782SGreg Clayton 
107244d93782SGreg Clayton   //----------------------------------------------------------------------
107344d93782SGreg Clayton   // Window drawing utilities
107444d93782SGreg Clayton   //----------------------------------------------------------------------
1075b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
107644d93782SGreg Clayton     attr_t attr = 0;
107744d93782SGreg Clayton     if (IsActive())
107844d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
107944d93782SGreg Clayton     else
108044d93782SGreg Clayton       attr = 0;
108144d93782SGreg Clayton     if (attr)
108244d93782SGreg Clayton       AttributeOn(attr);
108344d93782SGreg Clayton 
108444d93782SGreg Clayton     Box();
108544d93782SGreg Clayton     MoveCursor(3, 0);
108644d93782SGreg Clayton 
1087b9c1b51eSKate Stone     if (title && title[0]) {
108844d93782SGreg Clayton       PutChar('<');
108944d93782SGreg Clayton       PutCString(title);
109044d93782SGreg Clayton       PutChar('>');
109144d93782SGreg Clayton     }
109244d93782SGreg Clayton 
1093b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
109444d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
109544d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
109644d93782SGreg Clayton 
1097b9c1b51eSKate Stone       if (x > 0) {
109844d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
109944d93782SGreg Clayton         PutChar('[');
110044d93782SGreg Clayton         PutCString(bottom_message);
110144d93782SGreg Clayton         PutChar(']');
1102b9c1b51eSKate Stone       } else {
110344d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
110444d93782SGreg Clayton         PutChar('[');
110544d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
110644d93782SGreg Clayton       }
110744d93782SGreg Clayton     }
110844d93782SGreg Clayton     if (attr)
110944d93782SGreg Clayton       AttributeOff(attr);
111044d93782SGreg Clayton   }
111144d93782SGreg Clayton 
1112b9c1b51eSKate Stone   virtual void Draw(bool force) {
111344d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
111444d93782SGreg Clayton       return;
111544d93782SGreg Clayton 
111644d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
111744d93782SGreg Clayton       subwindow_sp->Draw(force);
111844d93782SGreg Clayton   }
111944d93782SGreg Clayton 
1120b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1121b9c1b51eSKate Stone     if (m_delegate_sp) {
112244d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
112344d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1124b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1125d5b44036SJonas Devlieghere         std::unique_ptr<HelpDialogDelegate> help_delegate_up(
1126b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
1127d5b44036SJonas Devlieghere         const size_t num_lines = help_delegate_up->GetNumLines();
1128d5b44036SJonas Devlieghere         const size_t max_length = help_delegate_up->GetMaxLineLength();
112944d93782SGreg Clayton         Rect bounds = GetBounds();
113044d93782SGreg Clayton         bounds.Inset(1, 1);
1131b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
113244d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
113344d93782SGreg Clayton           bounds.size.width = max_length + 4;
1134b9c1b51eSKate Stone         } else {
1135b9c1b51eSKate Stone           if (bounds.size.width > 100) {
113644d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
113744d93782SGreg Clayton             bounds.origin.x += inset_w;
113844d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
113944d93782SGreg Clayton           }
114044d93782SGreg Clayton         }
114144d93782SGreg Clayton 
1142b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
114344d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
114444d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1145b9c1b51eSKate Stone         } else {
1146b9c1b51eSKate Stone           if (bounds.size.height > 100) {
114744d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
114844d93782SGreg Clayton             bounds.origin.y += inset_h;
114944d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
115044d93782SGreg Clayton           }
115144d93782SGreg Clayton         }
11525fdb09bbSGreg Clayton         WindowSP help_window_sp;
11535fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11545fdb09bbSGreg Clayton         if (parent_window)
11555fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11565fdb09bbSGreg Clayton         else
11575fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1158b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1159d5b44036SJonas Devlieghere             WindowDelegateSP(help_delegate_up.release()));
116044d93782SGreg Clayton         return true;
116144d93782SGreg Clayton       }
116244d93782SGreg Clayton     }
116344d93782SGreg Clayton     return false;
116444d93782SGreg Clayton   }
116544d93782SGreg Clayton 
1166b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
116744d93782SGreg Clayton     // Always check the active window first
116844d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
116944d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1170b9c1b51eSKate Stone     if (active_window_sp) {
117144d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
117244d93782SGreg Clayton       if (result != eKeyNotHandled)
117344d93782SGreg Clayton         return result;
117444d93782SGreg Clayton     }
117544d93782SGreg Clayton 
1176b9c1b51eSKate Stone     if (m_delegate_sp) {
117744d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
117844d93782SGreg Clayton       if (result != eKeyNotHandled)
117944d93782SGreg Clayton         return result;
118044d93782SGreg Clayton     }
118144d93782SGreg Clayton 
118205097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
118305097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
118405097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
118505097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
118644d93782SGreg Clayton     Windows subwindows(m_subwindows);
1187b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1188b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
118944d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
119044d93782SGreg Clayton         if (result != eKeyNotHandled)
119144d93782SGreg Clayton           return result;
119244d93782SGreg Clayton       }
119344d93782SGreg Clayton     }
119444d93782SGreg Clayton 
119544d93782SGreg Clayton     return eKeyNotHandled;
119644d93782SGreg Clayton   }
119744d93782SGreg Clayton 
1198b9c1b51eSKate Stone   bool SetActiveWindow(Window *window) {
119944d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1200b9c1b51eSKate Stone     for (size_t i = 0; i < num_subwindows; ++i) {
1201b9c1b51eSKate Stone       if (m_subwindows[i].get() == window) {
120244d93782SGreg Clayton         m_prev_active_window_idx = m_curr_active_window_idx;
120344d93782SGreg Clayton         ::top_panel(window->m_panel);
120444d93782SGreg Clayton         m_curr_active_window_idx = i;
120544d93782SGreg Clayton         return true;
120644d93782SGreg Clayton       }
120744d93782SGreg Clayton     }
120844d93782SGreg Clayton     return false;
120944d93782SGreg Clayton   }
121044d93782SGreg Clayton 
1211b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1212b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1213b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1214b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
121544d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
121644d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1217b9c1b51eSKate Stone         } else if (IsActive()) {
121844d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
121944d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
122044d93782SGreg Clayton 
122144d93782SGreg Clayton           // Find first window that wants to be active if this window is active
122244d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1223b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1224b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
122544d93782SGreg Clayton               m_curr_active_window_idx = i;
122644d93782SGreg Clayton               break;
122744d93782SGreg Clayton             }
122844d93782SGreg Clayton           }
122944d93782SGreg Clayton         }
123044d93782SGreg Clayton       }
123144d93782SGreg Clayton 
123244d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
123344d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
123444d93782SGreg Clayton     }
123544d93782SGreg Clayton     return WindowSP();
123644d93782SGreg Clayton   }
123744d93782SGreg Clayton 
1238b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
123944d93782SGreg Clayton 
1240b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
124144d93782SGreg Clayton 
1242b9c1b51eSKate Stone   const WindowDelegateSP &GetDelegate() const { return m_delegate_sp; }
124344d93782SGreg Clayton 
1244b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
124544d93782SGreg Clayton     m_delegate_sp = delegate_sp;
124644d93782SGreg Clayton   }
124744d93782SGreg Clayton 
1248b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
124944d93782SGreg Clayton 
1250b9c1b51eSKate Stone   bool IsActive() const {
125144d93782SGreg Clayton     if (m_parent)
125244d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
125344d93782SGreg Clayton     else
125444d93782SGreg Clayton       return true; // Top level window is always active
125544d93782SGreg Clayton   }
125644d93782SGreg Clayton 
1257b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
125844d93782SGreg Clayton     // Move active focus to next window
125944d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1260b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
126144d93782SGreg Clayton       uint32_t idx = 0;
1262b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1263b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
126444d93782SGreg Clayton           m_curr_active_window_idx = idx;
126544d93782SGreg Clayton           break;
126644d93782SGreg Clayton         }
126744d93782SGreg Clayton         ++idx;
126844d93782SGreg Clayton       }
1269b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
127044d93782SGreg Clayton       bool handled = false;
127144d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1272b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1273b9c1b51eSKate Stone            ++idx) {
1274b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
127544d93782SGreg Clayton           m_curr_active_window_idx = idx;
127644d93782SGreg Clayton           handled = true;
127744d93782SGreg Clayton           break;
127844d93782SGreg Clayton         }
127944d93782SGreg Clayton       }
1280b9c1b51eSKate Stone       if (!handled) {
1281b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1282b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
128344d93782SGreg Clayton             m_curr_active_window_idx = idx;
128444d93782SGreg Clayton             break;
128544d93782SGreg Clayton           }
128644d93782SGreg Clayton         }
128744d93782SGreg Clayton       }
1288b9c1b51eSKate Stone     } else {
128944d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1290b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1291b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
129244d93782SGreg Clayton           m_curr_active_window_idx = idx;
129344d93782SGreg Clayton           break;
129444d93782SGreg Clayton         }
129544d93782SGreg Clayton       }
129644d93782SGreg Clayton     }
129744d93782SGreg Clayton   }
129844d93782SGreg Clayton 
1299b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1300315b6884SEugene Zelenko 
130144d93782SGreg Clayton protected:
130244d93782SGreg Clayton   std::string m_name;
130344d93782SGreg Clayton   WINDOW *m_window;
130444d93782SGreg Clayton   PANEL *m_panel;
130544d93782SGreg Clayton   Window *m_parent;
130644d93782SGreg Clayton   Windows m_subwindows;
130744d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
130844d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
130944d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
131044d93782SGreg Clayton   bool m_delete;
131144d93782SGreg Clayton   bool m_needs_update;
131244d93782SGreg Clayton   bool m_can_activate;
131344d93782SGreg Clayton   bool m_is_subwin;
131444d93782SGreg Clayton 
131544d93782SGreg Clayton private:
131644d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
131744d93782SGreg Clayton };
131844d93782SGreg Clayton 
1319b9c1b51eSKate Stone class MenuDelegate {
132044d93782SGreg Clayton public:
1321315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
132244d93782SGreg Clayton 
1323b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
132444d93782SGreg Clayton };
132544d93782SGreg Clayton 
1326b9c1b51eSKate Stone class Menu : public WindowDelegate {
132744d93782SGreg Clayton public:
1328b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
132944d93782SGreg Clayton 
133044d93782SGreg Clayton   // Menubar or separator constructor
133144d93782SGreg Clayton   Menu(Type type);
133244d93782SGreg Clayton 
133344d93782SGreg Clayton   // Menuitem constructor
1334b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
133544d93782SGreg Clayton        uint64_t identifier);
133644d93782SGreg Clayton 
1337315b6884SEugene Zelenko   ~Menu() override = default;
133844d93782SGreg Clayton 
1339b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
134044d93782SGreg Clayton 
1341b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
134244d93782SGreg Clayton     m_delegate_sp = delegate_sp;
134344d93782SGreg Clayton   }
134444d93782SGreg Clayton 
1345b9c1b51eSKate Stone   void RecalculateNameLengths();
134644d93782SGreg Clayton 
1347b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
134844d93782SGreg Clayton 
1349b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
135044d93782SGreg Clayton 
1351b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
135244d93782SGreg Clayton 
1353b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
135444d93782SGreg Clayton 
1355b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
135644d93782SGreg Clayton 
1357b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
135844d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1359b9c1b51eSKate Stone     if (m_delegate_sp) {
136044d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
136144d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
136244d93782SGreg Clayton         return result;
1363b9c1b51eSKate Stone     } else if (m_parent) {
136444d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
136544d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
136644d93782SGreg Clayton         return result;
136744d93782SGreg Clayton     }
136844d93782SGreg Clayton     return m_canned_result;
136944d93782SGreg Clayton   }
137044d93782SGreg Clayton 
1371b9c1b51eSKate Stone   MenuActionResult Action() {
137205097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
137305097246SAdrian Prantl     // delegate, and if not, try our parent menu
137444d93782SGreg Clayton     return ActionPrivate(*this);
137544d93782SGreg Clayton   }
137644d93782SGreg Clayton 
1377b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
137844d93782SGreg Clayton 
1379b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
138044d93782SGreg Clayton 
1381b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
138244d93782SGreg Clayton 
1383b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
138444d93782SGreg Clayton 
1385b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
138644d93782SGreg Clayton 
1387b9c1b51eSKate Stone   Type GetType() const { return m_type; }
138844d93782SGreg Clayton 
1389b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
139044d93782SGreg Clayton 
1391b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
139244d93782SGreg Clayton 
1393b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
139444d93782SGreg Clayton 
1395b9c1b51eSKate Stone   void SetKeyValue(int key_value) { m_key_value = key_value; }
139644d93782SGreg Clayton 
1397b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
139844d93782SGreg Clayton 
1399b9c1b51eSKate Stone   std::string &GetKeyName() { return m_key_name; }
140044d93782SGreg Clayton 
1401b9c1b51eSKate Stone   int GetDrawWidth() const {
140244d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
140344d93782SGreg Clayton   }
140444d93782SGreg Clayton 
1405b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
140644d93782SGreg Clayton 
1407b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
140844d93782SGreg Clayton 
140944d93782SGreg Clayton protected:
141044d93782SGreg Clayton   std::string m_name;
141144d93782SGreg Clayton   std::string m_key_name;
141244d93782SGreg Clayton   uint64_t m_identifier;
141344d93782SGreg Clayton   Type m_type;
141444d93782SGreg Clayton   int m_key_value;
141544d93782SGreg Clayton   int m_start_col;
141644d93782SGreg Clayton   int m_max_submenu_name_length;
141744d93782SGreg Clayton   int m_max_submenu_key_name_length;
141844d93782SGreg Clayton   int m_selected;
141944d93782SGreg Clayton   Menu *m_parent;
142044d93782SGreg Clayton   Menus m_submenus;
142144d93782SGreg Clayton   WindowSP m_menu_window_sp;
142244d93782SGreg Clayton   MenuActionResult m_canned_result;
142344d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
142444d93782SGreg Clayton };
142544d93782SGreg Clayton 
142644d93782SGreg Clayton // Menubar or separator constructor
1427b9c1b51eSKate Stone Menu::Menu(Type type)
1428b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1429b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1430b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1431b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1432b9c1b51eSKate Stone       m_delegate_sp() {}
143344d93782SGreg Clayton 
143444d93782SGreg Clayton // Menuitem constructor
1435b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1436b9c1b51eSKate Stone            uint64_t identifier)
1437b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1438b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1439b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1440b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1441b9c1b51eSKate Stone       m_delegate_sp() {
1442b9c1b51eSKate Stone   if (name && name[0]) {
144344d93782SGreg Clayton     m_name = name;
144444d93782SGreg Clayton     m_type = Type::Item;
144544d93782SGreg Clayton     if (key_name && key_name[0])
144644d93782SGreg Clayton       m_key_name = key_name;
1447b9c1b51eSKate Stone   } else {
144844d93782SGreg Clayton     m_type = Type::Separator;
144944d93782SGreg Clayton   }
145044d93782SGreg Clayton }
145144d93782SGreg Clayton 
1452b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
145344d93782SGreg Clayton   m_max_submenu_name_length = 0;
145444d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
145544d93782SGreg Clayton   Menus &submenus = GetSubmenus();
145644d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1457b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
145844d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14593985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
146044d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1461b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1462b9c1b51eSKate Stone         submenu->m_key_name.size())
146344d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
146444d93782SGreg Clayton   }
146544d93782SGreg Clayton }
146644d93782SGreg Clayton 
1467b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
146844d93782SGreg Clayton   menu_sp->m_parent = this;
14693985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
147044d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1471b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1472b9c1b51eSKate Stone       menu_sp->m_key_name.size())
147344d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
147444d93782SGreg Clayton   m_submenus.push_back(menu_sp);
147544d93782SGreg Clayton }
147644d93782SGreg Clayton 
1477b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1478b9c1b51eSKate Stone   if (m_type == Type::Separator) {
147944d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
148044d93782SGreg Clayton     window.PutChar(ACS_LTEE);
148144d93782SGreg Clayton     int width = window.GetWidth();
1482b9c1b51eSKate Stone     if (width > 2) {
148344d93782SGreg Clayton       width -= 2;
14843985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
148544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
148644d93782SGreg Clayton     }
148744d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1488b9c1b51eSKate Stone   } else {
148944d93782SGreg Clayton     const int shortcut_key = m_key_value;
149044d93782SGreg Clayton     bool underlined_shortcut = false;
149144d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
149244d93782SGreg Clayton     if (highlight)
149344d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1494b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
149544d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
149644d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
149744d93782SGreg Clayton       const char *name = m_name.c_str();
149844d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1499b9c1b51eSKate Stone       if (pos != std::string::npos) {
150044d93782SGreg Clayton         underlined_shortcut = true;
1501b9c1b51eSKate Stone         if (pos > 0) {
150244d93782SGreg Clayton           window.PutCString(name, pos);
150344d93782SGreg Clayton           name += pos;
150444d93782SGreg Clayton         }
150544d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
150644d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
150744d93782SGreg Clayton         window.PutChar(name[0]);
150844d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
150944d93782SGreg Clayton         name++;
151044d93782SGreg Clayton         if (name[0])
151144d93782SGreg Clayton           window.PutCString(name);
151244d93782SGreg Clayton       }
151344d93782SGreg Clayton     }
151444d93782SGreg Clayton 
1515b9c1b51eSKate Stone     if (!underlined_shortcut) {
151644d93782SGreg Clayton       window.PutCString(m_name.c_str());
151744d93782SGreg Clayton     }
151844d93782SGreg Clayton 
151944d93782SGreg Clayton     if (highlight)
152044d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
152144d93782SGreg Clayton 
1522b9c1b51eSKate Stone     if (m_key_name.empty()) {
1523b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
152444d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
152544d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
152644d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
152744d93782SGreg Clayton       }
1528b9c1b51eSKate Stone     } else {
152944d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
153044d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
153144d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
153244d93782SGreg Clayton     }
153344d93782SGreg Clayton   }
153444d93782SGreg Clayton }
153544d93782SGreg Clayton 
1536b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
153744d93782SGreg Clayton   Menus &submenus = GetSubmenus();
153844d93782SGreg Clayton   const size_t num_submenus = submenus.size();
153944d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
154044d93782SGreg Clayton   Menu::Type menu_type = GetType();
1541b9c1b51eSKate Stone   switch (menu_type) {
1542b9c1b51eSKate Stone   case Menu::Type::Bar: {
154344d93782SGreg Clayton     window.SetBackground(2);
154444d93782SGreg Clayton     window.MoveCursor(0, 0);
1545b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
154644d93782SGreg Clayton       Menu *menu = submenus[i].get();
154744d93782SGreg Clayton       if (i > 0)
154844d93782SGreg Clayton         window.PutChar(' ');
154944d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
155044d93782SGreg Clayton       window.PutCString("| ");
155144d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
155244d93782SGreg Clayton     }
155344d93782SGreg Clayton     window.PutCString(" |");
155444d93782SGreg Clayton     window.DeferredRefresh();
1555b9c1b51eSKate Stone   } break;
155644d93782SGreg Clayton 
1557b9c1b51eSKate Stone   case Menu::Type::Item: {
155844d93782SGreg Clayton     int y = 1;
155944d93782SGreg Clayton     int x = 3;
156044d93782SGreg Clayton     // Draw the menu
156144d93782SGreg Clayton     int cursor_x = 0;
156244d93782SGreg Clayton     int cursor_y = 0;
156344d93782SGreg Clayton     window.Erase();
156444d93782SGreg Clayton     window.SetBackground(2);
156544d93782SGreg Clayton     window.Box();
1566b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1567b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
156844d93782SGreg Clayton       window.MoveCursor(x, y + i);
1569b9c1b51eSKate Stone       if (is_selected) {
157044d93782SGreg Clayton         // Remember where we want the cursor to be
157144d93782SGreg Clayton         cursor_x = x - 1;
157244d93782SGreg Clayton         cursor_y = y + i;
157344d93782SGreg Clayton       }
157444d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
157544d93782SGreg Clayton     }
157644d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
157744d93782SGreg Clayton     window.DeferredRefresh();
1578b9c1b51eSKate Stone   } break;
157944d93782SGreg Clayton 
158044d93782SGreg Clayton   default:
158144d93782SGreg Clayton   case Menu::Type::Separator:
158244d93782SGreg Clayton     break;
158344d93782SGreg Clayton   }
158444d93782SGreg Clayton   return true; // Drawing handled...
158544d93782SGreg Clayton }
158644d93782SGreg Clayton 
1587b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
158844d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
158944d93782SGreg Clayton 
159044d93782SGreg Clayton   Menus &submenus = GetSubmenus();
159144d93782SGreg Clayton   const size_t num_submenus = submenus.size();
159244d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
159344d93782SGreg Clayton   Menu::Type menu_type = GetType();
1594b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
159544d93782SGreg Clayton     MenuSP run_menu_sp;
1596b9c1b51eSKate Stone     switch (key) {
159744d93782SGreg Clayton     case KEY_DOWN:
159844d93782SGreg Clayton     case KEY_UP:
159944d93782SGreg Clayton       // Show last menu or first menu
16003985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
160144d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
160244d93782SGreg Clayton       else if (!submenus.empty())
160344d93782SGreg Clayton         run_menu_sp = submenus.front();
160444d93782SGreg Clayton       result = eKeyHandled;
160544d93782SGreg Clayton       break;
160644d93782SGreg Clayton 
160744d93782SGreg Clayton     case KEY_RIGHT:
160844d93782SGreg Clayton       ++m_selected;
16093985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
161044d93782SGreg Clayton         m_selected = 0;
16113985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
161244d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
161344d93782SGreg Clayton       else if (!submenus.empty())
161444d93782SGreg Clayton         run_menu_sp = submenus.front();
161544d93782SGreg Clayton       result = eKeyHandled;
161644d93782SGreg Clayton       break;
161744d93782SGreg Clayton 
161844d93782SGreg Clayton     case KEY_LEFT:
161944d93782SGreg Clayton       --m_selected;
162044d93782SGreg Clayton       if (m_selected < 0)
162144d93782SGreg Clayton         m_selected = num_submenus - 1;
16223985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
162344d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
162444d93782SGreg Clayton       else if (!submenus.empty())
162544d93782SGreg Clayton         run_menu_sp = submenus.front();
162644d93782SGreg Clayton       result = eKeyHandled;
162744d93782SGreg Clayton       break;
162844d93782SGreg Clayton 
162944d93782SGreg Clayton     default:
1630b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1631b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
163244d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
163344d93782SGreg Clayton           run_menu_sp = submenus[i];
163444d93782SGreg Clayton           result = eKeyHandled;
163544d93782SGreg Clayton           break;
163644d93782SGreg Clayton         }
163744d93782SGreg Clayton       }
163844d93782SGreg Clayton       break;
163944d93782SGreg Clayton     }
164044d93782SGreg Clayton 
1641b9c1b51eSKate Stone     if (run_menu_sp) {
164205097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
164305097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
164405097246SAdrian Prantl       // decorations need to be calculated
164544d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
164644d93782SGreg Clayton         return eQuitApplication;
164744d93782SGreg Clayton 
164844d93782SGreg Clayton       Rect menu_bounds;
164944d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
165044d93782SGreg Clayton       menu_bounds.origin.y = 1;
165144d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
165244d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
165344d93782SGreg Clayton       if (m_menu_window_sp)
165444d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
165544d93782SGreg Clayton 
1656b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1657b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
165844d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
165944d93782SGreg Clayton     }
1660b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1661b9c1b51eSKate Stone     switch (key) {
166244d93782SGreg Clayton     case KEY_DOWN:
1663b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
166444d93782SGreg Clayton         const int start_select = m_selected;
1665b9c1b51eSKate Stone         while (++m_selected != start_select) {
16663985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
166744d93782SGreg Clayton             m_selected = 0;
166844d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
166944d93782SGreg Clayton             continue;
167044d93782SGreg Clayton           else
167144d93782SGreg Clayton             break;
167244d93782SGreg Clayton         }
167344d93782SGreg Clayton         return eKeyHandled;
167444d93782SGreg Clayton       }
167544d93782SGreg Clayton       break;
167644d93782SGreg Clayton 
167744d93782SGreg Clayton     case KEY_UP:
1678b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
167944d93782SGreg Clayton         const int start_select = m_selected;
1680b9c1b51eSKate Stone         while (--m_selected != start_select) {
16813985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
168244d93782SGreg Clayton             m_selected = num_submenus - 1;
168344d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
168444d93782SGreg Clayton             continue;
168544d93782SGreg Clayton           else
168644d93782SGreg Clayton             break;
168744d93782SGreg Clayton         }
168844d93782SGreg Clayton         return eKeyHandled;
168944d93782SGreg Clayton       }
169044d93782SGreg Clayton       break;
169144d93782SGreg Clayton 
169244d93782SGreg Clayton     case KEY_RETURN:
1693b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
169444d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
169544d93782SGreg Clayton           return eQuitApplication;
169644d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
169744d93782SGreg Clayton         return eKeyHandled;
169844d93782SGreg Clayton       }
169944d93782SGreg Clayton       break;
170044d93782SGreg Clayton 
1701b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1702b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
170344d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
170444d93782SGreg Clayton       return eKeyHandled;
170544d93782SGreg Clayton 
170644d93782SGreg Clayton     default:
1707b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
170844d93782SGreg Clayton         Menu *menu = submenus[i].get();
1709b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
171044d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
171144d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
171244d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
171344d93782SGreg Clayton             return eQuitApplication;
171444d93782SGreg Clayton           return eKeyHandled;
171544d93782SGreg Clayton         }
171644d93782SGreg Clayton       }
171744d93782SGreg Clayton       break;
171844d93782SGreg Clayton     }
1719b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
172044d93782SGreg Clayton   }
172144d93782SGreg Clayton   return result;
172244d93782SGreg Clayton }
172344d93782SGreg Clayton 
1724b9c1b51eSKate Stone class Application {
172544d93782SGreg Clayton public:
1726b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1727b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
172844d93782SGreg Clayton 
1729b9c1b51eSKate Stone   ~Application() {
173044d93782SGreg Clayton     m_window_delegates.clear();
173144d93782SGreg Clayton     m_window_sp.reset();
1732b9c1b51eSKate Stone     if (m_screen) {
173344d93782SGreg Clayton       ::delscreen(m_screen);
1734c5dac77aSEugene Zelenko       m_screen = nullptr;
173544d93782SGreg Clayton     }
173644d93782SGreg Clayton   }
173744d93782SGreg Clayton 
1738b9c1b51eSKate Stone   void Initialize() {
173944d93782SGreg Clayton     ::setlocale(LC_ALL, "");
174044d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
1741c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
174244d93782SGreg Clayton     ::start_color();
174344d93782SGreg Clayton     ::curs_set(0);
174444d93782SGreg Clayton     ::noecho();
174544d93782SGreg Clayton     ::keypad(stdscr, TRUE);
174644d93782SGreg Clayton   }
174744d93782SGreg Clayton 
1748b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
174944d93782SGreg Clayton 
1750b9c1b51eSKate Stone   void Run(Debugger &debugger) {
175144d93782SGreg Clayton     bool done = false;
175244d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
175344d93782SGreg Clayton 
175405097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
175505097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
175605097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
175705097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
175805097246SAdrian Prantl     // to receive async process events while in this loop.
175944d93782SGreg Clayton 
1760b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1761b9c1b51eSKate Stone                                             // of seconds seconds when calling
1762b9c1b51eSKate Stone                                             // Window::GetChar()
176344d93782SGreg Clayton 
1764b9c1b51eSKate Stone     ListenerSP listener_sp(
1765b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
176644d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
176744d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
176844d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
176944d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
177044d93782SGreg Clayton 
177144d93782SGreg Clayton     bool update = true;
177244d93782SGreg Clayton #if defined(__APPLE__)
177344d93782SGreg Clayton     std::deque<int> escape_chars;
177444d93782SGreg Clayton #endif
177544d93782SGreg Clayton 
1776b9c1b51eSKate Stone     while (!done) {
1777b9c1b51eSKate Stone       if (update) {
177844d93782SGreg Clayton         m_window_sp->Draw(false);
177905097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
178005097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
178105097246SAdrian Prantl         // blinking
178244d93782SGreg Clayton         update_panels();
178344d93782SGreg Clayton 
1784b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1785b9c1b51eSKate Stone         // corner
178644d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
178744d93782SGreg Clayton 
178844d93782SGreg Clayton         doupdate();
178944d93782SGreg Clayton         update = false;
179044d93782SGreg Clayton       }
179144d93782SGreg Clayton 
179244d93782SGreg Clayton #if defined(__APPLE__)
179305097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
179405097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1795b9c1b51eSKate Stone       // possible
179644d93782SGreg Clayton       int ch;
179744d93782SGreg Clayton       if (escape_chars.empty())
179844d93782SGreg Clayton         ch = m_window_sp->GetChar();
1799b9c1b51eSKate Stone       else {
180044d93782SGreg Clayton         ch = escape_chars.front();
180144d93782SGreg Clayton         escape_chars.pop_front();
180244d93782SGreg Clayton       }
1803b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
180444d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1805b9c1b51eSKate Stone         if (ch2 == 'O') {
180644d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1807b9c1b51eSKate Stone           switch (ch3) {
1808b9c1b51eSKate Stone           case 'P':
1809b9c1b51eSKate Stone             ch = KEY_F(1);
1810b9c1b51eSKate Stone             break;
1811b9c1b51eSKate Stone           case 'Q':
1812b9c1b51eSKate Stone             ch = KEY_F(2);
1813b9c1b51eSKate Stone             break;
1814b9c1b51eSKate Stone           case 'R':
1815b9c1b51eSKate Stone             ch = KEY_F(3);
1816b9c1b51eSKate Stone             break;
1817b9c1b51eSKate Stone           case 'S':
1818b9c1b51eSKate Stone             ch = KEY_F(4);
1819b9c1b51eSKate Stone             break;
182044d93782SGreg Clayton           default:
182144d93782SGreg Clayton             escape_chars.push_back(ch2);
182244d93782SGreg Clayton             if (ch3 != -1)
182344d93782SGreg Clayton               escape_chars.push_back(ch3);
182444d93782SGreg Clayton             break;
182544d93782SGreg Clayton           }
1826b9c1b51eSKate Stone         } else if (ch2 != -1)
182744d93782SGreg Clayton           escape_chars.push_back(ch2);
182844d93782SGreg Clayton       }
182944d93782SGreg Clayton #else
183044d93782SGreg Clayton       int ch = m_window_sp->GetChar();
183144d93782SGreg Clayton 
183244d93782SGreg Clayton #endif
1833b9c1b51eSKate Stone       if (ch == -1) {
1834b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
183544d93782SGreg Clayton           done = true;
1836b9c1b51eSKate Stone         } else {
183744d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
183844d93782SGreg Clayton           EventSP event_sp;
1839b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1840d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
184144d93782SGreg Clayton 
1842b9c1b51eSKate Stone             if (event_sp) {
184344d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1844b9c1b51eSKate Stone               if (broadcaster) {
184544d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1846b9c1b51eSKate Stone                 ConstString broadcaster_class(
1847b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1848b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1849b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1850b9c1b51eSKate Stone                       nullptr);
185144d93782SGreg Clayton                   update = true;
185244d93782SGreg Clayton                   continue; // Don't get any key, just update our view
185344d93782SGreg Clayton                 }
185444d93782SGreg Clayton               }
185544d93782SGreg Clayton             }
185644d93782SGreg Clayton           }
185744d93782SGreg Clayton         }
1858b9c1b51eSKate Stone       } else {
185944d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1860b9c1b51eSKate Stone         switch (key_result) {
186144d93782SGreg Clayton         case eKeyHandled:
1862c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
186344d93782SGreg Clayton           update = true;
186444d93782SGreg Clayton           break;
186544d93782SGreg Clayton         case eKeyNotHandled:
186644d93782SGreg Clayton           break;
186744d93782SGreg Clayton         case eQuitApplication:
186844d93782SGreg Clayton           done = true;
186944d93782SGreg Clayton           break;
187044d93782SGreg Clayton         }
187144d93782SGreg Clayton       }
187244d93782SGreg Clayton     }
187344d93782SGreg Clayton 
187444d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
187544d93782SGreg Clayton   }
187644d93782SGreg Clayton 
1877b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
187844d93782SGreg Clayton     if (!m_window_sp)
1879796ac80bSJonas Devlieghere       m_window_sp = std::make_shared<Window>("main", stdscr, false);
188044d93782SGreg Clayton     return m_window_sp;
188144d93782SGreg Clayton   }
188244d93782SGreg Clayton 
1883b9c1b51eSKate Stone   WindowDelegates &GetWindowDelegates() { return m_window_delegates; }
188444d93782SGreg Clayton 
188544d93782SGreg Clayton protected:
188644d93782SGreg Clayton   WindowSP m_window_sp;
188744d93782SGreg Clayton   WindowDelegates m_window_delegates;
188844d93782SGreg Clayton   SCREEN *m_screen;
188944d93782SGreg Clayton   FILE *m_in;
189044d93782SGreg Clayton   FILE *m_out;
189144d93782SGreg Clayton };
189244d93782SGreg Clayton 
189344d93782SGreg Clayton } // namespace curses
189444d93782SGreg Clayton 
189544d93782SGreg Clayton using namespace curses;
189644d93782SGreg Clayton 
1897b9c1b51eSKate Stone struct Row {
18988369b28dSGreg Clayton   ValueObjectManager value;
189944d93782SGreg Clayton   Row *parent;
19008369b28dSGreg Clayton   // The process stop ID when the children were calculated.
19018369b28dSGreg Clayton   uint32_t children_stop_id;
190244d93782SGreg Clayton   int row_idx;
190344d93782SGreg Clayton   int x;
190444d93782SGreg Clayton   int y;
190544d93782SGreg Clayton   bool might_have_children;
190644d93782SGreg Clayton   bool expanded;
190744d93782SGreg Clayton   bool calculated_children;
190844d93782SGreg Clayton   std::vector<Row> children;
190944d93782SGreg Clayton 
1910b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
19118369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
19128369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1913b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
191444d93782SGreg Clayton 
1915b9c1b51eSKate Stone   size_t GetDepth() const {
191644d93782SGreg Clayton     if (parent)
191744d93782SGreg Clayton       return 1 + parent->GetDepth();
191844d93782SGreg Clayton     return 0;
191944d93782SGreg Clayton   }
192044d93782SGreg Clayton 
1921b9c1b51eSKate Stone   void Expand() {
192244d93782SGreg Clayton     expanded = true;
19238369b28dSGreg Clayton   }
19248369b28dSGreg Clayton 
19258369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
19268369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
19278369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
19288369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
19298369b28dSGreg Clayton       children_stop_id = stop_id;
19308369b28dSGreg Clayton       calculated_children = false;
19318369b28dSGreg Clayton     }
1932b9c1b51eSKate Stone     if (!calculated_children) {
19338369b28dSGreg Clayton       children.clear();
193444d93782SGreg Clayton       calculated_children = true;
19358369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1936b9c1b51eSKate Stone       if (valobj) {
193744d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1938b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
193944d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
194044d93782SGreg Clayton         }
194144d93782SGreg Clayton       }
194244d93782SGreg Clayton     }
19438369b28dSGreg Clayton     return children;
194444d93782SGreg Clayton   }
194544d93782SGreg Clayton 
19468369b28dSGreg Clayton   void Unexpand() {
19478369b28dSGreg Clayton     expanded = false;
19488369b28dSGreg Clayton     calculated_children = false;
19498369b28dSGreg Clayton     children.clear();
19508369b28dSGreg Clayton   }
195144d93782SGreg Clayton 
1952b9c1b51eSKate Stone   void DrawTree(Window &window) {
195344d93782SGreg Clayton     if (parent)
195444d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
195544d93782SGreg Clayton 
1956b9c1b51eSKate Stone     if (might_have_children) {
195705097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
195805097246SAdrian Prantl       // "symbol" UTF8 string below
195944d93782SGreg Clayton       //            const char *symbol = "";
196044d93782SGreg Clayton       //            if (row.expanded)
196144d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
196244d93782SGreg Clayton       //            else
196344d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
196444d93782SGreg Clayton       //            window.PutCString (symbol);
196544d93782SGreg Clayton 
196605097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
196705097246SAdrian Prantl       // or '>' character...
196844d93782SGreg Clayton       //            if (expanded)
196944d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
197044d93782SGreg Clayton       //            else
197144d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
197205097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
197305097246SAdrian Prantl       // just use a diamond...
197444d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
197544d93782SGreg Clayton       window.PutChar(ACS_HLINE);
197644d93782SGreg Clayton     }
197744d93782SGreg Clayton   }
197844d93782SGreg Clayton 
1979b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
198044d93782SGreg Clayton     if (parent)
198144d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
198244d93782SGreg Clayton 
19838369b28dSGreg Clayton     if (&GetChildren().back() == child) {
198444d93782SGreg Clayton       // Last child
1985b9c1b51eSKate Stone       if (reverse_depth == 0) {
198644d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
198744d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1988b9c1b51eSKate Stone       } else {
198944d93782SGreg Clayton         window.PutChar(' ');
199044d93782SGreg Clayton         window.PutChar(' ');
199144d93782SGreg Clayton       }
1992b9c1b51eSKate Stone     } else {
1993b9c1b51eSKate Stone       if (reverse_depth == 0) {
199444d93782SGreg Clayton         window.PutChar(ACS_LTEE);
199544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1996b9c1b51eSKate Stone       } else {
199744d93782SGreg Clayton         window.PutChar(ACS_VLINE);
199844d93782SGreg Clayton         window.PutChar(' ');
199944d93782SGreg Clayton       }
200044d93782SGreg Clayton     }
200144d93782SGreg Clayton   }
200244d93782SGreg Clayton };
200344d93782SGreg Clayton 
2004b9c1b51eSKate Stone struct DisplayOptions {
200544d93782SGreg Clayton   bool show_types;
200644d93782SGreg Clayton };
200744d93782SGreg Clayton 
200844d93782SGreg Clayton class TreeItem;
200944d93782SGreg Clayton 
2010b9c1b51eSKate Stone class TreeDelegate {
201144d93782SGreg Clayton public:
2012c5dac77aSEugene Zelenko   TreeDelegate() = default;
2013315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
2014315b6884SEugene Zelenko 
201544d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
201644d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2017b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2018b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
201944d93782SGreg Clayton };
2020315b6884SEugene Zelenko 
202144d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
202244d93782SGreg Clayton 
2023b9c1b51eSKate Stone class TreeItem {
202444d93782SGreg Clayton public:
2025b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2026b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2027b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2028b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
202944d93782SGreg Clayton 
2030b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2031b9c1b51eSKate Stone     if (this != &rhs) {
203244d93782SGreg Clayton       m_parent = rhs.m_parent;
203344d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2034ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
203544d93782SGreg Clayton       m_identifier = rhs.m_identifier;
203644d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
203744d93782SGreg Clayton       m_children = rhs.m_children;
203844d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
203944d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
204044d93782SGreg Clayton     }
204144d93782SGreg Clayton     return *this;
204244d93782SGreg Clayton   }
204344d93782SGreg Clayton 
2044b9c1b51eSKate Stone   size_t GetDepth() const {
204544d93782SGreg Clayton     if (m_parent)
204644d93782SGreg Clayton       return 1 + m_parent->GetDepth();
204744d93782SGreg Clayton     return 0;
204844d93782SGreg Clayton   }
204944d93782SGreg Clayton 
2050b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
205144d93782SGreg Clayton 
2052b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
205344d93782SGreg Clayton 
2054b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
205544d93782SGreg Clayton 
2056b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
205744d93782SGreg Clayton 
2058b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
205944d93782SGreg Clayton 
2060b9c1b51eSKate Stone   size_t GetNumChildren() {
206144d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
206244d93782SGreg Clayton     return m_children.size();
206344d93782SGreg Clayton   }
206444d93782SGreg Clayton 
2065b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2066315b6884SEugene Zelenko 
2067b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
206844d93782SGreg Clayton     SetRowIndex(row_idx);
206944d93782SGreg Clayton     ++row_idx;
207044d93782SGreg Clayton 
2071ec990867SGreg Clayton     const bool expanded = IsExpanded();
2072ec990867SGreg Clayton 
207305097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
207405097246SAdrian Prantl     // number of children if the item is expanded
2075c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
207644d93782SGreg Clayton       GetNumChildren();
207744d93782SGreg Clayton 
2078b9c1b51eSKate Stone     for (auto &item : m_children) {
207944d93782SGreg Clayton       if (expanded)
208044d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
208144d93782SGreg Clayton       else
208244d93782SGreg Clayton         item.SetRowIndex(-1);
208344d93782SGreg Clayton     }
208444d93782SGreg Clayton   }
208544d93782SGreg Clayton 
2086b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
208744d93782SGreg Clayton 
2088b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
208944d93782SGreg Clayton 
2090b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
209144d93782SGreg Clayton 
2092b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
209344d93782SGreg Clayton 
2094b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2095b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
209644d93782SGreg Clayton     if (num_rows_left <= 0)
209744d93782SGreg Clayton       return false;
209844d93782SGreg Clayton 
2099b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
210044d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
210144d93782SGreg Clayton 
210244d93782SGreg Clayton       if (m_parent)
210344d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
210444d93782SGreg Clayton 
2105b9c1b51eSKate Stone       if (m_might_have_children) {
2106b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
210705097246SAdrian Prantl         // "symbol" UTF8 string below
210844d93782SGreg Clayton         //            const char *symbol = "";
210944d93782SGreg Clayton         //            if (row.expanded)
211044d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
211144d93782SGreg Clayton         //            else
211244d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
211344d93782SGreg Clayton         //            window.PutCString (symbol);
211444d93782SGreg Clayton 
211544d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
211644d93782SGreg Clayton         // 'v' or '>' character...
211744d93782SGreg Clayton         //            if (expanded)
211844d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
211944d93782SGreg Clayton         //            else
212044d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
212105097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
212205097246SAdrian Prantl         // just use a diamond...
212344d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
212444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
212544d93782SGreg Clayton       }
2126b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2127b9c1b51eSKate Stone                        window.IsActive();
212844d93782SGreg Clayton 
212944d93782SGreg Clayton       if (highlight)
213044d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
213144d93782SGreg Clayton 
213244d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
213344d93782SGreg Clayton 
213444d93782SGreg Clayton       if (highlight)
213544d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
213644d93782SGreg Clayton       ++row_idx;
213744d93782SGreg Clayton       --num_rows_left;
213844d93782SGreg Clayton     }
213944d93782SGreg Clayton 
214044d93782SGreg Clayton     if (num_rows_left <= 0)
214144d93782SGreg Clayton       return false; // We are done drawing...
214244d93782SGreg Clayton 
2143b9c1b51eSKate Stone     if (IsExpanded()) {
2144b9c1b51eSKate Stone       for (auto &item : m_children) {
214505097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
214605097246SAdrian Prantl         // done drawing and can exit this for loop
2147b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2148b9c1b51eSKate Stone                        num_rows_left))
214944d93782SGreg Clayton           break;
215044d93782SGreg Clayton       }
215144d93782SGreg Clayton     }
215244d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
215344d93782SGreg Clayton   }
215444d93782SGreg Clayton 
2155b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2156b9c1b51eSKate Stone                         uint32_t reverse_depth) {
215744d93782SGreg Clayton     if (m_parent)
215844d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
215944d93782SGreg Clayton 
2160b9c1b51eSKate Stone     if (&m_children.back() == child) {
216144d93782SGreg Clayton       // Last child
2162b9c1b51eSKate Stone       if (reverse_depth == 0) {
216344d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
216444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2165b9c1b51eSKate Stone       } else {
216644d93782SGreg Clayton         window.PutChar(' ');
216744d93782SGreg Clayton         window.PutChar(' ');
216844d93782SGreg Clayton       }
2169b9c1b51eSKate Stone     } else {
2170b9c1b51eSKate Stone       if (reverse_depth == 0) {
217144d93782SGreg Clayton         window.PutChar(ACS_LTEE);
217244d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2173b9c1b51eSKate Stone       } else {
217444d93782SGreg Clayton         window.PutChar(ACS_VLINE);
217544d93782SGreg Clayton         window.PutChar(' ');
217644d93782SGreg Clayton       }
217744d93782SGreg Clayton     }
217844d93782SGreg Clayton   }
217944d93782SGreg Clayton 
2180b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21813985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
218244d93782SGreg Clayton       return this;
218344d93782SGreg Clayton     if (m_children.empty())
2184c5dac77aSEugene Zelenko       return nullptr;
2185b9c1b51eSKate Stone     if (IsExpanded()) {
2186b9c1b51eSKate Stone       for (auto &item : m_children) {
218744d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
218844d93782SGreg Clayton         if (selected_item_ptr)
218944d93782SGreg Clayton           return selected_item_ptr;
219044d93782SGreg Clayton       }
219144d93782SGreg Clayton     }
2192c5dac77aSEugene Zelenko     return nullptr;
219344d93782SGreg Clayton   }
219444d93782SGreg Clayton 
2195b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2196ec990867SGreg Clayton 
2197b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2198ec990867SGreg Clayton 
2199b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
220044d93782SGreg Clayton 
2201b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
220244d93782SGreg Clayton 
2203b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2204ec990867SGreg Clayton 
220544d93782SGreg Clayton protected:
220644d93782SGreg Clayton   TreeItem *m_parent;
220744d93782SGreg Clayton   TreeDelegate &m_delegate;
2208ec990867SGreg Clayton   void *m_user_data;
220944d93782SGreg Clayton   uint64_t m_identifier;
2210b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2211b9c1b51eSKate Stone                  // root item
221244d93782SGreg Clayton   std::vector<TreeItem> m_children;
221344d93782SGreg Clayton   bool m_might_have_children;
221444d93782SGreg Clayton   bool m_is_expanded;
221544d93782SGreg Clayton };
221644d93782SGreg Clayton 
2217b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
221844d93782SGreg Clayton public:
2219b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2220b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2221b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2222b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2223b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
222444d93782SGreg Clayton 
2225b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
222644d93782SGreg Clayton 
2227b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2228b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2229b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
223044d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
223144d93782SGreg Clayton 
223244d93782SGreg Clayton     bool display_content = false;
2233b9c1b51eSKate Stone     if (process) {
223444d93782SGreg Clayton       StateType state = process->GetState();
2235b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
223644d93782SGreg Clayton         // We are stopped, so it is ok to
223744d93782SGreg Clayton         display_content = true;
2238b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
223944d93782SGreg Clayton         return true; // Don't do any updating when we are running
224044d93782SGreg Clayton       }
224144d93782SGreg Clayton     }
224244d93782SGreg Clayton 
224344d93782SGreg Clayton     m_min_x = 2;
224444d93782SGreg Clayton     m_min_y = 1;
224544d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
224644d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
224744d93782SGreg Clayton 
224844d93782SGreg Clayton     window.Erase();
224944d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
225044d93782SGreg Clayton 
2251b9c1b51eSKate Stone     if (display_content) {
225244d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
225344d93782SGreg Clayton       m_num_rows = 0;
225444d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
225544d93782SGreg Clayton 
225605097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
225705097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
225805097246SAdrian Prantl       // rows by setting the first visible row accordingly.
225944d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
226044d93782SGreg Clayton         m_first_visible_row = 0;
226144d93782SGreg Clayton 
226244d93782SGreg Clayton       // Make sure the selected row is always visible
226344d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
226444d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
226544d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
226644d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
226744d93782SGreg Clayton 
226844d93782SGreg Clayton       int row_idx = 0;
226944d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2270b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2271b9c1b51eSKate Stone                   num_rows_left);
227244d93782SGreg Clayton       // Get the selected row
227344d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2274b9c1b51eSKate Stone     } else {
2275c5dac77aSEugene Zelenko       m_selected_item = nullptr;
227644d93782SGreg Clayton     }
227744d93782SGreg Clayton 
227844d93782SGreg Clayton     window.DeferredRefresh();
227944d93782SGreg Clayton 
228044d93782SGreg Clayton     return true; // Drawing handled
228144d93782SGreg Clayton   }
228244d93782SGreg Clayton 
2283b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
228444d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
228544d93782SGreg Clayton   }
228644d93782SGreg Clayton 
2287b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
228844d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
228944d93782SGreg Clayton         {KEY_UP, "Select previous item"},
229044d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
229144d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2292b9c1b51eSKate Stone         {KEY_LEFT,
2293b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
229444d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
229544d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
229644d93782SGreg Clayton         {'h', "Show help dialog"},
229744d93782SGreg Clayton         {' ', "Toggle item expansion"},
229844d93782SGreg Clayton         {',', "Page up"},
229944d93782SGreg Clayton         {'.', "Page down"},
2300b9c1b51eSKate Stone         {'\0', nullptr}};
230144d93782SGreg Clayton     return g_source_view_key_help;
230244d93782SGreg Clayton   }
230344d93782SGreg Clayton 
2304b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2305b9c1b51eSKate Stone     switch (c) {
230644d93782SGreg Clayton     case ',':
230744d93782SGreg Clayton     case KEY_PPAGE:
230844d93782SGreg Clayton       // Page up key
2309b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
231044d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
231144d93782SGreg Clayton           m_first_visible_row -= m_max_y;
231244d93782SGreg Clayton         else
231344d93782SGreg Clayton           m_first_visible_row = 0;
231444d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
231544d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
231644d93782SGreg Clayton         if (m_selected_item)
231744d93782SGreg Clayton           m_selected_item->ItemWasSelected();
231844d93782SGreg Clayton       }
231944d93782SGreg Clayton       return eKeyHandled;
232044d93782SGreg Clayton 
232144d93782SGreg Clayton     case '.':
232244d93782SGreg Clayton     case KEY_NPAGE:
232344d93782SGreg Clayton       // Page down key
2324b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2325b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
232644d93782SGreg Clayton           m_first_visible_row += m_max_y;
232744d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
232844d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
232944d93782SGreg Clayton           if (m_selected_item)
233044d93782SGreg Clayton             m_selected_item->ItemWasSelected();
233144d93782SGreg Clayton         }
233244d93782SGreg Clayton       }
233344d93782SGreg Clayton       return eKeyHandled;
233444d93782SGreg Clayton 
233544d93782SGreg Clayton     case KEY_UP:
2336b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
233744d93782SGreg Clayton         --m_selected_row_idx;
233844d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
233944d93782SGreg Clayton         if (m_selected_item)
234044d93782SGreg Clayton           m_selected_item->ItemWasSelected();
234144d93782SGreg Clayton       }
234244d93782SGreg Clayton       return eKeyHandled;
2343315b6884SEugene Zelenko 
234444d93782SGreg Clayton     case KEY_DOWN:
2345b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
234644d93782SGreg Clayton         ++m_selected_row_idx;
234744d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
234844d93782SGreg Clayton         if (m_selected_item)
234944d93782SGreg Clayton           m_selected_item->ItemWasSelected();
235044d93782SGreg Clayton       }
235144d93782SGreg Clayton       return eKeyHandled;
235244d93782SGreg Clayton 
235344d93782SGreg Clayton     case KEY_RIGHT:
2354b9c1b51eSKate Stone       if (m_selected_item) {
235544d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
235644d93782SGreg Clayton           m_selected_item->Expand();
235744d93782SGreg Clayton       }
235844d93782SGreg Clayton       return eKeyHandled;
235944d93782SGreg Clayton 
236044d93782SGreg Clayton     case KEY_LEFT:
2361b9c1b51eSKate Stone       if (m_selected_item) {
236244d93782SGreg Clayton         if (m_selected_item->IsExpanded())
236344d93782SGreg Clayton           m_selected_item->Unexpand();
2364b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
236544d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
236644d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
236744d93782SGreg Clayton           if (m_selected_item)
236844d93782SGreg Clayton             m_selected_item->ItemWasSelected();
236944d93782SGreg Clayton         }
237044d93782SGreg Clayton       }
237144d93782SGreg Clayton       return eKeyHandled;
237244d93782SGreg Clayton 
237344d93782SGreg Clayton     case ' ':
237444d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2375b9c1b51eSKate Stone       if (m_selected_item) {
237644d93782SGreg Clayton         if (m_selected_item->IsExpanded())
237744d93782SGreg Clayton           m_selected_item->Unexpand();
237844d93782SGreg Clayton         else
237944d93782SGreg Clayton           m_selected_item->Expand();
238044d93782SGreg Clayton       }
238144d93782SGreg Clayton       return eKeyHandled;
238244d93782SGreg Clayton 
238344d93782SGreg Clayton     case 'h':
238444d93782SGreg Clayton       window.CreateHelpSubwindow();
238544d93782SGreg Clayton       return eKeyHandled;
238644d93782SGreg Clayton 
238744d93782SGreg Clayton     default:
238844d93782SGreg Clayton       break;
238944d93782SGreg Clayton     }
239044d93782SGreg Clayton     return eKeyNotHandled;
239144d93782SGreg Clayton   }
239244d93782SGreg Clayton 
239344d93782SGreg Clayton protected:
239444d93782SGreg Clayton   Debugger &m_debugger;
239544d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
239644d93782SGreg Clayton   TreeItem m_root;
239744d93782SGreg Clayton   TreeItem *m_selected_item;
239844d93782SGreg Clayton   int m_num_rows;
239944d93782SGreg Clayton   int m_selected_row_idx;
240044d93782SGreg Clayton   int m_first_visible_row;
240144d93782SGreg Clayton   int m_min_x;
240244d93782SGreg Clayton   int m_min_y;
240344d93782SGreg Clayton   int m_max_x;
240444d93782SGreg Clayton   int m_max_y;
240544d93782SGreg Clayton };
240644d93782SGreg Clayton 
2407b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
240844d93782SGreg Clayton public:
2409b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2410b9c1b51eSKate Stone     FormatEntity::Parse(
2411b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2412554f68d3SGreg Clayton         m_format);
241344d93782SGreg Clayton   }
241444d93782SGreg Clayton 
2415315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
241644d93782SGreg Clayton 
2417b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2418ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2419b9c1b51eSKate Stone     if (thread) {
242044d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2421ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2422b9c1b51eSKate Stone       if (frame_sp) {
242344d93782SGreg Clayton         StreamString strm;
2424b9c1b51eSKate Stone         const SymbolContext &sc =
2425b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
242644d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2427b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2428b9c1b51eSKate Stone                                  nullptr, false, false)) {
242944d93782SGreg Clayton           int right_pad = 1;
2430c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
243144d93782SGreg Clayton         }
243244d93782SGreg Clayton       }
243344d93782SGreg Clayton     }
243444d93782SGreg Clayton   }
2435315b6884SEugene Zelenko 
2436b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
243744d93782SGreg Clayton     // No children for frames yet...
243844d93782SGreg Clayton   }
243944d93782SGreg Clayton 
2440b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2441ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2442b9c1b51eSKate Stone     if (thread) {
2443b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2444b9c1b51eSKate Stone           thread->GetID());
244544d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2446ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
244744d93782SGreg Clayton       return true;
244844d93782SGreg Clayton     }
244944d93782SGreg Clayton     return false;
245044d93782SGreg Clayton   }
2451315b6884SEugene Zelenko 
2452554f68d3SGreg Clayton protected:
2453554f68d3SGreg Clayton   FormatEntity::Entry m_format;
245444d93782SGreg Clayton };
245544d93782SGreg Clayton 
2456b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
245744d93782SGreg Clayton public:
2458b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2459b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2460b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2461b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2462b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2463554f68d3SGreg Clayton                         m_format);
246444d93782SGreg Clayton   }
246544d93782SGreg Clayton 
2466315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
246744d93782SGreg Clayton 
2468b9c1b51eSKate Stone   ProcessSP GetProcess() {
2469b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2470b9c1b51eSKate Stone         .GetExecutionContext()
2471b9c1b51eSKate Stone         .GetProcessSP();
2472ec990867SGreg Clayton   }
2473ec990867SGreg Clayton 
2474b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2475ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2476ec990867SGreg Clayton     if (process_sp)
2477ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2478ec990867SGreg Clayton     return ThreadSP();
2479ec990867SGreg Clayton   }
2480ec990867SGreg Clayton 
2481b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2482ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2483b9c1b51eSKate Stone     if (thread_sp) {
248444d93782SGreg Clayton       StreamString strm;
248544d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2486b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2487b9c1b51eSKate Stone                                nullptr, false, false)) {
248844d93782SGreg Clayton         int right_pad = 1;
2489c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
249044d93782SGreg Clayton       }
249144d93782SGreg Clayton     }
249244d93782SGreg Clayton   }
2493315b6884SEugene Zelenko 
2494b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2495ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2496b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
249744d93782SGreg Clayton       StateType state = process_sp->GetState();
2498b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2499ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2500b9c1b51eSKate Stone         if (thread_sp) {
2501b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2502b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
250344d93782SGreg Clayton             return; // Children are already up to date
2504b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
250544d93782SGreg Clayton             // Always expand the thread item the first time we show it
2506796ac80bSJonas Devlieghere             m_frame_delegate_sp = std::make_shared<FrameTreeDelegate>();
250744d93782SGreg Clayton           }
250844d93782SGreg Clayton 
250944d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
251044d93782SGreg Clayton           m_tid = thread_sp->GetID();
251144d93782SGreg Clayton 
251244d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
251344d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
251444d93782SGreg Clayton           item.Resize(num_frames, t);
2515b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2516ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
251744d93782SGreg Clayton             item[i].SetIdentifier(i);
251844d93782SGreg Clayton           }
251944d93782SGreg Clayton         }
252044d93782SGreg Clayton         return;
252144d93782SGreg Clayton       }
252244d93782SGreg Clayton     }
252344d93782SGreg Clayton     item.ClearChildren();
252444d93782SGreg Clayton   }
252544d93782SGreg Clayton 
2526b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2527ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2528b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2529ec990867SGreg Clayton       StateType state = process_sp->GetState();
2530b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2531ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2532b9c1b51eSKate Stone         if (thread_sp) {
253344d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2534bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
253544d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2536b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
253744d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
253844d93782SGreg Clayton             return true;
253944d93782SGreg Clayton           }
254044d93782SGreg Clayton         }
2541ec990867SGreg Clayton       }
2542ec990867SGreg Clayton     }
254344d93782SGreg Clayton     return false;
254444d93782SGreg Clayton   }
254544d93782SGreg Clayton 
254644d93782SGreg Clayton protected:
254744d93782SGreg Clayton   Debugger &m_debugger;
254844d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
254944d93782SGreg Clayton   lldb::user_id_t m_tid;
255044d93782SGreg Clayton   uint32_t m_stop_id;
2551554f68d3SGreg Clayton   FormatEntity::Entry m_format;
255244d93782SGreg Clayton };
255344d93782SGreg Clayton 
2554b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2555ec990867SGreg Clayton public:
2556b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2557b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2558b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2559554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2560554f68d3SGreg Clayton                         m_format);
2561ec990867SGreg Clayton   }
2562ec990867SGreg Clayton 
2563315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2564ec990867SGreg Clayton 
2565b9c1b51eSKate Stone   ProcessSP GetProcess() {
2566b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2567b9c1b51eSKate Stone         .GetExecutionContext()
2568b9c1b51eSKate Stone         .GetProcessSP();
2569ec990867SGreg Clayton   }
2570ec990867SGreg Clayton 
2571b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2572ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2573b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2574ec990867SGreg Clayton       StreamString strm;
2575ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2576b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2577b9c1b51eSKate Stone                                nullptr, false, false)) {
2578ec990867SGreg Clayton         int right_pad = 1;
2579c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2580ec990867SGreg Clayton       }
2581ec990867SGreg Clayton     }
2582ec990867SGreg Clayton   }
2583ec990867SGreg Clayton 
2584b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2585ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2586b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2587ec990867SGreg Clayton       StateType state = process_sp->GetState();
2588b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2589ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2590ec990867SGreg Clayton         if (m_stop_id == stop_id)
2591ec990867SGreg Clayton           return; // Children are already up to date
2592ec990867SGreg Clayton 
2593ec990867SGreg Clayton         m_stop_id = stop_id;
2594ec990867SGreg Clayton 
2595b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2596ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2597ec990867SGreg Clayton           // item.Expand();
2598796ac80bSJonas Devlieghere           m_thread_delegate_sp =
2599796ac80bSJonas Devlieghere               std::make_shared<ThreadTreeDelegate>(m_debugger);
2600ec990867SGreg Clayton         }
2601ec990867SGreg Clayton 
2602ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2603ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2604bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2605ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2606ec990867SGreg Clayton         item.Resize(num_threads, t);
2607b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2608ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2609ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2610ec990867SGreg Clayton         }
2611ec990867SGreg Clayton         return;
2612ec990867SGreg Clayton       }
2613ec990867SGreg Clayton     }
2614ec990867SGreg Clayton     item.ClearChildren();
2615ec990867SGreg Clayton   }
2616ec990867SGreg Clayton 
2617b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2618ec990867SGreg Clayton 
2619ec990867SGreg Clayton protected:
2620ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2621ec990867SGreg Clayton   Debugger &m_debugger;
2622ec990867SGreg Clayton   uint32_t m_stop_id;
2623554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2624ec990867SGreg Clayton };
2625ec990867SGreg Clayton 
2626b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
262744d93782SGreg Clayton public:
2628b9c1b51eSKate Stone   ValueObjectListDelegate()
26298369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2630b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2631b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {}
263244d93782SGreg Clayton 
2633b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
26348369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2635b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2636b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {
263744d93782SGreg Clayton     SetValues(valobj_list);
263844d93782SGreg Clayton   }
263944d93782SGreg Clayton 
2640315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
264144d93782SGreg Clayton 
2642b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2643c5dac77aSEugene Zelenko     m_selected_row = nullptr;
264444d93782SGreg Clayton     m_selected_row_idx = 0;
264544d93782SGreg Clayton     m_first_visible_row = 0;
264644d93782SGreg Clayton     m_num_rows = 0;
264744d93782SGreg Clayton     m_rows.clear();
26488369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
26498369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
265044d93782SGreg Clayton   }
265144d93782SGreg Clayton 
2652b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
265344d93782SGreg Clayton     m_num_rows = 0;
265444d93782SGreg Clayton     m_min_x = 2;
265544d93782SGreg Clayton     m_min_y = 1;
265644d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
265744d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
265844d93782SGreg Clayton 
265944d93782SGreg Clayton     window.Erase();
266044d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
266144d93782SGreg Clayton 
266244d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
266344d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
266444d93782SGreg Clayton 
266505097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
266605097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
266705097246SAdrian Prantl     // rows by setting the first visible row accordingly.
266844d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
266944d93782SGreg Clayton       m_first_visible_row = 0;
267044d93782SGreg Clayton 
267144d93782SGreg Clayton     // Make sure the selected row is always visible
267244d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
267344d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
267444d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
267544d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
267644d93782SGreg Clayton 
267744d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
267844d93782SGreg Clayton 
267944d93782SGreg Clayton     window.DeferredRefresh();
268044d93782SGreg Clayton 
268144d93782SGreg Clayton     // Get the selected row
268244d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
268305097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
268405097246SAdrian Prantl     // always on the same line
268544d93782SGreg Clayton     if (m_selected_row)
2686b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
268744d93782SGreg Clayton 
268844d93782SGreg Clayton     return true; // Drawing handled
268944d93782SGreg Clayton   }
269044d93782SGreg Clayton 
2691b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
269244d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
269344d93782SGreg Clayton         {KEY_UP, "Select previous item"},
269444d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
269544d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
269644d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
269744d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
269844d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
269944d93782SGreg Clayton         {'A', "Format as annotated address"},
270044d93782SGreg Clayton         {'b', "Format as binary"},
270144d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
270244d93782SGreg Clayton         {'c', "Format as character"},
270344d93782SGreg Clayton         {'d', "Format as a signed integer"},
270444d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
270544d93782SGreg Clayton         {'f', "Format as float"},
270644d93782SGreg Clayton         {'h', "Show help dialog"},
270744d93782SGreg Clayton         {'i', "Format as instructions"},
270844d93782SGreg Clayton         {'o', "Format as octal"},
270944d93782SGreg Clayton         {'p', "Format as pointer"},
271044d93782SGreg Clayton         {'s', "Format as C string"},
271144d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
271244d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
271344d93782SGreg Clayton         {'x', "Format as hex"},
271444d93782SGreg Clayton         {'X', "Format as uppercase hex"},
271544d93782SGreg Clayton         {' ', "Toggle item expansion"},
271644d93782SGreg Clayton         {',', "Page up"},
271744d93782SGreg Clayton         {'.', "Page down"},
2718b9c1b51eSKate Stone         {'\0', nullptr}};
271944d93782SGreg Clayton     return g_source_view_key_help;
272044d93782SGreg Clayton   }
272144d93782SGreg Clayton 
2722b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2723b9c1b51eSKate Stone     switch (c) {
272444d93782SGreg Clayton     case 'x':
272544d93782SGreg Clayton     case 'X':
272644d93782SGreg Clayton     case 'o':
272744d93782SGreg Clayton     case 's':
272844d93782SGreg Clayton     case 'u':
272944d93782SGreg Clayton     case 'd':
273044d93782SGreg Clayton     case 'D':
273144d93782SGreg Clayton     case 'i':
273244d93782SGreg Clayton     case 'A':
273344d93782SGreg Clayton     case 'p':
273444d93782SGreg Clayton     case 'c':
273544d93782SGreg Clayton     case 'b':
273644d93782SGreg Clayton     case 'B':
273744d93782SGreg Clayton     case 'f':
273844d93782SGreg Clayton       // Change the format for the currently selected item
27398369b28dSGreg Clayton       if (m_selected_row) {
27408369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
27418369b28dSGreg Clayton         if (valobj_sp)
27428369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
27438369b28dSGreg Clayton       }
274444d93782SGreg Clayton       return eKeyHandled;
274544d93782SGreg Clayton 
274644d93782SGreg Clayton     case 't':
274744d93782SGreg Clayton       // Toggle showing type names
274844d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
274944d93782SGreg Clayton       return eKeyHandled;
275044d93782SGreg Clayton 
275144d93782SGreg Clayton     case ',':
275244d93782SGreg Clayton     case KEY_PPAGE:
275344d93782SGreg Clayton       // Page up key
2754b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27553985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
275644d93782SGreg Clayton           m_first_visible_row -= m_max_y;
275744d93782SGreg Clayton         else
275844d93782SGreg Clayton           m_first_visible_row = 0;
275944d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
276044d93782SGreg Clayton       }
276144d93782SGreg Clayton       return eKeyHandled;
276244d93782SGreg Clayton 
276344d93782SGreg Clayton     case '.':
276444d93782SGreg Clayton     case KEY_NPAGE:
276544d93782SGreg Clayton       // Page down key
2766b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2767b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
276844d93782SGreg Clayton           m_first_visible_row += m_max_y;
276944d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
277044d93782SGreg Clayton         }
277144d93782SGreg Clayton       }
277244d93782SGreg Clayton       return eKeyHandled;
277344d93782SGreg Clayton 
277444d93782SGreg Clayton     case KEY_UP:
277544d93782SGreg Clayton       if (m_selected_row_idx > 0)
277644d93782SGreg Clayton         --m_selected_row_idx;
277744d93782SGreg Clayton       return eKeyHandled;
2778315b6884SEugene Zelenko 
277944d93782SGreg Clayton     case KEY_DOWN:
278044d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
278144d93782SGreg Clayton         ++m_selected_row_idx;
278244d93782SGreg Clayton       return eKeyHandled;
278344d93782SGreg Clayton 
278444d93782SGreg Clayton     case KEY_RIGHT:
2785b9c1b51eSKate Stone       if (m_selected_row) {
278644d93782SGreg Clayton         if (!m_selected_row->expanded)
278744d93782SGreg Clayton           m_selected_row->Expand();
278844d93782SGreg Clayton       }
278944d93782SGreg Clayton       return eKeyHandled;
279044d93782SGreg Clayton 
279144d93782SGreg Clayton     case KEY_LEFT:
2792b9c1b51eSKate Stone       if (m_selected_row) {
279344d93782SGreg Clayton         if (m_selected_row->expanded)
279444d93782SGreg Clayton           m_selected_row->Unexpand();
279544d93782SGreg Clayton         else if (m_selected_row->parent)
279644d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
279744d93782SGreg Clayton       }
279844d93782SGreg Clayton       return eKeyHandled;
279944d93782SGreg Clayton 
280044d93782SGreg Clayton     case ' ':
280144d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2802b9c1b51eSKate Stone       if (m_selected_row) {
280344d93782SGreg Clayton         if (m_selected_row->expanded)
280444d93782SGreg Clayton           m_selected_row->Unexpand();
280544d93782SGreg Clayton         else
280644d93782SGreg Clayton           m_selected_row->Expand();
280744d93782SGreg Clayton       }
280844d93782SGreg Clayton       return eKeyHandled;
280944d93782SGreg Clayton 
281044d93782SGreg Clayton     case 'h':
281144d93782SGreg Clayton       window.CreateHelpSubwindow();
281244d93782SGreg Clayton       return eKeyHandled;
281344d93782SGreg Clayton 
281444d93782SGreg Clayton     default:
281544d93782SGreg Clayton       break;
281644d93782SGreg Clayton     }
281744d93782SGreg Clayton     return eKeyNotHandled;
281844d93782SGreg Clayton   }
281944d93782SGreg Clayton 
282044d93782SGreg Clayton protected:
282144d93782SGreg Clayton   std::vector<Row> m_rows;
282244d93782SGreg Clayton   Row *m_selected_row;
282344d93782SGreg Clayton   uint32_t m_selected_row_idx;
282444d93782SGreg Clayton   uint32_t m_first_visible_row;
282544d93782SGreg Clayton   uint32_t m_num_rows;
282644d93782SGreg Clayton   int m_min_x;
282744d93782SGreg Clayton   int m_min_y;
282844d93782SGreg Clayton   int m_max_x;
282944d93782SGreg Clayton   int m_max_y;
283044d93782SGreg Clayton 
2831b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2832b9c1b51eSKate Stone     switch (c) {
2833b9c1b51eSKate Stone     case 'x':
2834b9c1b51eSKate Stone       return eFormatHex;
2835b9c1b51eSKate Stone     case 'X':
2836b9c1b51eSKate Stone       return eFormatHexUppercase;
2837b9c1b51eSKate Stone     case 'o':
2838b9c1b51eSKate Stone       return eFormatOctal;
2839b9c1b51eSKate Stone     case 's':
2840b9c1b51eSKate Stone       return eFormatCString;
2841b9c1b51eSKate Stone     case 'u':
2842b9c1b51eSKate Stone       return eFormatUnsigned;
2843b9c1b51eSKate Stone     case 'd':
2844b9c1b51eSKate Stone       return eFormatDecimal;
2845b9c1b51eSKate Stone     case 'D':
2846b9c1b51eSKate Stone       return eFormatDefault;
2847b9c1b51eSKate Stone     case 'i':
2848b9c1b51eSKate Stone       return eFormatInstruction;
2849b9c1b51eSKate Stone     case 'A':
2850b9c1b51eSKate Stone       return eFormatAddressInfo;
2851b9c1b51eSKate Stone     case 'p':
2852b9c1b51eSKate Stone       return eFormatPointer;
2853b9c1b51eSKate Stone     case 'c':
2854b9c1b51eSKate Stone       return eFormatChar;
2855b9c1b51eSKate Stone     case 'b':
2856b9c1b51eSKate Stone       return eFormatBinary;
2857b9c1b51eSKate Stone     case 'B':
2858b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2859b9c1b51eSKate Stone     case 'f':
2860b9c1b51eSKate Stone       return eFormatFloat;
286144d93782SGreg Clayton     }
286244d93782SGreg Clayton     return eFormatDefault;
286344d93782SGreg Clayton   }
286444d93782SGreg Clayton 
2865b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2866b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28678369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
286844d93782SGreg Clayton 
2869c5dac77aSEugene Zelenko     if (valobj == nullptr)
287044d93782SGreg Clayton       return false;
287144d93782SGreg Clayton 
2872b9c1b51eSKate Stone     const char *type_name =
2873b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
287444d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
287544d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
287644d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
287744d93782SGreg Clayton 
287844d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
287944d93782SGreg Clayton 
288044d93782SGreg Clayton     row.DrawTree(window);
288144d93782SGreg Clayton 
288244d93782SGreg Clayton     if (highlight)
288344d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
288444d93782SGreg Clayton 
288544d93782SGreg Clayton     if (type_name && type_name[0])
288644d93782SGreg Clayton       window.Printf("(%s) ", type_name);
288744d93782SGreg Clayton 
288844d93782SGreg Clayton     if (name && name[0])
288944d93782SGreg Clayton       window.PutCString(name);
289044d93782SGreg Clayton 
289144d93782SGreg Clayton     attr_t changd_attr = 0;
289244d93782SGreg Clayton     if (valobj->GetValueDidChange())
289344d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
289444d93782SGreg Clayton 
2895b9c1b51eSKate Stone     if (value && value[0]) {
289644d93782SGreg Clayton       window.PutCString(" = ");
289744d93782SGreg Clayton       if (changd_attr)
289844d93782SGreg Clayton         window.AttributeOn(changd_attr);
289944d93782SGreg Clayton       window.PutCString(value);
290044d93782SGreg Clayton       if (changd_attr)
290144d93782SGreg Clayton         window.AttributeOff(changd_attr);
290244d93782SGreg Clayton     }
290344d93782SGreg Clayton 
2904b9c1b51eSKate Stone     if (summary && summary[0]) {
290544d93782SGreg Clayton       window.PutChar(' ');
290644d93782SGreg Clayton       if (changd_attr)
290744d93782SGreg Clayton         window.AttributeOn(changd_attr);
290844d93782SGreg Clayton       window.PutCString(summary);
290944d93782SGreg Clayton       if (changd_attr)
291044d93782SGreg Clayton         window.AttributeOff(changd_attr);
291144d93782SGreg Clayton     }
291244d93782SGreg Clayton 
291344d93782SGreg Clayton     if (highlight)
291444d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
291544d93782SGreg Clayton 
291644d93782SGreg Clayton     return true;
291744d93782SGreg Clayton   }
2918315b6884SEugene Zelenko 
2919b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2920b9c1b51eSKate Stone                    DisplayOptions &options) {
292144d93782SGreg Clayton     // >   0x25B7
292244d93782SGreg Clayton     // \/  0x25BD
292344d93782SGreg Clayton 
292444d93782SGreg Clayton     bool window_is_active = window.IsActive();
2925b9c1b51eSKate Stone     for (auto &row : rows) {
292644d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
292744d93782SGreg Clayton       // Save the row index in each Row structure
292844d93782SGreg Clayton       row.row_idx = m_num_rows;
292944d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2930b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2931b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
293244d93782SGreg Clayton         row.x = m_min_x;
293344d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2934b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2935b9c1b51eSKate Stone                              window_is_active &&
2936b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2937b9c1b51eSKate Stone                              last_child)) {
293844d93782SGreg Clayton           ++m_num_rows;
2939b9c1b51eSKate Stone         } else {
294044d93782SGreg Clayton           row.x = 0;
294144d93782SGreg Clayton           row.y = 0;
294244d93782SGreg Clayton         }
2943b9c1b51eSKate Stone       } else {
294444d93782SGreg Clayton         row.x = 0;
294544d93782SGreg Clayton         row.y = 0;
294644d93782SGreg Clayton         ++m_num_rows;
294744d93782SGreg Clayton       }
294844d93782SGreg Clayton 
29498369b28dSGreg Clayton       auto &children = row.GetChildren();
29508369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
29518369b28dSGreg Clayton         DisplayRows(window, children, options);
295244d93782SGreg Clayton       }
295344d93782SGreg Clayton     }
295444d93782SGreg Clayton   }
295544d93782SGreg Clayton 
29568369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
295744d93782SGreg Clayton     int row_count = 0;
29588369b28dSGreg Clayton     for (auto &row : rows) {
295944d93782SGreg Clayton       ++row_count;
296044d93782SGreg Clayton       if (row.expanded)
29618369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
296244d93782SGreg Clayton     }
296344d93782SGreg Clayton     return row_count;
296444d93782SGreg Clayton   }
2965315b6884SEugene Zelenko 
2966b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2967b9c1b51eSKate Stone     for (auto &row : rows) {
296844d93782SGreg Clayton       if (row_index == 0)
296944d93782SGreg Clayton         return &row;
2970b9c1b51eSKate Stone       else {
297144d93782SGreg Clayton         --row_index;
29728369b28dSGreg Clayton         auto &children = row.GetChildren();
29738369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29748369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
297544d93782SGreg Clayton           if (result)
297644d93782SGreg Clayton             return result;
297744d93782SGreg Clayton         }
297844d93782SGreg Clayton       }
297944d93782SGreg Clayton     }
2980c5dac77aSEugene Zelenko     return nullptr;
298144d93782SGreg Clayton   }
298244d93782SGreg Clayton 
2983b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
298444d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
298544d93782SGreg Clayton   }
298644d93782SGreg Clayton 
2987b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
298844d93782SGreg Clayton 
298944d93782SGreg Clayton   static DisplayOptions g_options;
299044d93782SGreg Clayton };
299144d93782SGreg Clayton 
2992b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
299344d93782SGreg Clayton public:
2994b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
2995b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
2996b9c1b51eSKate Stone         m_frame_block(nullptr) {}
299744d93782SGreg Clayton 
2998315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
299944d93782SGreg Clayton 
3000b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
300144d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
300244d93782SGreg Clayton   }
300344d93782SGreg Clayton 
3004b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3005b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3006b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
300744d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3008c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
3009c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
301044d93782SGreg Clayton 
3011b9c1b51eSKate Stone     if (process) {
301244d93782SGreg Clayton       StateType state = process->GetState();
3013b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
301444d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
301544d93782SGreg Clayton         if (frame)
301644d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
3017b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
301844d93782SGreg Clayton         return true; // Don't do any updating when we are running
301944d93782SGreg Clayton       }
302044d93782SGreg Clayton     }
302144d93782SGreg Clayton 
302244d93782SGreg Clayton     ValueObjectList local_values;
3023b9c1b51eSKate Stone     if (frame_block) {
302444d93782SGreg Clayton       // Only update the variables if they have changed
3025b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
302644d93782SGreg Clayton         m_frame_block = frame_block;
302744d93782SGreg Clayton 
302844d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3029b9c1b51eSKate Stone         if (locals) {
303044d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
303144d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3032b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3033b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3034b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3035b9c1b51eSKate Stone             if (value_sp) {
3036eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3037eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3038eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3039eb72dc7dSGreg Clayton               else
3040eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3041eb72dc7dSGreg Clayton             }
3042eb72dc7dSGreg Clayton           }
304344d93782SGreg Clayton           // Update the values
304444d93782SGreg Clayton           SetValues(local_values);
304544d93782SGreg Clayton         }
304644d93782SGreg Clayton       }
3047b9c1b51eSKate Stone     } else {
3048c5dac77aSEugene Zelenko       m_frame_block = nullptr;
304944d93782SGreg Clayton       // Update the values with an empty list if there is no frame
305044d93782SGreg Clayton       SetValues(local_values);
305144d93782SGreg Clayton     }
305244d93782SGreg Clayton 
305344d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
305444d93782SGreg Clayton   }
305544d93782SGreg Clayton 
305644d93782SGreg Clayton protected:
305744d93782SGreg Clayton   Debugger &m_debugger;
305844d93782SGreg Clayton   Block *m_frame_block;
305944d93782SGreg Clayton };
306044d93782SGreg Clayton 
3061b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
306244d93782SGreg Clayton public:
3063b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3064b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
306544d93782SGreg Clayton 
3066315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
306744d93782SGreg Clayton 
3068b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
306944d93782SGreg Clayton     return "Register window keyboard shortcuts:";
307044d93782SGreg Clayton   }
307144d93782SGreg Clayton 
3072b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3073b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3074b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
307544d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
307644d93782SGreg Clayton 
307744d93782SGreg Clayton     ValueObjectList value_list;
3078b9c1b51eSKate Stone     if (frame) {
3079b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
308044d93782SGreg Clayton         m_stack_id = frame->GetStackID();
308144d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3082b9c1b51eSKate Stone         if (reg_ctx) {
308344d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3084b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3085b9c1b51eSKate Stone             value_list.Append(
3086b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
308744d93782SGreg Clayton           }
308844d93782SGreg Clayton         }
308944d93782SGreg Clayton         SetValues(value_list);
309044d93782SGreg Clayton       }
3091b9c1b51eSKate Stone     } else {
309244d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
309344d93782SGreg Clayton       if (process && process->IsAlive())
309444d93782SGreg Clayton         return true; // Don't do any updating if we are running
3095b9c1b51eSKate Stone       else {
309605097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
309705097246SAdrian Prantl         // process isn't alive anymore
309844d93782SGreg Clayton         SetValues(value_list);
309944d93782SGreg Clayton       }
310044d93782SGreg Clayton     }
310144d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
310244d93782SGreg Clayton   }
310344d93782SGreg Clayton 
310444d93782SGreg Clayton protected:
310544d93782SGreg Clayton   Debugger &m_debugger;
310644d93782SGreg Clayton   StackID m_stack_id;
310744d93782SGreg Clayton };
310844d93782SGreg Clayton 
3109b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
311044d93782SGreg Clayton   static char g_desc[32];
3111b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
311244d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
311344d93782SGreg Clayton     return g_desc;
311444d93782SGreg Clayton   }
3115b9c1b51eSKate Stone   switch (ch) {
3116b9c1b51eSKate Stone   case KEY_DOWN:
3117b9c1b51eSKate Stone     return "down";
3118b9c1b51eSKate Stone   case KEY_UP:
3119b9c1b51eSKate Stone     return "up";
3120b9c1b51eSKate Stone   case KEY_LEFT:
3121b9c1b51eSKate Stone     return "left";
3122b9c1b51eSKate Stone   case KEY_RIGHT:
3123b9c1b51eSKate Stone     return "right";
3124b9c1b51eSKate Stone   case KEY_HOME:
3125b9c1b51eSKate Stone     return "home";
3126b9c1b51eSKate Stone   case KEY_BACKSPACE:
3127b9c1b51eSKate Stone     return "backspace";
3128b9c1b51eSKate Stone   case KEY_DL:
3129b9c1b51eSKate Stone     return "delete-line";
3130b9c1b51eSKate Stone   case KEY_IL:
3131b9c1b51eSKate Stone     return "insert-line";
3132b9c1b51eSKate Stone   case KEY_DC:
3133b9c1b51eSKate Stone     return "delete-char";
3134b9c1b51eSKate Stone   case KEY_IC:
3135b9c1b51eSKate Stone     return "insert-char";
3136b9c1b51eSKate Stone   case KEY_CLEAR:
3137b9c1b51eSKate Stone     return "clear";
3138b9c1b51eSKate Stone   case KEY_EOS:
3139b9c1b51eSKate Stone     return "clear-to-eos";
3140b9c1b51eSKate Stone   case KEY_EOL:
3141b9c1b51eSKate Stone     return "clear-to-eol";
3142b9c1b51eSKate Stone   case KEY_SF:
3143b9c1b51eSKate Stone     return "scroll-forward";
3144b9c1b51eSKate Stone   case KEY_SR:
3145b9c1b51eSKate Stone     return "scroll-backward";
3146b9c1b51eSKate Stone   case KEY_NPAGE:
3147b9c1b51eSKate Stone     return "page-down";
3148b9c1b51eSKate Stone   case KEY_PPAGE:
3149b9c1b51eSKate Stone     return "page-up";
3150b9c1b51eSKate Stone   case KEY_STAB:
3151b9c1b51eSKate Stone     return "set-tab";
3152b9c1b51eSKate Stone   case KEY_CTAB:
3153b9c1b51eSKate Stone     return "clear-tab";
3154b9c1b51eSKate Stone   case KEY_CATAB:
3155b9c1b51eSKate Stone     return "clear-all-tabs";
3156b9c1b51eSKate Stone   case KEY_ENTER:
3157b9c1b51eSKate Stone     return "enter";
3158b9c1b51eSKate Stone   case KEY_PRINT:
3159b9c1b51eSKate Stone     return "print";
3160b9c1b51eSKate Stone   case KEY_LL:
3161b9c1b51eSKate Stone     return "lower-left key";
3162b9c1b51eSKate Stone   case KEY_A1:
3163b9c1b51eSKate Stone     return "upper left of keypad";
3164b9c1b51eSKate Stone   case KEY_A3:
3165b9c1b51eSKate Stone     return "upper right of keypad";
3166b9c1b51eSKate Stone   case KEY_B2:
3167b9c1b51eSKate Stone     return "center of keypad";
3168b9c1b51eSKate Stone   case KEY_C1:
3169b9c1b51eSKate Stone     return "lower left of keypad";
3170b9c1b51eSKate Stone   case KEY_C3:
3171b9c1b51eSKate Stone     return "lower right of keypad";
3172b9c1b51eSKate Stone   case KEY_BTAB:
3173b9c1b51eSKate Stone     return "back-tab key";
3174b9c1b51eSKate Stone   case KEY_BEG:
3175b9c1b51eSKate Stone     return "begin key";
3176b9c1b51eSKate Stone   case KEY_CANCEL:
3177b9c1b51eSKate Stone     return "cancel key";
3178b9c1b51eSKate Stone   case KEY_CLOSE:
3179b9c1b51eSKate Stone     return "close key";
3180b9c1b51eSKate Stone   case KEY_COMMAND:
3181b9c1b51eSKate Stone     return "command key";
3182b9c1b51eSKate Stone   case KEY_COPY:
3183b9c1b51eSKate Stone     return "copy key";
3184b9c1b51eSKate Stone   case KEY_CREATE:
3185b9c1b51eSKate Stone     return "create key";
3186b9c1b51eSKate Stone   case KEY_END:
3187b9c1b51eSKate Stone     return "end key";
3188b9c1b51eSKate Stone   case KEY_EXIT:
3189b9c1b51eSKate Stone     return "exit key";
3190b9c1b51eSKate Stone   case KEY_FIND:
3191b9c1b51eSKate Stone     return "find key";
3192b9c1b51eSKate Stone   case KEY_HELP:
3193b9c1b51eSKate Stone     return "help key";
3194b9c1b51eSKate Stone   case KEY_MARK:
3195b9c1b51eSKate Stone     return "mark key";
3196b9c1b51eSKate Stone   case KEY_MESSAGE:
3197b9c1b51eSKate Stone     return "message key";
3198b9c1b51eSKate Stone   case KEY_MOVE:
3199b9c1b51eSKate Stone     return "move key";
3200b9c1b51eSKate Stone   case KEY_NEXT:
3201b9c1b51eSKate Stone     return "next key";
3202b9c1b51eSKate Stone   case KEY_OPEN:
3203b9c1b51eSKate Stone     return "open key";
3204b9c1b51eSKate Stone   case KEY_OPTIONS:
3205b9c1b51eSKate Stone     return "options key";
3206b9c1b51eSKate Stone   case KEY_PREVIOUS:
3207b9c1b51eSKate Stone     return "previous key";
3208b9c1b51eSKate Stone   case KEY_REDO:
3209b9c1b51eSKate Stone     return "redo key";
3210b9c1b51eSKate Stone   case KEY_REFERENCE:
3211b9c1b51eSKate Stone     return "reference key";
3212b9c1b51eSKate Stone   case KEY_REFRESH:
3213b9c1b51eSKate Stone     return "refresh key";
3214b9c1b51eSKate Stone   case KEY_REPLACE:
3215b9c1b51eSKate Stone     return "replace key";
3216b9c1b51eSKate Stone   case KEY_RESTART:
3217b9c1b51eSKate Stone     return "restart key";
3218b9c1b51eSKate Stone   case KEY_RESUME:
3219b9c1b51eSKate Stone     return "resume key";
3220b9c1b51eSKate Stone   case KEY_SAVE:
3221b9c1b51eSKate Stone     return "save key";
3222b9c1b51eSKate Stone   case KEY_SBEG:
3223b9c1b51eSKate Stone     return "shifted begin key";
3224b9c1b51eSKate Stone   case KEY_SCANCEL:
3225b9c1b51eSKate Stone     return "shifted cancel key";
3226b9c1b51eSKate Stone   case KEY_SCOMMAND:
3227b9c1b51eSKate Stone     return "shifted command key";
3228b9c1b51eSKate Stone   case KEY_SCOPY:
3229b9c1b51eSKate Stone     return "shifted copy key";
3230b9c1b51eSKate Stone   case KEY_SCREATE:
3231b9c1b51eSKate Stone     return "shifted create key";
3232b9c1b51eSKate Stone   case KEY_SDC:
3233b9c1b51eSKate Stone     return "shifted delete-character key";
3234b9c1b51eSKate Stone   case KEY_SDL:
3235b9c1b51eSKate Stone     return "shifted delete-line key";
3236b9c1b51eSKate Stone   case KEY_SELECT:
3237b9c1b51eSKate Stone     return "select key";
3238b9c1b51eSKate Stone   case KEY_SEND:
3239b9c1b51eSKate Stone     return "shifted end key";
3240b9c1b51eSKate Stone   case KEY_SEOL:
3241b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3242b9c1b51eSKate Stone   case KEY_SEXIT:
3243b9c1b51eSKate Stone     return "shifted exit key";
3244b9c1b51eSKate Stone   case KEY_SFIND:
3245b9c1b51eSKate Stone     return "shifted find key";
3246b9c1b51eSKate Stone   case KEY_SHELP:
3247b9c1b51eSKate Stone     return "shifted help key";
3248b9c1b51eSKate Stone   case KEY_SHOME:
3249b9c1b51eSKate Stone     return "shifted home key";
3250b9c1b51eSKate Stone   case KEY_SIC:
3251b9c1b51eSKate Stone     return "shifted insert-character key";
3252b9c1b51eSKate Stone   case KEY_SLEFT:
3253b9c1b51eSKate Stone     return "shifted left-arrow key";
3254b9c1b51eSKate Stone   case KEY_SMESSAGE:
3255b9c1b51eSKate Stone     return "shifted message key";
3256b9c1b51eSKate Stone   case KEY_SMOVE:
3257b9c1b51eSKate Stone     return "shifted move key";
3258b9c1b51eSKate Stone   case KEY_SNEXT:
3259b9c1b51eSKate Stone     return "shifted next key";
3260b9c1b51eSKate Stone   case KEY_SOPTIONS:
3261b9c1b51eSKate Stone     return "shifted options key";
3262b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3263b9c1b51eSKate Stone     return "shifted previous key";
3264b9c1b51eSKate Stone   case KEY_SPRINT:
3265b9c1b51eSKate Stone     return "shifted print key";
3266b9c1b51eSKate Stone   case KEY_SREDO:
3267b9c1b51eSKate Stone     return "shifted redo key";
3268b9c1b51eSKate Stone   case KEY_SREPLACE:
3269b9c1b51eSKate Stone     return "shifted replace key";
3270b9c1b51eSKate Stone   case KEY_SRIGHT:
3271b9c1b51eSKate Stone     return "shifted right-arrow key";
3272b9c1b51eSKate Stone   case KEY_SRSUME:
3273b9c1b51eSKate Stone     return "shifted resume key";
3274b9c1b51eSKate Stone   case KEY_SSAVE:
3275b9c1b51eSKate Stone     return "shifted save key";
3276b9c1b51eSKate Stone   case KEY_SSUSPEND:
3277b9c1b51eSKate Stone     return "shifted suspend key";
3278b9c1b51eSKate Stone   case KEY_SUNDO:
3279b9c1b51eSKate Stone     return "shifted undo key";
3280b9c1b51eSKate Stone   case KEY_SUSPEND:
3281b9c1b51eSKate Stone     return "suspend key";
3282b9c1b51eSKate Stone   case KEY_UNDO:
3283b9c1b51eSKate Stone     return "undo key";
3284b9c1b51eSKate Stone   case KEY_MOUSE:
3285b9c1b51eSKate Stone     return "Mouse event has occurred";
3286b9c1b51eSKate Stone   case KEY_RESIZE:
3287b9c1b51eSKate Stone     return "Terminal resize event";
328827801f4fSBruce Mitchener #ifdef KEY_EVENT
3289b9c1b51eSKate Stone   case KEY_EVENT:
3290b9c1b51eSKate Stone     return "We were interrupted by an event";
329127801f4fSBruce Mitchener #endif
3292b9c1b51eSKate Stone   case KEY_RETURN:
3293b9c1b51eSKate Stone     return "return";
3294b9c1b51eSKate Stone   case ' ':
3295b9c1b51eSKate Stone     return "space";
3296b9c1b51eSKate Stone   case '\t':
3297b9c1b51eSKate Stone     return "tab";
3298b9c1b51eSKate Stone   case KEY_ESCAPE:
3299b9c1b51eSKate Stone     return "escape";
330044d93782SGreg Clayton   default:
330144d93782SGreg Clayton     if (isprint(ch))
330244d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
330344d93782SGreg Clayton     else
330444d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
330544d93782SGreg Clayton     return g_desc;
330644d93782SGreg Clayton   }
3307c5dac77aSEugene Zelenko   return nullptr;
330844d93782SGreg Clayton }
330944d93782SGreg Clayton 
3310b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3311b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3312b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3313b9c1b51eSKate Stone   if (text && text[0]) {
331444d93782SGreg Clayton     m_text.SplitIntoLines(text);
331544d93782SGreg Clayton     m_text.AppendString("");
331644d93782SGreg Clayton   }
3317b9c1b51eSKate Stone   if (key_help_array) {
3318b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
331944d93782SGreg Clayton       StreamString key_description;
3320b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3321b9c1b51eSKate Stone                              key->description);
3322c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
332344d93782SGreg Clayton     }
332444d93782SGreg Clayton   }
332544d93782SGreg Clayton }
332644d93782SGreg Clayton 
3327315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
332844d93782SGreg Clayton 
3329b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
333044d93782SGreg Clayton   window.Erase();
333144d93782SGreg Clayton   const int window_height = window.GetHeight();
333244d93782SGreg Clayton   int x = 2;
333344d93782SGreg Clayton   int y = 1;
333444d93782SGreg Clayton   const int min_y = y;
333544d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33363985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
333744d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
333844d93782SGreg Clayton   const char *bottom_message;
333944d93782SGreg Clayton   if (num_lines <= num_visible_lines)
334044d93782SGreg Clayton     bottom_message = "Press any key to exit";
334144d93782SGreg Clayton   else
334244d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
334344d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3344b9c1b51eSKate Stone   while (y <= max_y) {
334544d93782SGreg Clayton     window.MoveCursor(x, y);
3346b9c1b51eSKate Stone     window.PutCStringTruncated(
3347b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
334844d93782SGreg Clayton     ++y;
334944d93782SGreg Clayton   }
335044d93782SGreg Clayton   return true;
335144d93782SGreg Clayton }
335244d93782SGreg Clayton 
3353b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3354b9c1b51eSKate Stone                                                               int key) {
335544d93782SGreg Clayton   bool done = false;
335644d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
335744d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
335844d93782SGreg Clayton 
3359b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
336044d93782SGreg Clayton     done = true;
336105097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
336205097246SAdrian Prantl     // press will cause us to exit
3363b9c1b51eSKate Stone   } else {
3364b9c1b51eSKate Stone     switch (key) {
336544d93782SGreg Clayton     case KEY_UP:
336644d93782SGreg Clayton       if (m_first_visible_line > 0)
336744d93782SGreg Clayton         --m_first_visible_line;
336844d93782SGreg Clayton       break;
336944d93782SGreg Clayton 
337044d93782SGreg Clayton     case KEY_DOWN:
337144d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
337244d93782SGreg Clayton         ++m_first_visible_line;
337344d93782SGreg Clayton       break;
337444d93782SGreg Clayton 
337544d93782SGreg Clayton     case KEY_PPAGE:
337644d93782SGreg Clayton     case ',':
3377b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33783985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
337944d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
338044d93782SGreg Clayton         else
338144d93782SGreg Clayton           m_first_visible_line = 0;
338244d93782SGreg Clayton       }
338344d93782SGreg Clayton       break;
3384315b6884SEugene Zelenko 
338544d93782SGreg Clayton     case KEY_NPAGE:
338644d93782SGreg Clayton     case '.':
3387b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
338844d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
33893985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
339044d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
339144d93782SGreg Clayton       }
339244d93782SGreg Clayton       break;
3393315b6884SEugene Zelenko 
339444d93782SGreg Clayton     default:
339544d93782SGreg Clayton       done = true;
339644d93782SGreg Clayton       break;
339744d93782SGreg Clayton     }
339844d93782SGreg Clayton   }
339944d93782SGreg Clayton   if (done)
340044d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
340144d93782SGreg Clayton   return eKeyHandled;
340244d93782SGreg Clayton }
340344d93782SGreg Clayton 
3404b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
340544d93782SGreg Clayton public:
340644d93782SGreg Clayton   enum {
340744d93782SGreg Clayton     eMenuID_LLDB = 1,
340844d93782SGreg Clayton     eMenuID_LLDBAbout,
340944d93782SGreg Clayton     eMenuID_LLDBExit,
341044d93782SGreg Clayton 
341144d93782SGreg Clayton     eMenuID_Target,
341244d93782SGreg Clayton     eMenuID_TargetCreate,
341344d93782SGreg Clayton     eMenuID_TargetDelete,
341444d93782SGreg Clayton 
341544d93782SGreg Clayton     eMenuID_Process,
341644d93782SGreg Clayton     eMenuID_ProcessAttach,
341744d93782SGreg Clayton     eMenuID_ProcessDetach,
341844d93782SGreg Clayton     eMenuID_ProcessLaunch,
341944d93782SGreg Clayton     eMenuID_ProcessContinue,
342044d93782SGreg Clayton     eMenuID_ProcessHalt,
342144d93782SGreg Clayton     eMenuID_ProcessKill,
342244d93782SGreg Clayton 
342344d93782SGreg Clayton     eMenuID_Thread,
342444d93782SGreg Clayton     eMenuID_ThreadStepIn,
342544d93782SGreg Clayton     eMenuID_ThreadStepOver,
342644d93782SGreg Clayton     eMenuID_ThreadStepOut,
342744d93782SGreg Clayton 
342844d93782SGreg Clayton     eMenuID_View,
342944d93782SGreg Clayton     eMenuID_ViewBacktrace,
343044d93782SGreg Clayton     eMenuID_ViewRegisters,
343144d93782SGreg Clayton     eMenuID_ViewSource,
343244d93782SGreg Clayton     eMenuID_ViewVariables,
343344d93782SGreg Clayton 
343444d93782SGreg Clayton     eMenuID_Help,
343544d93782SGreg Clayton     eMenuID_HelpGUIHelp
343644d93782SGreg Clayton   };
343744d93782SGreg Clayton 
3438b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3439b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
344044d93782SGreg Clayton 
3441315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3442bd5ae6b4SGreg Clayton 
3443b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
344444d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
344544d93782SGreg Clayton   }
344644d93782SGreg Clayton 
3447b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3448b9c1b51eSKate Stone     switch (key) {
34495fdb09bbSGreg Clayton     case '\t':
345044d93782SGreg Clayton       window.SelectNextWindowAsActive();
345144d93782SGreg Clayton       return eKeyHandled;
34525fdb09bbSGreg Clayton 
34535fdb09bbSGreg Clayton     case 'h':
34545fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34555fdb09bbSGreg Clayton       return eKeyHandled;
34565fdb09bbSGreg Clayton 
34575fdb09bbSGreg Clayton     case KEY_ESCAPE:
34585fdb09bbSGreg Clayton       return eQuitApplication;
34595fdb09bbSGreg Clayton 
34605fdb09bbSGreg Clayton     default:
34615fdb09bbSGreg Clayton       break;
346244d93782SGreg Clayton     }
346344d93782SGreg Clayton     return eKeyNotHandled;
346444d93782SGreg Clayton   }
346544d93782SGreg Clayton 
3466b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34675fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34685fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3469b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3470b9c1b51eSKate Stone            "dialog to display them.\n\n"
34715fdb09bbSGreg Clayton            "Common key bindings for all views:";
34725fdb09bbSGreg Clayton   }
34735fdb09bbSGreg Clayton 
3474b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34755fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34765fdb09bbSGreg Clayton         {'\t', "Select next view"},
34775fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34785fdb09bbSGreg Clayton         {',', "Page up"},
34795fdb09bbSGreg Clayton         {'.', "Page down"},
34805fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34815fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34825fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
34835fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
34845fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
34855fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3486b9c1b51eSKate Stone         {'\0', nullptr}};
34875fdb09bbSGreg Clayton     return g_source_view_key_help;
34885fdb09bbSGreg Clayton   }
34895fdb09bbSGreg Clayton 
3490b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3491b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3492b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3493b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3494b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3495b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
349644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3497b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3498b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
34994b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
350044d93782SGreg Clayton       }
350144d93782SGreg Clayton     }
350244d93782SGreg Clayton       return MenuActionResult::Handled;
350344d93782SGreg Clayton 
3504b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3505b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3506b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3507b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
350844d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3509b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3510b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
351144d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
351244d93782SGreg Clayton       }
351344d93782SGreg Clayton     }
351444d93782SGreg Clayton       return MenuActionResult::Handled;
351544d93782SGreg Clayton 
3516b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3517b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3518b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3519b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
352044d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3521b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3522b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
352344d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
352444d93782SGreg Clayton       }
352544d93782SGreg Clayton     }
352644d93782SGreg Clayton       return MenuActionResult::Handled;
352744d93782SGreg Clayton 
3528b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3529b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3530b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3531b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
353244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3533b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3534b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
353544d93782SGreg Clayton           process->Resume();
353644d93782SGreg Clayton       }
353744d93782SGreg Clayton     }
353844d93782SGreg Clayton       return MenuActionResult::Handled;
353944d93782SGreg Clayton 
3540b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3541b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3542b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3543b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
354444d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
354544d93782SGreg Clayton         if (process && process->IsAlive())
3546ede3193bSJason Molenda           process->Destroy(false);
354744d93782SGreg Clayton       }
354844d93782SGreg Clayton     }
354944d93782SGreg Clayton       return MenuActionResult::Handled;
355044d93782SGreg Clayton 
3551b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3552b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3553b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3554b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
355544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
355644d93782SGreg Clayton         if (process && process->IsAlive())
355744d93782SGreg Clayton           process->Halt();
355844d93782SGreg Clayton       }
355944d93782SGreg Clayton     }
356044d93782SGreg Clayton       return MenuActionResult::Handled;
356144d93782SGreg Clayton 
3562b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3563b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3564b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3565b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
356644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
356744d93782SGreg Clayton         if (process && process->IsAlive())
356844d93782SGreg Clayton           process->Detach(false);
356944d93782SGreg Clayton       }
357044d93782SGreg Clayton     }
357144d93782SGreg Clayton       return MenuActionResult::Handled;
357244d93782SGreg Clayton 
3573b9c1b51eSKate Stone     case eMenuID_Process: {
3574b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
357505097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
357605097246SAdrian Prantl       // submenu.
357744d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3578b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3579b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
358044d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3581b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3582b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
358344d93782SGreg Clayton         if (submenus.size() == 7)
358444d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
358544d93782SGreg Clayton         else if (submenus.size() > 8)
358644d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
358744d93782SGreg Clayton 
358844d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3589bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
359044d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3591b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
359244d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
359344d93782SGreg Clayton           char menu_char = '\0';
359444d93782SGreg Clayton           if (i < 9)
359544d93782SGreg Clayton             menu_char = '1' + i;
359644d93782SGreg Clayton           StreamString thread_menu_title;
359744d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
359844d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
359944d93782SGreg Clayton           if (thread_name && thread_name[0])
360044d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3601b9c1b51eSKate Stone           else {
360244d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
360344d93782SGreg Clayton             if (queue_name && queue_name[0])
360444d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
360544d93782SGreg Clayton           }
3606b9c1b51eSKate Stone           menu.AddSubmenu(
3607c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3608c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
360944d93782SGreg Clayton         }
3610b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
361105097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
361205097246SAdrian Prantl         // previously added
361344d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
361444d93782SGreg Clayton       }
3615b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3616b9c1b51eSKate Stone       // lengths
361744d93782SGreg Clayton       menu.RecalculateNameLengths();
361844d93782SGreg Clayton     }
361944d93782SGreg Clayton       return MenuActionResult::Handled;
362044d93782SGreg Clayton 
3621b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
362244d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
362344d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
362444d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
362544d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
362644d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
362744d93782SGreg Clayton 
3628b9c1b51eSKate Stone       if (variables_window_sp) {
362944d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
363044d93782SGreg Clayton 
363144d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
363244d93782SGreg Clayton 
3633b9c1b51eSKate Stone         if (registers_window_sp) {
3634b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3635b9c1b51eSKate Stone           // registers window
363644d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
363744d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
363844d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3639b9c1b51eSKate Stone         } else {
364005097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
364105097246SAdrian Prantl           // to the source view
364244d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3643b9c1b51eSKate Stone                                    source_bounds.size.height +
3644b9c1b51eSKate Stone                                        variables_bounds.size.height);
364544d93782SGreg Clayton         }
3646b9c1b51eSKate Stone       } else {
364744d93782SGreg Clayton         Rect new_variables_rect;
3648b9c1b51eSKate Stone         if (registers_window_sp) {
364944d93782SGreg Clayton           // We have a registers window so split the area of the registers
365044d93782SGreg Clayton           // window into two columns where the left hand side will be the
365144d93782SGreg Clayton           // variables and the right hand side will be the registers
365244d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
365344d93782SGreg Clayton           Rect new_registers_rect;
3654b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3655b9c1b51eSKate Stone                                                    new_registers_rect);
365644d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3657b9c1b51eSKate Stone         } else {
365844d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
365944d93782SGreg Clayton           Rect new_source_rect;
3660b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3661b9c1b51eSKate Stone                                                   new_variables_rect);
366244d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
366344d93782SGreg Clayton         }
3664b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3665b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3666b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3667b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
366844d93782SGreg Clayton       }
366944d93782SGreg Clayton       touchwin(stdscr);
367044d93782SGreg Clayton     }
367144d93782SGreg Clayton       return MenuActionResult::Handled;
367244d93782SGreg Clayton 
3673b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
367444d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
367544d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
367644d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
367744d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
367844d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
367944d93782SGreg Clayton 
3680b9c1b51eSKate Stone       if (registers_window_sp) {
3681b9c1b51eSKate Stone         if (variables_window_sp) {
368244d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
368344d93782SGreg Clayton 
3684b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3685b9c1b51eSKate Stone           // variables window
3686b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3687b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
368844d93782SGreg Clayton                                       variables_bounds.size.height);
3689b9c1b51eSKate Stone         } else {
369005097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
369105097246SAdrian Prantl           // to the source view
369244d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3693b9c1b51eSKate Stone                                    source_bounds.size.height +
3694b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
369544d93782SGreg Clayton         }
369644d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3697b9c1b51eSKate Stone       } else {
369844d93782SGreg Clayton         Rect new_regs_rect;
3699b9c1b51eSKate Stone         if (variables_window_sp) {
370005097246SAdrian Prantl           // We have a variables window, split it into two columns where the
370105097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
370205097246SAdrian Prantl           // be the registers
370344d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
370444d93782SGreg Clayton           Rect new_vars_rect;
3705b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3706b9c1b51eSKate Stone                                                    new_regs_rect);
370744d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3708b9c1b51eSKate Stone         } else {
370944d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
371044d93782SGreg Clayton           Rect new_source_rect;
3711b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3712b9c1b51eSKate Stone                                                   new_regs_rect);
371344d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
371444d93782SGreg Clayton         }
3715b9c1b51eSKate Stone         WindowSP new_window_sp =
3716b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3717b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3718b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
371944d93782SGreg Clayton       }
372044d93782SGreg Clayton       touchwin(stdscr);
372144d93782SGreg Clayton     }
372244d93782SGreg Clayton       return MenuActionResult::Handled;
372344d93782SGreg Clayton 
372444d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37255fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
372644d93782SGreg Clayton       return MenuActionResult::Handled;
372744d93782SGreg Clayton 
372844d93782SGreg Clayton     default:
372944d93782SGreg Clayton       break;
373044d93782SGreg Clayton     }
373144d93782SGreg Clayton 
373244d93782SGreg Clayton     return MenuActionResult::NotHandled;
373344d93782SGreg Clayton   }
3734b9c1b51eSKate Stone 
373544d93782SGreg Clayton protected:
373644d93782SGreg Clayton   Application &m_app;
373744d93782SGreg Clayton   Debugger &m_debugger;
373844d93782SGreg Clayton };
373944d93782SGreg Clayton 
3740b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
374144d93782SGreg Clayton public:
3742b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3743b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
374444d93782SGreg Clayton   }
374544d93782SGreg Clayton 
3746315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3747bd5ae6b4SGreg Clayton 
3748b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3749b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3750b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
375144d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
375244d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
375344d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
375444d93782SGreg Clayton     window.Erase();
375544d93782SGreg Clayton     window.SetBackground(2);
375644d93782SGreg Clayton     window.MoveCursor(0, 0);
3757b9c1b51eSKate Stone     if (process) {
375844d93782SGreg Clayton       const StateType state = process->GetState();
3759b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3760b9c1b51eSKate Stone                     StateAsCString(state));
376144d93782SGreg Clayton 
3762b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37635b031ebcSEd Maste         StreamString strm;
3764b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3765b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
376644d93782SGreg Clayton           window.MoveCursor(40, 0);
3767c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37685b031ebcSEd Maste         }
376944d93782SGreg Clayton 
377044d93782SGreg Clayton         window.MoveCursor(60, 0);
377144d93782SGreg Clayton         if (frame)
3772b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3773b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3774b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3775b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3776b9c1b51eSKate Stone       } else if (state == eStateExited) {
377744d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
377844d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
377944d93782SGreg Clayton         if (exit_desc && exit_desc[0])
378044d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
378144d93782SGreg Clayton         else
378244d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
378344d93782SGreg Clayton       }
378444d93782SGreg Clayton     }
378544d93782SGreg Clayton     window.DeferredRefresh();
378644d93782SGreg Clayton     return true;
378744d93782SGreg Clayton   }
378844d93782SGreg Clayton 
378944d93782SGreg Clayton protected:
379044d93782SGreg Clayton   Debugger &m_debugger;
3791554f68d3SGreg Clayton   FormatEntity::Entry m_format;
379244d93782SGreg Clayton };
379344d93782SGreg Clayton 
3794b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
379544d93782SGreg Clayton public:
3796b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3797b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3798b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3799b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3800b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3801b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
380244d93782SGreg Clayton 
3803315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
380444d93782SGreg Clayton 
3805b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
380644d93782SGreg Clayton 
3807b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
380844d93782SGreg Clayton 
3809b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
381044d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
381144d93782SGreg Clayton   }
381244d93782SGreg Clayton 
3813b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
381444d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
381544d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
381644d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
381744d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
381844d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
381944d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
382044d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
382144d93782SGreg Clayton         {'c', "Continue process"},
382244d93782SGreg Clayton         {'d', "Detach and resume process"},
382344d93782SGreg Clayton         {'D', "Detach with process suspended"},
382444d93782SGreg Clayton         {'h', "Show help dialog"},
382544d93782SGreg Clayton         {'k', "Kill process"},
382644d93782SGreg Clayton         {'n', "Step over (source line)"},
382744d93782SGreg Clayton         {'N', "Step over (single instruction)"},
382844d93782SGreg Clayton         {'o', "Step out"},
382944d93782SGreg Clayton         {'s', "Step in (source line)"},
383044d93782SGreg Clayton         {'S', "Step in (single instruction)"},
383144d93782SGreg Clayton         {',', "Page up"},
383244d93782SGreg Clayton         {'.', "Page down"},
3833b9c1b51eSKate Stone         {'\0', nullptr}};
383444d93782SGreg Clayton     return g_source_view_key_help;
383544d93782SGreg Clayton   }
383644d93782SGreg Clayton 
3837b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3838b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3839b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
384044d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3841c5dac77aSEugene Zelenko     Thread *thread = nullptr;
384244d93782SGreg Clayton 
384344d93782SGreg Clayton     bool update_location = false;
3844b9c1b51eSKate Stone     if (process) {
384544d93782SGreg Clayton       StateType state = process->GetState();
3846b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
384744d93782SGreg Clayton         // We are stopped, so it is ok to
384844d93782SGreg Clayton         update_location = true;
384944d93782SGreg Clayton       }
385044d93782SGreg Clayton     }
385144d93782SGreg Clayton 
385244d93782SGreg Clayton     m_min_x = 1;
3853ec990867SGreg Clayton     m_min_y = 2;
385444d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
385544d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
385644d93782SGreg Clayton 
385744d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
385844d93782SGreg Clayton     StackFrameSP frame_sp;
385944d93782SGreg Clayton     bool set_selected_line_to_pc = false;
386044d93782SGreg Clayton 
3861b9c1b51eSKate Stone     if (update_location) {
386244d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
386344d93782SGreg Clayton       bool thread_changed = false;
3864b9c1b51eSKate Stone       if (process_alive) {
386544d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3866b9c1b51eSKate Stone         if (thread) {
386744d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
386844d93782SGreg Clayton           auto tid = thread->GetID();
386944d93782SGreg Clayton           thread_changed = tid != m_tid;
387044d93782SGreg Clayton           m_tid = tid;
3871b9c1b51eSKate Stone         } else {
3872b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
387344d93782SGreg Clayton             thread_changed = true;
387444d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
387544d93782SGreg Clayton           }
387644d93782SGreg Clayton         }
387744d93782SGreg Clayton       }
387844d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
387944d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
388044d93782SGreg Clayton       bool frame_changed = false;
388144d93782SGreg Clayton       m_stop_id = stop_id;
3882ec990867SGreg Clayton       m_title.Clear();
3883b9c1b51eSKate Stone       if (frame_sp) {
388444d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3885b9c1b51eSKate Stone         if (m_sc.module_sp) {
3886b9c1b51eSKate Stone           m_title.Printf(
3887b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3888ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3889ec990867SGreg Clayton           if (func_name)
3890ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3891ec990867SGreg Clayton         }
389244d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
389344d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
389444d93782SGreg Clayton         m_frame_idx = frame_idx;
3895b9c1b51eSKate Stone       } else {
389644d93782SGreg Clayton         m_sc.Clear(true);
389744d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
389844d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
389944d93782SGreg Clayton       }
390044d93782SGreg Clayton 
3901b9c1b51eSKate Stone       const bool context_changed =
3902b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
390344d93782SGreg Clayton 
3904b9c1b51eSKate Stone       if (process_alive) {
3905b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
390644d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
390744d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
390844d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
390944d93782SGreg Clayton           // Update the selected line if the stop ID changed...
391044d93782SGreg Clayton           if (context_changed)
391144d93782SGreg Clayton             m_selected_line = m_pc_line;
391244d93782SGreg Clayton 
3913b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
391405097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
391505097246SAdrian Prantl             // (source file missing)
3916b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
391744d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
391844d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3919b9c1b51eSKate Stone             } else {
392044d93782SGreg Clayton               if (m_selected_line > 10)
392144d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
392244d93782SGreg Clayton               else
392344d93782SGreg Clayton                 m_first_visible_line = 0;
392444d93782SGreg Clayton             }
3925b9c1b51eSKate Stone           } else {
392644d93782SGreg Clayton             // File changed, set selected line to the line with the PC
392744d93782SGreg Clayton             m_selected_line = m_pc_line;
3928b9c1b51eSKate Stone             m_file_sp =
3929b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3930b9c1b51eSKate Stone             if (m_file_sp) {
393144d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
3932*eebf32faSPavel Labath               m_line_width = 1;
393344d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
393444d93782SGreg Clayton                 ++m_line_width;
393544d93782SGreg Clayton 
3936b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3937b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
393844d93782SGreg Clayton                 m_first_visible_line = 0;
393944d93782SGreg Clayton               else
394044d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
394144d93782SGreg Clayton             }
394244d93782SGreg Clayton           }
3943b9c1b51eSKate Stone         } else {
394444d93782SGreg Clayton           m_file_sp.reset();
394544d93782SGreg Clayton         }
394644d93782SGreg Clayton 
3947b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
394844d93782SGreg Clayton           // Show disassembly
394944d93782SGreg Clayton           bool prefer_file_cache = false;
3950b9c1b51eSKate Stone           if (m_sc.function) {
3951b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
395244d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3953b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3954b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3955b9c1b51eSKate Stone               if (m_disassembly_sp) {
395644d93782SGreg Clayton                 set_selected_line_to_pc = true;
395744d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3958b9c1b51eSKate Stone               } else {
395944d93782SGreg Clayton                 m_disassembly_range.Clear();
396044d93782SGreg Clayton               }
3961b9c1b51eSKate Stone             } else {
396244d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
396344d93782SGreg Clayton             }
3964b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3965b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
396644d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3967b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3968b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3969b9c1b51eSKate Stone               if (m_disassembly_sp) {
397044d93782SGreg Clayton                 set_selected_line_to_pc = true;
3971b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3972b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
397344d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3974b9c1b51eSKate Stone               } else {
397544d93782SGreg Clayton                 m_disassembly_range.Clear();
397644d93782SGreg Clayton               }
3977b9c1b51eSKate Stone             } else {
397844d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
397944d93782SGreg Clayton             }
398044d93782SGreg Clayton           }
398144d93782SGreg Clayton         }
3982b9c1b51eSKate Stone       } else {
398344d93782SGreg Clayton         m_pc_line = UINT32_MAX;
398444d93782SGreg Clayton       }
398544d93782SGreg Clayton     }
398644d93782SGreg Clayton 
3987ec990867SGreg Clayton     const int window_width = window.GetWidth();
398844d93782SGreg Clayton     window.Erase();
398944d93782SGreg Clayton     window.DrawTitleBox("Sources");
3990b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
3991ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
3992ec990867SGreg Clayton       window.MoveCursor(1, 1);
3993ec990867SGreg Clayton       window.PutChar(' ');
3994c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
3995ec990867SGreg Clayton       int x = window.GetCursorX();
3996b9c1b51eSKate Stone       if (x < window_width - 1) {
3997ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
3998ec990867SGreg Clayton       }
3999ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
4000ec990867SGreg Clayton     }
400144d93782SGreg Clayton 
400244d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
400344d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
4004b9c1b51eSKate Stone     if (num_source_lines > 0) {
400544d93782SGreg Clayton       // Display source
400644d93782SGreg Clayton       BreakpointLines bp_lines;
4007b9c1b51eSKate Stone       if (target) {
400844d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
400944d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
4010b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
401144d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
401244d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
4013b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
4014b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
4015b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
401644d93782SGreg Clayton             LineEntry bp_loc_line_entry;
4017b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
4018b9c1b51eSKate Stone                     bp_loc_line_entry)) {
4019b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
402044d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
402144d93782SGreg Clayton               }
402244d93782SGreg Clayton             }
402344d93782SGreg Clayton           }
402444d93782SGreg Clayton         }
402544d93782SGreg Clayton       }
402644d93782SGreg Clayton 
402744d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
402844d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
402944d93782SGreg Clayton 
4030b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
403144d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4032b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4033ec990867SGreg Clayton           const int line_y = m_min_y + i;
403444d93782SGreg Clayton           window.MoveCursor(1, line_y);
403544d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
403644d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
403744d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
403844d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
403944d93782SGreg Clayton           attr_t highlight_attr = 0;
404044d93782SGreg Clayton           attr_t bp_attr = 0;
404144d93782SGreg Clayton           if (is_pc_line)
404244d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
404344d93782SGreg Clayton           else if (line_is_selected)
404444d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
404544d93782SGreg Clayton 
404644d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
404744d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
404844d93782SGreg Clayton 
404944d93782SGreg Clayton           if (bp_attr)
405044d93782SGreg Clayton             window.AttributeOn(bp_attr);
405144d93782SGreg Clayton 
4052*eebf32faSPavel Labath           window.Printf(" %*u ", m_line_width, curr_line + 1);
405344d93782SGreg Clayton 
405444d93782SGreg Clayton           if (bp_attr)
405544d93782SGreg Clayton             window.AttributeOff(bp_attr);
405644d93782SGreg Clayton 
405744d93782SGreg Clayton           window.PutChar(ACS_VLINE);
405844d93782SGreg Clayton           // Mark the line with the PC with a diamond
405944d93782SGreg Clayton           if (is_pc_line)
406044d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
406144d93782SGreg Clayton           else
406244d93782SGreg Clayton             window.PutChar(' ');
406344d93782SGreg Clayton 
406444d93782SGreg Clayton           if (highlight_attr)
406544d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4066b9c1b51eSKate Stone           const uint32_t line_len =
4067b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
406844d93782SGreg Clayton           if (line_len > 0)
406944d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
407044d93782SGreg Clayton 
4071b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4072b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
407344d93782SGreg Clayton             StopInfoSP stop_info_sp;
407444d93782SGreg Clayton             if (thread)
407544d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4076b9c1b51eSKate Stone             if (stop_info_sp) {
407744d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4078b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
407944d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4080ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
408144d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4082b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4083b9c1b51eSKate Stone                 // line_y);
4084b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4085b9c1b51eSKate Stone                               stop_description);
408644d93782SGreg Clayton               }
4087b9c1b51eSKate Stone             } else {
4088ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
408944d93782SGreg Clayton             }
409044d93782SGreg Clayton           }
409144d93782SGreg Clayton           if (highlight_attr)
409244d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4093b9c1b51eSKate Stone         } else {
409444d93782SGreg Clayton           break;
409544d93782SGreg Clayton         }
409644d93782SGreg Clayton       }
4097b9c1b51eSKate Stone     } else {
409844d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4099b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
410044d93782SGreg Clayton         // Display disassembly
410144d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
410244d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4103b9c1b51eSKate Stone         if (target) {
410444d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
410544d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4106b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
410744d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
410844d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4109b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4110b9c1b51eSKate Stone                  ++bp_loc_idx) {
4111b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4112b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
411344d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4114b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4115b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4116b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
411744d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
411844d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
411944d93782SGreg Clayton               }
412044d93782SGreg Clayton             }
412144d93782SGreg Clayton           }
412244d93782SGreg Clayton         }
412344d93782SGreg Clayton 
412444d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
412544d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
412644d93782SGreg Clayton 
412744d93782SGreg Clayton         StreamString strm;
412844d93782SGreg Clayton 
412944d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
413044d93782SGreg Clayton         Address pc_address;
413144d93782SGreg Clayton 
413244d93782SGreg Clayton         if (frame_sp)
413344d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4134b9c1b51eSKate Stone         const uint32_t pc_idx =
4135b9c1b51eSKate Stone             pc_address.IsValid()
4136b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4137b9c1b51eSKate Stone                 : UINT32_MAX;
4138b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
413944d93782SGreg Clayton           m_selected_line = pc_idx;
414044d93782SGreg Clayton         }
414144d93782SGreg Clayton 
414244d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41433985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
414444d93782SGreg Clayton           m_first_visible_line = 0;
414544d93782SGreg Clayton 
4146b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41473985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
414844d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
414944d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
415044d93782SGreg Clayton         }
415144d93782SGreg Clayton 
4152b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
415344d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
415444d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
415544d93782SGreg Clayton           if (!inst)
415644d93782SGreg Clayton             break;
415744d93782SGreg Clayton 
4158ec990867SGreg Clayton           const int line_y = m_min_y + i;
4159ec990867SGreg Clayton           window.MoveCursor(1, line_y);
416044d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
416144d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
416244d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
416344d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
416444d93782SGreg Clayton           attr_t highlight_attr = 0;
416544d93782SGreg Clayton           attr_t bp_attr = 0;
416644d93782SGreg Clayton           if (is_pc_line)
416744d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
416844d93782SGreg Clayton           else if (line_is_selected)
416944d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
417044d93782SGreg Clayton 
4171b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4172b9c1b51eSKate Stone               bp_file_addrs.end())
417344d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
417444d93782SGreg Clayton 
417544d93782SGreg Clayton           if (bp_attr)
417644d93782SGreg Clayton             window.AttributeOn(bp_attr);
417744d93782SGreg Clayton 
4178324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4179b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4180b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
418144d93782SGreg Clayton 
418244d93782SGreg Clayton           if (bp_attr)
418344d93782SGreg Clayton             window.AttributeOff(bp_attr);
418444d93782SGreg Clayton 
418544d93782SGreg Clayton           window.PutChar(ACS_VLINE);
418644d93782SGreg Clayton           // Mark the line with the PC with a diamond
418744d93782SGreg Clayton           if (is_pc_line)
418844d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
418944d93782SGreg Clayton           else
419044d93782SGreg Clayton             window.PutChar(' ');
419144d93782SGreg Clayton 
419244d93782SGreg Clayton           if (highlight_attr)
419344d93782SGreg Clayton             window.AttributeOn(highlight_attr);
419444d93782SGreg Clayton 
419544d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
419644d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
419744d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
419844d93782SGreg Clayton 
4199c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4200c5dac77aSEugene Zelenko             mnemonic = nullptr;
4201c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4202c5dac77aSEugene Zelenko             operands = nullptr;
4203c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4204c5dac77aSEugene Zelenko             comment = nullptr;
420544d93782SGreg Clayton 
420644d93782SGreg Clayton           strm.Clear();
420744d93782SGreg Clayton 
4208c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
420944d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4210c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
421144d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4212c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
421344d93782SGreg Clayton             strm.Printf("%s", mnemonic);
421444d93782SGreg Clayton 
421544d93782SGreg Clayton           int right_pad = 1;
4216c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
421744d93782SGreg Clayton 
4218b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4219b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
422044d93782SGreg Clayton             StopInfoSP stop_info_sp;
422144d93782SGreg Clayton             if (thread)
422244d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4223b9c1b51eSKate Stone             if (stop_info_sp) {
422444d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4225b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
422644d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4227ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
422844d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4229b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4230b9c1b51eSKate Stone                 // line_y);
4231b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4232b9c1b51eSKate Stone                               stop_description);
423344d93782SGreg Clayton               }
4234b9c1b51eSKate Stone             } else {
4235ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
423644d93782SGreg Clayton             }
423744d93782SGreg Clayton           }
423844d93782SGreg Clayton           if (highlight_attr)
423944d93782SGreg Clayton             window.AttributeOff(highlight_attr);
424044d93782SGreg Clayton         }
424144d93782SGreg Clayton       }
424244d93782SGreg Clayton     }
424344d93782SGreg Clayton     window.DeferredRefresh();
424444d93782SGreg Clayton     return true; // Drawing handled
424544d93782SGreg Clayton   }
424644d93782SGreg Clayton 
4247b9c1b51eSKate Stone   size_t GetNumLines() {
424844d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
424944d93782SGreg Clayton     if (num_lines == 0)
425044d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
425144d93782SGreg Clayton     return num_lines;
425244d93782SGreg Clayton   }
425344d93782SGreg Clayton 
4254b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
425544d93782SGreg Clayton     if (m_file_sp)
425644d93782SGreg Clayton       return m_file_sp->GetNumLines();
425744d93782SGreg Clayton     return 0;
425844d93782SGreg Clayton   }
4259315b6884SEugene Zelenko 
4260b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
426144d93782SGreg Clayton     if (m_disassembly_sp)
426244d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
426344d93782SGreg Clayton     return 0;
426444d93782SGreg Clayton   }
426544d93782SGreg Clayton 
4266b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
426744d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
426844d93782SGreg Clayton     const size_t num_lines = GetNumLines();
426944d93782SGreg Clayton 
4270b9c1b51eSKate Stone     switch (c) {
427144d93782SGreg Clayton     case ',':
427244d93782SGreg Clayton     case KEY_PPAGE:
427344d93782SGreg Clayton       // Page up key
42743985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
427544d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
427644d93782SGreg Clayton       else
427744d93782SGreg Clayton         m_first_visible_line = 0;
427844d93782SGreg Clayton       m_selected_line = m_first_visible_line;
427944d93782SGreg Clayton       return eKeyHandled;
428044d93782SGreg Clayton 
428144d93782SGreg Clayton     case '.':
428244d93782SGreg Clayton     case KEY_NPAGE:
428344d93782SGreg Clayton       // Page down key
428444d93782SGreg Clayton       {
428544d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
428644d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
428744d93782SGreg Clayton         else if (num_lines < num_visible_lines)
428844d93782SGreg Clayton           m_first_visible_line = 0;
428944d93782SGreg Clayton         else
429044d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
429144d93782SGreg Clayton         m_selected_line = m_first_visible_line;
429244d93782SGreg Clayton       }
429344d93782SGreg Clayton       return eKeyHandled;
429444d93782SGreg Clayton 
429544d93782SGreg Clayton     case KEY_UP:
4296b9c1b51eSKate Stone       if (m_selected_line > 0) {
429744d93782SGreg Clayton         m_selected_line--;
42983985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
429944d93782SGreg Clayton           m_first_visible_line = m_selected_line;
430044d93782SGreg Clayton       }
430144d93782SGreg Clayton       return eKeyHandled;
430244d93782SGreg Clayton 
430344d93782SGreg Clayton     case KEY_DOWN:
4304b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
430544d93782SGreg Clayton         m_selected_line++;
430644d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
430744d93782SGreg Clayton           m_first_visible_line++;
430844d93782SGreg Clayton       }
430944d93782SGreg Clayton       return eKeyHandled;
431044d93782SGreg Clayton 
431144d93782SGreg Clayton     case '\r':
431244d93782SGreg Clayton     case '\n':
431344d93782SGreg Clayton     case KEY_ENTER:
431444d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4315b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4316b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4317b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4318b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4319b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4320b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
432144d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4322b9c1b51eSKate Stone               m_selected_line +
4323b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4324431b1584SAdrian Prantl               0,     // Unspecified column.
43252411167fSJim Ingham               0,     // No offset
432644d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
432744d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
432844d93782SGreg Clayton               false,               // internal
4329055ad9beSIlia K               false,               // request_hardware
4330055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
433144d93782SGreg Clayton           // Make breakpoint one shot
433244d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
433344d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
433444d93782SGreg Clayton         }
4335b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4336b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4337b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4338b9c1b51eSKate Stone                                       .get();
4339b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4340b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4341b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
434244d93782SGreg Clayton           Address addr = inst->GetAddress();
4343b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4344b9c1b51eSKate Stone               addr,   // lldb_private::Address
434544d93782SGreg Clayton               false,  // internal
434644d93782SGreg Clayton               false); // request_hardware
434744d93782SGreg Clayton           // Make breakpoint one shot
434844d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
434944d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
435044d93782SGreg Clayton         }
435144d93782SGreg Clayton       }
435244d93782SGreg Clayton       return eKeyHandled;
435344d93782SGreg Clayton 
435444d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4355b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4356b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4357b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4358b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4359b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4360b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
436144d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4362b9c1b51eSKate Stone               m_selected_line +
4363b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4364431b1584SAdrian Prantl               0,     // No column specified.
43652411167fSJim Ingham               0,     // No offset
436644d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
436744d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
436844d93782SGreg Clayton               false,               // internal
4369055ad9beSIlia K               false,               // request_hardware
4370055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
437144d93782SGreg Clayton         }
4372b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4373b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4374b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4375b9c1b51eSKate Stone                                       .get();
4376b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4377b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4378b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
437944d93782SGreg Clayton           Address addr = inst->GetAddress();
4380b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4381b9c1b51eSKate Stone               addr,   // lldb_private::Address
438244d93782SGreg Clayton               false,  // internal
438344d93782SGreg Clayton               false); // request_hardware
438444d93782SGreg Clayton         }
438544d93782SGreg Clayton       }
438644d93782SGreg Clayton       return eKeyHandled;
438744d93782SGreg Clayton 
438844d93782SGreg Clayton     case 'd': // 'd' == detach and let run
438944d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
439044d93782SGreg Clayton     {
4391b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4392b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
439344d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
439444d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
439544d93782SGreg Clayton     }
439644d93782SGreg Clayton       return eKeyHandled;
439744d93782SGreg Clayton 
439844d93782SGreg Clayton     case 'k':
439944d93782SGreg Clayton       // 'k' == kill
440044d93782SGreg Clayton       {
4401b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4402b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
440344d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4404ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
440544d93782SGreg Clayton       }
440644d93782SGreg Clayton       return eKeyHandled;
440744d93782SGreg Clayton 
440844d93782SGreg Clayton     case 'c':
440944d93782SGreg Clayton       // 'c' == continue
441044d93782SGreg Clayton       {
4411b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4412b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
441344d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
441444d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
441544d93782SGreg Clayton       }
441644d93782SGreg Clayton       return eKeyHandled;
441744d93782SGreg Clayton 
441844d93782SGreg Clayton     case 'o':
441944d93782SGreg Clayton       // 'o' == step out
442044d93782SGreg Clayton       {
4421b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4422b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4423b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4424b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
442544d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
442644d93782SGreg Clayton         }
442744d93782SGreg Clayton       }
442844d93782SGreg Clayton       return eKeyHandled;
4429315b6884SEugene Zelenko 
443044d93782SGreg Clayton     case 'n': // 'n' == step over
443144d93782SGreg Clayton     case 'N': // 'N' == step over instruction
443244d93782SGreg Clayton     {
4433b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4434b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4435b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4436b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
443744d93782SGreg Clayton         bool source_step = (c == 'n');
443844d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
443944d93782SGreg Clayton       }
444044d93782SGreg Clayton     }
444144d93782SGreg Clayton       return eKeyHandled;
4442315b6884SEugene Zelenko 
444344d93782SGreg Clayton     case 's': // 's' == step into
444444d93782SGreg Clayton     case 'S': // 'S' == step into instruction
444544d93782SGreg Clayton     {
4446b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4447b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4448b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4449b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
445044d93782SGreg Clayton         bool source_step = (c == 's');
44514b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
445244d93782SGreg Clayton       }
445344d93782SGreg Clayton     }
445444d93782SGreg Clayton       return eKeyHandled;
445544d93782SGreg Clayton 
445644d93782SGreg Clayton     case 'h':
445744d93782SGreg Clayton       window.CreateHelpSubwindow();
445844d93782SGreg Clayton       return eKeyHandled;
445944d93782SGreg Clayton 
446044d93782SGreg Clayton     default:
446144d93782SGreg Clayton       break;
446244d93782SGreg Clayton     }
446344d93782SGreg Clayton     return eKeyNotHandled;
446444d93782SGreg Clayton   }
446544d93782SGreg Clayton 
446644d93782SGreg Clayton protected:
446744d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
446844d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
446944d93782SGreg Clayton 
447044d93782SGreg Clayton   Debugger &m_debugger;
447144d93782SGreg Clayton   SymbolContext m_sc;
447244d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
447344d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
447444d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
447544d93782SGreg Clayton   AddressRange m_disassembly_range;
4476ec990867SGreg Clayton   StreamString m_title;
447744d93782SGreg Clayton   lldb::user_id_t m_tid;
447844d93782SGreg Clayton   int m_line_width;
447944d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
448044d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
448144d93782SGreg Clayton   uint32_t m_stop_id;
448244d93782SGreg Clayton   uint32_t m_frame_idx;
448344d93782SGreg Clayton   int m_first_visible_line;
448444d93782SGreg Clayton   int m_min_x;
448544d93782SGreg Clayton   int m_min_y;
448644d93782SGreg Clayton   int m_max_x;
448744d93782SGreg Clayton   int m_max_y;
448844d93782SGreg Clayton };
448944d93782SGreg Clayton 
449044d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
449144d93782SGreg Clayton 
4492b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4493b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
449444d93782SGreg Clayton 
4495b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
449644d93782SGreg Clayton   IOHandler::Activate();
4497b9c1b51eSKate Stone   if (!m_app_ap) {
449844d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
449944d93782SGreg Clayton 
450044d93782SGreg Clayton     // This is both a window and a menu delegate
4501b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4502b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
450344d93782SGreg Clayton 
4504b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4505b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4506b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4507b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4508b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4509b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
451044d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4511b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4512b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
451344d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
451444d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
451544d93782SGreg Clayton 
4516b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4517b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4518b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4519b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4520b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4521b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
452244d93782SGreg Clayton 
4523b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4524b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4525b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4526b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4527b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4528b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4529b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4530b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
453144d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4532b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4533b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4534b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4535b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4536b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4537b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4538b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
453944d93782SGreg Clayton 
4540b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4541b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4542b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4543b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4544b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4545b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4546b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4547b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4548b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
454944d93782SGreg Clayton 
4550b9c1b51eSKate Stone     MenuSP view_menu_sp(
4551b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4552b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4553b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4554b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4555b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4556b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4557b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4558b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4559b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4560b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4561b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4562b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
456344d93782SGreg Clayton 
4564b9c1b51eSKate Stone     MenuSP help_menu_sp(
4565b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4566b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4567b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
456844d93782SGreg Clayton 
456944d93782SGreg Clayton     m_app_ap->Initialize();
457044d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
457144d93782SGreg Clayton 
457244d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
457344d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
457444d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
457544d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
457644d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
457744d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
457844d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
457944d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
458044d93782SGreg Clayton 
458144d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
458244d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
458344d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
458444d93782SGreg Clayton     Rect source_bounds;
458544d93782SGreg Clayton     Rect variables_bounds;
458644d93782SGreg Clayton     Rect threads_bounds;
458744d93782SGreg Clayton     Rect source_variables_bounds;
4588b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4589b9c1b51eSKate Stone                                            threads_bounds);
4590b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4591b9c1b51eSKate Stone                                                       variables_bounds);
459244d93782SGreg Clayton 
4593b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4594b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
459505097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
459605097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4597b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4598b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
459944d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
460044d93782SGreg Clayton 
4601b9c1b51eSKate Stone     WindowSP source_window_sp(
4602b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4603b9c1b51eSKate Stone     WindowSP variables_window_sp(
4604b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4605b9c1b51eSKate Stone     WindowSP threads_window_sp(
4606b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4607b9c1b51eSKate Stone     WindowSP status_window_sp(
46086bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4609b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4610b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4611b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4612b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4613b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4614b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4615b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4616b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4617ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4618b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4619b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4620b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4621b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
462244d93782SGreg Clayton 
46235fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
46245fdb09bbSGreg Clayton     static bool g_showed_help = false;
4625b9c1b51eSKate Stone     if (!g_showed_help) {
46265fdb09bbSGreg Clayton       g_showed_help = true;
46275fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46285fdb09bbSGreg Clayton     }
46295fdb09bbSGreg Clayton 
463044d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
463144d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
463244d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
463344d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
463444d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
463544d93782SGreg Clayton   }
463644d93782SGreg Clayton }
463744d93782SGreg Clayton 
4638b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
463944d93782SGreg Clayton 
4640b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
464144d93782SGreg Clayton   m_app_ap->Run(m_debugger);
464244d93782SGreg Clayton   SetIsDone(true);
464344d93782SGreg Clayton }
464444d93782SGreg Clayton 
4645315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
464644d93782SGreg Clayton 
4647b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
464844d93782SGreg Clayton 
4649b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
465044d93782SGreg Clayton 
4651b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
465244d93782SGreg Clayton 
4653315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4654