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
79*d77c2e09SJonas Devlieghere                 0,              // Flags
80*d77c2e09SJonas Devlieghere                 nullptr         // Shadow file recorder
81*d77c2e09SJonas Devlieghere       ) {}
8244d93782SGreg Clayton 
83b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
8444d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
8544d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
86*d77c2e09SJonas Devlieghere                      const lldb::StreamFileSP &error_sp, uint32_t flags,
87*d77c2e09SJonas Devlieghere                      repro::DataRecorder *data_recorder)
88b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
89*d77c2e09SJonas Devlieghere       m_error_sp(error_sp), m_data_recorder(data_recorder), m_popped(false),
90*d77c2e09SJonas Devlieghere       m_flags(flags), m_type(type), m_user_data(nullptr), m_done(false),
91*d77c2e09SJonas Devlieghere       m_active(false) {
9244d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
9344d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
94b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
9544d93782SGreg Clayton                                              m_error_sp);
9644d93782SGreg Clayton }
9744d93782SGreg Clayton 
98315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
9944d93782SGreg Clayton 
100b9c1b51eSKate Stone int IOHandler::GetInputFD() {
101c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
10244d93782SGreg Clayton }
10344d93782SGreg Clayton 
104b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
105c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10644d93782SGreg Clayton }
10744d93782SGreg Clayton 
108b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
109c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
11044d93782SGreg Clayton }
11144d93782SGreg Clayton 
112b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
113c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11444d93782SGreg Clayton }
11544d93782SGreg Clayton 
116b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
117c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
11844d93782SGreg Clayton }
11944d93782SGreg Clayton 
120b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
121c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
12244d93782SGreg Clayton }
12344d93782SGreg Clayton 
124b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
12544d93782SGreg Clayton 
126b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
12744d93782SGreg Clayton 
128b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
12944d93782SGreg Clayton 
130b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
131340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
132340b0309SGreg Clayton }
133340b0309SGreg Clayton 
134b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
135340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
136340b0309SGreg Clayton }
13744d93782SGreg Clayton 
138b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
139e30f11d9SKate Stone 
140b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
141e30f11d9SKate Stone 
142b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
143b9c1b51eSKate Stone   if (stream) {
14416ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1454446487dSPavel Labath     if (m_top)
1464446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1474446487dSPavel Labath   }
1484446487dSPavel Labath }
1494446487dSPavel Labath 
1507a120c8bSZachary Turner IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, llvm::StringRef prompt,
151b9c1b51eSKate Stone                                    bool default_response)
152b9c1b51eSKate Stone     : IOHandlerEditline(
153b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
154c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
155514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
156514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
15744d93782SGreg Clayton           false,             // Multi-line
158e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
159*d77c2e09SJonas Devlieghere           0, *this, nullptr),
160b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
16144d93782SGreg Clayton   StreamString prompt_stream;
16244d93782SGreg Clayton   prompt_stream.PutCString(prompt);
16344d93782SGreg Clayton   if (m_default_response)
16444d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
16544d93782SGreg Clayton   else
16644d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
16744d93782SGreg Clayton 
168514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
16944d93782SGreg Clayton }
17044d93782SGreg Clayton 
171315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
17244d93782SGreg Clayton 
1737f88829cSRaphael Isemann int IOHandlerConfirm::IOHandlerComplete(
1747f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
1757f88829cSRaphael Isemann     const char *last_char, int skip_first_n_matches, int max_matches,
1767f88829cSRaphael Isemann     StringList &matches, StringList &descriptions) {
177b9c1b51eSKate Stone   if (current_line == cursor) {
178b9c1b51eSKate Stone     if (m_default_response) {
17944d93782SGreg Clayton       matches.AppendString("y");
180b9c1b51eSKate Stone     } else {
18144d93782SGreg Clayton       matches.AppendString("n");
18244d93782SGreg Clayton     }
18344d93782SGreg Clayton   }
18444d93782SGreg Clayton   return matches.GetSize();
18544d93782SGreg Clayton }
18644d93782SGreg Clayton 
187b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
188b9c1b51eSKate Stone                                               std::string &line) {
189b9c1b51eSKate Stone   if (line.empty()) {
19044d93782SGreg Clayton     // User just hit enter, set the response to the default
19144d93782SGreg Clayton     m_user_response = m_default_response;
19244d93782SGreg Clayton     io_handler.SetIsDone(true);
19344d93782SGreg Clayton     return;
19444d93782SGreg Clayton   }
19544d93782SGreg Clayton 
196b9c1b51eSKate Stone   if (line.size() == 1) {
197b9c1b51eSKate Stone     switch (line[0]) {
19844d93782SGreg Clayton     case 'y':
19944d93782SGreg Clayton     case 'Y':
20044d93782SGreg Clayton       m_user_response = true;
20144d93782SGreg Clayton       io_handler.SetIsDone(true);
20244d93782SGreg Clayton       return;
20344d93782SGreg Clayton     case 'n':
20444d93782SGreg Clayton     case 'N':
20544d93782SGreg Clayton       m_user_response = false;
20644d93782SGreg Clayton       io_handler.SetIsDone(true);
20744d93782SGreg Clayton       return;
20844d93782SGreg Clayton     default:
20944d93782SGreg Clayton       break;
21044d93782SGreg Clayton     }
21144d93782SGreg Clayton   }
21244d93782SGreg Clayton 
213b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
21444d93782SGreg Clayton     m_user_response = true;
21544d93782SGreg Clayton     io_handler.SetIsDone(true);
216b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
21744d93782SGreg Clayton     m_user_response = false;
21844d93782SGreg Clayton     io_handler.SetIsDone(true);
21944d93782SGreg Clayton   }
22044d93782SGreg Clayton }
22144d93782SGreg Clayton 
2227f88829cSRaphael Isemann int IOHandlerDelegate::IOHandlerComplete(
2237f88829cSRaphael Isemann     IOHandler &io_handler, const char *current_line, const char *cursor,
2247f88829cSRaphael Isemann     const char *last_char, int skip_first_n_matches, int max_matches,
2257f88829cSRaphael Isemann     StringList &matches, StringList &descriptions) {
226b9c1b51eSKate Stone   switch (m_completion) {
22744d93782SGreg Clayton   case Completion::None:
22844d93782SGreg Clayton     break;
22944d93782SGreg Clayton 
23044d93782SGreg Clayton   case Completion::LLDBCommand:
231b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
232b9c1b51eSKate Stone         current_line, cursor, last_char, skip_first_n_matches, max_matches,
2337f88829cSRaphael Isemann         matches, descriptions);
234b9c1b51eSKate Stone   case Completion::Expression: {
2357f88829cSRaphael Isemann     CompletionResult result;
236a2e76c0bSRaphael Isemann     CompletionRequest request(current_line, current_line - cursor,
2377f88829cSRaphael Isemann                               skip_first_n_matches, max_matches, result);
238b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
239b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
240a2e76c0bSRaphael Isemann         CommandCompletions::eVariablePathCompletion, request, nullptr);
2417f88829cSRaphael Isemann     result.GetMatches(matches);
2427f88829cSRaphael Isemann     result.GetDescriptions(descriptions);
24344d93782SGreg Clayton 
2441a6d7ab5SRaphael Isemann     size_t num_matches = request.GetNumberOfMatches();
245b9c1b51eSKate Stone     if (num_matches > 0) {
24644d93782SGreg Clayton       std::string common_prefix;
2471a6d7ab5SRaphael Isemann       matches.LongestCommonPrefix(common_prefix);
248a2e76c0bSRaphael Isemann       const size_t partial_name_len = request.GetCursorArgumentPrefix().size();
24944d93782SGreg Clayton 
25005097246SAdrian Prantl       // If we matched a unique single command, add a space... Only do this if
25105097246SAdrian Prantl       // the completer told us this was a complete word, however...
252a2e76c0bSRaphael Isemann       if (num_matches == 1 && request.GetWordComplete()) {
25344d93782SGreg Clayton         common_prefix.push_back(' ');
25444d93782SGreg Clayton       }
25544d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
25644d93782SGreg Clayton       matches.InsertStringAtIndex(0, std::move(common_prefix));
25744d93782SGreg Clayton     }
25844d93782SGreg Clayton     return num_matches;
259b9c1b51eSKate Stone   } break;
26044d93782SGreg Clayton   }
26144d93782SGreg Clayton 
26244d93782SGreg Clayton   return 0;
26344d93782SGreg Clayton }
26444d93782SGreg Clayton 
265b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
266b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
26744d93782SGreg Clayton     const char *editline_name, // Used for saving history files
268514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
269514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
270*d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
271b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
27244d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
27344d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
27444d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
275340b0309SGreg Clayton                         0,              // Flags
27644d93782SGreg Clayton                         editline_name,  // Used for saving history files
277b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
278*d77c2e09SJonas Devlieghere                         line_number_start, delegate, data_recorder) {}
27944d93782SGreg Clayton 
280b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
281b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
282b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
283b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
28444d93782SGreg Clayton     const char *editline_name, // Used for saving history files
285514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
286514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
287*d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
288*d77c2e09SJonas Devlieghere     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags,
289*d77c2e09SJonas Devlieghere                 data_recorder),
290cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
291d5b44036SJonas Devlieghere       m_editline_up(),
292cacde7dfSTodd Fiala #endif
293b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
294b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
295b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
296b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
297b9c1b51eSKate Stone       m_editing(false) {
29844d93782SGreg Clayton   SetPrompt(prompt);
29944d93782SGreg Clayton 
300cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
301914b8d98SDeepak Panickal   bool use_editline = false;
302340b0309SGreg Clayton 
303340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
30444d93782SGreg Clayton 
305b9c1b51eSKate Stone   if (use_editline) {
306d5b44036SJonas Devlieghere     m_editline_up.reset(new Editline(editline_name, GetInputFILE(),
307b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
308e30f11d9SKate Stone                                      m_color_prompts));
309d5b44036SJonas Devlieghere     m_editline_up->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
310d5b44036SJonas Devlieghere     m_editline_up->SetAutoCompleteCallback(AutoCompleteCallback, this);
311e30f11d9SKate Stone     // See if the delegate supports fixing indentation
312e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
313b9c1b51eSKate Stone     if (indent_chars) {
314b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
31505097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
316d5b44036SJonas Devlieghere       m_editline_up->SetFixIndentationCallback(FixIndentationCallback, this,
317b9c1b51eSKate Stone                                                indent_chars);
318e30f11d9SKate Stone     }
31944d93782SGreg Clayton   }
320cacde7dfSTodd Fiala #endif
321e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
322514d8cd8SZachary Turner   SetPrompt(prompt);
323e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
32444d93782SGreg Clayton }
32544d93782SGreg Clayton 
326b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
327cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
328d5b44036SJonas Devlieghere   m_editline_up.reset();
329cacde7dfSTodd Fiala #endif
33044d93782SGreg Clayton }
33144d93782SGreg Clayton 
332b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
333e30f11d9SKate Stone   IOHandler::Activate();
334e30f11d9SKate Stone   m_delegate.IOHandlerActivated(*this);
335e30f11d9SKate Stone }
336e30f11d9SKate Stone 
337b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
338e30f11d9SKate Stone   IOHandler::Deactivate();
339e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
340e30f11d9SKate Stone }
341e30f11d9SKate Stone 
342b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
343cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
344d5b44036SJonas Devlieghere   if (m_editline_up) {
345*d77c2e09SJonas Devlieghere     bool b = m_editline_up->GetLine(line, interrupted);
346*d77c2e09SJonas Devlieghere     if (m_data_recorder)
347*d77c2e09SJonas Devlieghere       m_data_recorder->Record(line, true);
348*d77c2e09SJonas Devlieghere     return b;
349b9c1b51eSKate Stone   } else {
350cacde7dfSTodd Fiala #endif
35144d93782SGreg Clayton     line.clear();
35244d93782SGreg Clayton 
35344d93782SGreg Clayton     FILE *in = GetInputFILE();
354b9c1b51eSKate Stone     if (in) {
355b9c1b51eSKate Stone       if (GetIsInteractive()) {
356c5dac77aSEugene Zelenko         const char *prompt = nullptr;
357e30f11d9SKate Stone 
358e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
359e30f11d9SKate Stone           prompt = GetContinuationPrompt();
360e30f11d9SKate Stone 
361c5dac77aSEugene Zelenko         if (prompt == nullptr)
362e30f11d9SKate Stone           prompt = GetPrompt();
363e30f11d9SKate Stone 
364b9c1b51eSKate Stone         if (prompt && prompt[0]) {
36544d93782SGreg Clayton           FILE *out = GetOutputFILE();
366b9c1b51eSKate Stone           if (out) {
36744d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
36844d93782SGreg Clayton             ::fflush(out);
36944d93782SGreg Clayton           }
37044d93782SGreg Clayton         }
37144d93782SGreg Clayton       }
37244d93782SGreg Clayton       char buffer[256];
37344d93782SGreg Clayton       bool done = false;
3740f86e6e7SGreg Clayton       bool got_line = false;
375e034a04eSGreg Clayton       m_editing = true;
376b9c1b51eSKate Stone       while (!done) {
377b9c1b51eSKate Stone         if (fgets(buffer, sizeof(buffer), in) == nullptr) {
378c7797accSGreg Clayton           const int saved_errno = errno;
379c9cf5798SGreg Clayton           if (feof(in))
38044d93782SGreg Clayton             done = true;
381b9c1b51eSKate Stone           else if (ferror(in)) {
382c7797accSGreg Clayton             if (saved_errno != EINTR)
383c7797accSGreg Clayton               done = true;
384c7797accSGreg Clayton           }
385b9c1b51eSKate Stone         } else {
3860f86e6e7SGreg Clayton           got_line = true;
38744d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
38844d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
38944d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
390b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
39144d93782SGreg Clayton             done = true;
39244d93782SGreg Clayton             // Strip trailing newlines
393b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
39444d93782SGreg Clayton               --buffer_len;
39544d93782SGreg Clayton               if (buffer_len == 0)
39644d93782SGreg Clayton                 break;
39744d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
39844d93782SGreg Clayton             }
39944d93782SGreg Clayton           }
40044d93782SGreg Clayton           line.append(buffer, buffer_len);
40144d93782SGreg Clayton         }
40244d93782SGreg Clayton       }
403e034a04eSGreg Clayton       m_editing = false;
404*d77c2e09SJonas Devlieghere       if (m_data_recorder && got_line)
405*d77c2e09SJonas Devlieghere         m_data_recorder->Record(line, true);
40605097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
40705097246SAdrian Prantl       // true in this case.
4080f86e6e7SGreg Clayton       return got_line;
409b9c1b51eSKate Stone     } else {
41044d93782SGreg Clayton       // No more input file, we are done...
41144d93782SGreg Clayton       SetIsDone(true);
41244d93782SGreg Clayton     }
413340b0309SGreg Clayton     return false;
414cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
41544d93782SGreg Clayton   }
416cacde7dfSTodd Fiala #endif
41744d93782SGreg Clayton }
41844d93782SGreg Clayton 
419cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
420b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
42144d93782SGreg Clayton                                                 StringList &lines,
422b9c1b51eSKate Stone                                                 void *baton) {
42344d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
424b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
425b9c1b51eSKate Stone                                                               lines);
426e30f11d9SKate Stone }
427e30f11d9SKate Stone 
428b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
429e30f11d9SKate Stone                                               const StringList &lines,
430e30f11d9SKate Stone                                               int cursor_position,
431b9c1b51eSKate Stone                                               void *baton) {
432e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
433b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
434b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
43544d93782SGreg Clayton }
43644d93782SGreg Clayton 
4377f88829cSRaphael Isemann int IOHandlerEditline::AutoCompleteCallback(
4387f88829cSRaphael Isemann     const char *current_line, const char *cursor, const char *last_char,
4397f88829cSRaphael Isemann     int skip_first_n_matches, int max_matches, StringList &matches,
4407f88829cSRaphael Isemann     StringList &descriptions, void *baton) {
44144d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
44244d93782SGreg Clayton   if (editline_reader)
443b9c1b51eSKate Stone     return editline_reader->m_delegate.IOHandlerComplete(
444b9c1b51eSKate Stone         *editline_reader, current_line, cursor, last_char, skip_first_n_matches,
4457f88829cSRaphael Isemann         max_matches, matches, descriptions);
44644d93782SGreg Clayton   return 0;
44744d93782SGreg Clayton }
448cacde7dfSTodd Fiala #endif
44944d93782SGreg Clayton 
450b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
451cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
452d5b44036SJonas Devlieghere   if (m_editline_up) {
453d5b44036SJonas Devlieghere     return m_editline_up->GetPrompt();
454b9c1b51eSKate Stone   } else {
455cacde7dfSTodd Fiala #endif
456cacde7dfSTodd Fiala     if (m_prompt.empty())
457c5dac77aSEugene Zelenko       return nullptr;
458cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
459cacde7dfSTodd Fiala   }
460cacde7dfSTodd Fiala #endif
46144d93782SGreg Clayton   return m_prompt.c_str();
46244d93782SGreg Clayton }
46344d93782SGreg Clayton 
464514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
465514d8cd8SZachary Turner   m_prompt = prompt;
466514d8cd8SZachary Turner 
467cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
468d5b44036SJonas Devlieghere   if (m_editline_up)
469d5b44036SJonas Devlieghere     m_editline_up->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
470cacde7dfSTodd Fiala #endif
47144d93782SGreg Clayton   return true;
47244d93782SGreg Clayton }
47344d93782SGreg Clayton 
474b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
475b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
476b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
477e30f11d9SKate Stone }
478e30f11d9SKate Stone 
479514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
480514d8cd8SZachary Turner   m_continuation_prompt = prompt;
481e30f11d9SKate Stone 
482d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
483d5b44036SJonas Devlieghere   if (m_editline_up)
484d5b44036SJonas Devlieghere     m_editline_up->SetContinuationPrompt(m_continuation_prompt.empty()
485b9c1b51eSKate Stone                                              ? nullptr
486b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
487d553d00cSZachary Turner #endif
488e30f11d9SKate Stone }
489e30f11d9SKate Stone 
490b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
491f6913cd7SGreg Clayton   m_base_line_number = line;
492f6913cd7SGreg Clayton }
493e30f11d9SKate Stone 
494b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
495d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
496d5b44036SJonas Devlieghere   if (m_editline_up)
497d5b44036SJonas Devlieghere     return m_editline_up->GetCurrentLine();
498e30f11d9SKate Stone #endif
499e30f11d9SKate Stone   return m_curr_line_idx;
500e30f11d9SKate Stone }
501e30f11d9SKate Stone 
502b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
503e30f11d9SKate Stone   m_current_lines_ptr = &lines;
504e30f11d9SKate Stone 
50544d93782SGreg Clayton   bool success = false;
506cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
507d5b44036SJonas Devlieghere   if (m_editline_up) {
508d5b44036SJonas Devlieghere     return m_editline_up->GetLines(m_base_line_number, lines, interrupted);
509b9c1b51eSKate Stone   } else {
510cacde7dfSTodd Fiala #endif
511e30f11d9SKate Stone     bool done = false;
51297206d57SZachary Turner     Status error;
51344d93782SGreg Clayton 
514b9c1b51eSKate Stone     while (!done) {
515f6913cd7SGreg Clayton       // Show line numbers if we are asked to
51644d93782SGreg Clayton       std::string line;
517b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
518f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
519f6913cd7SGreg Clayton         if (out)
520b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
521b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
522f6913cd7SGreg Clayton       }
523f6913cd7SGreg Clayton 
524e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
525e30f11d9SKate Stone 
526f0066ad0SGreg Clayton       bool interrupted = false;
527b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
52844d93782SGreg Clayton         lines.AppendString(line);
529e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
530b9c1b51eSKate Stone       } else {
531e30f11d9SKate Stone         done = true;
53244d93782SGreg Clayton       }
53344d93782SGreg Clayton     }
53444d93782SGreg Clayton     success = lines.GetSize() > 0;
535cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
53644d93782SGreg Clayton   }
537cacde7dfSTodd Fiala #endif
53844d93782SGreg Clayton   return success;
53944d93782SGreg Clayton }
54044d93782SGreg Clayton 
54105097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
54205097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
543b9c1b51eSKate Stone void IOHandlerEditline::Run() {
54444d93782SGreg Clayton   std::string line;
545b9c1b51eSKate Stone   while (IsActive()) {
546f0066ad0SGreg Clayton     bool interrupted = false;
547b9c1b51eSKate Stone     if (m_multi_line) {
54844d93782SGreg Clayton       StringList lines;
549b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
550b9c1b51eSKate Stone         if (interrupted) {
551e30f11d9SKate Stone           m_done = m_interrupt_exits;
552e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
553e30f11d9SKate Stone 
554b9c1b51eSKate Stone         } else {
55544d93782SGreg Clayton           line = lines.CopyList();
55644d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
55744d93782SGreg Clayton         }
558b9c1b51eSKate Stone       } else {
55944d93782SGreg Clayton         m_done = true;
56044d93782SGreg Clayton       }
561b9c1b51eSKate Stone     } else {
562b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
563e30f11d9SKate Stone         if (interrupted)
564e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
565e30f11d9SKate Stone         else
56644d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
567b9c1b51eSKate Stone       } else {
56844d93782SGreg Clayton         m_done = true;
56944d93782SGreg Clayton       }
57044d93782SGreg Clayton     }
57144d93782SGreg Clayton   }
57244d93782SGreg Clayton }
57344d93782SGreg Clayton 
574b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
575cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
576d5b44036SJonas Devlieghere   if (m_editline_up)
577d5b44036SJonas Devlieghere     m_editline_up->Cancel();
578cacde7dfSTodd Fiala #endif
579e68f5d6bSGreg Clayton }
580e68f5d6bSGreg Clayton 
581b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
582f0066ad0SGreg Clayton   // Let the delgate handle it first
583f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
584f0066ad0SGreg Clayton     return true;
585f0066ad0SGreg Clayton 
586cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
587d5b44036SJonas Devlieghere   if (m_editline_up)
588d5b44036SJonas Devlieghere     return m_editline_up->Interrupt();
589cacde7dfSTodd Fiala #endif
590f0066ad0SGreg Clayton   return false;
59144d93782SGreg Clayton }
59244d93782SGreg Clayton 
593b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
594cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
595d5b44036SJonas Devlieghere   if (m_editline_up)
596d5b44036SJonas Devlieghere     m_editline_up->Interrupt();
597cacde7dfSTodd Fiala #endif
59844d93782SGreg Clayton }
59944d93782SGreg Clayton 
600b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
6014446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
602d5b44036SJonas Devlieghere   if (m_editline_up)
603d5b44036SJonas Devlieghere     m_editline_up->PrintAsync(stream, s, len);
6044446487dSPavel Labath   else
6054446487dSPavel Labath #endif
606fab31220STed Woodward   {
607fab31220STed Woodward #ifdef _MSC_VER
608341e4789SDawn Perchik     const char *prompt = GetPrompt();
609b9c1b51eSKate Stone     if (prompt) {
610fab31220STed Woodward       // Back up over previous prompt using Windows API
611fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
612fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
613fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
614fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
615fab31220STed Woodward       coord.X -= strlen(prompt);
616fab31220STed Woodward       if (coord.X < 0)
617fab31220STed Woodward         coord.X = 0;
618fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
619fab31220STed Woodward     }
620fab31220STed Woodward #endif
6214446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
622341e4789SDawn Perchik #ifdef _MSC_VER
623fab31220STed Woodward     if (prompt)
624b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
625b9c1b51eSKate Stone                             strlen(prompt));
626341e4789SDawn Perchik #endif
627fab31220STed Woodward   }
6284446487dSPavel Labath }
6294446487dSPavel Labath 
63005097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
631914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
632914b8d98SDeepak Panickal 
63344d93782SGreg Clayton #define KEY_RETURN 10
63444d93782SGreg Clayton #define KEY_ESCAPE 27
63544d93782SGreg Clayton 
636b9c1b51eSKate Stone namespace curses {
63744d93782SGreg Clayton class Menu;
63844d93782SGreg Clayton class MenuDelegate;
63944d93782SGreg Clayton class Window;
64044d93782SGreg Clayton class WindowDelegate;
64144d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
64244d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
64344d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
64444d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
64544d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
64644d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
64744d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
64844d93782SGreg Clayton 
64944d93782SGreg Clayton #if 0
65044d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
65144d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
65244d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
65344d93782SGreg Clayton #endif
654315b6884SEugene Zelenko 
655b9c1b51eSKate Stone struct Point {
65644d93782SGreg Clayton   int x;
65744d93782SGreg Clayton   int y;
65844d93782SGreg Clayton 
659b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
66044d93782SGreg Clayton 
661b9c1b51eSKate Stone   void Clear() {
66244d93782SGreg Clayton     x = 0;
66344d93782SGreg Clayton     y = 0;
66444d93782SGreg Clayton   }
66544d93782SGreg Clayton 
666b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
66744d93782SGreg Clayton     x += rhs.x;
66844d93782SGreg Clayton     y += rhs.y;
66944d93782SGreg Clayton     return *this;
67044d93782SGreg Clayton   }
67144d93782SGreg Clayton 
672b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
67344d93782SGreg Clayton };
67444d93782SGreg Clayton 
675b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
67644d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
67744d93782SGreg Clayton }
678315b6884SEugene Zelenko 
679b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
68044d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
68144d93782SGreg Clayton }
68244d93782SGreg Clayton 
683b9c1b51eSKate Stone struct Size {
68444d93782SGreg Clayton   int width;
68544d93782SGreg Clayton   int height;
686b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
68744d93782SGreg Clayton 
688b9c1b51eSKate Stone   void Clear() {
68944d93782SGreg Clayton     width = 0;
69044d93782SGreg Clayton     height = 0;
69144d93782SGreg Clayton   }
69244d93782SGreg Clayton 
693b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
69444d93782SGreg Clayton };
69544d93782SGreg Clayton 
696b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
69744d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
69844d93782SGreg Clayton }
699315b6884SEugene Zelenko 
700b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
70144d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
70244d93782SGreg Clayton }
70344d93782SGreg Clayton 
704b9c1b51eSKate Stone struct Rect {
70544d93782SGreg Clayton   Point origin;
70644d93782SGreg Clayton   Size size;
70744d93782SGreg Clayton 
708b9c1b51eSKate Stone   Rect() : origin(), size() {}
70944d93782SGreg Clayton 
710b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
71144d93782SGreg Clayton 
712b9c1b51eSKate Stone   void Clear() {
71344d93782SGreg Clayton     origin.Clear();
71444d93782SGreg Clayton     size.Clear();
71544d93782SGreg Clayton   }
71644d93782SGreg Clayton 
717b9c1b51eSKate Stone   void Dump() {
718b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
719b9c1b51eSKate Stone            size.height);
72044d93782SGreg Clayton   }
72144d93782SGreg Clayton 
722b9c1b51eSKate Stone   void Inset(int w, int h) {
72344d93782SGreg Clayton     if (size.width > w * 2)
72444d93782SGreg Clayton       size.width -= w * 2;
72544d93782SGreg Clayton     origin.x += w;
72644d93782SGreg Clayton 
72744d93782SGreg Clayton     if (size.height > h * 2)
72844d93782SGreg Clayton       size.height -= h * 2;
72944d93782SGreg Clayton     origin.y += h;
73044d93782SGreg Clayton   }
731315b6884SEugene Zelenko 
73205097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
73305097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
734b9c1b51eSKate Stone   Rect MakeStatusBar() {
73544d93782SGreg Clayton     Rect status_bar;
736b9c1b51eSKate Stone     if (size.height > 1) {
73744d93782SGreg Clayton       status_bar.origin.x = origin.x;
73844d93782SGreg Clayton       status_bar.origin.y = size.height;
73944d93782SGreg Clayton       status_bar.size.width = size.width;
74044d93782SGreg Clayton       status_bar.size.height = 1;
74144d93782SGreg Clayton       --size.height;
74244d93782SGreg Clayton     }
74344d93782SGreg Clayton     return status_bar;
74444d93782SGreg Clayton   }
74544d93782SGreg Clayton 
74605097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
74705097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
748b9c1b51eSKate Stone   Rect MakeMenuBar() {
74944d93782SGreg Clayton     Rect menubar;
750b9c1b51eSKate Stone     if (size.height > 1) {
75144d93782SGreg Clayton       menubar.origin.x = origin.x;
75244d93782SGreg Clayton       menubar.origin.y = origin.y;
75344d93782SGreg Clayton       menubar.size.width = size.width;
75444d93782SGreg Clayton       menubar.size.height = 1;
75544d93782SGreg Clayton       ++origin.y;
75644d93782SGreg Clayton       --size.height;
75744d93782SGreg Clayton     }
75844d93782SGreg Clayton     return menubar;
75944d93782SGreg Clayton   }
76044d93782SGreg Clayton 
761b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
762b9c1b51eSKate Stone                                  Rect &bottom) const {
76344d93782SGreg Clayton     float top_height = top_percentage * size.height;
76444d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
76544d93782SGreg Clayton   }
76644d93782SGreg Clayton 
767b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
76844d93782SGreg Clayton     top = *this;
769b9c1b51eSKate Stone     if (top_height < size.height) {
77044d93782SGreg Clayton       top.size.height = top_height;
77144d93782SGreg Clayton       bottom.origin.x = origin.x;
77244d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
77344d93782SGreg Clayton       bottom.size.width = size.width;
77444d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
775b9c1b51eSKate Stone     } else {
77644d93782SGreg Clayton       bottom.Clear();
77744d93782SGreg Clayton     }
77844d93782SGreg Clayton   }
77944d93782SGreg Clayton 
780b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
781b9c1b51eSKate Stone                                Rect &right) const {
78244d93782SGreg Clayton     float left_width = left_percentage * size.width;
78344d93782SGreg Clayton     VerticalSplit(left_width, left, right);
78444d93782SGreg Clayton   }
78544d93782SGreg Clayton 
786b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
78744d93782SGreg Clayton     left = *this;
788b9c1b51eSKate Stone     if (left_width < size.width) {
78944d93782SGreg Clayton       left.size.width = left_width;
79044d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
79144d93782SGreg Clayton       right.origin.y = origin.y;
79244d93782SGreg Clayton       right.size.width = size.width - left.size.width;
79344d93782SGreg Clayton       right.size.height = size.height;
794b9c1b51eSKate Stone     } else {
79544d93782SGreg Clayton       right.Clear();
79644d93782SGreg Clayton     }
79744d93782SGreg Clayton   }
79844d93782SGreg Clayton };
79944d93782SGreg Clayton 
800b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
80144d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
80244d93782SGreg Clayton }
803315b6884SEugene Zelenko 
804b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
80544d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
80644d93782SGreg Clayton }
80744d93782SGreg Clayton 
808b9c1b51eSKate Stone enum HandleCharResult {
80944d93782SGreg Clayton   eKeyNotHandled = 0,
81044d93782SGreg Clayton   eKeyHandled = 1,
81144d93782SGreg Clayton   eQuitApplication = 2
81244d93782SGreg Clayton };
81344d93782SGreg Clayton 
814b9c1b51eSKate Stone enum class MenuActionResult {
81544d93782SGreg Clayton   Handled,
81644d93782SGreg Clayton   NotHandled,
81744d93782SGreg Clayton   Quit // Exit all menus and quit
81844d93782SGreg Clayton };
81944d93782SGreg Clayton 
820b9c1b51eSKate Stone struct KeyHelp {
82144d93782SGreg Clayton   int ch;
82244d93782SGreg Clayton   const char *description;
82344d93782SGreg Clayton };
82444d93782SGreg Clayton 
825b9c1b51eSKate Stone class WindowDelegate {
82644d93782SGreg Clayton public:
827b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
82844d93782SGreg Clayton 
829b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
83044d93782SGreg Clayton     return false; // Drawing not handled
83144d93782SGreg Clayton   }
83244d93782SGreg Clayton 
833b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
83444d93782SGreg Clayton     return eKeyNotHandled;
83544d93782SGreg Clayton   }
83644d93782SGreg Clayton 
837b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
83844d93782SGreg Clayton 
839b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
84044d93782SGreg Clayton };
84144d93782SGreg Clayton 
842b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
84344d93782SGreg Clayton public:
84444d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
84544d93782SGreg Clayton 
846bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
84744d93782SGreg Clayton 
848b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
84944d93782SGreg Clayton 
850b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
85144d93782SGreg Clayton 
852b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
85344d93782SGreg Clayton 
854b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
85544d93782SGreg Clayton 
85644d93782SGreg Clayton protected:
85744d93782SGreg Clayton   StringList m_text;
85844d93782SGreg Clayton   int m_first_visible_line;
85944d93782SGreg Clayton };
86044d93782SGreg Clayton 
861b9c1b51eSKate Stone class Window {
86244d93782SGreg Clayton public:
863b9c1b51eSKate Stone   Window(const char *name)
864b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
865b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
866b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
867b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
86844d93782SGreg Clayton 
869b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
870b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
871b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
872b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
873b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
87444d93782SGreg Clayton     if (w)
87544d93782SGreg Clayton       Reset(w);
87644d93782SGreg Clayton   }
87744d93782SGreg Clayton 
878b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
879b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
880b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
881b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
882b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
883b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
884b9c1b51eSKate Stone                    bounds.origin.y));
88544d93782SGreg Clayton   }
88644d93782SGreg Clayton 
887b9c1b51eSKate Stone   virtual ~Window() {
88844d93782SGreg Clayton     RemoveSubWindows();
88944d93782SGreg Clayton     Reset();
89044d93782SGreg Clayton   }
89144d93782SGreg Clayton 
892b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
89344d93782SGreg Clayton     if (m_window == w)
89444d93782SGreg Clayton       return;
89544d93782SGreg Clayton 
896b9c1b51eSKate Stone     if (m_panel) {
89744d93782SGreg Clayton       ::del_panel(m_panel);
898c5dac77aSEugene Zelenko       m_panel = nullptr;
89944d93782SGreg Clayton     }
900b9c1b51eSKate Stone     if (m_window && m_delete) {
90144d93782SGreg Clayton       ::delwin(m_window);
902c5dac77aSEugene Zelenko       m_window = nullptr;
90344d93782SGreg Clayton       m_delete = false;
90444d93782SGreg Clayton     }
905b9c1b51eSKate Stone     if (w) {
90644d93782SGreg Clayton       m_window = w;
90744d93782SGreg Clayton       m_panel = ::new_panel(m_window);
90844d93782SGreg Clayton       m_delete = del;
90944d93782SGreg Clayton     }
91044d93782SGreg Clayton   }
91144d93782SGreg Clayton 
91244d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
91344d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
914b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
915b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
916b9c1b51eSKate Stone   }
91744d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
91844d93782SGreg Clayton   void Erase() { ::werase(m_window); }
919b9c1b51eSKate Stone   Rect GetBounds() {
920b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
921b9c1b51eSKate Stone   } // Get the rectangle in our parent window
92244d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
92344d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
92444d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
925b9c1b51eSKate Stone   Rect GetFrame() {
926b9c1b51eSKate Stone     return Rect(Point(), GetSize());
927b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
92844d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
92944d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
93044d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
93144d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
93244d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
93344d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
93444d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
93544d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
93644d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
93744d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
93844d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
939b9c1b51eSKate Stone   void Resize(const Size &size) {
940b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
941b9c1b51eSKate Stone   }
94244d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
94344d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
94444d93782SGreg Clayton   void Refresh() { ::wrefresh(m_window); }
945b9c1b51eSKate Stone   void DeferredRefresh() {
94644d93782SGreg Clayton     // We are using panels, so we don't need to call this...
94744d93782SGreg Clayton     //::wnoutrefresh(m_window);
94844d93782SGreg Clayton   }
949b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
950b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
951b9c1b51eSKate Stone   }
95244d93782SGreg Clayton   void UnderlineOn() { AttributeOn(A_UNDERLINE); }
95344d93782SGreg Clayton   void UnderlineOff() { AttributeOff(A_UNDERLINE); }
95444d93782SGreg Clayton 
955b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
95644d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
957b9c1b51eSKate Stone     if (bytes_left > right_pad) {
95844d93782SGreg Clayton       bytes_left -= right_pad;
95944d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
96044d93782SGreg Clayton     }
96144d93782SGreg Clayton   }
96244d93782SGreg Clayton 
963b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
96444d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
965b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
96644d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
96744d93782SGreg Clayton       Size size = GetSize();
968b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
969b9c1b51eSKate Stone                      origin.x),
970b9c1b51eSKate Stone             true);
971b9c1b51eSKate Stone     } else {
97244d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
97344d93782SGreg Clayton     }
97444d93782SGreg Clayton   }
97544d93782SGreg Clayton 
976b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
97744d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
978b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
97944d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
980b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
981b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
982b9c1b51eSKate Stone             true);
983b9c1b51eSKate Stone     } else {
98444d93782SGreg Clayton       if (moving_window)
98544d93782SGreg Clayton         MoveWindow(bounds.origin);
98644d93782SGreg Clayton       Resize(bounds.size);
98744d93782SGreg Clayton     }
98844d93782SGreg Clayton   }
98944d93782SGreg Clayton 
990b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
99144d93782SGreg Clayton     va_list args;
99244d93782SGreg Clayton     va_start(args, format);
99344d93782SGreg Clayton     vwprintw(m_window, format, args);
99444d93782SGreg Clayton     va_end(args);
99544d93782SGreg Clayton   }
99644d93782SGreg Clayton 
997b9c1b51eSKate Stone   void Touch() {
99844d93782SGreg Clayton     ::touchwin(m_window);
99944d93782SGreg Clayton     if (m_parent)
100044d93782SGreg Clayton       m_parent->Touch();
100144d93782SGreg Clayton   }
100244d93782SGreg Clayton 
1003b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
1004b9c1b51eSKate Stone                            bool make_active) {
1005c6091d2bSJonas Devlieghere     auto get_window = [this, &bounds]() {
1006c6091d2bSJonas Devlieghere       return m_window
1007c6091d2bSJonas Devlieghere                  ? ::subwin(m_window, bounds.size.height, bounds.size.width,
1008c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x)
1009c6091d2bSJonas Devlieghere                  : ::newwin(bounds.size.height, bounds.size.width,
1010c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x);
1011c6091d2bSJonas Devlieghere     };
1012c6091d2bSJonas Devlieghere     WindowSP subwindow_sp = std::make_shared<Window>(name, get_window(), true);
1013c6091d2bSJonas Devlieghere     subwindow_sp->m_is_subwin = subwindow_sp.operator bool();
101444d93782SGreg Clayton     subwindow_sp->m_parent = this;
1015b9c1b51eSKate Stone     if (make_active) {
101644d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
101744d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
101844d93782SGreg Clayton     }
101944d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
102044d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
102144d93782SGreg Clayton     m_needs_update = true;
102244d93782SGreg Clayton     return subwindow_sp;
102344d93782SGreg Clayton   }
102444d93782SGreg Clayton 
1025b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
102644d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
102744d93782SGreg Clayton     size_t i = 0;
1028b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1029b9c1b51eSKate Stone       if ((*pos).get() == window) {
103044d93782SGreg Clayton         if (m_prev_active_window_idx == i)
103144d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1032b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1033b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
103444d93782SGreg Clayton           --m_prev_active_window_idx;
103544d93782SGreg Clayton 
103644d93782SGreg Clayton         if (m_curr_active_window_idx == i)
103744d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1038b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1039b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
104044d93782SGreg Clayton           --m_curr_active_window_idx;
104144d93782SGreg Clayton         window->Erase();
104244d93782SGreg Clayton         m_subwindows.erase(pos);
104344d93782SGreg Clayton         m_needs_update = true;
104444d93782SGreg Clayton         if (m_parent)
104544d93782SGreg Clayton           m_parent->Touch();
104644d93782SGreg Clayton         else
104744d93782SGreg Clayton           ::touchwin(stdscr);
104844d93782SGreg Clayton         return true;
104944d93782SGreg Clayton       }
105044d93782SGreg Clayton     }
105144d93782SGreg Clayton     return false;
105244d93782SGreg Clayton   }
105344d93782SGreg Clayton 
1054b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
105544d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
105644d93782SGreg Clayton     size_t i = 0;
1057b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
10588d20cfdfSJonas Devlieghere       if ((*pos)->m_name == name)
105944d93782SGreg Clayton         return *pos;
106044d93782SGreg Clayton     }
106144d93782SGreg Clayton     return WindowSP();
106244d93782SGreg Clayton   }
106344d93782SGreg Clayton 
1064b9c1b51eSKate Stone   void RemoveSubWindows() {
106544d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
106644d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
106744d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1068b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
106944d93782SGreg Clayton       (*pos)->Erase();
107044d93782SGreg Clayton     }
107144d93782SGreg Clayton     if (m_parent)
107244d93782SGreg Clayton       m_parent->Touch();
107344d93782SGreg Clayton     else
107444d93782SGreg Clayton       ::touchwin(stdscr);
107544d93782SGreg Clayton   }
107644d93782SGreg Clayton 
1077b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
107844d93782SGreg Clayton 
1079b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
108044d93782SGreg Clayton 
108144d93782SGreg Clayton   //----------------------------------------------------------------------
108244d93782SGreg Clayton   // Window drawing utilities
108344d93782SGreg Clayton   //----------------------------------------------------------------------
1084b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
108544d93782SGreg Clayton     attr_t attr = 0;
108644d93782SGreg Clayton     if (IsActive())
108744d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
108844d93782SGreg Clayton     else
108944d93782SGreg Clayton       attr = 0;
109044d93782SGreg Clayton     if (attr)
109144d93782SGreg Clayton       AttributeOn(attr);
109244d93782SGreg Clayton 
109344d93782SGreg Clayton     Box();
109444d93782SGreg Clayton     MoveCursor(3, 0);
109544d93782SGreg Clayton 
1096b9c1b51eSKate Stone     if (title && title[0]) {
109744d93782SGreg Clayton       PutChar('<');
109844d93782SGreg Clayton       PutCString(title);
109944d93782SGreg Clayton       PutChar('>');
110044d93782SGreg Clayton     }
110144d93782SGreg Clayton 
1102b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
110344d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
110444d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
110544d93782SGreg Clayton 
1106b9c1b51eSKate Stone       if (x > 0) {
110744d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
110844d93782SGreg Clayton         PutChar('[');
110944d93782SGreg Clayton         PutCString(bottom_message);
111044d93782SGreg Clayton         PutChar(']');
1111b9c1b51eSKate Stone       } else {
111244d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
111344d93782SGreg Clayton         PutChar('[');
111444d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
111544d93782SGreg Clayton       }
111644d93782SGreg Clayton     }
111744d93782SGreg Clayton     if (attr)
111844d93782SGreg Clayton       AttributeOff(attr);
111944d93782SGreg Clayton   }
112044d93782SGreg Clayton 
1121b9c1b51eSKate Stone   virtual void Draw(bool force) {
112244d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
112344d93782SGreg Clayton       return;
112444d93782SGreg Clayton 
112544d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
112644d93782SGreg Clayton       subwindow_sp->Draw(force);
112744d93782SGreg Clayton   }
112844d93782SGreg Clayton 
1129b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1130b9c1b51eSKate Stone     if (m_delegate_sp) {
113144d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
113244d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1133b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1134d5b44036SJonas Devlieghere         std::unique_ptr<HelpDialogDelegate> help_delegate_up(
1135b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
1136d5b44036SJonas Devlieghere         const size_t num_lines = help_delegate_up->GetNumLines();
1137d5b44036SJonas Devlieghere         const size_t max_length = help_delegate_up->GetMaxLineLength();
113844d93782SGreg Clayton         Rect bounds = GetBounds();
113944d93782SGreg Clayton         bounds.Inset(1, 1);
1140b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
114144d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
114244d93782SGreg Clayton           bounds.size.width = max_length + 4;
1143b9c1b51eSKate Stone         } else {
1144b9c1b51eSKate Stone           if (bounds.size.width > 100) {
114544d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
114644d93782SGreg Clayton             bounds.origin.x += inset_w;
114744d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
114844d93782SGreg Clayton           }
114944d93782SGreg Clayton         }
115044d93782SGreg Clayton 
1151b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
115244d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
115344d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1154b9c1b51eSKate Stone         } else {
1155b9c1b51eSKate Stone           if (bounds.size.height > 100) {
115644d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
115744d93782SGreg Clayton             bounds.origin.y += inset_h;
115844d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
115944d93782SGreg Clayton           }
116044d93782SGreg Clayton         }
11615fdb09bbSGreg Clayton         WindowSP help_window_sp;
11625fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11635fdb09bbSGreg Clayton         if (parent_window)
11645fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11655fdb09bbSGreg Clayton         else
11665fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1167b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1168d5b44036SJonas Devlieghere             WindowDelegateSP(help_delegate_up.release()));
116944d93782SGreg Clayton         return true;
117044d93782SGreg Clayton       }
117144d93782SGreg Clayton     }
117244d93782SGreg Clayton     return false;
117344d93782SGreg Clayton   }
117444d93782SGreg Clayton 
1175b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
117644d93782SGreg Clayton     // Always check the active window first
117744d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
117844d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1179b9c1b51eSKate Stone     if (active_window_sp) {
118044d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
118144d93782SGreg Clayton       if (result != eKeyNotHandled)
118244d93782SGreg Clayton         return result;
118344d93782SGreg Clayton     }
118444d93782SGreg Clayton 
1185b9c1b51eSKate Stone     if (m_delegate_sp) {
118644d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
118744d93782SGreg Clayton       if (result != eKeyNotHandled)
118844d93782SGreg Clayton         return result;
118944d93782SGreg Clayton     }
119044d93782SGreg Clayton 
119105097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
119205097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
119305097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
119405097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
119544d93782SGreg Clayton     Windows subwindows(m_subwindows);
1196b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1197b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
119844d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
119944d93782SGreg Clayton         if (result != eKeyNotHandled)
120044d93782SGreg Clayton           return result;
120144d93782SGreg Clayton       }
120244d93782SGreg Clayton     }
120344d93782SGreg Clayton 
120444d93782SGreg Clayton     return eKeyNotHandled;
120544d93782SGreg Clayton   }
120644d93782SGreg Clayton 
1207b9c1b51eSKate Stone   bool SetActiveWindow(Window *window) {
120844d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1209b9c1b51eSKate Stone     for (size_t i = 0; i < num_subwindows; ++i) {
1210b9c1b51eSKate Stone       if (m_subwindows[i].get() == window) {
121144d93782SGreg Clayton         m_prev_active_window_idx = m_curr_active_window_idx;
121244d93782SGreg Clayton         ::top_panel(window->m_panel);
121344d93782SGreg Clayton         m_curr_active_window_idx = i;
121444d93782SGreg Clayton         return true;
121544d93782SGreg Clayton       }
121644d93782SGreg Clayton     }
121744d93782SGreg Clayton     return false;
121844d93782SGreg Clayton   }
121944d93782SGreg Clayton 
1220b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1221b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1222b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1223b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
122444d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
122544d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1226b9c1b51eSKate Stone         } else if (IsActive()) {
122744d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
122844d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
122944d93782SGreg Clayton 
123044d93782SGreg Clayton           // Find first window that wants to be active if this window is active
123144d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1232b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1233b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
123444d93782SGreg Clayton               m_curr_active_window_idx = i;
123544d93782SGreg Clayton               break;
123644d93782SGreg Clayton             }
123744d93782SGreg Clayton           }
123844d93782SGreg Clayton         }
123944d93782SGreg Clayton       }
124044d93782SGreg Clayton 
124144d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
124244d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
124344d93782SGreg Clayton     }
124444d93782SGreg Clayton     return WindowSP();
124544d93782SGreg Clayton   }
124644d93782SGreg Clayton 
1247b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
124844d93782SGreg Clayton 
1249b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
125044d93782SGreg Clayton 
1251b9c1b51eSKate Stone   const WindowDelegateSP &GetDelegate() const { return m_delegate_sp; }
125244d93782SGreg Clayton 
1253b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
125444d93782SGreg Clayton     m_delegate_sp = delegate_sp;
125544d93782SGreg Clayton   }
125644d93782SGreg Clayton 
1257b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
125844d93782SGreg Clayton 
1259b9c1b51eSKate Stone   bool IsActive() const {
126044d93782SGreg Clayton     if (m_parent)
126144d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
126244d93782SGreg Clayton     else
126344d93782SGreg Clayton       return true; // Top level window is always active
126444d93782SGreg Clayton   }
126544d93782SGreg Clayton 
1266b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
126744d93782SGreg Clayton     // Move active focus to next window
126844d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1269b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
127044d93782SGreg Clayton       uint32_t idx = 0;
1271b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1272b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
127344d93782SGreg Clayton           m_curr_active_window_idx = idx;
127444d93782SGreg Clayton           break;
127544d93782SGreg Clayton         }
127644d93782SGreg Clayton         ++idx;
127744d93782SGreg Clayton       }
1278b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
127944d93782SGreg Clayton       bool handled = false;
128044d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1281b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1282b9c1b51eSKate Stone            ++idx) {
1283b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
128444d93782SGreg Clayton           m_curr_active_window_idx = idx;
128544d93782SGreg Clayton           handled = true;
128644d93782SGreg Clayton           break;
128744d93782SGreg Clayton         }
128844d93782SGreg Clayton       }
1289b9c1b51eSKate Stone       if (!handled) {
1290b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++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       }
1297b9c1b51eSKate Stone     } else {
129844d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1299b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1300b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
130144d93782SGreg Clayton           m_curr_active_window_idx = idx;
130244d93782SGreg Clayton           break;
130344d93782SGreg Clayton         }
130444d93782SGreg Clayton       }
130544d93782SGreg Clayton     }
130644d93782SGreg Clayton   }
130744d93782SGreg Clayton 
1308b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1309315b6884SEugene Zelenko 
131044d93782SGreg Clayton protected:
131144d93782SGreg Clayton   std::string m_name;
131244d93782SGreg Clayton   WINDOW *m_window;
131344d93782SGreg Clayton   PANEL *m_panel;
131444d93782SGreg Clayton   Window *m_parent;
131544d93782SGreg Clayton   Windows m_subwindows;
131644d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
131744d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
131844d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
131944d93782SGreg Clayton   bool m_delete;
132044d93782SGreg Clayton   bool m_needs_update;
132144d93782SGreg Clayton   bool m_can_activate;
132244d93782SGreg Clayton   bool m_is_subwin;
132344d93782SGreg Clayton 
132444d93782SGreg Clayton private:
132544d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
132644d93782SGreg Clayton };
132744d93782SGreg Clayton 
1328b9c1b51eSKate Stone class MenuDelegate {
132944d93782SGreg Clayton public:
1330315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
133144d93782SGreg Clayton 
1332b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
133344d93782SGreg Clayton };
133444d93782SGreg Clayton 
1335b9c1b51eSKate Stone class Menu : public WindowDelegate {
133644d93782SGreg Clayton public:
1337b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
133844d93782SGreg Clayton 
133944d93782SGreg Clayton   // Menubar or separator constructor
134044d93782SGreg Clayton   Menu(Type type);
134144d93782SGreg Clayton 
134244d93782SGreg Clayton   // Menuitem constructor
1343b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
134444d93782SGreg Clayton        uint64_t identifier);
134544d93782SGreg Clayton 
1346315b6884SEugene Zelenko   ~Menu() override = default;
134744d93782SGreg Clayton 
1348b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
134944d93782SGreg Clayton 
1350b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
135144d93782SGreg Clayton     m_delegate_sp = delegate_sp;
135244d93782SGreg Clayton   }
135344d93782SGreg Clayton 
1354b9c1b51eSKate Stone   void RecalculateNameLengths();
135544d93782SGreg Clayton 
1356b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
135744d93782SGreg Clayton 
1358b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
135944d93782SGreg Clayton 
1360b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
136144d93782SGreg Clayton 
1362b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
136344d93782SGreg Clayton 
1364b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
136544d93782SGreg Clayton 
1366b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
136744d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1368b9c1b51eSKate Stone     if (m_delegate_sp) {
136944d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
137044d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
137144d93782SGreg Clayton         return result;
1372b9c1b51eSKate Stone     } else if (m_parent) {
137344d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
137444d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
137544d93782SGreg Clayton         return result;
137644d93782SGreg Clayton     }
137744d93782SGreg Clayton     return m_canned_result;
137844d93782SGreg Clayton   }
137944d93782SGreg Clayton 
1380b9c1b51eSKate Stone   MenuActionResult Action() {
138105097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
138205097246SAdrian Prantl     // delegate, and if not, try our parent menu
138344d93782SGreg Clayton     return ActionPrivate(*this);
138444d93782SGreg Clayton   }
138544d93782SGreg Clayton 
1386b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
138744d93782SGreg Clayton 
1388b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
138944d93782SGreg Clayton 
1390b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
139144d93782SGreg Clayton 
1392b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
139344d93782SGreg Clayton 
1394b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
139544d93782SGreg Clayton 
1396b9c1b51eSKate Stone   Type GetType() const { return m_type; }
139744d93782SGreg Clayton 
1398b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
139944d93782SGreg Clayton 
1400b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
140144d93782SGreg Clayton 
1402b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
140344d93782SGreg Clayton 
1404b9c1b51eSKate Stone   void SetKeyValue(int key_value) { m_key_value = key_value; }
140544d93782SGreg Clayton 
1406b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
140744d93782SGreg Clayton 
1408b9c1b51eSKate Stone   std::string &GetKeyName() { return m_key_name; }
140944d93782SGreg Clayton 
1410b9c1b51eSKate Stone   int GetDrawWidth() const {
141144d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
141244d93782SGreg Clayton   }
141344d93782SGreg Clayton 
1414b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
141544d93782SGreg Clayton 
1416b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
141744d93782SGreg Clayton 
141844d93782SGreg Clayton protected:
141944d93782SGreg Clayton   std::string m_name;
142044d93782SGreg Clayton   std::string m_key_name;
142144d93782SGreg Clayton   uint64_t m_identifier;
142244d93782SGreg Clayton   Type m_type;
142344d93782SGreg Clayton   int m_key_value;
142444d93782SGreg Clayton   int m_start_col;
142544d93782SGreg Clayton   int m_max_submenu_name_length;
142644d93782SGreg Clayton   int m_max_submenu_key_name_length;
142744d93782SGreg Clayton   int m_selected;
142844d93782SGreg Clayton   Menu *m_parent;
142944d93782SGreg Clayton   Menus m_submenus;
143044d93782SGreg Clayton   WindowSP m_menu_window_sp;
143144d93782SGreg Clayton   MenuActionResult m_canned_result;
143244d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
143344d93782SGreg Clayton };
143444d93782SGreg Clayton 
143544d93782SGreg Clayton // Menubar or separator constructor
1436b9c1b51eSKate Stone Menu::Menu(Type type)
1437b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1438b9c1b51eSKate Stone       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() {}
144244d93782SGreg Clayton 
144344d93782SGreg Clayton // Menuitem constructor
1444b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1445b9c1b51eSKate Stone            uint64_t identifier)
1446b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1447b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1448b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1449b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1450b9c1b51eSKate Stone       m_delegate_sp() {
1451b9c1b51eSKate Stone   if (name && name[0]) {
145244d93782SGreg Clayton     m_name = name;
145344d93782SGreg Clayton     m_type = Type::Item;
145444d93782SGreg Clayton     if (key_name && key_name[0])
145544d93782SGreg Clayton       m_key_name = key_name;
1456b9c1b51eSKate Stone   } else {
145744d93782SGreg Clayton     m_type = Type::Separator;
145844d93782SGreg Clayton   }
145944d93782SGreg Clayton }
146044d93782SGreg Clayton 
1461b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
146244d93782SGreg Clayton   m_max_submenu_name_length = 0;
146344d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
146444d93782SGreg Clayton   Menus &submenus = GetSubmenus();
146544d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1466b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
146744d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14683985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
146944d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1470b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1471b9c1b51eSKate Stone         submenu->m_key_name.size())
147244d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
147344d93782SGreg Clayton   }
147444d93782SGreg Clayton }
147544d93782SGreg Clayton 
1476b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
147744d93782SGreg Clayton   menu_sp->m_parent = this;
14783985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
147944d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1480b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1481b9c1b51eSKate Stone       menu_sp->m_key_name.size())
148244d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
148344d93782SGreg Clayton   m_submenus.push_back(menu_sp);
148444d93782SGreg Clayton }
148544d93782SGreg Clayton 
1486b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1487b9c1b51eSKate Stone   if (m_type == Type::Separator) {
148844d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
148944d93782SGreg Clayton     window.PutChar(ACS_LTEE);
149044d93782SGreg Clayton     int width = window.GetWidth();
1491b9c1b51eSKate Stone     if (width > 2) {
149244d93782SGreg Clayton       width -= 2;
14933985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
149444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
149544d93782SGreg Clayton     }
149644d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1497b9c1b51eSKate Stone   } else {
149844d93782SGreg Clayton     const int shortcut_key = m_key_value;
149944d93782SGreg Clayton     bool underlined_shortcut = false;
150044d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
150144d93782SGreg Clayton     if (highlight)
150244d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1503b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
150444d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
150544d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
150644d93782SGreg Clayton       const char *name = m_name.c_str();
150744d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1508b9c1b51eSKate Stone       if (pos != std::string::npos) {
150944d93782SGreg Clayton         underlined_shortcut = true;
1510b9c1b51eSKate Stone         if (pos > 0) {
151144d93782SGreg Clayton           window.PutCString(name, pos);
151244d93782SGreg Clayton           name += pos;
151344d93782SGreg Clayton         }
151444d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
151544d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
151644d93782SGreg Clayton         window.PutChar(name[0]);
151744d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
151844d93782SGreg Clayton         name++;
151944d93782SGreg Clayton         if (name[0])
152044d93782SGreg Clayton           window.PutCString(name);
152144d93782SGreg Clayton       }
152244d93782SGreg Clayton     }
152344d93782SGreg Clayton 
1524b9c1b51eSKate Stone     if (!underlined_shortcut) {
152544d93782SGreg Clayton       window.PutCString(m_name.c_str());
152644d93782SGreg Clayton     }
152744d93782SGreg Clayton 
152844d93782SGreg Clayton     if (highlight)
152944d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
153044d93782SGreg Clayton 
1531b9c1b51eSKate Stone     if (m_key_name.empty()) {
1532b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
153344d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
153444d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
153544d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
153644d93782SGreg Clayton       }
1537b9c1b51eSKate Stone     } else {
153844d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
153944d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
154044d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
154144d93782SGreg Clayton     }
154244d93782SGreg Clayton   }
154344d93782SGreg Clayton }
154444d93782SGreg Clayton 
1545b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
154644d93782SGreg Clayton   Menus &submenus = GetSubmenus();
154744d93782SGreg Clayton   const size_t num_submenus = submenus.size();
154844d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
154944d93782SGreg Clayton   Menu::Type menu_type = GetType();
1550b9c1b51eSKate Stone   switch (menu_type) {
1551b9c1b51eSKate Stone   case Menu::Type::Bar: {
155244d93782SGreg Clayton     window.SetBackground(2);
155344d93782SGreg Clayton     window.MoveCursor(0, 0);
1554b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
155544d93782SGreg Clayton       Menu *menu = submenus[i].get();
155644d93782SGreg Clayton       if (i > 0)
155744d93782SGreg Clayton         window.PutChar(' ');
155844d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
155944d93782SGreg Clayton       window.PutCString("| ");
156044d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
156144d93782SGreg Clayton     }
156244d93782SGreg Clayton     window.PutCString(" |");
156344d93782SGreg Clayton     window.DeferredRefresh();
1564b9c1b51eSKate Stone   } break;
156544d93782SGreg Clayton 
1566b9c1b51eSKate Stone   case Menu::Type::Item: {
156744d93782SGreg Clayton     int y = 1;
156844d93782SGreg Clayton     int x = 3;
156944d93782SGreg Clayton     // Draw the menu
157044d93782SGreg Clayton     int cursor_x = 0;
157144d93782SGreg Clayton     int cursor_y = 0;
157244d93782SGreg Clayton     window.Erase();
157344d93782SGreg Clayton     window.SetBackground(2);
157444d93782SGreg Clayton     window.Box();
1575b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1576b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
157744d93782SGreg Clayton       window.MoveCursor(x, y + i);
1578b9c1b51eSKate Stone       if (is_selected) {
157944d93782SGreg Clayton         // Remember where we want the cursor to be
158044d93782SGreg Clayton         cursor_x = x - 1;
158144d93782SGreg Clayton         cursor_y = y + i;
158244d93782SGreg Clayton       }
158344d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
158444d93782SGreg Clayton     }
158544d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
158644d93782SGreg Clayton     window.DeferredRefresh();
1587b9c1b51eSKate Stone   } break;
158844d93782SGreg Clayton 
158944d93782SGreg Clayton   default:
159044d93782SGreg Clayton   case Menu::Type::Separator:
159144d93782SGreg Clayton     break;
159244d93782SGreg Clayton   }
159344d93782SGreg Clayton   return true; // Drawing handled...
159444d93782SGreg Clayton }
159544d93782SGreg Clayton 
1596b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
159744d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
159844d93782SGreg Clayton 
159944d93782SGreg Clayton   Menus &submenus = GetSubmenus();
160044d93782SGreg Clayton   const size_t num_submenus = submenus.size();
160144d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
160244d93782SGreg Clayton   Menu::Type menu_type = GetType();
1603b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
160444d93782SGreg Clayton     MenuSP run_menu_sp;
1605b9c1b51eSKate Stone     switch (key) {
160644d93782SGreg Clayton     case KEY_DOWN:
160744d93782SGreg Clayton     case KEY_UP:
160844d93782SGreg Clayton       // Show last menu or first menu
16093985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
161044d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
161144d93782SGreg Clayton       else if (!submenus.empty())
161244d93782SGreg Clayton         run_menu_sp = submenus.front();
161344d93782SGreg Clayton       result = eKeyHandled;
161444d93782SGreg Clayton       break;
161544d93782SGreg Clayton 
161644d93782SGreg Clayton     case KEY_RIGHT:
161744d93782SGreg Clayton       ++m_selected;
16183985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
161944d93782SGreg Clayton         m_selected = 0;
16203985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
162144d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
162244d93782SGreg Clayton       else if (!submenus.empty())
162344d93782SGreg Clayton         run_menu_sp = submenus.front();
162444d93782SGreg Clayton       result = eKeyHandled;
162544d93782SGreg Clayton       break;
162644d93782SGreg Clayton 
162744d93782SGreg Clayton     case KEY_LEFT:
162844d93782SGreg Clayton       --m_selected;
162944d93782SGreg Clayton       if (m_selected < 0)
163044d93782SGreg Clayton         m_selected = num_submenus - 1;
16313985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
163244d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
163344d93782SGreg Clayton       else if (!submenus.empty())
163444d93782SGreg Clayton         run_menu_sp = submenus.front();
163544d93782SGreg Clayton       result = eKeyHandled;
163644d93782SGreg Clayton       break;
163744d93782SGreg Clayton 
163844d93782SGreg Clayton     default:
1639b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1640b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
164144d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
164244d93782SGreg Clayton           run_menu_sp = submenus[i];
164344d93782SGreg Clayton           result = eKeyHandled;
164444d93782SGreg Clayton           break;
164544d93782SGreg Clayton         }
164644d93782SGreg Clayton       }
164744d93782SGreg Clayton       break;
164844d93782SGreg Clayton     }
164944d93782SGreg Clayton 
1650b9c1b51eSKate Stone     if (run_menu_sp) {
165105097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
165205097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
165305097246SAdrian Prantl       // decorations need to be calculated
165444d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
165544d93782SGreg Clayton         return eQuitApplication;
165644d93782SGreg Clayton 
165744d93782SGreg Clayton       Rect menu_bounds;
165844d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
165944d93782SGreg Clayton       menu_bounds.origin.y = 1;
166044d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
166144d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
166244d93782SGreg Clayton       if (m_menu_window_sp)
166344d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
166444d93782SGreg Clayton 
1665b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1666b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
166744d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
166844d93782SGreg Clayton     }
1669b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1670b9c1b51eSKate Stone     switch (key) {
167144d93782SGreg Clayton     case KEY_DOWN:
1672b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
167344d93782SGreg Clayton         const int start_select = m_selected;
1674b9c1b51eSKate Stone         while (++m_selected != start_select) {
16753985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
167644d93782SGreg Clayton             m_selected = 0;
167744d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
167844d93782SGreg Clayton             continue;
167944d93782SGreg Clayton           else
168044d93782SGreg Clayton             break;
168144d93782SGreg Clayton         }
168244d93782SGreg Clayton         return eKeyHandled;
168344d93782SGreg Clayton       }
168444d93782SGreg Clayton       break;
168544d93782SGreg Clayton 
168644d93782SGreg Clayton     case KEY_UP:
1687b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
168844d93782SGreg Clayton         const int start_select = m_selected;
1689b9c1b51eSKate Stone         while (--m_selected != start_select) {
16903985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
169144d93782SGreg Clayton             m_selected = num_submenus - 1;
169244d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
169344d93782SGreg Clayton             continue;
169444d93782SGreg Clayton           else
169544d93782SGreg Clayton             break;
169644d93782SGreg Clayton         }
169744d93782SGreg Clayton         return eKeyHandled;
169844d93782SGreg Clayton       }
169944d93782SGreg Clayton       break;
170044d93782SGreg Clayton 
170144d93782SGreg Clayton     case KEY_RETURN:
1702b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
170344d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
170444d93782SGreg Clayton           return eQuitApplication;
170544d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
170644d93782SGreg Clayton         return eKeyHandled;
170744d93782SGreg Clayton       }
170844d93782SGreg Clayton       break;
170944d93782SGreg Clayton 
1710b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1711b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
171244d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
171344d93782SGreg Clayton       return eKeyHandled;
171444d93782SGreg Clayton 
171544d93782SGreg Clayton     default:
1716b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
171744d93782SGreg Clayton         Menu *menu = submenus[i].get();
1718b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
171944d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
172044d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
172144d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
172244d93782SGreg Clayton             return eQuitApplication;
172344d93782SGreg Clayton           return eKeyHandled;
172444d93782SGreg Clayton         }
172544d93782SGreg Clayton       }
172644d93782SGreg Clayton       break;
172744d93782SGreg Clayton     }
1728b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
172944d93782SGreg Clayton   }
173044d93782SGreg Clayton   return result;
173144d93782SGreg Clayton }
173244d93782SGreg Clayton 
1733b9c1b51eSKate Stone class Application {
173444d93782SGreg Clayton public:
1735b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1736b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
173744d93782SGreg Clayton 
1738b9c1b51eSKate Stone   ~Application() {
173944d93782SGreg Clayton     m_window_delegates.clear();
174044d93782SGreg Clayton     m_window_sp.reset();
1741b9c1b51eSKate Stone     if (m_screen) {
174244d93782SGreg Clayton       ::delscreen(m_screen);
1743c5dac77aSEugene Zelenko       m_screen = nullptr;
174444d93782SGreg Clayton     }
174544d93782SGreg Clayton   }
174644d93782SGreg Clayton 
1747b9c1b51eSKate Stone   void Initialize() {
174844d93782SGreg Clayton     ::setlocale(LC_ALL, "");
174944d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
1750c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
175144d93782SGreg Clayton     ::start_color();
175244d93782SGreg Clayton     ::curs_set(0);
175344d93782SGreg Clayton     ::noecho();
175444d93782SGreg Clayton     ::keypad(stdscr, TRUE);
175544d93782SGreg Clayton   }
175644d93782SGreg Clayton 
1757b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
175844d93782SGreg Clayton 
1759b9c1b51eSKate Stone   void Run(Debugger &debugger) {
176044d93782SGreg Clayton     bool done = false;
176144d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
176244d93782SGreg Clayton 
176305097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
176405097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
176505097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
176605097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
176705097246SAdrian Prantl     // to receive async process events while in this loop.
176844d93782SGreg Clayton 
1769b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1770b9c1b51eSKate Stone                                             // of seconds seconds when calling
1771b9c1b51eSKate Stone                                             // Window::GetChar()
177244d93782SGreg Clayton 
1773b9c1b51eSKate Stone     ListenerSP listener_sp(
1774b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
177544d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
177644d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
177744d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
177844d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
177944d93782SGreg Clayton 
178044d93782SGreg Clayton     bool update = true;
178144d93782SGreg Clayton #if defined(__APPLE__)
178244d93782SGreg Clayton     std::deque<int> escape_chars;
178344d93782SGreg Clayton #endif
178444d93782SGreg Clayton 
1785b9c1b51eSKate Stone     while (!done) {
1786b9c1b51eSKate Stone       if (update) {
178744d93782SGreg Clayton         m_window_sp->Draw(false);
178805097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
178905097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
179005097246SAdrian Prantl         // blinking
179144d93782SGreg Clayton         update_panels();
179244d93782SGreg Clayton 
1793b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1794b9c1b51eSKate Stone         // corner
179544d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
179644d93782SGreg Clayton 
179744d93782SGreg Clayton         doupdate();
179844d93782SGreg Clayton         update = false;
179944d93782SGreg Clayton       }
180044d93782SGreg Clayton 
180144d93782SGreg Clayton #if defined(__APPLE__)
180205097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
180305097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1804b9c1b51eSKate Stone       // possible
180544d93782SGreg Clayton       int ch;
180644d93782SGreg Clayton       if (escape_chars.empty())
180744d93782SGreg Clayton         ch = m_window_sp->GetChar();
1808b9c1b51eSKate Stone       else {
180944d93782SGreg Clayton         ch = escape_chars.front();
181044d93782SGreg Clayton         escape_chars.pop_front();
181144d93782SGreg Clayton       }
1812b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
181344d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1814b9c1b51eSKate Stone         if (ch2 == 'O') {
181544d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1816b9c1b51eSKate Stone           switch (ch3) {
1817b9c1b51eSKate Stone           case 'P':
1818b9c1b51eSKate Stone             ch = KEY_F(1);
1819b9c1b51eSKate Stone             break;
1820b9c1b51eSKate Stone           case 'Q':
1821b9c1b51eSKate Stone             ch = KEY_F(2);
1822b9c1b51eSKate Stone             break;
1823b9c1b51eSKate Stone           case 'R':
1824b9c1b51eSKate Stone             ch = KEY_F(3);
1825b9c1b51eSKate Stone             break;
1826b9c1b51eSKate Stone           case 'S':
1827b9c1b51eSKate Stone             ch = KEY_F(4);
1828b9c1b51eSKate Stone             break;
182944d93782SGreg Clayton           default:
183044d93782SGreg Clayton             escape_chars.push_back(ch2);
183144d93782SGreg Clayton             if (ch3 != -1)
183244d93782SGreg Clayton               escape_chars.push_back(ch3);
183344d93782SGreg Clayton             break;
183444d93782SGreg Clayton           }
1835b9c1b51eSKate Stone         } else if (ch2 != -1)
183644d93782SGreg Clayton           escape_chars.push_back(ch2);
183744d93782SGreg Clayton       }
183844d93782SGreg Clayton #else
183944d93782SGreg Clayton       int ch = m_window_sp->GetChar();
184044d93782SGreg Clayton 
184144d93782SGreg Clayton #endif
1842b9c1b51eSKate Stone       if (ch == -1) {
1843b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
184444d93782SGreg Clayton           done = true;
1845b9c1b51eSKate Stone         } else {
184644d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
184744d93782SGreg Clayton           EventSP event_sp;
1848b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1849d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
185044d93782SGreg Clayton 
1851b9c1b51eSKate Stone             if (event_sp) {
185244d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1853b9c1b51eSKate Stone               if (broadcaster) {
185444d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1855b9c1b51eSKate Stone                 ConstString broadcaster_class(
1856b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1857b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1858b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1859b9c1b51eSKate Stone                       nullptr);
186044d93782SGreg Clayton                   update = true;
186144d93782SGreg Clayton                   continue; // Don't get any key, just update our view
186244d93782SGreg Clayton                 }
186344d93782SGreg Clayton               }
186444d93782SGreg Clayton             }
186544d93782SGreg Clayton           }
186644d93782SGreg Clayton         }
1867b9c1b51eSKate Stone       } else {
186844d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1869b9c1b51eSKate Stone         switch (key_result) {
187044d93782SGreg Clayton         case eKeyHandled:
1871c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
187244d93782SGreg Clayton           update = true;
187344d93782SGreg Clayton           break;
187444d93782SGreg Clayton         case eKeyNotHandled:
187544d93782SGreg Clayton           break;
187644d93782SGreg Clayton         case eQuitApplication:
187744d93782SGreg Clayton           done = true;
187844d93782SGreg Clayton           break;
187944d93782SGreg Clayton         }
188044d93782SGreg Clayton       }
188144d93782SGreg Clayton     }
188244d93782SGreg Clayton 
188344d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
188444d93782SGreg Clayton   }
188544d93782SGreg Clayton 
1886b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
188744d93782SGreg Clayton     if (!m_window_sp)
1888796ac80bSJonas Devlieghere       m_window_sp = std::make_shared<Window>("main", stdscr, false);
188944d93782SGreg Clayton     return m_window_sp;
189044d93782SGreg Clayton   }
189144d93782SGreg Clayton 
1892b9c1b51eSKate Stone   WindowDelegates &GetWindowDelegates() { return m_window_delegates; }
189344d93782SGreg Clayton 
189444d93782SGreg Clayton protected:
189544d93782SGreg Clayton   WindowSP m_window_sp;
189644d93782SGreg Clayton   WindowDelegates m_window_delegates;
189744d93782SGreg Clayton   SCREEN *m_screen;
189844d93782SGreg Clayton   FILE *m_in;
189944d93782SGreg Clayton   FILE *m_out;
190044d93782SGreg Clayton };
190144d93782SGreg Clayton 
190244d93782SGreg Clayton } // namespace curses
190344d93782SGreg Clayton 
190444d93782SGreg Clayton using namespace curses;
190544d93782SGreg Clayton 
1906b9c1b51eSKate Stone struct Row {
19078369b28dSGreg Clayton   ValueObjectManager value;
190844d93782SGreg Clayton   Row *parent;
19098369b28dSGreg Clayton   // The process stop ID when the children were calculated.
19108369b28dSGreg Clayton   uint32_t children_stop_id;
191144d93782SGreg Clayton   int row_idx;
191244d93782SGreg Clayton   int x;
191344d93782SGreg Clayton   int y;
191444d93782SGreg Clayton   bool might_have_children;
191544d93782SGreg Clayton   bool expanded;
191644d93782SGreg Clayton   bool calculated_children;
191744d93782SGreg Clayton   std::vector<Row> children;
191844d93782SGreg Clayton 
1919b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
19208369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
19218369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1922b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
192344d93782SGreg Clayton 
1924b9c1b51eSKate Stone   size_t GetDepth() const {
192544d93782SGreg Clayton     if (parent)
192644d93782SGreg Clayton       return 1 + parent->GetDepth();
192744d93782SGreg Clayton     return 0;
192844d93782SGreg Clayton   }
192944d93782SGreg Clayton 
1930b9c1b51eSKate Stone   void Expand() {
193144d93782SGreg Clayton     expanded = true;
19328369b28dSGreg Clayton   }
19338369b28dSGreg Clayton 
19348369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
19358369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
19368369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
19378369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
19388369b28dSGreg Clayton       children_stop_id = stop_id;
19398369b28dSGreg Clayton       calculated_children = false;
19408369b28dSGreg Clayton     }
1941b9c1b51eSKate Stone     if (!calculated_children) {
19428369b28dSGreg Clayton       children.clear();
194344d93782SGreg Clayton       calculated_children = true;
19448369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1945b9c1b51eSKate Stone       if (valobj) {
194644d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1947b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
194844d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
194944d93782SGreg Clayton         }
195044d93782SGreg Clayton       }
195144d93782SGreg Clayton     }
19528369b28dSGreg Clayton     return children;
195344d93782SGreg Clayton   }
195444d93782SGreg Clayton 
19558369b28dSGreg Clayton   void Unexpand() {
19568369b28dSGreg Clayton     expanded = false;
19578369b28dSGreg Clayton     calculated_children = false;
19588369b28dSGreg Clayton     children.clear();
19598369b28dSGreg Clayton   }
196044d93782SGreg Clayton 
1961b9c1b51eSKate Stone   void DrawTree(Window &window) {
196244d93782SGreg Clayton     if (parent)
196344d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
196444d93782SGreg Clayton 
1965b9c1b51eSKate Stone     if (might_have_children) {
196605097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
196705097246SAdrian Prantl       // "symbol" UTF8 string below
196844d93782SGreg Clayton       //            const char *symbol = "";
196944d93782SGreg Clayton       //            if (row.expanded)
197044d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
197144d93782SGreg Clayton       //            else
197244d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
197344d93782SGreg Clayton       //            window.PutCString (symbol);
197444d93782SGreg Clayton 
197505097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
197605097246SAdrian Prantl       // or '>' character...
197744d93782SGreg Clayton       //            if (expanded)
197844d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
197944d93782SGreg Clayton       //            else
198044d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
198105097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
198205097246SAdrian Prantl       // just use a diamond...
198344d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
198444d93782SGreg Clayton       window.PutChar(ACS_HLINE);
198544d93782SGreg Clayton     }
198644d93782SGreg Clayton   }
198744d93782SGreg Clayton 
1988b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
198944d93782SGreg Clayton     if (parent)
199044d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
199144d93782SGreg Clayton 
19928369b28dSGreg Clayton     if (&GetChildren().back() == child) {
199344d93782SGreg Clayton       // Last child
1994b9c1b51eSKate Stone       if (reverse_depth == 0) {
199544d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
199644d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1997b9c1b51eSKate Stone       } else {
199844d93782SGreg Clayton         window.PutChar(' ');
199944d93782SGreg Clayton         window.PutChar(' ');
200044d93782SGreg Clayton       }
2001b9c1b51eSKate Stone     } else {
2002b9c1b51eSKate Stone       if (reverse_depth == 0) {
200344d93782SGreg Clayton         window.PutChar(ACS_LTEE);
200444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2005b9c1b51eSKate Stone       } else {
200644d93782SGreg Clayton         window.PutChar(ACS_VLINE);
200744d93782SGreg Clayton         window.PutChar(' ');
200844d93782SGreg Clayton       }
200944d93782SGreg Clayton     }
201044d93782SGreg Clayton   }
201144d93782SGreg Clayton };
201244d93782SGreg Clayton 
2013b9c1b51eSKate Stone struct DisplayOptions {
201444d93782SGreg Clayton   bool show_types;
201544d93782SGreg Clayton };
201644d93782SGreg Clayton 
201744d93782SGreg Clayton class TreeItem;
201844d93782SGreg Clayton 
2019b9c1b51eSKate Stone class TreeDelegate {
202044d93782SGreg Clayton public:
2021c5dac77aSEugene Zelenko   TreeDelegate() = default;
2022315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
2023315b6884SEugene Zelenko 
202444d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
202544d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2026b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2027b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
202844d93782SGreg Clayton };
2029315b6884SEugene Zelenko 
203044d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
203144d93782SGreg Clayton 
2032b9c1b51eSKate Stone class TreeItem {
203344d93782SGreg Clayton public:
2034b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2035b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2036b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2037b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
203844d93782SGreg Clayton 
2039b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2040b9c1b51eSKate Stone     if (this != &rhs) {
204144d93782SGreg Clayton       m_parent = rhs.m_parent;
204244d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2043ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
204444d93782SGreg Clayton       m_identifier = rhs.m_identifier;
204544d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
204644d93782SGreg Clayton       m_children = rhs.m_children;
204744d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
204844d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
204944d93782SGreg Clayton     }
205044d93782SGreg Clayton     return *this;
205144d93782SGreg Clayton   }
205244d93782SGreg Clayton 
2053b9c1b51eSKate Stone   size_t GetDepth() const {
205444d93782SGreg Clayton     if (m_parent)
205544d93782SGreg Clayton       return 1 + m_parent->GetDepth();
205644d93782SGreg Clayton     return 0;
205744d93782SGreg Clayton   }
205844d93782SGreg Clayton 
2059b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
206044d93782SGreg Clayton 
2061b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
206244d93782SGreg Clayton 
2063b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
206444d93782SGreg Clayton 
2065b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
206644d93782SGreg Clayton 
2067b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
206844d93782SGreg Clayton 
2069b9c1b51eSKate Stone   size_t GetNumChildren() {
207044d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
207144d93782SGreg Clayton     return m_children.size();
207244d93782SGreg Clayton   }
207344d93782SGreg Clayton 
2074b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2075315b6884SEugene Zelenko 
2076b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
207744d93782SGreg Clayton     SetRowIndex(row_idx);
207844d93782SGreg Clayton     ++row_idx;
207944d93782SGreg Clayton 
2080ec990867SGreg Clayton     const bool expanded = IsExpanded();
2081ec990867SGreg Clayton 
208205097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
208305097246SAdrian Prantl     // number of children if the item is expanded
2084c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
208544d93782SGreg Clayton       GetNumChildren();
208644d93782SGreg Clayton 
2087b9c1b51eSKate Stone     for (auto &item : m_children) {
208844d93782SGreg Clayton       if (expanded)
208944d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
209044d93782SGreg Clayton       else
209144d93782SGreg Clayton         item.SetRowIndex(-1);
209244d93782SGreg Clayton     }
209344d93782SGreg Clayton   }
209444d93782SGreg Clayton 
2095b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
209644d93782SGreg Clayton 
2097b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
209844d93782SGreg Clayton 
2099b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
210044d93782SGreg Clayton 
2101b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
210244d93782SGreg Clayton 
2103b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2104b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
210544d93782SGreg Clayton     if (num_rows_left <= 0)
210644d93782SGreg Clayton       return false;
210744d93782SGreg Clayton 
2108b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
210944d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
211044d93782SGreg Clayton 
211144d93782SGreg Clayton       if (m_parent)
211244d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
211344d93782SGreg Clayton 
2114b9c1b51eSKate Stone       if (m_might_have_children) {
2115b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
211605097246SAdrian Prantl         // "symbol" UTF8 string below
211744d93782SGreg Clayton         //            const char *symbol = "";
211844d93782SGreg Clayton         //            if (row.expanded)
211944d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
212044d93782SGreg Clayton         //            else
212144d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
212244d93782SGreg Clayton         //            window.PutCString (symbol);
212344d93782SGreg Clayton 
212444d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
212544d93782SGreg Clayton         // 'v' or '>' character...
212644d93782SGreg Clayton         //            if (expanded)
212744d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
212844d93782SGreg Clayton         //            else
212944d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
213005097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
213105097246SAdrian Prantl         // just use a diamond...
213244d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
213344d93782SGreg Clayton         window.PutChar(ACS_HLINE);
213444d93782SGreg Clayton       }
2135b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2136b9c1b51eSKate Stone                        window.IsActive();
213744d93782SGreg Clayton 
213844d93782SGreg Clayton       if (highlight)
213944d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
214044d93782SGreg Clayton 
214144d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
214244d93782SGreg Clayton 
214344d93782SGreg Clayton       if (highlight)
214444d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
214544d93782SGreg Clayton       ++row_idx;
214644d93782SGreg Clayton       --num_rows_left;
214744d93782SGreg Clayton     }
214844d93782SGreg Clayton 
214944d93782SGreg Clayton     if (num_rows_left <= 0)
215044d93782SGreg Clayton       return false; // We are done drawing...
215144d93782SGreg Clayton 
2152b9c1b51eSKate Stone     if (IsExpanded()) {
2153b9c1b51eSKate Stone       for (auto &item : m_children) {
215405097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
215505097246SAdrian Prantl         // done drawing and can exit this for loop
2156b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2157b9c1b51eSKate Stone                        num_rows_left))
215844d93782SGreg Clayton           break;
215944d93782SGreg Clayton       }
216044d93782SGreg Clayton     }
216144d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
216244d93782SGreg Clayton   }
216344d93782SGreg Clayton 
2164b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2165b9c1b51eSKate Stone                         uint32_t reverse_depth) {
216644d93782SGreg Clayton     if (m_parent)
216744d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
216844d93782SGreg Clayton 
2169b9c1b51eSKate Stone     if (&m_children.back() == child) {
217044d93782SGreg Clayton       // Last child
2171b9c1b51eSKate Stone       if (reverse_depth == 0) {
217244d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
217344d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2174b9c1b51eSKate Stone       } else {
217544d93782SGreg Clayton         window.PutChar(' ');
217644d93782SGreg Clayton         window.PutChar(' ');
217744d93782SGreg Clayton       }
2178b9c1b51eSKate Stone     } else {
2179b9c1b51eSKate Stone       if (reverse_depth == 0) {
218044d93782SGreg Clayton         window.PutChar(ACS_LTEE);
218144d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2182b9c1b51eSKate Stone       } else {
218344d93782SGreg Clayton         window.PutChar(ACS_VLINE);
218444d93782SGreg Clayton         window.PutChar(' ');
218544d93782SGreg Clayton       }
218644d93782SGreg Clayton     }
218744d93782SGreg Clayton   }
218844d93782SGreg Clayton 
2189b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21903985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
219144d93782SGreg Clayton       return this;
219244d93782SGreg Clayton     if (m_children.empty())
2193c5dac77aSEugene Zelenko       return nullptr;
2194b9c1b51eSKate Stone     if (IsExpanded()) {
2195b9c1b51eSKate Stone       for (auto &item : m_children) {
219644d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
219744d93782SGreg Clayton         if (selected_item_ptr)
219844d93782SGreg Clayton           return selected_item_ptr;
219944d93782SGreg Clayton       }
220044d93782SGreg Clayton     }
2201c5dac77aSEugene Zelenko     return nullptr;
220244d93782SGreg Clayton   }
220344d93782SGreg Clayton 
2204b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2205ec990867SGreg Clayton 
2206b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2207ec990867SGreg Clayton 
2208b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
220944d93782SGreg Clayton 
2210b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
221144d93782SGreg Clayton 
2212b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2213ec990867SGreg Clayton 
221444d93782SGreg Clayton protected:
221544d93782SGreg Clayton   TreeItem *m_parent;
221644d93782SGreg Clayton   TreeDelegate &m_delegate;
2217ec990867SGreg Clayton   void *m_user_data;
221844d93782SGreg Clayton   uint64_t m_identifier;
2219b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2220b9c1b51eSKate Stone                  // root item
222144d93782SGreg Clayton   std::vector<TreeItem> m_children;
222244d93782SGreg Clayton   bool m_might_have_children;
222344d93782SGreg Clayton   bool m_is_expanded;
222444d93782SGreg Clayton };
222544d93782SGreg Clayton 
2226b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
222744d93782SGreg Clayton public:
2228b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2229b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2230b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2231b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2232b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
223344d93782SGreg Clayton 
2234b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
223544d93782SGreg Clayton 
2236b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2237b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2238b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
223944d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
224044d93782SGreg Clayton 
224144d93782SGreg Clayton     bool display_content = false;
2242b9c1b51eSKate Stone     if (process) {
224344d93782SGreg Clayton       StateType state = process->GetState();
2244b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
224544d93782SGreg Clayton         // We are stopped, so it is ok to
224644d93782SGreg Clayton         display_content = true;
2247b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
224844d93782SGreg Clayton         return true; // Don't do any updating when we are running
224944d93782SGreg Clayton       }
225044d93782SGreg Clayton     }
225144d93782SGreg Clayton 
225244d93782SGreg Clayton     m_min_x = 2;
225344d93782SGreg Clayton     m_min_y = 1;
225444d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
225544d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
225644d93782SGreg Clayton 
225744d93782SGreg Clayton     window.Erase();
225844d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
225944d93782SGreg Clayton 
2260b9c1b51eSKate Stone     if (display_content) {
226144d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
226244d93782SGreg Clayton       m_num_rows = 0;
226344d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
226444d93782SGreg Clayton 
226505097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
226605097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
226705097246SAdrian Prantl       // rows by setting the first visible row accordingly.
226844d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
226944d93782SGreg Clayton         m_first_visible_row = 0;
227044d93782SGreg Clayton 
227144d93782SGreg Clayton       // Make sure the selected row is always visible
227244d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
227344d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
227444d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
227544d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
227644d93782SGreg Clayton 
227744d93782SGreg Clayton       int row_idx = 0;
227844d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2279b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2280b9c1b51eSKate Stone                   num_rows_left);
228144d93782SGreg Clayton       // Get the selected row
228244d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2283b9c1b51eSKate Stone     } else {
2284c5dac77aSEugene Zelenko       m_selected_item = nullptr;
228544d93782SGreg Clayton     }
228644d93782SGreg Clayton 
228744d93782SGreg Clayton     window.DeferredRefresh();
228844d93782SGreg Clayton 
228944d93782SGreg Clayton     return true; // Drawing handled
229044d93782SGreg Clayton   }
229144d93782SGreg Clayton 
2292b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
229344d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
229444d93782SGreg Clayton   }
229544d93782SGreg Clayton 
2296b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
229744d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
229844d93782SGreg Clayton         {KEY_UP, "Select previous item"},
229944d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
230044d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2301b9c1b51eSKate Stone         {KEY_LEFT,
2302b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
230344d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
230444d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
230544d93782SGreg Clayton         {'h', "Show help dialog"},
230644d93782SGreg Clayton         {' ', "Toggle item expansion"},
230744d93782SGreg Clayton         {',', "Page up"},
230844d93782SGreg Clayton         {'.', "Page down"},
2309b9c1b51eSKate Stone         {'\0', nullptr}};
231044d93782SGreg Clayton     return g_source_view_key_help;
231144d93782SGreg Clayton   }
231244d93782SGreg Clayton 
2313b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2314b9c1b51eSKate Stone     switch (c) {
231544d93782SGreg Clayton     case ',':
231644d93782SGreg Clayton     case KEY_PPAGE:
231744d93782SGreg Clayton       // Page up key
2318b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
231944d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
232044d93782SGreg Clayton           m_first_visible_row -= m_max_y;
232144d93782SGreg Clayton         else
232244d93782SGreg Clayton           m_first_visible_row = 0;
232344d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
232444d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
232544d93782SGreg Clayton         if (m_selected_item)
232644d93782SGreg Clayton           m_selected_item->ItemWasSelected();
232744d93782SGreg Clayton       }
232844d93782SGreg Clayton       return eKeyHandled;
232944d93782SGreg Clayton 
233044d93782SGreg Clayton     case '.':
233144d93782SGreg Clayton     case KEY_NPAGE:
233244d93782SGreg Clayton       // Page down key
2333b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2334b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
233544d93782SGreg Clayton           m_first_visible_row += m_max_y;
233644d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
233744d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
233844d93782SGreg Clayton           if (m_selected_item)
233944d93782SGreg Clayton             m_selected_item->ItemWasSelected();
234044d93782SGreg Clayton         }
234144d93782SGreg Clayton       }
234244d93782SGreg Clayton       return eKeyHandled;
234344d93782SGreg Clayton 
234444d93782SGreg Clayton     case KEY_UP:
2345b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
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;
2352315b6884SEugene Zelenko 
235344d93782SGreg Clayton     case KEY_DOWN:
2354b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
235544d93782SGreg Clayton         ++m_selected_row_idx;
235644d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
235744d93782SGreg Clayton         if (m_selected_item)
235844d93782SGreg Clayton           m_selected_item->ItemWasSelected();
235944d93782SGreg Clayton       }
236044d93782SGreg Clayton       return eKeyHandled;
236144d93782SGreg Clayton 
236244d93782SGreg Clayton     case KEY_RIGHT:
2363b9c1b51eSKate Stone       if (m_selected_item) {
236444d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
236544d93782SGreg Clayton           m_selected_item->Expand();
236644d93782SGreg Clayton       }
236744d93782SGreg Clayton       return eKeyHandled;
236844d93782SGreg Clayton 
236944d93782SGreg Clayton     case KEY_LEFT:
2370b9c1b51eSKate Stone       if (m_selected_item) {
237144d93782SGreg Clayton         if (m_selected_item->IsExpanded())
237244d93782SGreg Clayton           m_selected_item->Unexpand();
2373b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
237444d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
237544d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
237644d93782SGreg Clayton           if (m_selected_item)
237744d93782SGreg Clayton             m_selected_item->ItemWasSelected();
237844d93782SGreg Clayton         }
237944d93782SGreg Clayton       }
238044d93782SGreg Clayton       return eKeyHandled;
238144d93782SGreg Clayton 
238244d93782SGreg Clayton     case ' ':
238344d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2384b9c1b51eSKate Stone       if (m_selected_item) {
238544d93782SGreg Clayton         if (m_selected_item->IsExpanded())
238644d93782SGreg Clayton           m_selected_item->Unexpand();
238744d93782SGreg Clayton         else
238844d93782SGreg Clayton           m_selected_item->Expand();
238944d93782SGreg Clayton       }
239044d93782SGreg Clayton       return eKeyHandled;
239144d93782SGreg Clayton 
239244d93782SGreg Clayton     case 'h':
239344d93782SGreg Clayton       window.CreateHelpSubwindow();
239444d93782SGreg Clayton       return eKeyHandled;
239544d93782SGreg Clayton 
239644d93782SGreg Clayton     default:
239744d93782SGreg Clayton       break;
239844d93782SGreg Clayton     }
239944d93782SGreg Clayton     return eKeyNotHandled;
240044d93782SGreg Clayton   }
240144d93782SGreg Clayton 
240244d93782SGreg Clayton protected:
240344d93782SGreg Clayton   Debugger &m_debugger;
240444d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
240544d93782SGreg Clayton   TreeItem m_root;
240644d93782SGreg Clayton   TreeItem *m_selected_item;
240744d93782SGreg Clayton   int m_num_rows;
240844d93782SGreg Clayton   int m_selected_row_idx;
240944d93782SGreg Clayton   int m_first_visible_row;
241044d93782SGreg Clayton   int m_min_x;
241144d93782SGreg Clayton   int m_min_y;
241244d93782SGreg Clayton   int m_max_x;
241344d93782SGreg Clayton   int m_max_y;
241444d93782SGreg Clayton };
241544d93782SGreg Clayton 
2416b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
241744d93782SGreg Clayton public:
2418b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2419b9c1b51eSKate Stone     FormatEntity::Parse(
2420b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2421554f68d3SGreg Clayton         m_format);
242244d93782SGreg Clayton   }
242344d93782SGreg Clayton 
2424315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
242544d93782SGreg Clayton 
2426b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2427ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2428b9c1b51eSKate Stone     if (thread) {
242944d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2430ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2431b9c1b51eSKate Stone       if (frame_sp) {
243244d93782SGreg Clayton         StreamString strm;
2433b9c1b51eSKate Stone         const SymbolContext &sc =
2434b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
243544d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2436b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2437b9c1b51eSKate Stone                                  nullptr, false, false)) {
243844d93782SGreg Clayton           int right_pad = 1;
2439c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
244044d93782SGreg Clayton         }
244144d93782SGreg Clayton       }
244244d93782SGreg Clayton     }
244344d93782SGreg Clayton   }
2444315b6884SEugene Zelenko 
2445b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
244644d93782SGreg Clayton     // No children for frames yet...
244744d93782SGreg Clayton   }
244844d93782SGreg Clayton 
2449b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2450ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2451b9c1b51eSKate Stone     if (thread) {
2452b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2453b9c1b51eSKate Stone           thread->GetID());
245444d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2455ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
245644d93782SGreg Clayton       return true;
245744d93782SGreg Clayton     }
245844d93782SGreg Clayton     return false;
245944d93782SGreg Clayton   }
2460315b6884SEugene Zelenko 
2461554f68d3SGreg Clayton protected:
2462554f68d3SGreg Clayton   FormatEntity::Entry m_format;
246344d93782SGreg Clayton };
246444d93782SGreg Clayton 
2465b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
246644d93782SGreg Clayton public:
2467b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2468b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2469b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2470b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2471b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2472554f68d3SGreg Clayton                         m_format);
247344d93782SGreg Clayton   }
247444d93782SGreg Clayton 
2475315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
247644d93782SGreg Clayton 
2477b9c1b51eSKate Stone   ProcessSP GetProcess() {
2478b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2479b9c1b51eSKate Stone         .GetExecutionContext()
2480b9c1b51eSKate Stone         .GetProcessSP();
2481ec990867SGreg Clayton   }
2482ec990867SGreg Clayton 
2483b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2484ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2485ec990867SGreg Clayton     if (process_sp)
2486ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2487ec990867SGreg Clayton     return ThreadSP();
2488ec990867SGreg Clayton   }
2489ec990867SGreg Clayton 
2490b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2491ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2492b9c1b51eSKate Stone     if (thread_sp) {
249344d93782SGreg Clayton       StreamString strm;
249444d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2495b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2496b9c1b51eSKate Stone                                nullptr, false, false)) {
249744d93782SGreg Clayton         int right_pad = 1;
2498c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
249944d93782SGreg Clayton       }
250044d93782SGreg Clayton     }
250144d93782SGreg Clayton   }
2502315b6884SEugene Zelenko 
2503b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2504ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2505b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
250644d93782SGreg Clayton       StateType state = process_sp->GetState();
2507b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2508ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2509b9c1b51eSKate Stone         if (thread_sp) {
2510b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2511b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
251244d93782SGreg Clayton             return; // Children are already up to date
2513b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
251444d93782SGreg Clayton             // Always expand the thread item the first time we show it
2515796ac80bSJonas Devlieghere             m_frame_delegate_sp = std::make_shared<FrameTreeDelegate>();
251644d93782SGreg Clayton           }
251744d93782SGreg Clayton 
251844d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
251944d93782SGreg Clayton           m_tid = thread_sp->GetID();
252044d93782SGreg Clayton 
252144d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
252244d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
252344d93782SGreg Clayton           item.Resize(num_frames, t);
2524b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2525ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
252644d93782SGreg Clayton             item[i].SetIdentifier(i);
252744d93782SGreg Clayton           }
252844d93782SGreg Clayton         }
252944d93782SGreg Clayton         return;
253044d93782SGreg Clayton       }
253144d93782SGreg Clayton     }
253244d93782SGreg Clayton     item.ClearChildren();
253344d93782SGreg Clayton   }
253444d93782SGreg Clayton 
2535b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2536ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2537b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2538ec990867SGreg Clayton       StateType state = process_sp->GetState();
2539b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2540ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2541b9c1b51eSKate Stone         if (thread_sp) {
254244d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2543bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
254444d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2545b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
254644d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
254744d93782SGreg Clayton             return true;
254844d93782SGreg Clayton           }
254944d93782SGreg Clayton         }
2550ec990867SGreg Clayton       }
2551ec990867SGreg Clayton     }
255244d93782SGreg Clayton     return false;
255344d93782SGreg Clayton   }
255444d93782SGreg Clayton 
255544d93782SGreg Clayton protected:
255644d93782SGreg Clayton   Debugger &m_debugger;
255744d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
255844d93782SGreg Clayton   lldb::user_id_t m_tid;
255944d93782SGreg Clayton   uint32_t m_stop_id;
2560554f68d3SGreg Clayton   FormatEntity::Entry m_format;
256144d93782SGreg Clayton };
256244d93782SGreg Clayton 
2563b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2564ec990867SGreg Clayton public:
2565b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2566b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2567b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2568554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2569554f68d3SGreg Clayton                         m_format);
2570ec990867SGreg Clayton   }
2571ec990867SGreg Clayton 
2572315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2573ec990867SGreg Clayton 
2574b9c1b51eSKate Stone   ProcessSP GetProcess() {
2575b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2576b9c1b51eSKate Stone         .GetExecutionContext()
2577b9c1b51eSKate Stone         .GetProcessSP();
2578ec990867SGreg Clayton   }
2579ec990867SGreg Clayton 
2580b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2581ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2582b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2583ec990867SGreg Clayton       StreamString strm;
2584ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2585b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2586b9c1b51eSKate Stone                                nullptr, false, false)) {
2587ec990867SGreg Clayton         int right_pad = 1;
2588c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2589ec990867SGreg Clayton       }
2590ec990867SGreg Clayton     }
2591ec990867SGreg Clayton   }
2592ec990867SGreg Clayton 
2593b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2594ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2595b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2596ec990867SGreg Clayton       StateType state = process_sp->GetState();
2597b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2598ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2599ec990867SGreg Clayton         if (m_stop_id == stop_id)
2600ec990867SGreg Clayton           return; // Children are already up to date
2601ec990867SGreg Clayton 
2602ec990867SGreg Clayton         m_stop_id = stop_id;
2603ec990867SGreg Clayton 
2604b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2605ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2606ec990867SGreg Clayton           // item.Expand();
2607796ac80bSJonas Devlieghere           m_thread_delegate_sp =
2608796ac80bSJonas Devlieghere               std::make_shared<ThreadTreeDelegate>(m_debugger);
2609ec990867SGreg Clayton         }
2610ec990867SGreg Clayton 
2611ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2612ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2613bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2614ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2615ec990867SGreg Clayton         item.Resize(num_threads, t);
2616b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2617ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2618ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2619ec990867SGreg Clayton         }
2620ec990867SGreg Clayton         return;
2621ec990867SGreg Clayton       }
2622ec990867SGreg Clayton     }
2623ec990867SGreg Clayton     item.ClearChildren();
2624ec990867SGreg Clayton   }
2625ec990867SGreg Clayton 
2626b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2627ec990867SGreg Clayton 
2628ec990867SGreg Clayton protected:
2629ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2630ec990867SGreg Clayton   Debugger &m_debugger;
2631ec990867SGreg Clayton   uint32_t m_stop_id;
2632554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2633ec990867SGreg Clayton };
2634ec990867SGreg Clayton 
2635b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
263644d93782SGreg Clayton public:
2637b9c1b51eSKate Stone   ValueObjectListDelegate()
26388369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2639b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2640b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {}
264144d93782SGreg Clayton 
2642b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
26438369b28dSGreg Clayton       : m_rows(), m_selected_row(nullptr),
2644b9c1b51eSKate Stone         m_selected_row_idx(0), m_first_visible_row(0), m_num_rows(0),
2645b9c1b51eSKate Stone         m_max_x(0), m_max_y(0) {
264644d93782SGreg Clayton     SetValues(valobj_list);
264744d93782SGreg Clayton   }
264844d93782SGreg Clayton 
2649315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
265044d93782SGreg Clayton 
2651b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2652c5dac77aSEugene Zelenko     m_selected_row = nullptr;
265344d93782SGreg Clayton     m_selected_row_idx = 0;
265444d93782SGreg Clayton     m_first_visible_row = 0;
265544d93782SGreg Clayton     m_num_rows = 0;
265644d93782SGreg Clayton     m_rows.clear();
26578369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
26588369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
265944d93782SGreg Clayton   }
266044d93782SGreg Clayton 
2661b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
266244d93782SGreg Clayton     m_num_rows = 0;
266344d93782SGreg Clayton     m_min_x = 2;
266444d93782SGreg Clayton     m_min_y = 1;
266544d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
266644d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
266744d93782SGreg Clayton 
266844d93782SGreg Clayton     window.Erase();
266944d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
267044d93782SGreg Clayton 
267144d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
267244d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
267344d93782SGreg Clayton 
267405097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
267505097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
267605097246SAdrian Prantl     // rows by setting the first visible row accordingly.
267744d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
267844d93782SGreg Clayton       m_first_visible_row = 0;
267944d93782SGreg Clayton 
268044d93782SGreg Clayton     // Make sure the selected row is always visible
268144d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
268244d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
268344d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
268444d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
268544d93782SGreg Clayton 
268644d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
268744d93782SGreg Clayton 
268844d93782SGreg Clayton     window.DeferredRefresh();
268944d93782SGreg Clayton 
269044d93782SGreg Clayton     // Get the selected row
269144d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
269205097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
269305097246SAdrian Prantl     // always on the same line
269444d93782SGreg Clayton     if (m_selected_row)
2695b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
269644d93782SGreg Clayton 
269744d93782SGreg Clayton     return true; // Drawing handled
269844d93782SGreg Clayton   }
269944d93782SGreg Clayton 
2700b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
270144d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
270244d93782SGreg Clayton         {KEY_UP, "Select previous item"},
270344d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
270444d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
270544d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
270644d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
270744d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
270844d93782SGreg Clayton         {'A', "Format as annotated address"},
270944d93782SGreg Clayton         {'b', "Format as binary"},
271044d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
271144d93782SGreg Clayton         {'c', "Format as character"},
271244d93782SGreg Clayton         {'d', "Format as a signed integer"},
271344d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
271444d93782SGreg Clayton         {'f', "Format as float"},
271544d93782SGreg Clayton         {'h', "Show help dialog"},
271644d93782SGreg Clayton         {'i', "Format as instructions"},
271744d93782SGreg Clayton         {'o', "Format as octal"},
271844d93782SGreg Clayton         {'p', "Format as pointer"},
271944d93782SGreg Clayton         {'s', "Format as C string"},
272044d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
272144d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
272244d93782SGreg Clayton         {'x', "Format as hex"},
272344d93782SGreg Clayton         {'X', "Format as uppercase hex"},
272444d93782SGreg Clayton         {' ', "Toggle item expansion"},
272544d93782SGreg Clayton         {',', "Page up"},
272644d93782SGreg Clayton         {'.', "Page down"},
2727b9c1b51eSKate Stone         {'\0', nullptr}};
272844d93782SGreg Clayton     return g_source_view_key_help;
272944d93782SGreg Clayton   }
273044d93782SGreg Clayton 
2731b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2732b9c1b51eSKate Stone     switch (c) {
273344d93782SGreg Clayton     case 'x':
273444d93782SGreg Clayton     case 'X':
273544d93782SGreg Clayton     case 'o':
273644d93782SGreg Clayton     case 's':
273744d93782SGreg Clayton     case 'u':
273844d93782SGreg Clayton     case 'd':
273944d93782SGreg Clayton     case 'D':
274044d93782SGreg Clayton     case 'i':
274144d93782SGreg Clayton     case 'A':
274244d93782SGreg Clayton     case 'p':
274344d93782SGreg Clayton     case 'c':
274444d93782SGreg Clayton     case 'b':
274544d93782SGreg Clayton     case 'B':
274644d93782SGreg Clayton     case 'f':
274744d93782SGreg Clayton       // Change the format for the currently selected item
27488369b28dSGreg Clayton       if (m_selected_row) {
27498369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
27508369b28dSGreg Clayton         if (valobj_sp)
27518369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
27528369b28dSGreg Clayton       }
275344d93782SGreg Clayton       return eKeyHandled;
275444d93782SGreg Clayton 
275544d93782SGreg Clayton     case 't':
275644d93782SGreg Clayton       // Toggle showing type names
275744d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
275844d93782SGreg Clayton       return eKeyHandled;
275944d93782SGreg Clayton 
276044d93782SGreg Clayton     case ',':
276144d93782SGreg Clayton     case KEY_PPAGE:
276244d93782SGreg Clayton       // Page up key
2763b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27643985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
276544d93782SGreg Clayton           m_first_visible_row -= m_max_y;
276644d93782SGreg Clayton         else
276744d93782SGreg Clayton           m_first_visible_row = 0;
276844d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
276944d93782SGreg Clayton       }
277044d93782SGreg Clayton       return eKeyHandled;
277144d93782SGreg Clayton 
277244d93782SGreg Clayton     case '.':
277344d93782SGreg Clayton     case KEY_NPAGE:
277444d93782SGreg Clayton       // Page down key
2775b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2776b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
277744d93782SGreg Clayton           m_first_visible_row += m_max_y;
277844d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
277944d93782SGreg Clayton         }
278044d93782SGreg Clayton       }
278144d93782SGreg Clayton       return eKeyHandled;
278244d93782SGreg Clayton 
278344d93782SGreg Clayton     case KEY_UP:
278444d93782SGreg Clayton       if (m_selected_row_idx > 0)
278544d93782SGreg Clayton         --m_selected_row_idx;
278644d93782SGreg Clayton       return eKeyHandled;
2787315b6884SEugene Zelenko 
278844d93782SGreg Clayton     case KEY_DOWN:
278944d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
279044d93782SGreg Clayton         ++m_selected_row_idx;
279144d93782SGreg Clayton       return eKeyHandled;
279244d93782SGreg Clayton 
279344d93782SGreg Clayton     case KEY_RIGHT:
2794b9c1b51eSKate Stone       if (m_selected_row) {
279544d93782SGreg Clayton         if (!m_selected_row->expanded)
279644d93782SGreg Clayton           m_selected_row->Expand();
279744d93782SGreg Clayton       }
279844d93782SGreg Clayton       return eKeyHandled;
279944d93782SGreg Clayton 
280044d93782SGreg Clayton     case KEY_LEFT:
2801b9c1b51eSKate Stone       if (m_selected_row) {
280244d93782SGreg Clayton         if (m_selected_row->expanded)
280344d93782SGreg Clayton           m_selected_row->Unexpand();
280444d93782SGreg Clayton         else if (m_selected_row->parent)
280544d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
280644d93782SGreg Clayton       }
280744d93782SGreg Clayton       return eKeyHandled;
280844d93782SGreg Clayton 
280944d93782SGreg Clayton     case ' ':
281044d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2811b9c1b51eSKate Stone       if (m_selected_row) {
281244d93782SGreg Clayton         if (m_selected_row->expanded)
281344d93782SGreg Clayton           m_selected_row->Unexpand();
281444d93782SGreg Clayton         else
281544d93782SGreg Clayton           m_selected_row->Expand();
281644d93782SGreg Clayton       }
281744d93782SGreg Clayton       return eKeyHandled;
281844d93782SGreg Clayton 
281944d93782SGreg Clayton     case 'h':
282044d93782SGreg Clayton       window.CreateHelpSubwindow();
282144d93782SGreg Clayton       return eKeyHandled;
282244d93782SGreg Clayton 
282344d93782SGreg Clayton     default:
282444d93782SGreg Clayton       break;
282544d93782SGreg Clayton     }
282644d93782SGreg Clayton     return eKeyNotHandled;
282744d93782SGreg Clayton   }
282844d93782SGreg Clayton 
282944d93782SGreg Clayton protected:
283044d93782SGreg Clayton   std::vector<Row> m_rows;
283144d93782SGreg Clayton   Row *m_selected_row;
283244d93782SGreg Clayton   uint32_t m_selected_row_idx;
283344d93782SGreg Clayton   uint32_t m_first_visible_row;
283444d93782SGreg Clayton   uint32_t m_num_rows;
283544d93782SGreg Clayton   int m_min_x;
283644d93782SGreg Clayton   int m_min_y;
283744d93782SGreg Clayton   int m_max_x;
283844d93782SGreg Clayton   int m_max_y;
283944d93782SGreg Clayton 
2840b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2841b9c1b51eSKate Stone     switch (c) {
2842b9c1b51eSKate Stone     case 'x':
2843b9c1b51eSKate Stone       return eFormatHex;
2844b9c1b51eSKate Stone     case 'X':
2845b9c1b51eSKate Stone       return eFormatHexUppercase;
2846b9c1b51eSKate Stone     case 'o':
2847b9c1b51eSKate Stone       return eFormatOctal;
2848b9c1b51eSKate Stone     case 's':
2849b9c1b51eSKate Stone       return eFormatCString;
2850b9c1b51eSKate Stone     case 'u':
2851b9c1b51eSKate Stone       return eFormatUnsigned;
2852b9c1b51eSKate Stone     case 'd':
2853b9c1b51eSKate Stone       return eFormatDecimal;
2854b9c1b51eSKate Stone     case 'D':
2855b9c1b51eSKate Stone       return eFormatDefault;
2856b9c1b51eSKate Stone     case 'i':
2857b9c1b51eSKate Stone       return eFormatInstruction;
2858b9c1b51eSKate Stone     case 'A':
2859b9c1b51eSKate Stone       return eFormatAddressInfo;
2860b9c1b51eSKate Stone     case 'p':
2861b9c1b51eSKate Stone       return eFormatPointer;
2862b9c1b51eSKate Stone     case 'c':
2863b9c1b51eSKate Stone       return eFormatChar;
2864b9c1b51eSKate Stone     case 'b':
2865b9c1b51eSKate Stone       return eFormatBinary;
2866b9c1b51eSKate Stone     case 'B':
2867b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2868b9c1b51eSKate Stone     case 'f':
2869b9c1b51eSKate Stone       return eFormatFloat;
287044d93782SGreg Clayton     }
287144d93782SGreg Clayton     return eFormatDefault;
287244d93782SGreg Clayton   }
287344d93782SGreg Clayton 
2874b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2875b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28768369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
287744d93782SGreg Clayton 
2878c5dac77aSEugene Zelenko     if (valobj == nullptr)
287944d93782SGreg Clayton       return false;
288044d93782SGreg Clayton 
2881b9c1b51eSKate Stone     const char *type_name =
2882b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
288344d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
288444d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
288544d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
288644d93782SGreg Clayton 
288744d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
288844d93782SGreg Clayton 
288944d93782SGreg Clayton     row.DrawTree(window);
289044d93782SGreg Clayton 
289144d93782SGreg Clayton     if (highlight)
289244d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
289344d93782SGreg Clayton 
289444d93782SGreg Clayton     if (type_name && type_name[0])
289544d93782SGreg Clayton       window.Printf("(%s) ", type_name);
289644d93782SGreg Clayton 
289744d93782SGreg Clayton     if (name && name[0])
289844d93782SGreg Clayton       window.PutCString(name);
289944d93782SGreg Clayton 
290044d93782SGreg Clayton     attr_t changd_attr = 0;
290144d93782SGreg Clayton     if (valobj->GetValueDidChange())
290244d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
290344d93782SGreg Clayton 
2904b9c1b51eSKate Stone     if (value && value[0]) {
290544d93782SGreg Clayton       window.PutCString(" = ");
290644d93782SGreg Clayton       if (changd_attr)
290744d93782SGreg Clayton         window.AttributeOn(changd_attr);
290844d93782SGreg Clayton       window.PutCString(value);
290944d93782SGreg Clayton       if (changd_attr)
291044d93782SGreg Clayton         window.AttributeOff(changd_attr);
291144d93782SGreg Clayton     }
291244d93782SGreg Clayton 
2913b9c1b51eSKate Stone     if (summary && summary[0]) {
291444d93782SGreg Clayton       window.PutChar(' ');
291544d93782SGreg Clayton       if (changd_attr)
291644d93782SGreg Clayton         window.AttributeOn(changd_attr);
291744d93782SGreg Clayton       window.PutCString(summary);
291844d93782SGreg Clayton       if (changd_attr)
291944d93782SGreg Clayton         window.AttributeOff(changd_attr);
292044d93782SGreg Clayton     }
292144d93782SGreg Clayton 
292244d93782SGreg Clayton     if (highlight)
292344d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
292444d93782SGreg Clayton 
292544d93782SGreg Clayton     return true;
292644d93782SGreg Clayton   }
2927315b6884SEugene Zelenko 
2928b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2929b9c1b51eSKate Stone                    DisplayOptions &options) {
293044d93782SGreg Clayton     // >   0x25B7
293144d93782SGreg Clayton     // \/  0x25BD
293244d93782SGreg Clayton 
293344d93782SGreg Clayton     bool window_is_active = window.IsActive();
2934b9c1b51eSKate Stone     for (auto &row : rows) {
293544d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
293644d93782SGreg Clayton       // Save the row index in each Row structure
293744d93782SGreg Clayton       row.row_idx = m_num_rows;
293844d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2939b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2940b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
294144d93782SGreg Clayton         row.x = m_min_x;
294244d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2943b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2944b9c1b51eSKate Stone                              window_is_active &&
2945b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2946b9c1b51eSKate Stone                              last_child)) {
294744d93782SGreg Clayton           ++m_num_rows;
2948b9c1b51eSKate Stone         } else {
294944d93782SGreg Clayton           row.x = 0;
295044d93782SGreg Clayton           row.y = 0;
295144d93782SGreg Clayton         }
2952b9c1b51eSKate Stone       } else {
295344d93782SGreg Clayton         row.x = 0;
295444d93782SGreg Clayton         row.y = 0;
295544d93782SGreg Clayton         ++m_num_rows;
295644d93782SGreg Clayton       }
295744d93782SGreg Clayton 
29588369b28dSGreg Clayton       auto &children = row.GetChildren();
29598369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
29608369b28dSGreg Clayton         DisplayRows(window, children, options);
296144d93782SGreg Clayton       }
296244d93782SGreg Clayton     }
296344d93782SGreg Clayton   }
296444d93782SGreg Clayton 
29658369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
296644d93782SGreg Clayton     int row_count = 0;
29678369b28dSGreg Clayton     for (auto &row : rows) {
296844d93782SGreg Clayton       ++row_count;
296944d93782SGreg Clayton       if (row.expanded)
29708369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
297144d93782SGreg Clayton     }
297244d93782SGreg Clayton     return row_count;
297344d93782SGreg Clayton   }
2974315b6884SEugene Zelenko 
2975b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2976b9c1b51eSKate Stone     for (auto &row : rows) {
297744d93782SGreg Clayton       if (row_index == 0)
297844d93782SGreg Clayton         return &row;
2979b9c1b51eSKate Stone       else {
298044d93782SGreg Clayton         --row_index;
29818369b28dSGreg Clayton         auto &children = row.GetChildren();
29828369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29838369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
298444d93782SGreg Clayton           if (result)
298544d93782SGreg Clayton             return result;
298644d93782SGreg Clayton         }
298744d93782SGreg Clayton       }
298844d93782SGreg Clayton     }
2989c5dac77aSEugene Zelenko     return nullptr;
299044d93782SGreg Clayton   }
299144d93782SGreg Clayton 
2992b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
299344d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
299444d93782SGreg Clayton   }
299544d93782SGreg Clayton 
2996b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
299744d93782SGreg Clayton 
299844d93782SGreg Clayton   static DisplayOptions g_options;
299944d93782SGreg Clayton };
300044d93782SGreg Clayton 
3001b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
300244d93782SGreg Clayton public:
3003b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
3004b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
3005b9c1b51eSKate Stone         m_frame_block(nullptr) {}
300644d93782SGreg Clayton 
3007315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
300844d93782SGreg Clayton 
3009b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
301044d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
301144d93782SGreg Clayton   }
301244d93782SGreg Clayton 
3013b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3014b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3015b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
301644d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3017c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
3018c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
301944d93782SGreg Clayton 
3020b9c1b51eSKate Stone     if (process) {
302144d93782SGreg Clayton       StateType state = process->GetState();
3022b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
302344d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
302444d93782SGreg Clayton         if (frame)
302544d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
3026b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
302744d93782SGreg Clayton         return true; // Don't do any updating when we are running
302844d93782SGreg Clayton       }
302944d93782SGreg Clayton     }
303044d93782SGreg Clayton 
303144d93782SGreg Clayton     ValueObjectList local_values;
3032b9c1b51eSKate Stone     if (frame_block) {
303344d93782SGreg Clayton       // Only update the variables if they have changed
3034b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
303544d93782SGreg Clayton         m_frame_block = frame_block;
303644d93782SGreg Clayton 
303744d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3038b9c1b51eSKate Stone         if (locals) {
303944d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
304044d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3041b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3042b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3043b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3044b9c1b51eSKate Stone             if (value_sp) {
3045eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3046eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3047eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3048eb72dc7dSGreg Clayton               else
3049eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3050eb72dc7dSGreg Clayton             }
3051eb72dc7dSGreg Clayton           }
305244d93782SGreg Clayton           // Update the values
305344d93782SGreg Clayton           SetValues(local_values);
305444d93782SGreg Clayton         }
305544d93782SGreg Clayton       }
3056b9c1b51eSKate Stone     } else {
3057c5dac77aSEugene Zelenko       m_frame_block = nullptr;
305844d93782SGreg Clayton       // Update the values with an empty list if there is no frame
305944d93782SGreg Clayton       SetValues(local_values);
306044d93782SGreg Clayton     }
306144d93782SGreg Clayton 
306244d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
306344d93782SGreg Clayton   }
306444d93782SGreg Clayton 
306544d93782SGreg Clayton protected:
306644d93782SGreg Clayton   Debugger &m_debugger;
306744d93782SGreg Clayton   Block *m_frame_block;
306844d93782SGreg Clayton };
306944d93782SGreg Clayton 
3070b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
307144d93782SGreg Clayton public:
3072b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3073b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
307444d93782SGreg Clayton 
3075315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
307644d93782SGreg Clayton 
3077b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
307844d93782SGreg Clayton     return "Register window keyboard shortcuts:";
307944d93782SGreg Clayton   }
308044d93782SGreg Clayton 
3081b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3082b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3083b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
308444d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
308544d93782SGreg Clayton 
308644d93782SGreg Clayton     ValueObjectList value_list;
3087b9c1b51eSKate Stone     if (frame) {
3088b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
308944d93782SGreg Clayton         m_stack_id = frame->GetStackID();
309044d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3091b9c1b51eSKate Stone         if (reg_ctx) {
309244d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3093b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3094b9c1b51eSKate Stone             value_list.Append(
3095b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
309644d93782SGreg Clayton           }
309744d93782SGreg Clayton         }
309844d93782SGreg Clayton         SetValues(value_list);
309944d93782SGreg Clayton       }
3100b9c1b51eSKate Stone     } else {
310144d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
310244d93782SGreg Clayton       if (process && process->IsAlive())
310344d93782SGreg Clayton         return true; // Don't do any updating if we are running
3104b9c1b51eSKate Stone       else {
310505097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
310605097246SAdrian Prantl         // process isn't alive anymore
310744d93782SGreg Clayton         SetValues(value_list);
310844d93782SGreg Clayton       }
310944d93782SGreg Clayton     }
311044d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
311144d93782SGreg Clayton   }
311244d93782SGreg Clayton 
311344d93782SGreg Clayton protected:
311444d93782SGreg Clayton   Debugger &m_debugger;
311544d93782SGreg Clayton   StackID m_stack_id;
311644d93782SGreg Clayton };
311744d93782SGreg Clayton 
3118b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
311944d93782SGreg Clayton   static char g_desc[32];
3120b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
312144d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
312244d93782SGreg Clayton     return g_desc;
312344d93782SGreg Clayton   }
3124b9c1b51eSKate Stone   switch (ch) {
3125b9c1b51eSKate Stone   case KEY_DOWN:
3126b9c1b51eSKate Stone     return "down";
3127b9c1b51eSKate Stone   case KEY_UP:
3128b9c1b51eSKate Stone     return "up";
3129b9c1b51eSKate Stone   case KEY_LEFT:
3130b9c1b51eSKate Stone     return "left";
3131b9c1b51eSKate Stone   case KEY_RIGHT:
3132b9c1b51eSKate Stone     return "right";
3133b9c1b51eSKate Stone   case KEY_HOME:
3134b9c1b51eSKate Stone     return "home";
3135b9c1b51eSKate Stone   case KEY_BACKSPACE:
3136b9c1b51eSKate Stone     return "backspace";
3137b9c1b51eSKate Stone   case KEY_DL:
3138b9c1b51eSKate Stone     return "delete-line";
3139b9c1b51eSKate Stone   case KEY_IL:
3140b9c1b51eSKate Stone     return "insert-line";
3141b9c1b51eSKate Stone   case KEY_DC:
3142b9c1b51eSKate Stone     return "delete-char";
3143b9c1b51eSKate Stone   case KEY_IC:
3144b9c1b51eSKate Stone     return "insert-char";
3145b9c1b51eSKate Stone   case KEY_CLEAR:
3146b9c1b51eSKate Stone     return "clear";
3147b9c1b51eSKate Stone   case KEY_EOS:
3148b9c1b51eSKate Stone     return "clear-to-eos";
3149b9c1b51eSKate Stone   case KEY_EOL:
3150b9c1b51eSKate Stone     return "clear-to-eol";
3151b9c1b51eSKate Stone   case KEY_SF:
3152b9c1b51eSKate Stone     return "scroll-forward";
3153b9c1b51eSKate Stone   case KEY_SR:
3154b9c1b51eSKate Stone     return "scroll-backward";
3155b9c1b51eSKate Stone   case KEY_NPAGE:
3156b9c1b51eSKate Stone     return "page-down";
3157b9c1b51eSKate Stone   case KEY_PPAGE:
3158b9c1b51eSKate Stone     return "page-up";
3159b9c1b51eSKate Stone   case KEY_STAB:
3160b9c1b51eSKate Stone     return "set-tab";
3161b9c1b51eSKate Stone   case KEY_CTAB:
3162b9c1b51eSKate Stone     return "clear-tab";
3163b9c1b51eSKate Stone   case KEY_CATAB:
3164b9c1b51eSKate Stone     return "clear-all-tabs";
3165b9c1b51eSKate Stone   case KEY_ENTER:
3166b9c1b51eSKate Stone     return "enter";
3167b9c1b51eSKate Stone   case KEY_PRINT:
3168b9c1b51eSKate Stone     return "print";
3169b9c1b51eSKate Stone   case KEY_LL:
3170b9c1b51eSKate Stone     return "lower-left key";
3171b9c1b51eSKate Stone   case KEY_A1:
3172b9c1b51eSKate Stone     return "upper left of keypad";
3173b9c1b51eSKate Stone   case KEY_A3:
3174b9c1b51eSKate Stone     return "upper right of keypad";
3175b9c1b51eSKate Stone   case KEY_B2:
3176b9c1b51eSKate Stone     return "center of keypad";
3177b9c1b51eSKate Stone   case KEY_C1:
3178b9c1b51eSKate Stone     return "lower left of keypad";
3179b9c1b51eSKate Stone   case KEY_C3:
3180b9c1b51eSKate Stone     return "lower right of keypad";
3181b9c1b51eSKate Stone   case KEY_BTAB:
3182b9c1b51eSKate Stone     return "back-tab key";
3183b9c1b51eSKate Stone   case KEY_BEG:
3184b9c1b51eSKate Stone     return "begin key";
3185b9c1b51eSKate Stone   case KEY_CANCEL:
3186b9c1b51eSKate Stone     return "cancel key";
3187b9c1b51eSKate Stone   case KEY_CLOSE:
3188b9c1b51eSKate Stone     return "close key";
3189b9c1b51eSKate Stone   case KEY_COMMAND:
3190b9c1b51eSKate Stone     return "command key";
3191b9c1b51eSKate Stone   case KEY_COPY:
3192b9c1b51eSKate Stone     return "copy key";
3193b9c1b51eSKate Stone   case KEY_CREATE:
3194b9c1b51eSKate Stone     return "create key";
3195b9c1b51eSKate Stone   case KEY_END:
3196b9c1b51eSKate Stone     return "end key";
3197b9c1b51eSKate Stone   case KEY_EXIT:
3198b9c1b51eSKate Stone     return "exit key";
3199b9c1b51eSKate Stone   case KEY_FIND:
3200b9c1b51eSKate Stone     return "find key";
3201b9c1b51eSKate Stone   case KEY_HELP:
3202b9c1b51eSKate Stone     return "help key";
3203b9c1b51eSKate Stone   case KEY_MARK:
3204b9c1b51eSKate Stone     return "mark key";
3205b9c1b51eSKate Stone   case KEY_MESSAGE:
3206b9c1b51eSKate Stone     return "message key";
3207b9c1b51eSKate Stone   case KEY_MOVE:
3208b9c1b51eSKate Stone     return "move key";
3209b9c1b51eSKate Stone   case KEY_NEXT:
3210b9c1b51eSKate Stone     return "next key";
3211b9c1b51eSKate Stone   case KEY_OPEN:
3212b9c1b51eSKate Stone     return "open key";
3213b9c1b51eSKate Stone   case KEY_OPTIONS:
3214b9c1b51eSKate Stone     return "options key";
3215b9c1b51eSKate Stone   case KEY_PREVIOUS:
3216b9c1b51eSKate Stone     return "previous key";
3217b9c1b51eSKate Stone   case KEY_REDO:
3218b9c1b51eSKate Stone     return "redo key";
3219b9c1b51eSKate Stone   case KEY_REFERENCE:
3220b9c1b51eSKate Stone     return "reference key";
3221b9c1b51eSKate Stone   case KEY_REFRESH:
3222b9c1b51eSKate Stone     return "refresh key";
3223b9c1b51eSKate Stone   case KEY_REPLACE:
3224b9c1b51eSKate Stone     return "replace key";
3225b9c1b51eSKate Stone   case KEY_RESTART:
3226b9c1b51eSKate Stone     return "restart key";
3227b9c1b51eSKate Stone   case KEY_RESUME:
3228b9c1b51eSKate Stone     return "resume key";
3229b9c1b51eSKate Stone   case KEY_SAVE:
3230b9c1b51eSKate Stone     return "save key";
3231b9c1b51eSKate Stone   case KEY_SBEG:
3232b9c1b51eSKate Stone     return "shifted begin key";
3233b9c1b51eSKate Stone   case KEY_SCANCEL:
3234b9c1b51eSKate Stone     return "shifted cancel key";
3235b9c1b51eSKate Stone   case KEY_SCOMMAND:
3236b9c1b51eSKate Stone     return "shifted command key";
3237b9c1b51eSKate Stone   case KEY_SCOPY:
3238b9c1b51eSKate Stone     return "shifted copy key";
3239b9c1b51eSKate Stone   case KEY_SCREATE:
3240b9c1b51eSKate Stone     return "shifted create key";
3241b9c1b51eSKate Stone   case KEY_SDC:
3242b9c1b51eSKate Stone     return "shifted delete-character key";
3243b9c1b51eSKate Stone   case KEY_SDL:
3244b9c1b51eSKate Stone     return "shifted delete-line key";
3245b9c1b51eSKate Stone   case KEY_SELECT:
3246b9c1b51eSKate Stone     return "select key";
3247b9c1b51eSKate Stone   case KEY_SEND:
3248b9c1b51eSKate Stone     return "shifted end key";
3249b9c1b51eSKate Stone   case KEY_SEOL:
3250b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3251b9c1b51eSKate Stone   case KEY_SEXIT:
3252b9c1b51eSKate Stone     return "shifted exit key";
3253b9c1b51eSKate Stone   case KEY_SFIND:
3254b9c1b51eSKate Stone     return "shifted find key";
3255b9c1b51eSKate Stone   case KEY_SHELP:
3256b9c1b51eSKate Stone     return "shifted help key";
3257b9c1b51eSKate Stone   case KEY_SHOME:
3258b9c1b51eSKate Stone     return "shifted home key";
3259b9c1b51eSKate Stone   case KEY_SIC:
3260b9c1b51eSKate Stone     return "shifted insert-character key";
3261b9c1b51eSKate Stone   case KEY_SLEFT:
3262b9c1b51eSKate Stone     return "shifted left-arrow key";
3263b9c1b51eSKate Stone   case KEY_SMESSAGE:
3264b9c1b51eSKate Stone     return "shifted message key";
3265b9c1b51eSKate Stone   case KEY_SMOVE:
3266b9c1b51eSKate Stone     return "shifted move key";
3267b9c1b51eSKate Stone   case KEY_SNEXT:
3268b9c1b51eSKate Stone     return "shifted next key";
3269b9c1b51eSKate Stone   case KEY_SOPTIONS:
3270b9c1b51eSKate Stone     return "shifted options key";
3271b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3272b9c1b51eSKate Stone     return "shifted previous key";
3273b9c1b51eSKate Stone   case KEY_SPRINT:
3274b9c1b51eSKate Stone     return "shifted print key";
3275b9c1b51eSKate Stone   case KEY_SREDO:
3276b9c1b51eSKate Stone     return "shifted redo key";
3277b9c1b51eSKate Stone   case KEY_SREPLACE:
3278b9c1b51eSKate Stone     return "shifted replace key";
3279b9c1b51eSKate Stone   case KEY_SRIGHT:
3280b9c1b51eSKate Stone     return "shifted right-arrow key";
3281b9c1b51eSKate Stone   case KEY_SRSUME:
3282b9c1b51eSKate Stone     return "shifted resume key";
3283b9c1b51eSKate Stone   case KEY_SSAVE:
3284b9c1b51eSKate Stone     return "shifted save key";
3285b9c1b51eSKate Stone   case KEY_SSUSPEND:
3286b9c1b51eSKate Stone     return "shifted suspend key";
3287b9c1b51eSKate Stone   case KEY_SUNDO:
3288b9c1b51eSKate Stone     return "shifted undo key";
3289b9c1b51eSKate Stone   case KEY_SUSPEND:
3290b9c1b51eSKate Stone     return "suspend key";
3291b9c1b51eSKate Stone   case KEY_UNDO:
3292b9c1b51eSKate Stone     return "undo key";
3293b9c1b51eSKate Stone   case KEY_MOUSE:
3294b9c1b51eSKate Stone     return "Mouse event has occurred";
3295b9c1b51eSKate Stone   case KEY_RESIZE:
3296b9c1b51eSKate Stone     return "Terminal resize event";
329727801f4fSBruce Mitchener #ifdef KEY_EVENT
3298b9c1b51eSKate Stone   case KEY_EVENT:
3299b9c1b51eSKate Stone     return "We were interrupted by an event";
330027801f4fSBruce Mitchener #endif
3301b9c1b51eSKate Stone   case KEY_RETURN:
3302b9c1b51eSKate Stone     return "return";
3303b9c1b51eSKate Stone   case ' ':
3304b9c1b51eSKate Stone     return "space";
3305b9c1b51eSKate Stone   case '\t':
3306b9c1b51eSKate Stone     return "tab";
3307b9c1b51eSKate Stone   case KEY_ESCAPE:
3308b9c1b51eSKate Stone     return "escape";
330944d93782SGreg Clayton   default:
331044d93782SGreg Clayton     if (isprint(ch))
331144d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
331244d93782SGreg Clayton     else
331344d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
331444d93782SGreg Clayton     return g_desc;
331544d93782SGreg Clayton   }
3316c5dac77aSEugene Zelenko   return nullptr;
331744d93782SGreg Clayton }
331844d93782SGreg Clayton 
3319b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3320b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3321b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3322b9c1b51eSKate Stone   if (text && text[0]) {
332344d93782SGreg Clayton     m_text.SplitIntoLines(text);
332444d93782SGreg Clayton     m_text.AppendString("");
332544d93782SGreg Clayton   }
3326b9c1b51eSKate Stone   if (key_help_array) {
3327b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
332844d93782SGreg Clayton       StreamString key_description;
3329b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3330b9c1b51eSKate Stone                              key->description);
3331c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
333244d93782SGreg Clayton     }
333344d93782SGreg Clayton   }
333444d93782SGreg Clayton }
333544d93782SGreg Clayton 
3336315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
333744d93782SGreg Clayton 
3338b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
333944d93782SGreg Clayton   window.Erase();
334044d93782SGreg Clayton   const int window_height = window.GetHeight();
334144d93782SGreg Clayton   int x = 2;
334244d93782SGreg Clayton   int y = 1;
334344d93782SGreg Clayton   const int min_y = y;
334444d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33453985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
334644d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
334744d93782SGreg Clayton   const char *bottom_message;
334844d93782SGreg Clayton   if (num_lines <= num_visible_lines)
334944d93782SGreg Clayton     bottom_message = "Press any key to exit";
335044d93782SGreg Clayton   else
335144d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
335244d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3353b9c1b51eSKate Stone   while (y <= max_y) {
335444d93782SGreg Clayton     window.MoveCursor(x, y);
3355b9c1b51eSKate Stone     window.PutCStringTruncated(
3356b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
335744d93782SGreg Clayton     ++y;
335844d93782SGreg Clayton   }
335944d93782SGreg Clayton   return true;
336044d93782SGreg Clayton }
336144d93782SGreg Clayton 
3362b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3363b9c1b51eSKate Stone                                                               int key) {
336444d93782SGreg Clayton   bool done = false;
336544d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
336644d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
336744d93782SGreg Clayton 
3368b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
336944d93782SGreg Clayton     done = true;
337005097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
337105097246SAdrian Prantl     // press will cause us to exit
3372b9c1b51eSKate Stone   } else {
3373b9c1b51eSKate Stone     switch (key) {
337444d93782SGreg Clayton     case KEY_UP:
337544d93782SGreg Clayton       if (m_first_visible_line > 0)
337644d93782SGreg Clayton         --m_first_visible_line;
337744d93782SGreg Clayton       break;
337844d93782SGreg Clayton 
337944d93782SGreg Clayton     case KEY_DOWN:
338044d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
338144d93782SGreg Clayton         ++m_first_visible_line;
338244d93782SGreg Clayton       break;
338344d93782SGreg Clayton 
338444d93782SGreg Clayton     case KEY_PPAGE:
338544d93782SGreg Clayton     case ',':
3386b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33873985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
338844d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
338944d93782SGreg Clayton         else
339044d93782SGreg Clayton           m_first_visible_line = 0;
339144d93782SGreg Clayton       }
339244d93782SGreg Clayton       break;
3393315b6884SEugene Zelenko 
339444d93782SGreg Clayton     case KEY_NPAGE:
339544d93782SGreg Clayton     case '.':
3396b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
339744d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
33983985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
339944d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
340044d93782SGreg Clayton       }
340144d93782SGreg Clayton       break;
3402315b6884SEugene Zelenko 
340344d93782SGreg Clayton     default:
340444d93782SGreg Clayton       done = true;
340544d93782SGreg Clayton       break;
340644d93782SGreg Clayton     }
340744d93782SGreg Clayton   }
340844d93782SGreg Clayton   if (done)
340944d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
341044d93782SGreg Clayton   return eKeyHandled;
341144d93782SGreg Clayton }
341244d93782SGreg Clayton 
3413b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
341444d93782SGreg Clayton public:
341544d93782SGreg Clayton   enum {
341644d93782SGreg Clayton     eMenuID_LLDB = 1,
341744d93782SGreg Clayton     eMenuID_LLDBAbout,
341844d93782SGreg Clayton     eMenuID_LLDBExit,
341944d93782SGreg Clayton 
342044d93782SGreg Clayton     eMenuID_Target,
342144d93782SGreg Clayton     eMenuID_TargetCreate,
342244d93782SGreg Clayton     eMenuID_TargetDelete,
342344d93782SGreg Clayton 
342444d93782SGreg Clayton     eMenuID_Process,
342544d93782SGreg Clayton     eMenuID_ProcessAttach,
342644d93782SGreg Clayton     eMenuID_ProcessDetach,
342744d93782SGreg Clayton     eMenuID_ProcessLaunch,
342844d93782SGreg Clayton     eMenuID_ProcessContinue,
342944d93782SGreg Clayton     eMenuID_ProcessHalt,
343044d93782SGreg Clayton     eMenuID_ProcessKill,
343144d93782SGreg Clayton 
343244d93782SGreg Clayton     eMenuID_Thread,
343344d93782SGreg Clayton     eMenuID_ThreadStepIn,
343444d93782SGreg Clayton     eMenuID_ThreadStepOver,
343544d93782SGreg Clayton     eMenuID_ThreadStepOut,
343644d93782SGreg Clayton 
343744d93782SGreg Clayton     eMenuID_View,
343844d93782SGreg Clayton     eMenuID_ViewBacktrace,
343944d93782SGreg Clayton     eMenuID_ViewRegisters,
344044d93782SGreg Clayton     eMenuID_ViewSource,
344144d93782SGreg Clayton     eMenuID_ViewVariables,
344244d93782SGreg Clayton 
344344d93782SGreg Clayton     eMenuID_Help,
344444d93782SGreg Clayton     eMenuID_HelpGUIHelp
344544d93782SGreg Clayton   };
344644d93782SGreg Clayton 
3447b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3448b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
344944d93782SGreg Clayton 
3450315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3451bd5ae6b4SGreg Clayton 
3452b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
345344d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
345444d93782SGreg Clayton   }
345544d93782SGreg Clayton 
3456b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3457b9c1b51eSKate Stone     switch (key) {
34585fdb09bbSGreg Clayton     case '\t':
345944d93782SGreg Clayton       window.SelectNextWindowAsActive();
346044d93782SGreg Clayton       return eKeyHandled;
34615fdb09bbSGreg Clayton 
34625fdb09bbSGreg Clayton     case 'h':
34635fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34645fdb09bbSGreg Clayton       return eKeyHandled;
34655fdb09bbSGreg Clayton 
34665fdb09bbSGreg Clayton     case KEY_ESCAPE:
34675fdb09bbSGreg Clayton       return eQuitApplication;
34685fdb09bbSGreg Clayton 
34695fdb09bbSGreg Clayton     default:
34705fdb09bbSGreg Clayton       break;
347144d93782SGreg Clayton     }
347244d93782SGreg Clayton     return eKeyNotHandled;
347344d93782SGreg Clayton   }
347444d93782SGreg Clayton 
3475b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34765fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34775fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3478b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3479b9c1b51eSKate Stone            "dialog to display them.\n\n"
34805fdb09bbSGreg Clayton            "Common key bindings for all views:";
34815fdb09bbSGreg Clayton   }
34825fdb09bbSGreg Clayton 
3483b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34845fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34855fdb09bbSGreg Clayton         {'\t', "Select next view"},
34865fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34875fdb09bbSGreg Clayton         {',', "Page up"},
34885fdb09bbSGreg Clayton         {'.', "Page down"},
34895fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34905fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34915fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
34925fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
34935fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
34945fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3495b9c1b51eSKate Stone         {'\0', nullptr}};
34965fdb09bbSGreg Clayton     return g_source_view_key_help;
34975fdb09bbSGreg Clayton   }
34985fdb09bbSGreg Clayton 
3499b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3500b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3501b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3502b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3503b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3504b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
350544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3506b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3507b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
35084b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
350944d93782SGreg Clayton       }
351044d93782SGreg Clayton     }
351144d93782SGreg Clayton       return MenuActionResult::Handled;
351244d93782SGreg Clayton 
3513b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3514b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3515b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3516b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
351744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3518b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3519b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
352044d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
352144d93782SGreg Clayton       }
352244d93782SGreg Clayton     }
352344d93782SGreg Clayton       return MenuActionResult::Handled;
352444d93782SGreg Clayton 
3525b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3526b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3527b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3528b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
352944d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3530b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3531b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
353244d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
353344d93782SGreg Clayton       }
353444d93782SGreg Clayton     }
353544d93782SGreg Clayton       return MenuActionResult::Handled;
353644d93782SGreg Clayton 
3537b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3538b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3539b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3540b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
354144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3542b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3543b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
354444d93782SGreg Clayton           process->Resume();
354544d93782SGreg Clayton       }
354644d93782SGreg Clayton     }
354744d93782SGreg Clayton       return MenuActionResult::Handled;
354844d93782SGreg Clayton 
3549b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3550b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3551b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3552b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
355344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
355444d93782SGreg Clayton         if (process && process->IsAlive())
3555ede3193bSJason Molenda           process->Destroy(false);
355644d93782SGreg Clayton       }
355744d93782SGreg Clayton     }
355844d93782SGreg Clayton       return MenuActionResult::Handled;
355944d93782SGreg Clayton 
3560b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3561b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3562b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3563b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
356444d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
356544d93782SGreg Clayton         if (process && process->IsAlive())
356644d93782SGreg Clayton           process->Halt();
356744d93782SGreg Clayton       }
356844d93782SGreg Clayton     }
356944d93782SGreg Clayton       return MenuActionResult::Handled;
357044d93782SGreg Clayton 
3571b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3572b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3573b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3574b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
357544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
357644d93782SGreg Clayton         if (process && process->IsAlive())
357744d93782SGreg Clayton           process->Detach(false);
357844d93782SGreg Clayton       }
357944d93782SGreg Clayton     }
358044d93782SGreg Clayton       return MenuActionResult::Handled;
358144d93782SGreg Clayton 
3582b9c1b51eSKate Stone     case eMenuID_Process: {
3583b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
358405097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
358505097246SAdrian Prantl       // submenu.
358644d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3587b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3588b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
358944d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3590b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3591b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
359244d93782SGreg Clayton         if (submenus.size() == 7)
359344d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
359444d93782SGreg Clayton         else if (submenus.size() > 8)
359544d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
359644d93782SGreg Clayton 
359744d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3598bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
359944d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3600b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
360144d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
360244d93782SGreg Clayton           char menu_char = '\0';
360344d93782SGreg Clayton           if (i < 9)
360444d93782SGreg Clayton             menu_char = '1' + i;
360544d93782SGreg Clayton           StreamString thread_menu_title;
360644d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
360744d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
360844d93782SGreg Clayton           if (thread_name && thread_name[0])
360944d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3610b9c1b51eSKate Stone           else {
361144d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
361244d93782SGreg Clayton             if (queue_name && queue_name[0])
361344d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
361444d93782SGreg Clayton           }
3615b9c1b51eSKate Stone           menu.AddSubmenu(
3616c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3617c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
361844d93782SGreg Clayton         }
3619b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
362005097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
362105097246SAdrian Prantl         // previously added
362244d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
362344d93782SGreg Clayton       }
3624b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3625b9c1b51eSKate Stone       // lengths
362644d93782SGreg Clayton       menu.RecalculateNameLengths();
362744d93782SGreg Clayton     }
362844d93782SGreg Clayton       return MenuActionResult::Handled;
362944d93782SGreg Clayton 
3630b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
363144d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
363244d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
363344d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
363444d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
363544d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
363644d93782SGreg Clayton 
3637b9c1b51eSKate Stone       if (variables_window_sp) {
363844d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
363944d93782SGreg Clayton 
364044d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
364144d93782SGreg Clayton 
3642b9c1b51eSKate Stone         if (registers_window_sp) {
3643b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3644b9c1b51eSKate Stone           // registers window
364544d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
364644d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
364744d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3648b9c1b51eSKate Stone         } else {
364905097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
365005097246SAdrian Prantl           // to the source view
365144d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3652b9c1b51eSKate Stone                                    source_bounds.size.height +
3653b9c1b51eSKate Stone                                        variables_bounds.size.height);
365444d93782SGreg Clayton         }
3655b9c1b51eSKate Stone       } else {
365644d93782SGreg Clayton         Rect new_variables_rect;
3657b9c1b51eSKate Stone         if (registers_window_sp) {
365844d93782SGreg Clayton           // We have a registers window so split the area of the registers
365944d93782SGreg Clayton           // window into two columns where the left hand side will be the
366044d93782SGreg Clayton           // variables and the right hand side will be the registers
366144d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
366244d93782SGreg Clayton           Rect new_registers_rect;
3663b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3664b9c1b51eSKate Stone                                                    new_registers_rect);
366544d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3666b9c1b51eSKate Stone         } else {
366744d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
366844d93782SGreg Clayton           Rect new_source_rect;
3669b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3670b9c1b51eSKate Stone                                                   new_variables_rect);
367144d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
367244d93782SGreg Clayton         }
3673b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3674b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3675b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3676b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
367744d93782SGreg Clayton       }
367844d93782SGreg Clayton       touchwin(stdscr);
367944d93782SGreg Clayton     }
368044d93782SGreg Clayton       return MenuActionResult::Handled;
368144d93782SGreg Clayton 
3682b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
368344d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
368444d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
368544d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
368644d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
368744d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
368844d93782SGreg Clayton 
3689b9c1b51eSKate Stone       if (registers_window_sp) {
3690b9c1b51eSKate Stone         if (variables_window_sp) {
369144d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
369244d93782SGreg Clayton 
3693b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3694b9c1b51eSKate Stone           // variables window
3695b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3696b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
369744d93782SGreg Clayton                                       variables_bounds.size.height);
3698b9c1b51eSKate Stone         } else {
369905097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
370005097246SAdrian Prantl           // to the source view
370144d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3702b9c1b51eSKate Stone                                    source_bounds.size.height +
3703b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
370444d93782SGreg Clayton         }
370544d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3706b9c1b51eSKate Stone       } else {
370744d93782SGreg Clayton         Rect new_regs_rect;
3708b9c1b51eSKate Stone         if (variables_window_sp) {
370905097246SAdrian Prantl           // We have a variables window, split it into two columns where the
371005097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
371105097246SAdrian Prantl           // be the registers
371244d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
371344d93782SGreg Clayton           Rect new_vars_rect;
3714b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3715b9c1b51eSKate Stone                                                    new_regs_rect);
371644d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3717b9c1b51eSKate Stone         } else {
371844d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
371944d93782SGreg Clayton           Rect new_source_rect;
3720b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3721b9c1b51eSKate Stone                                                   new_regs_rect);
372244d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
372344d93782SGreg Clayton         }
3724b9c1b51eSKate Stone         WindowSP new_window_sp =
3725b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3726b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3727b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
372844d93782SGreg Clayton       }
372944d93782SGreg Clayton       touchwin(stdscr);
373044d93782SGreg Clayton     }
373144d93782SGreg Clayton       return MenuActionResult::Handled;
373244d93782SGreg Clayton 
373344d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37345fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
373544d93782SGreg Clayton       return MenuActionResult::Handled;
373644d93782SGreg Clayton 
373744d93782SGreg Clayton     default:
373844d93782SGreg Clayton       break;
373944d93782SGreg Clayton     }
374044d93782SGreg Clayton 
374144d93782SGreg Clayton     return MenuActionResult::NotHandled;
374244d93782SGreg Clayton   }
3743b9c1b51eSKate Stone 
374444d93782SGreg Clayton protected:
374544d93782SGreg Clayton   Application &m_app;
374644d93782SGreg Clayton   Debugger &m_debugger;
374744d93782SGreg Clayton };
374844d93782SGreg Clayton 
3749b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
375044d93782SGreg Clayton public:
3751b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3752b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
375344d93782SGreg Clayton   }
375444d93782SGreg Clayton 
3755315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3756bd5ae6b4SGreg Clayton 
3757b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3758b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3759b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
376044d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
376144d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
376244d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
376344d93782SGreg Clayton     window.Erase();
376444d93782SGreg Clayton     window.SetBackground(2);
376544d93782SGreg Clayton     window.MoveCursor(0, 0);
3766b9c1b51eSKate Stone     if (process) {
376744d93782SGreg Clayton       const StateType state = process->GetState();
3768b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3769b9c1b51eSKate Stone                     StateAsCString(state));
377044d93782SGreg Clayton 
3771b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37725b031ebcSEd Maste         StreamString strm;
3773b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3774b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
377544d93782SGreg Clayton           window.MoveCursor(40, 0);
3776c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37775b031ebcSEd Maste         }
377844d93782SGreg Clayton 
377944d93782SGreg Clayton         window.MoveCursor(60, 0);
378044d93782SGreg Clayton         if (frame)
3781b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3782b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3783b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3784b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3785b9c1b51eSKate Stone       } else if (state == eStateExited) {
378644d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
378744d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
378844d93782SGreg Clayton         if (exit_desc && exit_desc[0])
378944d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
379044d93782SGreg Clayton         else
379144d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
379244d93782SGreg Clayton       }
379344d93782SGreg Clayton     }
379444d93782SGreg Clayton     window.DeferredRefresh();
379544d93782SGreg Clayton     return true;
379644d93782SGreg Clayton   }
379744d93782SGreg Clayton 
379844d93782SGreg Clayton protected:
379944d93782SGreg Clayton   Debugger &m_debugger;
3800554f68d3SGreg Clayton   FormatEntity::Entry m_format;
380144d93782SGreg Clayton };
380244d93782SGreg Clayton 
3803b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
380444d93782SGreg Clayton public:
3805b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3806b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3807b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3808b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3809b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3810b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
381144d93782SGreg Clayton 
3812315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
381344d93782SGreg Clayton 
3814b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
381544d93782SGreg Clayton 
3816b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
381744d93782SGreg Clayton 
3818b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
381944d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
382044d93782SGreg Clayton   }
382144d93782SGreg Clayton 
3822b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
382344d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
382444d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
382544d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
382644d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
382744d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
382844d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
382944d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
383044d93782SGreg Clayton         {'c', "Continue process"},
383144d93782SGreg Clayton         {'d', "Detach and resume process"},
383244d93782SGreg Clayton         {'D', "Detach with process suspended"},
383344d93782SGreg Clayton         {'h', "Show help dialog"},
383444d93782SGreg Clayton         {'k', "Kill process"},
383544d93782SGreg Clayton         {'n', "Step over (source line)"},
383644d93782SGreg Clayton         {'N', "Step over (single instruction)"},
383744d93782SGreg Clayton         {'o', "Step out"},
383844d93782SGreg Clayton         {'s', "Step in (source line)"},
383944d93782SGreg Clayton         {'S', "Step in (single instruction)"},
384044d93782SGreg Clayton         {',', "Page up"},
384144d93782SGreg Clayton         {'.', "Page down"},
3842b9c1b51eSKate Stone         {'\0', nullptr}};
384344d93782SGreg Clayton     return g_source_view_key_help;
384444d93782SGreg Clayton   }
384544d93782SGreg Clayton 
3846b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3847b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3848b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
384944d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3850c5dac77aSEugene Zelenko     Thread *thread = nullptr;
385144d93782SGreg Clayton 
385244d93782SGreg Clayton     bool update_location = false;
3853b9c1b51eSKate Stone     if (process) {
385444d93782SGreg Clayton       StateType state = process->GetState();
3855b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
385644d93782SGreg Clayton         // We are stopped, so it is ok to
385744d93782SGreg Clayton         update_location = true;
385844d93782SGreg Clayton       }
385944d93782SGreg Clayton     }
386044d93782SGreg Clayton 
386144d93782SGreg Clayton     m_min_x = 1;
3862ec990867SGreg Clayton     m_min_y = 2;
386344d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
386444d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
386544d93782SGreg Clayton 
386644d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
386744d93782SGreg Clayton     StackFrameSP frame_sp;
386844d93782SGreg Clayton     bool set_selected_line_to_pc = false;
386944d93782SGreg Clayton 
3870b9c1b51eSKate Stone     if (update_location) {
387144d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
387244d93782SGreg Clayton       bool thread_changed = false;
3873b9c1b51eSKate Stone       if (process_alive) {
387444d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3875b9c1b51eSKate Stone         if (thread) {
387644d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
387744d93782SGreg Clayton           auto tid = thread->GetID();
387844d93782SGreg Clayton           thread_changed = tid != m_tid;
387944d93782SGreg Clayton           m_tid = tid;
3880b9c1b51eSKate Stone         } else {
3881b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
388244d93782SGreg Clayton             thread_changed = true;
388344d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
388444d93782SGreg Clayton           }
388544d93782SGreg Clayton         }
388644d93782SGreg Clayton       }
388744d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
388844d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
388944d93782SGreg Clayton       bool frame_changed = false;
389044d93782SGreg Clayton       m_stop_id = stop_id;
3891ec990867SGreg Clayton       m_title.Clear();
3892b9c1b51eSKate Stone       if (frame_sp) {
389344d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3894b9c1b51eSKate Stone         if (m_sc.module_sp) {
3895b9c1b51eSKate Stone           m_title.Printf(
3896b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3897ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3898ec990867SGreg Clayton           if (func_name)
3899ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3900ec990867SGreg Clayton         }
390144d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
390244d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
390344d93782SGreg Clayton         m_frame_idx = frame_idx;
3904b9c1b51eSKate Stone       } else {
390544d93782SGreg Clayton         m_sc.Clear(true);
390644d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
390744d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
390844d93782SGreg Clayton       }
390944d93782SGreg Clayton 
3910b9c1b51eSKate Stone       const bool context_changed =
3911b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
391244d93782SGreg Clayton 
3913b9c1b51eSKate Stone       if (process_alive) {
3914b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
391544d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
391644d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
391744d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
391844d93782SGreg Clayton           // Update the selected line if the stop ID changed...
391944d93782SGreg Clayton           if (context_changed)
392044d93782SGreg Clayton             m_selected_line = m_pc_line;
392144d93782SGreg Clayton 
3922b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
392305097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
392405097246SAdrian Prantl             // (source file missing)
3925b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
392644d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
392744d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3928b9c1b51eSKate Stone             } else {
392944d93782SGreg Clayton               if (m_selected_line > 10)
393044d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
393144d93782SGreg Clayton               else
393244d93782SGreg Clayton                 m_first_visible_line = 0;
393344d93782SGreg Clayton             }
3934b9c1b51eSKate Stone           } else {
393544d93782SGreg Clayton             // File changed, set selected line to the line with the PC
393644d93782SGreg Clayton             m_selected_line = m_pc_line;
3937b9c1b51eSKate Stone             m_file_sp =
3938b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3939b9c1b51eSKate Stone             if (m_file_sp) {
394044d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
3941eebf32faSPavel Labath               m_line_width = 1;
394244d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
394344d93782SGreg Clayton                 ++m_line_width;
394444d93782SGreg Clayton 
3945b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3946b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
394744d93782SGreg Clayton                 m_first_visible_line = 0;
394844d93782SGreg Clayton               else
394944d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
395044d93782SGreg Clayton             }
395144d93782SGreg Clayton           }
3952b9c1b51eSKate Stone         } else {
395344d93782SGreg Clayton           m_file_sp.reset();
395444d93782SGreg Clayton         }
395544d93782SGreg Clayton 
3956b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
395744d93782SGreg Clayton           // Show disassembly
395844d93782SGreg Clayton           bool prefer_file_cache = false;
3959b9c1b51eSKate Stone           if (m_sc.function) {
3960b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
396144d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3962b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3963b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3964b9c1b51eSKate Stone               if (m_disassembly_sp) {
396544d93782SGreg Clayton                 set_selected_line_to_pc = true;
396644d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3967b9c1b51eSKate Stone               } else {
396844d93782SGreg Clayton                 m_disassembly_range.Clear();
396944d93782SGreg Clayton               }
3970b9c1b51eSKate Stone             } else {
397144d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
397244d93782SGreg Clayton             }
3973b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3974b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
397544d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3976b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3977b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3978b9c1b51eSKate Stone               if (m_disassembly_sp) {
397944d93782SGreg Clayton                 set_selected_line_to_pc = true;
3980b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3981b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
398244d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3983b9c1b51eSKate Stone               } else {
398444d93782SGreg Clayton                 m_disassembly_range.Clear();
398544d93782SGreg Clayton               }
3986b9c1b51eSKate Stone             } else {
398744d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
398844d93782SGreg Clayton             }
398944d93782SGreg Clayton           }
399044d93782SGreg Clayton         }
3991b9c1b51eSKate Stone       } else {
399244d93782SGreg Clayton         m_pc_line = UINT32_MAX;
399344d93782SGreg Clayton       }
399444d93782SGreg Clayton     }
399544d93782SGreg Clayton 
3996ec990867SGreg Clayton     const int window_width = window.GetWidth();
399744d93782SGreg Clayton     window.Erase();
399844d93782SGreg Clayton     window.DrawTitleBox("Sources");
3999b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
4000ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
4001ec990867SGreg Clayton       window.MoveCursor(1, 1);
4002ec990867SGreg Clayton       window.PutChar(' ');
4003c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
4004ec990867SGreg Clayton       int x = window.GetCursorX();
4005b9c1b51eSKate Stone       if (x < window_width - 1) {
4006ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
4007ec990867SGreg Clayton       }
4008ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
4009ec990867SGreg Clayton     }
401044d93782SGreg Clayton 
401144d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
401244d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
4013b9c1b51eSKate Stone     if (num_source_lines > 0) {
401444d93782SGreg Clayton       // Display source
401544d93782SGreg Clayton       BreakpointLines bp_lines;
4016b9c1b51eSKate Stone       if (target) {
401744d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
401844d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
4019b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
402044d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
402144d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
4022b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
4023b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
4024b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
402544d93782SGreg Clayton             LineEntry bp_loc_line_entry;
4026b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
4027b9c1b51eSKate Stone                     bp_loc_line_entry)) {
4028b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
402944d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
403044d93782SGreg Clayton               }
403144d93782SGreg Clayton             }
403244d93782SGreg Clayton           }
403344d93782SGreg Clayton         }
403444d93782SGreg Clayton       }
403544d93782SGreg Clayton 
403644d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
403744d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
403844d93782SGreg Clayton 
4039b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
404044d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4041b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4042ec990867SGreg Clayton           const int line_y = m_min_y + i;
404344d93782SGreg Clayton           window.MoveCursor(1, line_y);
404444d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
404544d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
404644d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
404744d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
404844d93782SGreg Clayton           attr_t highlight_attr = 0;
404944d93782SGreg Clayton           attr_t bp_attr = 0;
405044d93782SGreg Clayton           if (is_pc_line)
405144d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
405244d93782SGreg Clayton           else if (line_is_selected)
405344d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
405444d93782SGreg Clayton 
405544d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
405644d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
405744d93782SGreg Clayton 
405844d93782SGreg Clayton           if (bp_attr)
405944d93782SGreg Clayton             window.AttributeOn(bp_attr);
406044d93782SGreg Clayton 
4061eebf32faSPavel Labath           window.Printf(" %*u ", m_line_width, curr_line + 1);
406244d93782SGreg Clayton 
406344d93782SGreg Clayton           if (bp_attr)
406444d93782SGreg Clayton             window.AttributeOff(bp_attr);
406544d93782SGreg Clayton 
406644d93782SGreg Clayton           window.PutChar(ACS_VLINE);
406744d93782SGreg Clayton           // Mark the line with the PC with a diamond
406844d93782SGreg Clayton           if (is_pc_line)
406944d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
407044d93782SGreg Clayton           else
407144d93782SGreg Clayton             window.PutChar(' ');
407244d93782SGreg Clayton 
407344d93782SGreg Clayton           if (highlight_attr)
407444d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4075b9c1b51eSKate Stone           const uint32_t line_len =
4076b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
407744d93782SGreg Clayton           if (line_len > 0)
407844d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
407944d93782SGreg Clayton 
4080b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4081b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
408244d93782SGreg Clayton             StopInfoSP stop_info_sp;
408344d93782SGreg Clayton             if (thread)
408444d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4085b9c1b51eSKate Stone             if (stop_info_sp) {
408644d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4087b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
408844d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4089ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
409044d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4091b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4092b9c1b51eSKate Stone                 // line_y);
4093b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4094b9c1b51eSKate Stone                               stop_description);
409544d93782SGreg Clayton               }
4096b9c1b51eSKate Stone             } else {
4097ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
409844d93782SGreg Clayton             }
409944d93782SGreg Clayton           }
410044d93782SGreg Clayton           if (highlight_attr)
410144d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4102b9c1b51eSKate Stone         } else {
410344d93782SGreg Clayton           break;
410444d93782SGreg Clayton         }
410544d93782SGreg Clayton       }
4106b9c1b51eSKate Stone     } else {
410744d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4108b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
410944d93782SGreg Clayton         // Display disassembly
411044d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
411144d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4112b9c1b51eSKate Stone         if (target) {
411344d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
411444d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4115b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
411644d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
411744d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4118b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4119b9c1b51eSKate Stone                  ++bp_loc_idx) {
4120b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4121b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
412244d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4123b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4124b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4125b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
412644d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
412744d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
412844d93782SGreg Clayton               }
412944d93782SGreg Clayton             }
413044d93782SGreg Clayton           }
413144d93782SGreg Clayton         }
413244d93782SGreg Clayton 
413344d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
413444d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
413544d93782SGreg Clayton 
413644d93782SGreg Clayton         StreamString strm;
413744d93782SGreg Clayton 
413844d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
413944d93782SGreg Clayton         Address pc_address;
414044d93782SGreg Clayton 
414144d93782SGreg Clayton         if (frame_sp)
414244d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4143b9c1b51eSKate Stone         const uint32_t pc_idx =
4144b9c1b51eSKate Stone             pc_address.IsValid()
4145b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4146b9c1b51eSKate Stone                 : UINT32_MAX;
4147b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
414844d93782SGreg Clayton           m_selected_line = pc_idx;
414944d93782SGreg Clayton         }
415044d93782SGreg Clayton 
415144d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41523985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
415344d93782SGreg Clayton           m_first_visible_line = 0;
415444d93782SGreg Clayton 
4155b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41563985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
415744d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
415844d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
415944d93782SGreg Clayton         }
416044d93782SGreg Clayton 
4161b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
416244d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
416344d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
416444d93782SGreg Clayton           if (!inst)
416544d93782SGreg Clayton             break;
416644d93782SGreg Clayton 
4167ec990867SGreg Clayton           const int line_y = m_min_y + i;
4168ec990867SGreg Clayton           window.MoveCursor(1, line_y);
416944d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
417044d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
417144d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
417244d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
417344d93782SGreg Clayton           attr_t highlight_attr = 0;
417444d93782SGreg Clayton           attr_t bp_attr = 0;
417544d93782SGreg Clayton           if (is_pc_line)
417644d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
417744d93782SGreg Clayton           else if (line_is_selected)
417844d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
417944d93782SGreg Clayton 
4180b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4181b9c1b51eSKate Stone               bp_file_addrs.end())
418244d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
418344d93782SGreg Clayton 
418444d93782SGreg Clayton           if (bp_attr)
418544d93782SGreg Clayton             window.AttributeOn(bp_attr);
418644d93782SGreg Clayton 
4187324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4188b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4189b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
419044d93782SGreg Clayton 
419144d93782SGreg Clayton           if (bp_attr)
419244d93782SGreg Clayton             window.AttributeOff(bp_attr);
419344d93782SGreg Clayton 
419444d93782SGreg Clayton           window.PutChar(ACS_VLINE);
419544d93782SGreg Clayton           // Mark the line with the PC with a diamond
419644d93782SGreg Clayton           if (is_pc_line)
419744d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
419844d93782SGreg Clayton           else
419944d93782SGreg Clayton             window.PutChar(' ');
420044d93782SGreg Clayton 
420144d93782SGreg Clayton           if (highlight_attr)
420244d93782SGreg Clayton             window.AttributeOn(highlight_attr);
420344d93782SGreg Clayton 
420444d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
420544d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
420644d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
420744d93782SGreg Clayton 
4208c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4209c5dac77aSEugene Zelenko             mnemonic = nullptr;
4210c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4211c5dac77aSEugene Zelenko             operands = nullptr;
4212c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4213c5dac77aSEugene Zelenko             comment = nullptr;
421444d93782SGreg Clayton 
421544d93782SGreg Clayton           strm.Clear();
421644d93782SGreg Clayton 
4217c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
421844d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4219c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
422044d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4221c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
422244d93782SGreg Clayton             strm.Printf("%s", mnemonic);
422344d93782SGreg Clayton 
422444d93782SGreg Clayton           int right_pad = 1;
4225c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
422644d93782SGreg Clayton 
4227b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4228b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
422944d93782SGreg Clayton             StopInfoSP stop_info_sp;
423044d93782SGreg Clayton             if (thread)
423144d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4232b9c1b51eSKate Stone             if (stop_info_sp) {
423344d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4234b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
423544d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4236ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
423744d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4238b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4239b9c1b51eSKate Stone                 // line_y);
4240b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4241b9c1b51eSKate Stone                               stop_description);
424244d93782SGreg Clayton               }
4243b9c1b51eSKate Stone             } else {
4244ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
424544d93782SGreg Clayton             }
424644d93782SGreg Clayton           }
424744d93782SGreg Clayton           if (highlight_attr)
424844d93782SGreg Clayton             window.AttributeOff(highlight_attr);
424944d93782SGreg Clayton         }
425044d93782SGreg Clayton       }
425144d93782SGreg Clayton     }
425244d93782SGreg Clayton     window.DeferredRefresh();
425344d93782SGreg Clayton     return true; // Drawing handled
425444d93782SGreg Clayton   }
425544d93782SGreg Clayton 
4256b9c1b51eSKate Stone   size_t GetNumLines() {
425744d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
425844d93782SGreg Clayton     if (num_lines == 0)
425944d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
426044d93782SGreg Clayton     return num_lines;
426144d93782SGreg Clayton   }
426244d93782SGreg Clayton 
4263b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
426444d93782SGreg Clayton     if (m_file_sp)
426544d93782SGreg Clayton       return m_file_sp->GetNumLines();
426644d93782SGreg Clayton     return 0;
426744d93782SGreg Clayton   }
4268315b6884SEugene Zelenko 
4269b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
427044d93782SGreg Clayton     if (m_disassembly_sp)
427144d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
427244d93782SGreg Clayton     return 0;
427344d93782SGreg Clayton   }
427444d93782SGreg Clayton 
4275b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
427644d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
427744d93782SGreg Clayton     const size_t num_lines = GetNumLines();
427844d93782SGreg Clayton 
4279b9c1b51eSKate Stone     switch (c) {
428044d93782SGreg Clayton     case ',':
428144d93782SGreg Clayton     case KEY_PPAGE:
428244d93782SGreg Clayton       // Page up key
42833985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
428444d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
428544d93782SGreg Clayton       else
428644d93782SGreg Clayton         m_first_visible_line = 0;
428744d93782SGreg Clayton       m_selected_line = m_first_visible_line;
428844d93782SGreg Clayton       return eKeyHandled;
428944d93782SGreg Clayton 
429044d93782SGreg Clayton     case '.':
429144d93782SGreg Clayton     case KEY_NPAGE:
429244d93782SGreg Clayton       // Page down key
429344d93782SGreg Clayton       {
429444d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
429544d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
429644d93782SGreg Clayton         else if (num_lines < num_visible_lines)
429744d93782SGreg Clayton           m_first_visible_line = 0;
429844d93782SGreg Clayton         else
429944d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
430044d93782SGreg Clayton         m_selected_line = m_first_visible_line;
430144d93782SGreg Clayton       }
430244d93782SGreg Clayton       return eKeyHandled;
430344d93782SGreg Clayton 
430444d93782SGreg Clayton     case KEY_UP:
4305b9c1b51eSKate Stone       if (m_selected_line > 0) {
430644d93782SGreg Clayton         m_selected_line--;
43073985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
430844d93782SGreg Clayton           m_first_visible_line = m_selected_line;
430944d93782SGreg Clayton       }
431044d93782SGreg Clayton       return eKeyHandled;
431144d93782SGreg Clayton 
431244d93782SGreg Clayton     case KEY_DOWN:
4313b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
431444d93782SGreg Clayton         m_selected_line++;
431544d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
431644d93782SGreg Clayton           m_first_visible_line++;
431744d93782SGreg Clayton       }
431844d93782SGreg Clayton       return eKeyHandled;
431944d93782SGreg Clayton 
432044d93782SGreg Clayton     case '\r':
432144d93782SGreg Clayton     case '\n':
432244d93782SGreg Clayton     case KEY_ENTER:
432344d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4324b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4325b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4326b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4327b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4328b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4329b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
433044d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4331b9c1b51eSKate Stone               m_selected_line +
4332b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4333431b1584SAdrian Prantl               0,     // Unspecified column.
43342411167fSJim Ingham               0,     // No offset
433544d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
433644d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
433744d93782SGreg Clayton               false,               // internal
4338055ad9beSIlia K               false,               // request_hardware
4339055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
434044d93782SGreg Clayton           // Make breakpoint one shot
434144d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
434244d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
434344d93782SGreg Clayton         }
4344b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4345b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4346b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4347b9c1b51eSKate Stone                                       .get();
4348b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4349b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4350b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
435144d93782SGreg Clayton           Address addr = inst->GetAddress();
4352b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4353b9c1b51eSKate Stone               addr,   // lldb_private::Address
435444d93782SGreg Clayton               false,  // internal
435544d93782SGreg Clayton               false); // request_hardware
435644d93782SGreg Clayton           // Make breakpoint one shot
435744d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
435844d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
435944d93782SGreg Clayton         }
436044d93782SGreg Clayton       }
436144d93782SGreg Clayton       return eKeyHandled;
436244d93782SGreg Clayton 
436344d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4364b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4365b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4366b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4367b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4368b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4369b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
437044d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4371b9c1b51eSKate Stone               m_selected_line +
4372b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4373431b1584SAdrian Prantl               0,     // No column specified.
43742411167fSJim Ingham               0,     // No offset
437544d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
437644d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
437744d93782SGreg Clayton               false,               // internal
4378055ad9beSIlia K               false,               // request_hardware
4379055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
438044d93782SGreg Clayton         }
4381b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4382b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4383b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4384b9c1b51eSKate Stone                                       .get();
4385b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4386b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4387b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
438844d93782SGreg Clayton           Address addr = inst->GetAddress();
4389b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4390b9c1b51eSKate Stone               addr,   // lldb_private::Address
439144d93782SGreg Clayton               false,  // internal
439244d93782SGreg Clayton               false); // request_hardware
439344d93782SGreg Clayton         }
439444d93782SGreg Clayton       }
439544d93782SGreg Clayton       return eKeyHandled;
439644d93782SGreg Clayton 
439744d93782SGreg Clayton     case 'd': // 'd' == detach and let run
439844d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
439944d93782SGreg Clayton     {
4400b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4401b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
440244d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
440344d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
440444d93782SGreg Clayton     }
440544d93782SGreg Clayton       return eKeyHandled;
440644d93782SGreg Clayton 
440744d93782SGreg Clayton     case 'k':
440844d93782SGreg Clayton       // 'k' == kill
440944d93782SGreg Clayton       {
4410b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4411b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
441244d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4413ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
441444d93782SGreg Clayton       }
441544d93782SGreg Clayton       return eKeyHandled;
441644d93782SGreg Clayton 
441744d93782SGreg Clayton     case 'c':
441844d93782SGreg Clayton       // 'c' == continue
441944d93782SGreg Clayton       {
4420b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4421b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
442244d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
442344d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
442444d93782SGreg Clayton       }
442544d93782SGreg Clayton       return eKeyHandled;
442644d93782SGreg Clayton 
442744d93782SGreg Clayton     case 'o':
442844d93782SGreg Clayton       // 'o' == step out
442944d93782SGreg Clayton       {
4430b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4431b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4432b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4433b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
443444d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
443544d93782SGreg Clayton         }
443644d93782SGreg Clayton       }
443744d93782SGreg Clayton       return eKeyHandled;
4438315b6884SEugene Zelenko 
443944d93782SGreg Clayton     case 'n': // 'n' == step over
444044d93782SGreg Clayton     case 'N': // 'N' == step over instruction
444144d93782SGreg Clayton     {
4442b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4443b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4444b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4445b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
444644d93782SGreg Clayton         bool source_step = (c == 'n');
444744d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
444844d93782SGreg Clayton       }
444944d93782SGreg Clayton     }
445044d93782SGreg Clayton       return eKeyHandled;
4451315b6884SEugene Zelenko 
445244d93782SGreg Clayton     case 's': // 's' == step into
445344d93782SGreg Clayton     case 'S': // 'S' == step into instruction
445444d93782SGreg Clayton     {
4455b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4456b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4457b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4458b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
445944d93782SGreg Clayton         bool source_step = (c == 's');
44604b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
446144d93782SGreg Clayton       }
446244d93782SGreg Clayton     }
446344d93782SGreg Clayton       return eKeyHandled;
446444d93782SGreg Clayton 
446544d93782SGreg Clayton     case 'h':
446644d93782SGreg Clayton       window.CreateHelpSubwindow();
446744d93782SGreg Clayton       return eKeyHandled;
446844d93782SGreg Clayton 
446944d93782SGreg Clayton     default:
447044d93782SGreg Clayton       break;
447144d93782SGreg Clayton     }
447244d93782SGreg Clayton     return eKeyNotHandled;
447344d93782SGreg Clayton   }
447444d93782SGreg Clayton 
447544d93782SGreg Clayton protected:
447644d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
447744d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
447844d93782SGreg Clayton 
447944d93782SGreg Clayton   Debugger &m_debugger;
448044d93782SGreg Clayton   SymbolContext m_sc;
448144d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
448244d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
448344d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
448444d93782SGreg Clayton   AddressRange m_disassembly_range;
4485ec990867SGreg Clayton   StreamString m_title;
448644d93782SGreg Clayton   lldb::user_id_t m_tid;
448744d93782SGreg Clayton   int m_line_width;
448844d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
448944d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
449044d93782SGreg Clayton   uint32_t m_stop_id;
449144d93782SGreg Clayton   uint32_t m_frame_idx;
449244d93782SGreg Clayton   int m_first_visible_line;
449344d93782SGreg Clayton   int m_min_x;
449444d93782SGreg Clayton   int m_min_y;
449544d93782SGreg Clayton   int m_max_x;
449644d93782SGreg Clayton   int m_max_y;
449744d93782SGreg Clayton };
449844d93782SGreg Clayton 
449944d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
450044d93782SGreg Clayton 
4501b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4502b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
450344d93782SGreg Clayton 
4504b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
450544d93782SGreg Clayton   IOHandler::Activate();
4506b9c1b51eSKate Stone   if (!m_app_ap) {
450744d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
450844d93782SGreg Clayton 
450944d93782SGreg Clayton     // This is both a window and a menu delegate
4510b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4511b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
451244d93782SGreg Clayton 
4513b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4514b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4515b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4516b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4517b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4518b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
451944d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4520b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4521b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
452244d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
452344d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
452444d93782SGreg Clayton 
4525b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4526b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4527b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4528b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4529b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4530b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
453144d93782SGreg Clayton 
4532b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4533b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4534b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4535b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4536b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4537b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4538b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4539b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
454044d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4541b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4542b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4543b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4544b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4545b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4546b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4547b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
454844d93782SGreg Clayton 
4549b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4550b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4551b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4552b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4553b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4554b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4555b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4556b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4557b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
455844d93782SGreg Clayton 
4559b9c1b51eSKate Stone     MenuSP view_menu_sp(
4560b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4561b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4562b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4563b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4564b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4565b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4566b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4567b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4568b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4569b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4570b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4571b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
457244d93782SGreg Clayton 
4573b9c1b51eSKate Stone     MenuSP help_menu_sp(
4574b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4575b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4576b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
457744d93782SGreg Clayton 
457844d93782SGreg Clayton     m_app_ap->Initialize();
457944d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
458044d93782SGreg Clayton 
458144d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
458244d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
458344d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
458444d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
458544d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
458644d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
458744d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
458844d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
458944d93782SGreg Clayton 
459044d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
459144d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
459244d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
459344d93782SGreg Clayton     Rect source_bounds;
459444d93782SGreg Clayton     Rect variables_bounds;
459544d93782SGreg Clayton     Rect threads_bounds;
459644d93782SGreg Clayton     Rect source_variables_bounds;
4597b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4598b9c1b51eSKate Stone                                            threads_bounds);
4599b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4600b9c1b51eSKate Stone                                                       variables_bounds);
460144d93782SGreg Clayton 
4602b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4603b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
460405097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
460505097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4606b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4607b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
460844d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
460944d93782SGreg Clayton 
4610b9c1b51eSKate Stone     WindowSP source_window_sp(
4611b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4612b9c1b51eSKate Stone     WindowSP variables_window_sp(
4613b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4614b9c1b51eSKate Stone     WindowSP threads_window_sp(
4615b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4616b9c1b51eSKate Stone     WindowSP status_window_sp(
46176bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4618b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4619b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4620b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4621b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4622b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4623b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4624b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4625b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4626ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4627b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4628b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4629b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4630b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
463144d93782SGreg Clayton 
46325fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
46335fdb09bbSGreg Clayton     static bool g_showed_help = false;
4634b9c1b51eSKate Stone     if (!g_showed_help) {
46355fdb09bbSGreg Clayton       g_showed_help = true;
46365fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46375fdb09bbSGreg Clayton     }
46385fdb09bbSGreg Clayton 
463944d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
464044d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
464144d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
464244d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
464344d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
464444d93782SGreg Clayton   }
464544d93782SGreg Clayton }
464644d93782SGreg Clayton 
4647b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
464844d93782SGreg Clayton 
4649b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
465044d93782SGreg Clayton   m_app_ap->Run(m_debugger);
465144d93782SGreg Clayton   SetIsDone(true);
465244d93782SGreg Clayton }
465344d93782SGreg Clayton 
4654315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
465544d93782SGreg Clayton 
4656b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
465744d93782SGreg Clayton 
4658b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
465944d93782SGreg Clayton 
4660b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
466144d93782SGreg Clayton 
4662315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4663