144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
644d93782SGreg Clayton //
744d93782SGreg Clayton //===----------------------------------------------------------------------===//
844d93782SGreg Clayton 
92f3df613SZachary Turner #include "lldb/Core/IOHandler.h"
102f3df613SZachary Turner 
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
167c9aa073STodd Fiala #if defined(__APPLE__)
177c9aa073STodd Fiala #include <deque>
187c9aa073STodd Fiala #endif
1944d93782SGreg Clayton #include <string>
2044d93782SGreg Clayton 
2144d93782SGreg Clayton #include "lldb/Core/Debugger.h"
2244d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
23672d2c12SJonas Devlieghere #include "lldb/Host/File.h"
24672d2c12SJonas Devlieghere #include "lldb/Utility/Predicate.h"
25672d2c12SJonas Devlieghere #include "lldb/Utility/Status.h"
26672d2c12SJonas Devlieghere #include "lldb/Utility/StreamString.h"
27672d2c12SJonas Devlieghere #include "lldb/Utility/StringList.h"
28672d2c12SJonas Devlieghere #include "lldb/lldb-forward.h"
292f3df613SZachary Turner 
30cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3144d93782SGreg Clayton #include "lldb/Host/Editline.h"
32cacde7dfSTodd Fiala #endif
3344d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
352f3df613SZachary Turner #ifndef LLDB_DISABLE_CURSES
362f3df613SZachary Turner #include "lldb/Breakpoint/BreakpointLocation.h"
372f3df613SZachary Turner #include "lldb/Core/Module.h"
382f3df613SZachary Turner #include "lldb/Core/ValueObject.h"
392f3df613SZachary Turner #include "lldb/Core/ValueObjectRegister.h"
4044d93782SGreg Clayton #include "lldb/Symbol/Block.h"
4144d93782SGreg Clayton #include "lldb/Symbol/Function.h"
4244d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
43c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
44c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
452f3df613SZachary Turner #include "lldb/Target/RegisterContext.h"
46c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
472f3df613SZachary Turner #include "lldb/Target/StopInfo.h"
48b9c1b51eSKate Stone #include "lldb/Target/Target.h"
49b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
50d821c997SPavel Labath #include "lldb/Utility/State.h"
51c5dac77aSEugene Zelenko #endif
527c9aa073STodd Fiala 
53672d2c12SJonas Devlieghere #include "llvm/ADT/StringRef.h"
542f3df613SZachary Turner 
55fab31220STed Woodward #ifdef _MSC_VER
56aaea8ee6SZachary Turner #include "lldb/Host/windows/windows.h"
57fab31220STed Woodward #endif
58fab31220STed Woodward 
59672d2c12SJonas Devlieghere #include <memory>
60672d2c12SJonas Devlieghere #include <mutex>
612f3df613SZachary Turner 
62672d2c12SJonas Devlieghere #include <assert.h>
63672d2c12SJonas Devlieghere #include <ctype.h>
64672d2c12SJonas Devlieghere #include <errno.h>
65672d2c12SJonas Devlieghere #include <locale.h>
66672d2c12SJonas Devlieghere #include <stdint.h>
67672d2c12SJonas Devlieghere #include <stdio.h>
68672d2c12SJonas Devlieghere #include <string.h>
69672d2c12SJonas Devlieghere #include <type_traits>
702f3df613SZachary Turner 
7144d93782SGreg Clayton using namespace lldb;
7244d93782SGreg Clayton using namespace lldb_private;
7344d93782SGreg Clayton 
74b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type)
75b9c1b51eSKate Stone     : IOHandler(debugger, type,
7644d93782SGreg Clayton                 StreamFileSP(), // Adopt STDIN from top input reader
7744d93782SGreg Clayton                 StreamFileSP(), // Adopt STDOUT from top input reader
78340b0309SGreg Clayton                 StreamFileSP(), // Adopt STDERR from top input reader
79d77c2e09SJonas Devlieghere                 0,              // Flags
80d77c2e09SJonas Devlieghere                 nullptr         // Shadow file recorder
81d77c2e09SJonas Devlieghere       ) {}
8244d93782SGreg Clayton 
83b9c1b51eSKate Stone IOHandler::IOHandler(Debugger &debugger, IOHandler::Type type,
8444d93782SGreg Clayton                      const lldb::StreamFileSP &input_sp,
8544d93782SGreg Clayton                      const lldb::StreamFileSP &output_sp,
86d77c2e09SJonas Devlieghere                      const lldb::StreamFileSP &error_sp, uint32_t flags,
87d77c2e09SJonas Devlieghere                      repro::DataRecorder *data_recorder)
88b9c1b51eSKate Stone     : m_debugger(debugger), m_input_sp(input_sp), m_output_sp(output_sp),
89d77c2e09SJonas Devlieghere       m_error_sp(error_sp), m_data_recorder(data_recorder), m_popped(false),
90d77c2e09SJonas Devlieghere       m_flags(flags), m_type(type), m_user_data(nullptr), m_done(false),
91d77c2e09SJonas Devlieghere       m_active(false) {
9244d93782SGreg Clayton   // If any files are not specified, then adopt them from the top input reader.
9344d93782SGreg Clayton   if (!m_input_sp || !m_output_sp || !m_error_sp)
94b9c1b51eSKate Stone     debugger.AdoptTopIOHandlerFilesIfInvalid(m_input_sp, m_output_sp,
9544d93782SGreg Clayton                                              m_error_sp);
9644d93782SGreg Clayton }
9744d93782SGreg Clayton 
98315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
9944d93782SGreg Clayton 
100b9c1b51eSKate Stone int IOHandler::GetInputFD() {
101c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
10244d93782SGreg Clayton }
10344d93782SGreg Clayton 
104b9c1b51eSKate Stone int IOHandler::GetOutputFD() {
105c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10644d93782SGreg Clayton }
10744d93782SGreg Clayton 
108b9c1b51eSKate Stone int IOHandler::GetErrorFD() {
109c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
11044d93782SGreg Clayton }
11144d93782SGreg Clayton 
112b9c1b51eSKate Stone FILE *IOHandler::GetInputFILE() {
113c5dac77aSEugene Zelenko   return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11444d93782SGreg Clayton }
11544d93782SGreg Clayton 
116b9c1b51eSKate Stone FILE *IOHandler::GetOutputFILE() {
117c5dac77aSEugene Zelenko   return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
11844d93782SGreg Clayton }
11944d93782SGreg Clayton 
120b9c1b51eSKate Stone FILE *IOHandler::GetErrorFILE() {
121c5dac77aSEugene Zelenko   return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
12244d93782SGreg Clayton }
12344d93782SGreg Clayton 
124b9c1b51eSKate Stone StreamFileSP &IOHandler::GetInputStreamFile() { return m_input_sp; }
12544d93782SGreg Clayton 
126b9c1b51eSKate Stone StreamFileSP &IOHandler::GetOutputStreamFile() { return m_output_sp; }
12744d93782SGreg Clayton 
128b9c1b51eSKate Stone StreamFileSP &IOHandler::GetErrorStreamFile() { return m_error_sp; }
12944d93782SGreg Clayton 
130b9c1b51eSKate Stone bool IOHandler::GetIsInteractive() {
131340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsInteractive();
132340b0309SGreg Clayton }
133340b0309SGreg Clayton 
134b9c1b51eSKate Stone bool IOHandler::GetIsRealTerminal() {
135340b0309SGreg Clayton   return GetInputStreamFile()->GetFile().GetIsRealTerminal();
136340b0309SGreg Clayton }
13744d93782SGreg Clayton 
138b9c1b51eSKate Stone void IOHandler::SetPopped(bool b) { m_popped.SetValue(b, eBroadcastOnChange); }
139e30f11d9SKate Stone 
140b9c1b51eSKate Stone void IOHandler::WaitForPop() { m_popped.WaitForValueEqualTo(true); }
141e30f11d9SKate Stone 
142b9c1b51eSKate Stone void IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len) {
143b9c1b51eSKate Stone   if (stream) {
14416ff8604SSaleem Abdulrasool     std::lock_guard<std::recursive_mutex> guard(m_mutex);
1454446487dSPavel Labath     if (m_top)
1464446487dSPavel Labath       m_top->PrintAsync(stream, s, len);
1474446487dSPavel Labath   }
1484446487dSPavel Labath }
1494446487dSPavel Labath 
1507a120c8bSZachary Turner IOHandlerConfirm::IOHandlerConfirm(Debugger &debugger, llvm::StringRef prompt,
151b9c1b51eSKate Stone                                    bool default_response)
152b9c1b51eSKate Stone     : IOHandlerEditline(
153b9c1b51eSKate Stone           debugger, IOHandler::Type::Confirm,
154c5dac77aSEugene Zelenko           nullptr, // nullptr editline_name means no history loaded/saved
155514d8cd8SZachary Turner           llvm::StringRef(), // No prompt
156514d8cd8SZachary Turner           llvm::StringRef(), // No continuation prompt
15744d93782SGreg Clayton           false,             // Multi-line
158e30f11d9SKate Stone           false, // Don't colorize the prompt (i.e. the confirm message.)
159d77c2e09SJonas Devlieghere           0, *this, nullptr),
160b9c1b51eSKate Stone       m_default_response(default_response), m_user_response(default_response) {
16144d93782SGreg Clayton   StreamString prompt_stream;
16244d93782SGreg Clayton   prompt_stream.PutCString(prompt);
16344d93782SGreg Clayton   if (m_default_response)
16444d93782SGreg Clayton     prompt_stream.Printf(": [Y/n] ");
16544d93782SGreg Clayton   else
16644d93782SGreg Clayton     prompt_stream.Printf(": [y/N] ");
16744d93782SGreg Clayton 
168514d8cd8SZachary Turner   SetPrompt(prompt_stream.GetString());
16944d93782SGreg Clayton }
17044d93782SGreg Clayton 
171315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
17244d93782SGreg Clayton 
173*2fc20f65SRaphael Isemann int IOHandlerConfirm::IOHandlerComplete(IOHandler &io_handler,
174*2fc20f65SRaphael Isemann                                         CompletionRequest &request) {
175*2fc20f65SRaphael Isemann   if (request.GetRawCursorPos() == 0) {
176*2fc20f65SRaphael Isemann     if (m_default_response)
177*2fc20f65SRaphael Isemann       request.AddCompletion("y");
178*2fc20f65SRaphael Isemann     else
179*2fc20f65SRaphael Isemann       request.AddCompletion("n");
18044d93782SGreg Clayton   }
181*2fc20f65SRaphael Isemann   return request.GetNumberOfMatches();
18244d93782SGreg Clayton }
18344d93782SGreg Clayton 
184b9c1b51eSKate Stone void IOHandlerConfirm::IOHandlerInputComplete(IOHandler &io_handler,
185b9c1b51eSKate Stone                                               std::string &line) {
186b9c1b51eSKate Stone   if (line.empty()) {
18744d93782SGreg Clayton     // User just hit enter, set the response to the default
18844d93782SGreg Clayton     m_user_response = m_default_response;
18944d93782SGreg Clayton     io_handler.SetIsDone(true);
19044d93782SGreg Clayton     return;
19144d93782SGreg Clayton   }
19244d93782SGreg Clayton 
193b9c1b51eSKate Stone   if (line.size() == 1) {
194b9c1b51eSKate Stone     switch (line[0]) {
19544d93782SGreg Clayton     case 'y':
19644d93782SGreg Clayton     case 'Y':
19744d93782SGreg Clayton       m_user_response = true;
19844d93782SGreg Clayton       io_handler.SetIsDone(true);
19944d93782SGreg Clayton       return;
20044d93782SGreg Clayton     case 'n':
20144d93782SGreg Clayton     case 'N':
20244d93782SGreg Clayton       m_user_response = false;
20344d93782SGreg Clayton       io_handler.SetIsDone(true);
20444d93782SGreg Clayton       return;
20544d93782SGreg Clayton     default:
20644d93782SGreg Clayton       break;
20744d93782SGreg Clayton     }
20844d93782SGreg Clayton   }
20944d93782SGreg Clayton 
210b9c1b51eSKate Stone   if (line == "yes" || line == "YES" || line == "Yes") {
21144d93782SGreg Clayton     m_user_response = true;
21244d93782SGreg Clayton     io_handler.SetIsDone(true);
213b9c1b51eSKate Stone   } else if (line == "no" || line == "NO" || line == "No") {
21444d93782SGreg Clayton     m_user_response = false;
21544d93782SGreg Clayton     io_handler.SetIsDone(true);
21644d93782SGreg Clayton   }
21744d93782SGreg Clayton }
21844d93782SGreg Clayton 
219*2fc20f65SRaphael Isemann int IOHandlerDelegate::IOHandlerComplete(IOHandler &io_handler,
220*2fc20f65SRaphael Isemann                                          CompletionRequest &request) {
221b9c1b51eSKate Stone   switch (m_completion) {
22244d93782SGreg Clayton   case Completion::None:
22344d93782SGreg Clayton     break;
22444d93782SGreg Clayton 
22544d93782SGreg Clayton   case Completion::LLDBCommand:
226b9c1b51eSKate Stone     return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion(
227*2fc20f65SRaphael Isemann         request);
228b9c1b51eSKate Stone   case Completion::Expression: {
2297f88829cSRaphael Isemann     CompletionResult result;
230*2fc20f65SRaphael Isemann     CompletionRequest subrequest(request.GetRawLine(),
231*2fc20f65SRaphael Isemann                                  request.GetRawCursorPos(), result);
232b9c1b51eSKate Stone     CommandCompletions::InvokeCommonCompletionCallbacks(
233b9c1b51eSKate Stone         io_handler.GetDebugger().GetCommandInterpreter(),
234*2fc20f65SRaphael Isemann         CommandCompletions::eVariablePathCompletion, subrequest, nullptr);
235*2fc20f65SRaphael Isemann     StringList matches;
236*2fc20f65SRaphael Isemann     StringList descriptions;
2377f88829cSRaphael Isemann     result.GetMatches(matches);
2387f88829cSRaphael Isemann     result.GetDescriptions(descriptions);
23944d93782SGreg Clayton 
240*2fc20f65SRaphael Isemann     size_t num_matches = subrequest.GetNumberOfMatches();
241b9c1b51eSKate Stone     if (num_matches > 0) {
242175f0930SJonas Devlieghere       std::string common_prefix = matches.LongestCommonPrefix();
243*2fc20f65SRaphael Isemann       const size_t partial_name_len =
244*2fc20f65SRaphael Isemann           subrequest.GetCursorArgumentPrefix().size();
24544d93782SGreg Clayton 
24605097246SAdrian Prantl       // If we matched a unique single command, add a space... Only do this if
24705097246SAdrian Prantl       // the completer told us this was a complete word, however...
248*2fc20f65SRaphael Isemann       if (num_matches == 1 && subrequest.GetWordComplete()) {
24944d93782SGreg Clayton         common_prefix.push_back(' ');
25044d93782SGreg Clayton       }
25144d93782SGreg Clayton       common_prefix.erase(0, partial_name_len);
252*2fc20f65SRaphael Isemann       request.AddCompletion(common_prefix);
25344d93782SGreg Clayton     }
254*2fc20f65SRaphael Isemann     request.AddCompletions(matches, descriptions);
255*2fc20f65SRaphael Isemann     return request.GetNumberOfMatches();
256b9c1b51eSKate Stone   } break;
25744d93782SGreg Clayton   }
25844d93782SGreg Clayton 
25944d93782SGreg Clayton   return 0;
26044d93782SGreg Clayton }
26144d93782SGreg Clayton 
262b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
263b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
26444d93782SGreg Clayton     const char *editline_name, // Used for saving history files
265514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
266514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
267d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
268b9c1b51eSKate Stone     : IOHandlerEditline(debugger, type,
26944d93782SGreg Clayton                         StreamFileSP(), // Inherit input from top input reader
27044d93782SGreg Clayton                         StreamFileSP(), // Inherit output from top input reader
27144d93782SGreg Clayton                         StreamFileSP(), // Inherit error from top input reader
272340b0309SGreg Clayton                         0,              // Flags
27344d93782SGreg Clayton                         editline_name,  // Used for saving history files
274b9c1b51eSKate Stone                         prompt, continuation_prompt, multi_line, color_prompts,
275d77c2e09SJonas Devlieghere                         line_number_start, delegate, data_recorder) {}
27644d93782SGreg Clayton 
277b9c1b51eSKate Stone IOHandlerEditline::IOHandlerEditline(
278b9c1b51eSKate Stone     Debugger &debugger, IOHandler::Type type,
279b9c1b51eSKate Stone     const lldb::StreamFileSP &input_sp, const lldb::StreamFileSP &output_sp,
280b9c1b51eSKate Stone     const lldb::StreamFileSP &error_sp, uint32_t flags,
28144d93782SGreg Clayton     const char *editline_name, // Used for saving history files
282514d8cd8SZachary Turner     llvm::StringRef prompt, llvm::StringRef continuation_prompt,
283514d8cd8SZachary Turner     bool multi_line, bool color_prompts, uint32_t line_number_start,
284d77c2e09SJonas Devlieghere     IOHandlerDelegate &delegate, repro::DataRecorder *data_recorder)
285d77c2e09SJonas Devlieghere     : IOHandler(debugger, type, input_sp, output_sp, error_sp, flags,
286d77c2e09SJonas Devlieghere                 data_recorder),
287cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
288d5b44036SJonas Devlieghere       m_editline_up(),
289cacde7dfSTodd Fiala #endif
290b9c1b51eSKate Stone       m_delegate(delegate), m_prompt(), m_continuation_prompt(),
291b9c1b51eSKate Stone       m_current_lines_ptr(nullptr), m_base_line_number(line_number_start),
292b9c1b51eSKate Stone       m_curr_line_idx(UINT32_MAX), m_multi_line(multi_line),
293b9c1b51eSKate Stone       m_color_prompts(color_prompts), m_interrupt_exits(true),
294b9c1b51eSKate Stone       m_editing(false) {
29544d93782SGreg Clayton   SetPrompt(prompt);
29644d93782SGreg Clayton 
297cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
298914b8d98SDeepak Panickal   bool use_editline = false;
299340b0309SGreg Clayton 
300340b0309SGreg Clayton   use_editline = m_input_sp->GetFile().GetIsRealTerminal();
30144d93782SGreg Clayton 
302b9c1b51eSKate Stone   if (use_editline) {
303d5b44036SJonas Devlieghere     m_editline_up.reset(new Editline(editline_name, GetInputFILE(),
304b9c1b51eSKate Stone                                      GetOutputFILE(), GetErrorFILE(),
305e30f11d9SKate Stone                                      m_color_prompts));
306d5b44036SJonas Devlieghere     m_editline_up->SetIsInputCompleteCallback(IsInputCompleteCallback, this);
307d5b44036SJonas Devlieghere     m_editline_up->SetAutoCompleteCallback(AutoCompleteCallback, this);
308e30f11d9SKate Stone     // See if the delegate supports fixing indentation
309e30f11d9SKate Stone     const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
310b9c1b51eSKate Stone     if (indent_chars) {
311b9c1b51eSKate Stone       // The delegate does support indentation, hook it up so when any
31205097246SAdrian Prantl       // indentation character is typed, the delegate gets a chance to fix it
313d5b44036SJonas Devlieghere       m_editline_up->SetFixIndentationCallback(FixIndentationCallback, this,
314b9c1b51eSKate Stone                                                indent_chars);
315e30f11d9SKate Stone     }
31644d93782SGreg Clayton   }
317cacde7dfSTodd Fiala #endif
318e30f11d9SKate Stone   SetBaseLineNumber(m_base_line_number);
319514d8cd8SZachary Turner   SetPrompt(prompt);
320e30f11d9SKate Stone   SetContinuationPrompt(continuation_prompt);
32144d93782SGreg Clayton }
32244d93782SGreg Clayton 
323b9c1b51eSKate Stone IOHandlerEditline::~IOHandlerEditline() {
324cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
325d5b44036SJonas Devlieghere   m_editline_up.reset();
326cacde7dfSTodd Fiala #endif
32744d93782SGreg Clayton }
32844d93782SGreg Clayton 
329b9c1b51eSKate Stone void IOHandlerEditline::Activate() {
330e30f11d9SKate Stone   IOHandler::Activate();
3310affb582SDave Lee   m_delegate.IOHandlerActivated(*this, GetIsInteractive());
332e30f11d9SKate Stone }
333e30f11d9SKate Stone 
334b9c1b51eSKate Stone void IOHandlerEditline::Deactivate() {
335e30f11d9SKate Stone   IOHandler::Deactivate();
336e30f11d9SKate Stone   m_delegate.IOHandlerDeactivated(*this);
337e30f11d9SKate Stone }
338e30f11d9SKate Stone 
339b9c1b51eSKate Stone bool IOHandlerEditline::GetLine(std::string &line, bool &interrupted) {
340cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
341d5b44036SJonas Devlieghere   if (m_editline_up) {
342d77c2e09SJonas Devlieghere     bool b = m_editline_up->GetLine(line, interrupted);
343d77c2e09SJonas Devlieghere     if (m_data_recorder)
344d77c2e09SJonas Devlieghere       m_data_recorder->Record(line, true);
345d77c2e09SJonas Devlieghere     return b;
346b9c1b51eSKate Stone   } else {
347cacde7dfSTodd Fiala #endif
34844d93782SGreg Clayton     line.clear();
34944d93782SGreg Clayton 
35044d93782SGreg Clayton     FILE *in = GetInputFILE();
351b9c1b51eSKate Stone     if (in) {
352b9c1b51eSKate Stone       if (GetIsInteractive()) {
353c5dac77aSEugene Zelenko         const char *prompt = nullptr;
354e30f11d9SKate Stone 
355e30f11d9SKate Stone         if (m_multi_line && m_curr_line_idx > 0)
356e30f11d9SKate Stone           prompt = GetContinuationPrompt();
357e30f11d9SKate Stone 
358c5dac77aSEugene Zelenko         if (prompt == nullptr)
359e30f11d9SKate Stone           prompt = GetPrompt();
360e30f11d9SKate Stone 
361b9c1b51eSKate Stone         if (prompt && prompt[0]) {
36244d93782SGreg Clayton           FILE *out = GetOutputFILE();
363b9c1b51eSKate Stone           if (out) {
36444d93782SGreg Clayton             ::fprintf(out, "%s", prompt);
36544d93782SGreg Clayton             ::fflush(out);
36644d93782SGreg Clayton           }
36744d93782SGreg Clayton         }
36844d93782SGreg Clayton       }
36944d93782SGreg Clayton       char buffer[256];
37044d93782SGreg Clayton       bool done = false;
3710f86e6e7SGreg Clayton       bool got_line = false;
372e034a04eSGreg Clayton       m_editing = true;
373b9c1b51eSKate Stone       while (!done) {
374e7167908SNathan Lanza #ifdef _WIN32
375e7167908SNathan Lanza         // ReadFile on Windows is supposed to set ERROR_OPERATION_ABORTED
376e7167908SNathan Lanza         // according to the docs on MSDN. However, this has evidently been a
377e7167908SNathan Lanza         // known bug since Windows 8. Therefore, we can't detect if a signal
378e7167908SNathan Lanza         // interrupted in the fgets. So pressing ctrl-c causes the repl to end
379e7167908SNathan Lanza         // and the process to exit. A temporary workaround is just to attempt to
380e7167908SNathan Lanza         // fgets twice until this bug is fixed.
381e7167908SNathan Lanza         if (fgets(buffer, sizeof(buffer), in) == nullptr &&
382e7167908SNathan Lanza             fgets(buffer, sizeof(buffer), in) == nullptr) {
383cb305205SNathan Lanza           // this is the equivalent of EINTR for Windows
384cb305205SNathan Lanza           if (GetLastError() == ERROR_OPERATION_ABORTED)
385cb305205SNathan Lanza             continue;
386e7167908SNathan Lanza #else
387b9c1b51eSKate Stone       if (fgets(buffer, sizeof(buffer), in) == nullptr) {
388e7167908SNathan Lanza #endif
389c7797accSGreg Clayton           const int saved_errno = errno;
390c9cf5798SGreg Clayton           if (feof(in))
39144d93782SGreg Clayton             done = true;
392b9c1b51eSKate Stone           else if (ferror(in)) {
393c7797accSGreg Clayton             if (saved_errno != EINTR)
394c7797accSGreg Clayton               done = true;
395c7797accSGreg Clayton           }
396b9c1b51eSKate Stone         } else {
3970f86e6e7SGreg Clayton           got_line = true;
39844d93782SGreg Clayton           size_t buffer_len = strlen(buffer);
39944d93782SGreg Clayton           assert(buffer[buffer_len] == '\0');
40044d93782SGreg Clayton           char last_char = buffer[buffer_len - 1];
401b9c1b51eSKate Stone           if (last_char == '\r' || last_char == '\n') {
40244d93782SGreg Clayton             done = true;
40344d93782SGreg Clayton             // Strip trailing newlines
404b9c1b51eSKate Stone             while (last_char == '\r' || last_char == '\n') {
40544d93782SGreg Clayton               --buffer_len;
40644d93782SGreg Clayton               if (buffer_len == 0)
40744d93782SGreg Clayton                 break;
40844d93782SGreg Clayton               last_char = buffer[buffer_len - 1];
40944d93782SGreg Clayton             }
41044d93782SGreg Clayton           }
41144d93782SGreg Clayton           line.append(buffer, buffer_len);
41244d93782SGreg Clayton         }
41344d93782SGreg Clayton       }
414e034a04eSGreg Clayton       m_editing = false;
415d77c2e09SJonas Devlieghere       if (m_data_recorder && got_line)
416d77c2e09SJonas Devlieghere         m_data_recorder->Record(line, true);
41705097246SAdrian Prantl       // We might have gotten a newline on a line by itself make sure to return
41805097246SAdrian Prantl       // true in this case.
4190f86e6e7SGreg Clayton       return got_line;
420b9c1b51eSKate Stone     } else {
42144d93782SGreg Clayton       // No more input file, we are done...
42244d93782SGreg Clayton       SetIsDone(true);
42344d93782SGreg Clayton     }
424340b0309SGreg Clayton     return false;
425cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
42644d93782SGreg Clayton   }
427cacde7dfSTodd Fiala #endif
42844d93782SGreg Clayton }
42944d93782SGreg Clayton 
430cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
431b9c1b51eSKate Stone bool IOHandlerEditline::IsInputCompleteCallback(Editline *editline,
43244d93782SGreg Clayton                                                 StringList &lines,
433b9c1b51eSKate Stone                                                 void *baton) {
43444d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
435b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader,
436b9c1b51eSKate Stone                                                               lines);
437e30f11d9SKate Stone }
438e30f11d9SKate Stone 
439b9c1b51eSKate Stone int IOHandlerEditline::FixIndentationCallback(Editline *editline,
440e30f11d9SKate Stone                                               const StringList &lines,
441e30f11d9SKate Stone                                               int cursor_position,
442b9c1b51eSKate Stone                                               void *baton) {
443e30f11d9SKate Stone   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
444b9c1b51eSKate Stone   return editline_reader->m_delegate.IOHandlerFixIndentation(
445b9c1b51eSKate Stone       *editline_reader, lines, cursor_position);
44644d93782SGreg Clayton }
44744d93782SGreg Clayton 
448*2fc20f65SRaphael Isemann int IOHandlerEditline::AutoCompleteCallback(CompletionRequest &request,
449*2fc20f65SRaphael Isemann                                             void *baton) {
45044d93782SGreg Clayton   IOHandlerEditline *editline_reader = (IOHandlerEditline *)baton;
45144d93782SGreg Clayton   if (editline_reader)
452*2fc20f65SRaphael Isemann     return editline_reader->m_delegate.IOHandlerComplete(*editline_reader,
453*2fc20f65SRaphael Isemann                                                          request);
45444d93782SGreg Clayton   return 0;
45544d93782SGreg Clayton }
456cacde7dfSTodd Fiala #endif
45744d93782SGreg Clayton 
458b9c1b51eSKate Stone const char *IOHandlerEditline::GetPrompt() {
459cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
460d5b44036SJonas Devlieghere   if (m_editline_up) {
461d5b44036SJonas Devlieghere     return m_editline_up->GetPrompt();
462b9c1b51eSKate Stone   } else {
463cacde7dfSTodd Fiala #endif
464cacde7dfSTodd Fiala     if (m_prompt.empty())
465c5dac77aSEugene Zelenko       return nullptr;
466cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
467cacde7dfSTodd Fiala   }
468cacde7dfSTodd Fiala #endif
46944d93782SGreg Clayton   return m_prompt.c_str();
47044d93782SGreg Clayton }
47144d93782SGreg Clayton 
472514d8cd8SZachary Turner bool IOHandlerEditline::SetPrompt(llvm::StringRef prompt) {
473514d8cd8SZachary Turner   m_prompt = prompt;
474514d8cd8SZachary Turner 
475cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
476d5b44036SJonas Devlieghere   if (m_editline_up)
477d5b44036SJonas Devlieghere     m_editline_up->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
478cacde7dfSTodd Fiala #endif
47944d93782SGreg Clayton   return true;
48044d93782SGreg Clayton }
48144d93782SGreg Clayton 
482b9c1b51eSKate Stone const char *IOHandlerEditline::GetContinuationPrompt() {
483b9c1b51eSKate Stone   return (m_continuation_prompt.empty() ? nullptr
484b9c1b51eSKate Stone                                         : m_continuation_prompt.c_str());
485e30f11d9SKate Stone }
486e30f11d9SKate Stone 
487514d8cd8SZachary Turner void IOHandlerEditline::SetContinuationPrompt(llvm::StringRef prompt) {
488514d8cd8SZachary Turner   m_continuation_prompt = prompt;
489e30f11d9SKate Stone 
490d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
491d5b44036SJonas Devlieghere   if (m_editline_up)
492d5b44036SJonas Devlieghere     m_editline_up->SetContinuationPrompt(m_continuation_prompt.empty()
493b9c1b51eSKate Stone                                              ? nullptr
494b9c1b51eSKate Stone                                              : m_continuation_prompt.c_str());
495d553d00cSZachary Turner #endif
496e30f11d9SKate Stone }
497e30f11d9SKate Stone 
498b9c1b51eSKate Stone void IOHandlerEditline::SetBaseLineNumber(uint32_t line) {
499f6913cd7SGreg Clayton   m_base_line_number = line;
500f6913cd7SGreg Clayton }
501e30f11d9SKate Stone 
502b9c1b51eSKate Stone uint32_t IOHandlerEditline::GetCurrentLineIndex() const {
503d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
504d5b44036SJonas Devlieghere   if (m_editline_up)
505d5b44036SJonas Devlieghere     return m_editline_up->GetCurrentLine();
506e30f11d9SKate Stone #endif
507e30f11d9SKate Stone   return m_curr_line_idx;
508e30f11d9SKate Stone }
509e30f11d9SKate Stone 
510b9c1b51eSKate Stone bool IOHandlerEditline::GetLines(StringList &lines, bool &interrupted) {
511e30f11d9SKate Stone   m_current_lines_ptr = &lines;
512e30f11d9SKate Stone 
51344d93782SGreg Clayton   bool success = false;
514cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
515d5b44036SJonas Devlieghere   if (m_editline_up) {
516d5b44036SJonas Devlieghere     return m_editline_up->GetLines(m_base_line_number, lines, interrupted);
517b9c1b51eSKate Stone   } else {
518cacde7dfSTodd Fiala #endif
519e30f11d9SKate Stone     bool done = false;
52097206d57SZachary Turner     Status error;
52144d93782SGreg Clayton 
522b9c1b51eSKate Stone     while (!done) {
523f6913cd7SGreg Clayton       // Show line numbers if we are asked to
52444d93782SGreg Clayton       std::string line;
525b9c1b51eSKate Stone       if (m_base_line_number > 0 && GetIsInteractive()) {
526f6913cd7SGreg Clayton         FILE *out = GetOutputFILE();
527f6913cd7SGreg Clayton         if (out)
528b9c1b51eSKate Stone           ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(),
529b9c1b51eSKate Stone                     GetPrompt() == nullptr ? " " : "");
530f6913cd7SGreg Clayton       }
531f6913cd7SGreg Clayton 
532e30f11d9SKate Stone       m_curr_line_idx = lines.GetSize();
533e30f11d9SKate Stone 
534f0066ad0SGreg Clayton       bool interrupted = false;
535b9c1b51eSKate Stone       if (GetLine(line, interrupted) && !interrupted) {
53644d93782SGreg Clayton         lines.AppendString(line);
537e30f11d9SKate Stone         done = m_delegate.IOHandlerIsInputComplete(*this, lines);
538b9c1b51eSKate Stone       } else {
539e30f11d9SKate Stone         done = true;
54044d93782SGreg Clayton       }
54144d93782SGreg Clayton     }
54244d93782SGreg Clayton     success = lines.GetSize() > 0;
543cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
54444d93782SGreg Clayton   }
545cacde7dfSTodd Fiala #endif
54644d93782SGreg Clayton   return success;
54744d93782SGreg Clayton }
54844d93782SGreg Clayton 
54905097246SAdrian Prantl // Each IOHandler gets to run until it is done. It should read data from the
55005097246SAdrian Prantl // "in" and place output into "out" and "err and return when done.
551b9c1b51eSKate Stone void IOHandlerEditline::Run() {
55244d93782SGreg Clayton   std::string line;
553b9c1b51eSKate Stone   while (IsActive()) {
554f0066ad0SGreg Clayton     bool interrupted = false;
555b9c1b51eSKate Stone     if (m_multi_line) {
55644d93782SGreg Clayton       StringList lines;
557b9c1b51eSKate Stone       if (GetLines(lines, interrupted)) {
558b9c1b51eSKate Stone         if (interrupted) {
559e30f11d9SKate Stone           m_done = m_interrupt_exits;
560e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
561e30f11d9SKate Stone 
562b9c1b51eSKate Stone         } else {
56344d93782SGreg Clayton           line = lines.CopyList();
56444d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
56544d93782SGreg Clayton         }
566b9c1b51eSKate Stone       } else {
56744d93782SGreg Clayton         m_done = true;
56844d93782SGreg Clayton       }
569b9c1b51eSKate Stone     } else {
570b9c1b51eSKate Stone       if (GetLine(line, interrupted)) {
571e30f11d9SKate Stone         if (interrupted)
572e30f11d9SKate Stone           m_delegate.IOHandlerInputInterrupted(*this, line);
573e30f11d9SKate Stone         else
57444d93782SGreg Clayton           m_delegate.IOHandlerInputComplete(*this, line);
575b9c1b51eSKate Stone       } else {
57644d93782SGreg Clayton         m_done = true;
57744d93782SGreg Clayton       }
57844d93782SGreg Clayton     }
57944d93782SGreg Clayton   }
58044d93782SGreg Clayton }
58144d93782SGreg Clayton 
582b9c1b51eSKate Stone void IOHandlerEditline::Cancel() {
583cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
584d5b44036SJonas Devlieghere   if (m_editline_up)
585d5b44036SJonas Devlieghere     m_editline_up->Cancel();
586cacde7dfSTodd Fiala #endif
587e68f5d6bSGreg Clayton }
588e68f5d6bSGreg Clayton 
589b9c1b51eSKate Stone bool IOHandlerEditline::Interrupt() {
590f0066ad0SGreg Clayton   // Let the delgate handle it first
591f0066ad0SGreg Clayton   if (m_delegate.IOHandlerInterrupt(*this))
592f0066ad0SGreg Clayton     return true;
593f0066ad0SGreg Clayton 
594cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
595d5b44036SJonas Devlieghere   if (m_editline_up)
596d5b44036SJonas Devlieghere     return m_editline_up->Interrupt();
597cacde7dfSTodd Fiala #endif
598f0066ad0SGreg Clayton   return false;
59944d93782SGreg Clayton }
60044d93782SGreg Clayton 
601b9c1b51eSKate Stone void IOHandlerEditline::GotEOF() {
602cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
603d5b44036SJonas Devlieghere   if (m_editline_up)
604d5b44036SJonas Devlieghere     m_editline_up->Interrupt();
605cacde7dfSTodd Fiala #endif
60644d93782SGreg Clayton }
60744d93782SGreg Clayton 
608b9c1b51eSKate Stone void IOHandlerEditline::PrintAsync(Stream *stream, const char *s, size_t len) {
6094446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
610d5b44036SJonas Devlieghere   if (m_editline_up)
611d5b44036SJonas Devlieghere     m_editline_up->PrintAsync(stream, s, len);
6124446487dSPavel Labath   else
6134446487dSPavel Labath #endif
614fab31220STed Woodward   {
615fab31220STed Woodward #ifdef _MSC_VER
616341e4789SDawn Perchik     const char *prompt = GetPrompt();
617b9c1b51eSKate Stone     if (prompt) {
618fab31220STed Woodward       // Back up over previous prompt using Windows API
619fab31220STed Woodward       CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
620fab31220STed Woodward       HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
621fab31220STed Woodward       GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
622fab31220STed Woodward       COORD coord = screen_buffer_info.dwCursorPosition;
623fab31220STed Woodward       coord.X -= strlen(prompt);
624fab31220STed Woodward       if (coord.X < 0)
625fab31220STed Woodward         coord.X = 0;
626fab31220STed Woodward       SetConsoleCursorPosition(console_handle, coord);
627fab31220STed Woodward     }
628fab31220STed Woodward #endif
6294446487dSPavel Labath     IOHandler::PrintAsync(stream, s, len);
630341e4789SDawn Perchik #ifdef _MSC_VER
631fab31220STed Woodward     if (prompt)
632b9c1b51eSKate Stone       IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt,
633b9c1b51eSKate Stone                             strlen(prompt));
634341e4789SDawn Perchik #endif
635fab31220STed Woodward   }
6364446487dSPavel Labath }
6374446487dSPavel Labath 
63805097246SAdrian Prantl // we may want curses to be disabled for some builds for instance, windows
639914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
640914b8d98SDeepak Panickal 
64144d93782SGreg Clayton #define KEY_RETURN 10
64244d93782SGreg Clayton #define KEY_ESCAPE 27
64344d93782SGreg Clayton 
644b9c1b51eSKate Stone namespace curses {
64544d93782SGreg Clayton class Menu;
64644d93782SGreg Clayton class MenuDelegate;
64744d93782SGreg Clayton class Window;
64844d93782SGreg Clayton class WindowDelegate;
64944d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP;
65044d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
65144d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP;
65244d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
65344d93782SGreg Clayton typedef std::vector<MenuSP> Menus;
65444d93782SGreg Clayton typedef std::vector<WindowSP> Windows;
65544d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates;
65644d93782SGreg Clayton 
65744d93782SGreg Clayton #if 0
65844d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
65944d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
66044d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
66144d93782SGreg Clayton #endif
662315b6884SEugene Zelenko 
663b9c1b51eSKate Stone struct Point {
66444d93782SGreg Clayton   int x;
66544d93782SGreg Clayton   int y;
66644d93782SGreg Clayton 
667b9c1b51eSKate Stone   Point(int _x = 0, int _y = 0) : x(_x), y(_y) {}
66844d93782SGreg Clayton 
669b9c1b51eSKate Stone   void Clear() {
67044d93782SGreg Clayton     x = 0;
67144d93782SGreg Clayton     y = 0;
67244d93782SGreg Clayton   }
67344d93782SGreg Clayton 
674b9c1b51eSKate Stone   Point &operator+=(const Point &rhs) {
67544d93782SGreg Clayton     x += rhs.x;
67644d93782SGreg Clayton     y += rhs.y;
67744d93782SGreg Clayton     return *this;
67844d93782SGreg Clayton   }
67944d93782SGreg Clayton 
680b9c1b51eSKate Stone   void Dump() { printf("(x=%i, y=%i)\n", x, y); }
68144d93782SGreg Clayton };
68244d93782SGreg Clayton 
683b9c1b51eSKate Stone bool operator==(const Point &lhs, const Point &rhs) {
68444d93782SGreg Clayton   return lhs.x == rhs.x && lhs.y == rhs.y;
68544d93782SGreg Clayton }
686315b6884SEugene Zelenko 
687b9c1b51eSKate Stone bool operator!=(const Point &lhs, const Point &rhs) {
68844d93782SGreg Clayton   return lhs.x != rhs.x || lhs.y != rhs.y;
68944d93782SGreg Clayton }
69044d93782SGreg Clayton 
691b9c1b51eSKate Stone struct Size {
69244d93782SGreg Clayton   int width;
69344d93782SGreg Clayton   int height;
694b9c1b51eSKate Stone   Size(int w = 0, int h = 0) : width(w), height(h) {}
69544d93782SGreg Clayton 
696b9c1b51eSKate Stone   void Clear() {
69744d93782SGreg Clayton     width = 0;
69844d93782SGreg Clayton     height = 0;
69944d93782SGreg Clayton   }
70044d93782SGreg Clayton 
701b9c1b51eSKate Stone   void Dump() { printf("(w=%i, h=%i)\n", width, height); }
70244d93782SGreg Clayton };
70344d93782SGreg Clayton 
704b9c1b51eSKate Stone bool operator==(const Size &lhs, const Size &rhs) {
70544d93782SGreg Clayton   return lhs.width == rhs.width && lhs.height == rhs.height;
70644d93782SGreg Clayton }
707315b6884SEugene Zelenko 
708b9c1b51eSKate Stone bool operator!=(const Size &lhs, const Size &rhs) {
70944d93782SGreg Clayton   return lhs.width != rhs.width || lhs.height != rhs.height;
71044d93782SGreg Clayton }
71144d93782SGreg Clayton 
712b9c1b51eSKate Stone struct Rect {
71344d93782SGreg Clayton   Point origin;
71444d93782SGreg Clayton   Size size;
71544d93782SGreg Clayton 
716b9c1b51eSKate Stone   Rect() : origin(), size() {}
71744d93782SGreg Clayton 
718b9c1b51eSKate Stone   Rect(const Point &p, const Size &s) : origin(p), size(s) {}
71944d93782SGreg Clayton 
720b9c1b51eSKate Stone   void Clear() {
72144d93782SGreg Clayton     origin.Clear();
72244d93782SGreg Clayton     size.Clear();
72344d93782SGreg Clayton   }
72444d93782SGreg Clayton 
725b9c1b51eSKate Stone   void Dump() {
726b9c1b51eSKate Stone     printf("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width,
727b9c1b51eSKate Stone            size.height);
72844d93782SGreg Clayton   }
72944d93782SGreg Clayton 
730b9c1b51eSKate Stone   void Inset(int w, int h) {
73144d93782SGreg Clayton     if (size.width > w * 2)
73244d93782SGreg Clayton       size.width -= w * 2;
73344d93782SGreg Clayton     origin.x += w;
73444d93782SGreg Clayton 
73544d93782SGreg Clayton     if (size.height > h * 2)
73644d93782SGreg Clayton       size.height -= h * 2;
73744d93782SGreg Clayton     origin.y += h;
73844d93782SGreg Clayton   }
739315b6884SEugene Zelenko 
74005097246SAdrian Prantl   // Return a status bar rectangle which is the last line of this rectangle.
74105097246SAdrian Prantl   // This rectangle will be modified to not include the status bar area.
742b9c1b51eSKate Stone   Rect MakeStatusBar() {
74344d93782SGreg Clayton     Rect status_bar;
744b9c1b51eSKate Stone     if (size.height > 1) {
74544d93782SGreg Clayton       status_bar.origin.x = origin.x;
74644d93782SGreg Clayton       status_bar.origin.y = size.height;
74744d93782SGreg Clayton       status_bar.size.width = size.width;
74844d93782SGreg Clayton       status_bar.size.height = 1;
74944d93782SGreg Clayton       --size.height;
75044d93782SGreg Clayton     }
75144d93782SGreg Clayton     return status_bar;
75244d93782SGreg Clayton   }
75344d93782SGreg Clayton 
75405097246SAdrian Prantl   // Return a menubar rectangle which is the first line of this rectangle. This
75505097246SAdrian Prantl   // rectangle will be modified to not include the menubar area.
756b9c1b51eSKate Stone   Rect MakeMenuBar() {
75744d93782SGreg Clayton     Rect menubar;
758b9c1b51eSKate Stone     if (size.height > 1) {
75944d93782SGreg Clayton       menubar.origin.x = origin.x;
76044d93782SGreg Clayton       menubar.origin.y = origin.y;
76144d93782SGreg Clayton       menubar.size.width = size.width;
76244d93782SGreg Clayton       menubar.size.height = 1;
76344d93782SGreg Clayton       ++origin.y;
76444d93782SGreg Clayton       --size.height;
76544d93782SGreg Clayton     }
76644d93782SGreg Clayton     return menubar;
76744d93782SGreg Clayton   }
76844d93782SGreg Clayton 
769b9c1b51eSKate Stone   void HorizontalSplitPercentage(float top_percentage, Rect &top,
770b9c1b51eSKate Stone                                  Rect &bottom) const {
77144d93782SGreg Clayton     float top_height = top_percentage * size.height;
77244d93782SGreg Clayton     HorizontalSplit(top_height, top, bottom);
77344d93782SGreg Clayton   }
77444d93782SGreg Clayton 
775b9c1b51eSKate Stone   void HorizontalSplit(int top_height, Rect &top, Rect &bottom) const {
77644d93782SGreg Clayton     top = *this;
777b9c1b51eSKate Stone     if (top_height < size.height) {
77844d93782SGreg Clayton       top.size.height = top_height;
77944d93782SGreg Clayton       bottom.origin.x = origin.x;
78044d93782SGreg Clayton       bottom.origin.y = origin.y + top.size.height;
78144d93782SGreg Clayton       bottom.size.width = size.width;
78244d93782SGreg Clayton       bottom.size.height = size.height - top.size.height;
783b9c1b51eSKate Stone     } else {
78444d93782SGreg Clayton       bottom.Clear();
78544d93782SGreg Clayton     }
78644d93782SGreg Clayton   }
78744d93782SGreg Clayton 
788b9c1b51eSKate Stone   void VerticalSplitPercentage(float left_percentage, Rect &left,
789b9c1b51eSKate Stone                                Rect &right) const {
79044d93782SGreg Clayton     float left_width = left_percentage * size.width;
79144d93782SGreg Clayton     VerticalSplit(left_width, left, right);
79244d93782SGreg Clayton   }
79344d93782SGreg Clayton 
794b9c1b51eSKate Stone   void VerticalSplit(int left_width, Rect &left, Rect &right) const {
79544d93782SGreg Clayton     left = *this;
796b9c1b51eSKate Stone     if (left_width < size.width) {
79744d93782SGreg Clayton       left.size.width = left_width;
79844d93782SGreg Clayton       right.origin.x = origin.x + left.size.width;
79944d93782SGreg Clayton       right.origin.y = origin.y;
80044d93782SGreg Clayton       right.size.width = size.width - left.size.width;
80144d93782SGreg Clayton       right.size.height = size.height;
802b9c1b51eSKate Stone     } else {
80344d93782SGreg Clayton       right.Clear();
80444d93782SGreg Clayton     }
80544d93782SGreg Clayton   }
80644d93782SGreg Clayton };
80744d93782SGreg Clayton 
808b9c1b51eSKate Stone bool operator==(const Rect &lhs, const Rect &rhs) {
80944d93782SGreg Clayton   return lhs.origin == rhs.origin && lhs.size == rhs.size;
81044d93782SGreg Clayton }
811315b6884SEugene Zelenko 
812b9c1b51eSKate Stone bool operator!=(const Rect &lhs, const Rect &rhs) {
81344d93782SGreg Clayton   return lhs.origin != rhs.origin || lhs.size != rhs.size;
81444d93782SGreg Clayton }
81544d93782SGreg Clayton 
816b9c1b51eSKate Stone enum HandleCharResult {
81744d93782SGreg Clayton   eKeyNotHandled = 0,
81844d93782SGreg Clayton   eKeyHandled = 1,
81944d93782SGreg Clayton   eQuitApplication = 2
82044d93782SGreg Clayton };
82144d93782SGreg Clayton 
822b9c1b51eSKate Stone enum class MenuActionResult {
82344d93782SGreg Clayton   Handled,
82444d93782SGreg Clayton   NotHandled,
82544d93782SGreg Clayton   Quit // Exit all menus and quit
82644d93782SGreg Clayton };
82744d93782SGreg Clayton 
828b9c1b51eSKate Stone struct KeyHelp {
82944d93782SGreg Clayton   int ch;
83044d93782SGreg Clayton   const char *description;
83144d93782SGreg Clayton };
83244d93782SGreg Clayton 
833b9c1b51eSKate Stone class WindowDelegate {
83444d93782SGreg Clayton public:
835b9c1b51eSKate Stone   virtual ~WindowDelegate() = default;
83644d93782SGreg Clayton 
837b9c1b51eSKate Stone   virtual bool WindowDelegateDraw(Window &window, bool force) {
83844d93782SGreg Clayton     return false; // Drawing not handled
83944d93782SGreg Clayton   }
84044d93782SGreg Clayton 
841b9c1b51eSKate Stone   virtual HandleCharResult WindowDelegateHandleChar(Window &window, int key) {
84244d93782SGreg Clayton     return eKeyNotHandled;
84344d93782SGreg Clayton   }
84444d93782SGreg Clayton 
845b9c1b51eSKate Stone   virtual const char *WindowDelegateGetHelpText() { return nullptr; }
84644d93782SGreg Clayton 
847b9c1b51eSKate Stone   virtual KeyHelp *WindowDelegateGetKeyHelp() { return nullptr; }
84844d93782SGreg Clayton };
84944d93782SGreg Clayton 
850b9c1b51eSKate Stone class HelpDialogDelegate : public WindowDelegate {
85144d93782SGreg Clayton public:
85244d93782SGreg Clayton   HelpDialogDelegate(const char *text, KeyHelp *key_help_array);
85344d93782SGreg Clayton 
854bd5ae6b4SGreg Clayton   ~HelpDialogDelegate() override;
85544d93782SGreg Clayton 
856b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
85744d93782SGreg Clayton 
858b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
85944d93782SGreg Clayton 
860b9c1b51eSKate Stone   size_t GetNumLines() const { return m_text.GetSize(); }
86144d93782SGreg Clayton 
862b9c1b51eSKate Stone   size_t GetMaxLineLength() const { return m_text.GetMaxStringLength(); }
86344d93782SGreg Clayton 
86444d93782SGreg Clayton protected:
86544d93782SGreg Clayton   StringList m_text;
86644d93782SGreg Clayton   int m_first_visible_line;
86744d93782SGreg Clayton };
86844d93782SGreg Clayton 
869b9c1b51eSKate Stone class Window {
87044d93782SGreg Clayton public:
871b9c1b51eSKate Stone   Window(const char *name)
872b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
873b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
874b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(false),
875b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {}
87644d93782SGreg Clayton 
877b9c1b51eSKate Stone   Window(const char *name, WINDOW *w, bool del = true)
878b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_panel(nullptr), m_parent(nullptr),
879b9c1b51eSKate Stone         m_subwindows(), m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
880b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(del),
881b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
88244d93782SGreg Clayton     if (w)
88344d93782SGreg Clayton       Reset(w);
88444d93782SGreg Clayton   }
88544d93782SGreg Clayton 
886b9c1b51eSKate Stone   Window(const char *name, const Rect &bounds)
887b9c1b51eSKate Stone       : m_name(name), m_window(nullptr), m_parent(nullptr), m_subwindows(),
888b9c1b51eSKate Stone         m_delegate_sp(), m_curr_active_window_idx(UINT32_MAX),
889b9c1b51eSKate Stone         m_prev_active_window_idx(UINT32_MAX), m_delete(true),
890b9c1b51eSKate Stone         m_needs_update(true), m_can_activate(true), m_is_subwin(false) {
891b9c1b51eSKate Stone     Reset(::newwin(bounds.size.height, bounds.size.width, bounds.origin.y,
892b9c1b51eSKate Stone                    bounds.origin.y));
89344d93782SGreg Clayton   }
89444d93782SGreg Clayton 
895b9c1b51eSKate Stone   virtual ~Window() {
89644d93782SGreg Clayton     RemoveSubWindows();
89744d93782SGreg Clayton     Reset();
89844d93782SGreg Clayton   }
89944d93782SGreg Clayton 
900b9c1b51eSKate Stone   void Reset(WINDOW *w = nullptr, bool del = true) {
90144d93782SGreg Clayton     if (m_window == w)
90244d93782SGreg Clayton       return;
90344d93782SGreg Clayton 
904b9c1b51eSKate Stone     if (m_panel) {
90544d93782SGreg Clayton       ::del_panel(m_panel);
906c5dac77aSEugene Zelenko       m_panel = nullptr;
90744d93782SGreg Clayton     }
908b9c1b51eSKate Stone     if (m_window && m_delete) {
90944d93782SGreg Clayton       ::delwin(m_window);
910c5dac77aSEugene Zelenko       m_window = nullptr;
91144d93782SGreg Clayton       m_delete = false;
91244d93782SGreg Clayton     }
913b9c1b51eSKate Stone     if (w) {
91444d93782SGreg Clayton       m_window = w;
91544d93782SGreg Clayton       m_panel = ::new_panel(m_window);
91644d93782SGreg Clayton       m_delete = del;
91744d93782SGreg Clayton     }
91844d93782SGreg Clayton   }
91944d93782SGreg Clayton 
92044d93782SGreg Clayton   void AttributeOn(attr_t attr) { ::wattron(m_window, attr); }
92144d93782SGreg Clayton   void AttributeOff(attr_t attr) { ::wattroff(m_window, attr); }
922b9c1b51eSKate Stone   void Box(chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) {
923b9c1b51eSKate Stone     ::box(m_window, v_char, h_char);
924b9c1b51eSKate Stone   }
92544d93782SGreg Clayton   void Clear() { ::wclear(m_window); }
92644d93782SGreg Clayton   void Erase() { ::werase(m_window); }
927b9c1b51eSKate Stone   Rect GetBounds() {
928b9c1b51eSKate Stone     return Rect(GetParentOrigin(), GetSize());
929b9c1b51eSKate Stone   } // Get the rectangle in our parent window
93044d93782SGreg Clayton   int GetChar() { return ::wgetch(m_window); }
93144d93782SGreg Clayton   int GetCursorX() { return getcurx(m_window); }
93244d93782SGreg Clayton   int GetCursorY() { return getcury(m_window); }
933b9c1b51eSKate Stone   Rect GetFrame() {
934b9c1b51eSKate Stone     return Rect(Point(), GetSize());
935b9c1b51eSKate Stone   } // Get our rectangle in our own coordinate system
93644d93782SGreg Clayton   Point GetParentOrigin() { return Point(GetParentX(), GetParentY()); }
93744d93782SGreg Clayton   Size GetSize() { return Size(GetWidth(), GetHeight()); }
93844d93782SGreg Clayton   int GetParentX() { return getparx(m_window); }
93944d93782SGreg Clayton   int GetParentY() { return getpary(m_window); }
94044d93782SGreg Clayton   int GetMaxX() { return getmaxx(m_window); }
94144d93782SGreg Clayton   int GetMaxY() { return getmaxy(m_window); }
94244d93782SGreg Clayton   int GetWidth() { return GetMaxX(); }
94344d93782SGreg Clayton   int GetHeight() { return GetMaxY(); }
94444d93782SGreg Clayton   void MoveCursor(int x, int y) { ::wmove(m_window, y, x); }
94544d93782SGreg Clayton   void MoveWindow(int x, int y) { MoveWindow(Point(x, y)); }
94644d93782SGreg Clayton   void Resize(int w, int h) { ::wresize(m_window, h, w); }
947b9c1b51eSKate Stone   void Resize(const Size &size) {
948b9c1b51eSKate Stone     ::wresize(m_window, size.height, size.width);
949b9c1b51eSKate Stone   }
95044d93782SGreg Clayton   void PutChar(int ch) { ::waddch(m_window, ch); }
95144d93782SGreg Clayton   void PutCString(const char *s, int len = -1) { ::waddnstr(m_window, s, len); }
952b9c1b51eSKate Stone   void SetBackground(int color_pair_idx) {
953b9c1b51eSKate Stone     ::wbkgd(m_window, COLOR_PAIR(color_pair_idx));
954b9c1b51eSKate Stone   }
95544d93782SGreg Clayton 
956b9c1b51eSKate Stone   void PutCStringTruncated(const char *s, int right_pad) {
95744d93782SGreg Clayton     int bytes_left = GetWidth() - GetCursorX();
958b9c1b51eSKate Stone     if (bytes_left > right_pad) {
95944d93782SGreg Clayton       bytes_left -= right_pad;
96044d93782SGreg Clayton       ::waddnstr(m_window, s, bytes_left);
96144d93782SGreg Clayton     }
96244d93782SGreg Clayton   }
96344d93782SGreg Clayton 
964b9c1b51eSKate Stone   void MoveWindow(const Point &origin) {
96544d93782SGreg Clayton     const bool moving_window = origin != GetParentOrigin();
966b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
96744d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
96844d93782SGreg Clayton       Size size = GetSize();
969b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, size.height, size.width, origin.y,
970b9c1b51eSKate Stone                      origin.x),
971b9c1b51eSKate Stone             true);
972b9c1b51eSKate Stone     } else {
97344d93782SGreg Clayton       ::mvwin(m_window, origin.y, origin.x);
97444d93782SGreg Clayton     }
97544d93782SGreg Clayton   }
97644d93782SGreg Clayton 
977b9c1b51eSKate Stone   void SetBounds(const Rect &bounds) {
97844d93782SGreg Clayton     const bool moving_window = bounds.origin != GetParentOrigin();
979b9c1b51eSKate Stone     if (m_is_subwin && moving_window) {
98044d93782SGreg Clayton       // Can't move subwindows, must delete and re-create
981b9c1b51eSKate Stone       Reset(::subwin(m_parent->m_window, bounds.size.height, bounds.size.width,
982b9c1b51eSKate Stone                      bounds.origin.y, bounds.origin.x),
983b9c1b51eSKate Stone             true);
984b9c1b51eSKate Stone     } else {
98544d93782SGreg Clayton       if (moving_window)
98644d93782SGreg Clayton         MoveWindow(bounds.origin);
98744d93782SGreg Clayton       Resize(bounds.size);
98844d93782SGreg Clayton     }
98944d93782SGreg Clayton   }
99044d93782SGreg Clayton 
991b9c1b51eSKate Stone   void Printf(const char *format, ...) __attribute__((format(printf, 2, 3))) {
99244d93782SGreg Clayton     va_list args;
99344d93782SGreg Clayton     va_start(args, format);
99444d93782SGreg Clayton     vwprintw(m_window, format, args);
99544d93782SGreg Clayton     va_end(args);
99644d93782SGreg Clayton   }
99744d93782SGreg Clayton 
998b9c1b51eSKate Stone   void Touch() {
99944d93782SGreg Clayton     ::touchwin(m_window);
100044d93782SGreg Clayton     if (m_parent)
100144d93782SGreg Clayton       m_parent->Touch();
100244d93782SGreg Clayton   }
100344d93782SGreg Clayton 
1004b9c1b51eSKate Stone   WindowSP CreateSubWindow(const char *name, const Rect &bounds,
1005b9c1b51eSKate Stone                            bool make_active) {
1006c6091d2bSJonas Devlieghere     auto get_window = [this, &bounds]() {
1007c6091d2bSJonas Devlieghere       return m_window
1008c6091d2bSJonas Devlieghere                  ? ::subwin(m_window, bounds.size.height, bounds.size.width,
1009c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x)
1010c6091d2bSJonas Devlieghere                  : ::newwin(bounds.size.height, bounds.size.width,
1011c6091d2bSJonas Devlieghere                             bounds.origin.y, bounds.origin.x);
1012c6091d2bSJonas Devlieghere     };
1013c6091d2bSJonas Devlieghere     WindowSP subwindow_sp = std::make_shared<Window>(name, get_window(), true);
1014c6091d2bSJonas Devlieghere     subwindow_sp->m_is_subwin = subwindow_sp.operator bool();
101544d93782SGreg Clayton     subwindow_sp->m_parent = this;
1016b9c1b51eSKate Stone     if (make_active) {
101744d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
101844d93782SGreg Clayton       m_curr_active_window_idx = m_subwindows.size();
101944d93782SGreg Clayton     }
102044d93782SGreg Clayton     m_subwindows.push_back(subwindow_sp);
102144d93782SGreg Clayton     ::top_panel(subwindow_sp->m_panel);
102244d93782SGreg Clayton     m_needs_update = true;
102344d93782SGreg Clayton     return subwindow_sp;
102444d93782SGreg Clayton   }
102544d93782SGreg Clayton 
1026b9c1b51eSKate Stone   bool RemoveSubWindow(Window *window) {
102744d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
102844d93782SGreg Clayton     size_t i = 0;
1029b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
1030b9c1b51eSKate Stone       if ((*pos).get() == window) {
103144d93782SGreg Clayton         if (m_prev_active_window_idx == i)
103244d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1033b9c1b51eSKate Stone         else if (m_prev_active_window_idx != UINT32_MAX &&
1034b9c1b51eSKate Stone                  m_prev_active_window_idx > i)
103544d93782SGreg Clayton           --m_prev_active_window_idx;
103644d93782SGreg Clayton 
103744d93782SGreg Clayton         if (m_curr_active_window_idx == i)
103844d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
1039b9c1b51eSKate Stone         else if (m_curr_active_window_idx != UINT32_MAX &&
1040b9c1b51eSKate Stone                  m_curr_active_window_idx > i)
104144d93782SGreg Clayton           --m_curr_active_window_idx;
104244d93782SGreg Clayton         window->Erase();
104344d93782SGreg Clayton         m_subwindows.erase(pos);
104444d93782SGreg Clayton         m_needs_update = true;
104544d93782SGreg Clayton         if (m_parent)
104644d93782SGreg Clayton           m_parent->Touch();
104744d93782SGreg Clayton         else
104844d93782SGreg Clayton           ::touchwin(stdscr);
104944d93782SGreg Clayton         return true;
105044d93782SGreg Clayton       }
105144d93782SGreg Clayton     }
105244d93782SGreg Clayton     return false;
105344d93782SGreg Clayton   }
105444d93782SGreg Clayton 
1055b9c1b51eSKate Stone   WindowSP FindSubWindow(const char *name) {
105644d93782SGreg Clayton     Windows::iterator pos, end = m_subwindows.end();
105744d93782SGreg Clayton     size_t i = 0;
1058b9c1b51eSKate Stone     for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) {
10598d20cfdfSJonas Devlieghere       if ((*pos)->m_name == name)
106044d93782SGreg Clayton         return *pos;
106144d93782SGreg Clayton     }
106244d93782SGreg Clayton     return WindowSP();
106344d93782SGreg Clayton   }
106444d93782SGreg Clayton 
1065b9c1b51eSKate Stone   void RemoveSubWindows() {
106644d93782SGreg Clayton     m_curr_active_window_idx = UINT32_MAX;
106744d93782SGreg Clayton     m_prev_active_window_idx = UINT32_MAX;
106844d93782SGreg Clayton     for (Windows::iterator pos = m_subwindows.begin();
1069b9c1b51eSKate Stone          pos != m_subwindows.end(); pos = m_subwindows.erase(pos)) {
107044d93782SGreg Clayton       (*pos)->Erase();
107144d93782SGreg Clayton     }
107244d93782SGreg Clayton     if (m_parent)
107344d93782SGreg Clayton       m_parent->Touch();
107444d93782SGreg Clayton     else
107544d93782SGreg Clayton       ::touchwin(stdscr);
107644d93782SGreg Clayton   }
107744d93782SGreg Clayton 
1078b9c1b51eSKate Stone   WINDOW *get() { return m_window; }
107944d93782SGreg Clayton 
1080b9c1b51eSKate Stone   operator WINDOW *() { return m_window; }
108144d93782SGreg Clayton 
108244d93782SGreg Clayton   // Window drawing utilities
1083b9c1b51eSKate Stone   void DrawTitleBox(const char *title, const char *bottom_message = nullptr) {
108444d93782SGreg Clayton     attr_t attr = 0;
108544d93782SGreg Clayton     if (IsActive())
108644d93782SGreg Clayton       attr = A_BOLD | COLOR_PAIR(2);
108744d93782SGreg Clayton     else
108844d93782SGreg Clayton       attr = 0;
108944d93782SGreg Clayton     if (attr)
109044d93782SGreg Clayton       AttributeOn(attr);
109144d93782SGreg Clayton 
109244d93782SGreg Clayton     Box();
109344d93782SGreg Clayton     MoveCursor(3, 0);
109444d93782SGreg Clayton 
1095b9c1b51eSKate Stone     if (title && title[0]) {
109644d93782SGreg Clayton       PutChar('<');
109744d93782SGreg Clayton       PutCString(title);
109844d93782SGreg Clayton       PutChar('>');
109944d93782SGreg Clayton     }
110044d93782SGreg Clayton 
1101b9c1b51eSKate Stone     if (bottom_message && bottom_message[0]) {
110244d93782SGreg Clayton       int bottom_message_length = strlen(bottom_message);
110344d93782SGreg Clayton       int x = GetWidth() - 3 - (bottom_message_length + 2);
110444d93782SGreg Clayton 
1105b9c1b51eSKate Stone       if (x > 0) {
110644d93782SGreg Clayton         MoveCursor(x, GetHeight() - 1);
110744d93782SGreg Clayton         PutChar('[');
110844d93782SGreg Clayton         PutCString(bottom_message);
110944d93782SGreg Clayton         PutChar(']');
1110b9c1b51eSKate Stone       } else {
111144d93782SGreg Clayton         MoveCursor(1, GetHeight() - 1);
111244d93782SGreg Clayton         PutChar('[');
111344d93782SGreg Clayton         PutCStringTruncated(bottom_message, 1);
111444d93782SGreg Clayton       }
111544d93782SGreg Clayton     }
111644d93782SGreg Clayton     if (attr)
111744d93782SGreg Clayton       AttributeOff(attr);
111844d93782SGreg Clayton   }
111944d93782SGreg Clayton 
1120b9c1b51eSKate Stone   virtual void Draw(bool force) {
112144d93782SGreg Clayton     if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw(*this, force))
112244d93782SGreg Clayton       return;
112344d93782SGreg Clayton 
112444d93782SGreg Clayton     for (auto &subwindow_sp : m_subwindows)
112544d93782SGreg Clayton       subwindow_sp->Draw(force);
112644d93782SGreg Clayton   }
112744d93782SGreg Clayton 
1128b9c1b51eSKate Stone   bool CreateHelpSubwindow() {
1129b9c1b51eSKate Stone     if (m_delegate_sp) {
113044d93782SGreg Clayton       const char *text = m_delegate_sp->WindowDelegateGetHelpText();
113144d93782SGreg Clayton       KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp();
1132b9c1b51eSKate Stone       if ((text && text[0]) || key_help) {
1133d5b44036SJonas Devlieghere         std::unique_ptr<HelpDialogDelegate> help_delegate_up(
1134b9c1b51eSKate Stone             new HelpDialogDelegate(text, key_help));
1135d5b44036SJonas Devlieghere         const size_t num_lines = help_delegate_up->GetNumLines();
1136d5b44036SJonas Devlieghere         const size_t max_length = help_delegate_up->GetMaxLineLength();
113744d93782SGreg Clayton         Rect bounds = GetBounds();
113844d93782SGreg Clayton         bounds.Inset(1, 1);
1139b9c1b51eSKate Stone         if (max_length + 4 < static_cast<size_t>(bounds.size.width)) {
114044d93782SGreg Clayton           bounds.origin.x += (bounds.size.width - max_length + 4) / 2;
114144d93782SGreg Clayton           bounds.size.width = max_length + 4;
1142b9c1b51eSKate Stone         } else {
1143b9c1b51eSKate Stone           if (bounds.size.width > 100) {
114444d93782SGreg Clayton             const int inset_w = bounds.size.width / 4;
114544d93782SGreg Clayton             bounds.origin.x += inset_w;
114644d93782SGreg Clayton             bounds.size.width -= 2 * inset_w;
114744d93782SGreg Clayton           }
114844d93782SGreg Clayton         }
114944d93782SGreg Clayton 
1150b9c1b51eSKate Stone         if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) {
115144d93782SGreg Clayton           bounds.origin.y += (bounds.size.height - num_lines + 2) / 2;
115244d93782SGreg Clayton           bounds.size.height = num_lines + 2;
1153b9c1b51eSKate Stone         } else {
1154b9c1b51eSKate Stone           if (bounds.size.height > 100) {
115544d93782SGreg Clayton             const int inset_h = bounds.size.height / 4;
115644d93782SGreg Clayton             bounds.origin.y += inset_h;
115744d93782SGreg Clayton             bounds.size.height -= 2 * inset_h;
115844d93782SGreg Clayton           }
115944d93782SGreg Clayton         }
11605fdb09bbSGreg Clayton         WindowSP help_window_sp;
11615fdb09bbSGreg Clayton         Window *parent_window = GetParent();
11625fdb09bbSGreg Clayton         if (parent_window)
11635fdb09bbSGreg Clayton           help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
11645fdb09bbSGreg Clayton         else
11655fdb09bbSGreg Clayton           help_window_sp = CreateSubWindow("Help", bounds, true);
1166b9c1b51eSKate Stone         help_window_sp->SetDelegate(
1167d5b44036SJonas Devlieghere             WindowDelegateSP(help_delegate_up.release()));
116844d93782SGreg Clayton         return true;
116944d93782SGreg Clayton       }
117044d93782SGreg Clayton     }
117144d93782SGreg Clayton     return false;
117244d93782SGreg Clayton   }
117344d93782SGreg Clayton 
1174b9c1b51eSKate Stone   virtual HandleCharResult HandleChar(int key) {
117544d93782SGreg Clayton     // Always check the active window first
117644d93782SGreg Clayton     HandleCharResult result = eKeyNotHandled;
117744d93782SGreg Clayton     WindowSP active_window_sp = GetActiveWindow();
1178b9c1b51eSKate Stone     if (active_window_sp) {
117944d93782SGreg Clayton       result = active_window_sp->HandleChar(key);
118044d93782SGreg Clayton       if (result != eKeyNotHandled)
118144d93782SGreg Clayton         return result;
118244d93782SGreg Clayton     }
118344d93782SGreg Clayton 
1184b9c1b51eSKate Stone     if (m_delegate_sp) {
118544d93782SGreg Clayton       result = m_delegate_sp->WindowDelegateHandleChar(*this, key);
118644d93782SGreg Clayton       if (result != eKeyNotHandled)
118744d93782SGreg Clayton         return result;
118844d93782SGreg Clayton     }
118944d93782SGreg Clayton 
119005097246SAdrian Prantl     // Then check for any windows that want any keys that weren't handled. This
119105097246SAdrian Prantl     // is typically only for a menubar. Make a copy of the subwindows in case
119205097246SAdrian Prantl     // any HandleChar() functions muck with the subwindows. If we don't do
119305097246SAdrian Prantl     // this, we can crash when iterating over the subwindows.
119444d93782SGreg Clayton     Windows subwindows(m_subwindows);
1195b9c1b51eSKate Stone     for (auto subwindow_sp : subwindows) {
1196b9c1b51eSKate Stone       if (!subwindow_sp->m_can_activate) {
119744d93782SGreg Clayton         HandleCharResult result = subwindow_sp->HandleChar(key);
119844d93782SGreg Clayton         if (result != eKeyNotHandled)
119944d93782SGreg Clayton           return result;
120044d93782SGreg Clayton       }
120144d93782SGreg Clayton     }
120244d93782SGreg Clayton 
120344d93782SGreg Clayton     return eKeyNotHandled;
120444d93782SGreg Clayton   }
120544d93782SGreg Clayton 
1206b9c1b51eSKate Stone   WindowSP GetActiveWindow() {
1207b9c1b51eSKate Stone     if (!m_subwindows.empty()) {
1208b9c1b51eSKate Stone       if (m_curr_active_window_idx >= m_subwindows.size()) {
1209b9c1b51eSKate Stone         if (m_prev_active_window_idx < m_subwindows.size()) {
121044d93782SGreg Clayton           m_curr_active_window_idx = m_prev_active_window_idx;
121144d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
1212b9c1b51eSKate Stone         } else if (IsActive()) {
121344d93782SGreg Clayton           m_prev_active_window_idx = UINT32_MAX;
121444d93782SGreg Clayton           m_curr_active_window_idx = UINT32_MAX;
121544d93782SGreg Clayton 
121644d93782SGreg Clayton           // Find first window that wants to be active if this window is active
121744d93782SGreg Clayton           const size_t num_subwindows = m_subwindows.size();
1218b9c1b51eSKate Stone           for (size_t i = 0; i < num_subwindows; ++i) {
1219b9c1b51eSKate Stone             if (m_subwindows[i]->GetCanBeActive()) {
122044d93782SGreg Clayton               m_curr_active_window_idx = i;
122144d93782SGreg Clayton               break;
122244d93782SGreg Clayton             }
122344d93782SGreg Clayton           }
122444d93782SGreg Clayton         }
122544d93782SGreg Clayton       }
122644d93782SGreg Clayton 
122744d93782SGreg Clayton       if (m_curr_active_window_idx < m_subwindows.size())
122844d93782SGreg Clayton         return m_subwindows[m_curr_active_window_idx];
122944d93782SGreg Clayton     }
123044d93782SGreg Clayton     return WindowSP();
123144d93782SGreg Clayton   }
123244d93782SGreg Clayton 
1233b9c1b51eSKate Stone   bool GetCanBeActive() const { return m_can_activate; }
123444d93782SGreg Clayton 
1235b9c1b51eSKate Stone   void SetCanBeActive(bool b) { m_can_activate = b; }
123644d93782SGreg Clayton 
1237b9c1b51eSKate Stone   void SetDelegate(const WindowDelegateSP &delegate_sp) {
123844d93782SGreg Clayton     m_delegate_sp = delegate_sp;
123944d93782SGreg Clayton   }
124044d93782SGreg Clayton 
1241b9c1b51eSKate Stone   Window *GetParent() const { return m_parent; }
124244d93782SGreg Clayton 
1243b9c1b51eSKate Stone   bool IsActive() const {
124444d93782SGreg Clayton     if (m_parent)
124544d93782SGreg Clayton       return m_parent->GetActiveWindow().get() == this;
124644d93782SGreg Clayton     else
124744d93782SGreg Clayton       return true; // Top level window is always active
124844d93782SGreg Clayton   }
124944d93782SGreg Clayton 
1250b9c1b51eSKate Stone   void SelectNextWindowAsActive() {
125144d93782SGreg Clayton     // Move active focus to next window
125244d93782SGreg Clayton     const size_t num_subwindows = m_subwindows.size();
1253b9c1b51eSKate Stone     if (m_curr_active_window_idx == UINT32_MAX) {
125444d93782SGreg Clayton       uint32_t idx = 0;
1255b9c1b51eSKate Stone       for (auto subwindow_sp : m_subwindows) {
1256b9c1b51eSKate Stone         if (subwindow_sp->GetCanBeActive()) {
125744d93782SGreg Clayton           m_curr_active_window_idx = idx;
125844d93782SGreg Clayton           break;
125944d93782SGreg Clayton         }
126044d93782SGreg Clayton         ++idx;
126144d93782SGreg Clayton       }
1262b9c1b51eSKate Stone     } else if (m_curr_active_window_idx + 1 < num_subwindows) {
126344d93782SGreg Clayton       bool handled = false;
126444d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1265b9c1b51eSKate Stone       for (size_t idx = m_curr_active_window_idx + 1; idx < num_subwindows;
1266b9c1b51eSKate Stone            ++idx) {
1267b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
126844d93782SGreg Clayton           m_curr_active_window_idx = idx;
126944d93782SGreg Clayton           handled = true;
127044d93782SGreg Clayton           break;
127144d93782SGreg Clayton         }
127244d93782SGreg Clayton       }
1273b9c1b51eSKate Stone       if (!handled) {
1274b9c1b51eSKate Stone         for (size_t idx = 0; idx <= m_prev_active_window_idx; ++idx) {
1275b9c1b51eSKate Stone           if (m_subwindows[idx]->GetCanBeActive()) {
127644d93782SGreg Clayton             m_curr_active_window_idx = idx;
127744d93782SGreg Clayton             break;
127844d93782SGreg Clayton           }
127944d93782SGreg Clayton         }
128044d93782SGreg Clayton       }
1281b9c1b51eSKate Stone     } else {
128244d93782SGreg Clayton       m_prev_active_window_idx = m_curr_active_window_idx;
1283b9c1b51eSKate Stone       for (size_t idx = 0; idx < num_subwindows; ++idx) {
1284b9c1b51eSKate Stone         if (m_subwindows[idx]->GetCanBeActive()) {
128544d93782SGreg Clayton           m_curr_active_window_idx = idx;
128644d93782SGreg Clayton           break;
128744d93782SGreg Clayton         }
128844d93782SGreg Clayton       }
128944d93782SGreg Clayton     }
129044d93782SGreg Clayton   }
129144d93782SGreg Clayton 
1292b9c1b51eSKate Stone   const char *GetName() const { return m_name.c_str(); }
1293315b6884SEugene Zelenko 
129444d93782SGreg Clayton protected:
129544d93782SGreg Clayton   std::string m_name;
129644d93782SGreg Clayton   WINDOW *m_window;
129744d93782SGreg Clayton   PANEL *m_panel;
129844d93782SGreg Clayton   Window *m_parent;
129944d93782SGreg Clayton   Windows m_subwindows;
130044d93782SGreg Clayton   WindowDelegateSP m_delegate_sp;
130144d93782SGreg Clayton   uint32_t m_curr_active_window_idx;
130244d93782SGreg Clayton   uint32_t m_prev_active_window_idx;
130344d93782SGreg Clayton   bool m_delete;
130444d93782SGreg Clayton   bool m_needs_update;
130544d93782SGreg Clayton   bool m_can_activate;
130644d93782SGreg Clayton   bool m_is_subwin;
130744d93782SGreg Clayton 
130844d93782SGreg Clayton private:
130944d93782SGreg Clayton   DISALLOW_COPY_AND_ASSIGN(Window);
131044d93782SGreg Clayton };
131144d93782SGreg Clayton 
1312b9c1b51eSKate Stone class MenuDelegate {
131344d93782SGreg Clayton public:
1314315b6884SEugene Zelenko   virtual ~MenuDelegate() = default;
131544d93782SGreg Clayton 
1316b9c1b51eSKate Stone   virtual MenuActionResult MenuDelegateAction(Menu &menu) = 0;
131744d93782SGreg Clayton };
131844d93782SGreg Clayton 
1319b9c1b51eSKate Stone class Menu : public WindowDelegate {
132044d93782SGreg Clayton public:
1321b9c1b51eSKate Stone   enum class Type { Invalid, Bar, Item, Separator };
132244d93782SGreg Clayton 
132344d93782SGreg Clayton   // Menubar or separator constructor
132444d93782SGreg Clayton   Menu(Type type);
132544d93782SGreg Clayton 
132644d93782SGreg Clayton   // Menuitem constructor
1327b9c1b51eSKate Stone   Menu(const char *name, const char *key_name, int key_value,
132844d93782SGreg Clayton        uint64_t identifier);
132944d93782SGreg Clayton 
1330315b6884SEugene Zelenko   ~Menu() override = default;
133144d93782SGreg Clayton 
1332b9c1b51eSKate Stone   const MenuDelegateSP &GetDelegate() const { return m_delegate_sp; }
133344d93782SGreg Clayton 
1334b9c1b51eSKate Stone   void SetDelegate(const MenuDelegateSP &delegate_sp) {
133544d93782SGreg Clayton     m_delegate_sp = delegate_sp;
133644d93782SGreg Clayton   }
133744d93782SGreg Clayton 
1338b9c1b51eSKate Stone   void RecalculateNameLengths();
133944d93782SGreg Clayton 
1340b9c1b51eSKate Stone   void AddSubmenu(const MenuSP &menu_sp);
134144d93782SGreg Clayton 
1342b9c1b51eSKate Stone   int DrawAndRunMenu(Window &window);
134344d93782SGreg Clayton 
1344b9c1b51eSKate Stone   void DrawMenuTitle(Window &window, bool highlight);
134544d93782SGreg Clayton 
1346b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override;
134744d93782SGreg Clayton 
1348b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override;
134944d93782SGreg Clayton 
1350b9c1b51eSKate Stone   MenuActionResult ActionPrivate(Menu &menu) {
135144d93782SGreg Clayton     MenuActionResult result = MenuActionResult::NotHandled;
1352b9c1b51eSKate Stone     if (m_delegate_sp) {
135344d93782SGreg Clayton       result = m_delegate_sp->MenuDelegateAction(menu);
135444d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
135544d93782SGreg Clayton         return result;
1356b9c1b51eSKate Stone     } else if (m_parent) {
135744d93782SGreg Clayton       result = m_parent->ActionPrivate(menu);
135844d93782SGreg Clayton       if (result != MenuActionResult::NotHandled)
135944d93782SGreg Clayton         return result;
136044d93782SGreg Clayton     }
136144d93782SGreg Clayton     return m_canned_result;
136244d93782SGreg Clayton   }
136344d93782SGreg Clayton 
1364b9c1b51eSKate Stone   MenuActionResult Action() {
136505097246SAdrian Prantl     // Call the recursive action so it can try to handle it with the menu
136605097246SAdrian Prantl     // delegate, and if not, try our parent menu
136744d93782SGreg Clayton     return ActionPrivate(*this);
136844d93782SGreg Clayton   }
136944d93782SGreg Clayton 
1370b9c1b51eSKate Stone   void SetCannedResult(MenuActionResult result) { m_canned_result = result; }
137144d93782SGreg Clayton 
1372b9c1b51eSKate Stone   Menus &GetSubmenus() { return m_submenus; }
137344d93782SGreg Clayton 
1374b9c1b51eSKate Stone   const Menus &GetSubmenus() const { return m_submenus; }
137544d93782SGreg Clayton 
1376b9c1b51eSKate Stone   int GetSelectedSubmenuIndex() const { return m_selected; }
137744d93782SGreg Clayton 
1378b9c1b51eSKate Stone   void SetSelectedSubmenuIndex(int idx) { m_selected = idx; }
137944d93782SGreg Clayton 
1380b9c1b51eSKate Stone   Type GetType() const { return m_type; }
138144d93782SGreg Clayton 
1382b9c1b51eSKate Stone   int GetStartingColumn() const { return m_start_col; }
138344d93782SGreg Clayton 
1384b9c1b51eSKate Stone   void SetStartingColumn(int col) { m_start_col = col; }
138544d93782SGreg Clayton 
1386b9c1b51eSKate Stone   int GetKeyValue() const { return m_key_value; }
138744d93782SGreg Clayton 
1388b9c1b51eSKate Stone   std::string &GetName() { return m_name; }
138944d93782SGreg Clayton 
1390b9c1b51eSKate Stone   int GetDrawWidth() const {
139144d93782SGreg Clayton     return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
139244d93782SGreg Clayton   }
139344d93782SGreg Clayton 
1394b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
139544d93782SGreg Clayton 
1396b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
139744d93782SGreg Clayton 
139844d93782SGreg Clayton protected:
139944d93782SGreg Clayton   std::string m_name;
140044d93782SGreg Clayton   std::string m_key_name;
140144d93782SGreg Clayton   uint64_t m_identifier;
140244d93782SGreg Clayton   Type m_type;
140344d93782SGreg Clayton   int m_key_value;
140444d93782SGreg Clayton   int m_start_col;
140544d93782SGreg Clayton   int m_max_submenu_name_length;
140644d93782SGreg Clayton   int m_max_submenu_key_name_length;
140744d93782SGreg Clayton   int m_selected;
140844d93782SGreg Clayton   Menu *m_parent;
140944d93782SGreg Clayton   Menus m_submenus;
141044d93782SGreg Clayton   WindowSP m_menu_window_sp;
141144d93782SGreg Clayton   MenuActionResult m_canned_result;
141244d93782SGreg Clayton   MenuDelegateSP m_delegate_sp;
141344d93782SGreg Clayton };
141444d93782SGreg Clayton 
141544d93782SGreg Clayton // Menubar or separator constructor
1416b9c1b51eSKate Stone Menu::Menu(Type type)
1417b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(0), m_type(type), m_key_value(0),
1418b9c1b51eSKate Stone       m_start_col(0), m_max_submenu_name_length(0),
1419b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1420b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1421b9c1b51eSKate Stone       m_delegate_sp() {}
142244d93782SGreg Clayton 
142344d93782SGreg Clayton // Menuitem constructor
1424b9c1b51eSKate Stone Menu::Menu(const char *name, const char *key_name, int key_value,
1425b9c1b51eSKate Stone            uint64_t identifier)
1426b9c1b51eSKate Stone     : m_name(), m_key_name(), m_identifier(identifier), m_type(Type::Invalid),
1427b9c1b51eSKate Stone       m_key_value(key_value), m_start_col(0), m_max_submenu_name_length(0),
1428b9c1b51eSKate Stone       m_max_submenu_key_name_length(0), m_selected(0), m_parent(nullptr),
1429b9c1b51eSKate Stone       m_submenus(), m_canned_result(MenuActionResult::NotHandled),
1430b9c1b51eSKate Stone       m_delegate_sp() {
1431b9c1b51eSKate Stone   if (name && name[0]) {
143244d93782SGreg Clayton     m_name = name;
143344d93782SGreg Clayton     m_type = Type::Item;
143444d93782SGreg Clayton     if (key_name && key_name[0])
143544d93782SGreg Clayton       m_key_name = key_name;
1436b9c1b51eSKate Stone   } else {
143744d93782SGreg Clayton     m_type = Type::Separator;
143844d93782SGreg Clayton   }
143944d93782SGreg Clayton }
144044d93782SGreg Clayton 
1441b9c1b51eSKate Stone void Menu::RecalculateNameLengths() {
144244d93782SGreg Clayton   m_max_submenu_name_length = 0;
144344d93782SGreg Clayton   m_max_submenu_key_name_length = 0;
144444d93782SGreg Clayton   Menus &submenus = GetSubmenus();
144544d93782SGreg Clayton   const size_t num_submenus = submenus.size();
1446b9c1b51eSKate Stone   for (size_t i = 0; i < num_submenus; ++i) {
144744d93782SGreg Clayton     Menu *submenu = submenus[i].get();
14483985c8c6SSaleem Abdulrasool     if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
144944d93782SGreg Clayton       m_max_submenu_name_length = submenu->m_name.size();
1450b9c1b51eSKate Stone     if (static_cast<size_t>(m_max_submenu_key_name_length) <
1451b9c1b51eSKate Stone         submenu->m_key_name.size())
145244d93782SGreg Clayton       m_max_submenu_key_name_length = submenu->m_key_name.size();
145344d93782SGreg Clayton   }
145444d93782SGreg Clayton }
145544d93782SGreg Clayton 
1456b9c1b51eSKate Stone void Menu::AddSubmenu(const MenuSP &menu_sp) {
145744d93782SGreg Clayton   menu_sp->m_parent = this;
14583985c8c6SSaleem Abdulrasool   if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
145944d93782SGreg Clayton     m_max_submenu_name_length = menu_sp->m_name.size();
1460b9c1b51eSKate Stone   if (static_cast<size_t>(m_max_submenu_key_name_length) <
1461b9c1b51eSKate Stone       menu_sp->m_key_name.size())
146244d93782SGreg Clayton     m_max_submenu_key_name_length = menu_sp->m_key_name.size();
146344d93782SGreg Clayton   m_submenus.push_back(menu_sp);
146444d93782SGreg Clayton }
146544d93782SGreg Clayton 
1466b9c1b51eSKate Stone void Menu::DrawMenuTitle(Window &window, bool highlight) {
1467b9c1b51eSKate Stone   if (m_type == Type::Separator) {
146844d93782SGreg Clayton     window.MoveCursor(0, window.GetCursorY());
146944d93782SGreg Clayton     window.PutChar(ACS_LTEE);
147044d93782SGreg Clayton     int width = window.GetWidth();
1471b9c1b51eSKate Stone     if (width > 2) {
147244d93782SGreg Clayton       width -= 2;
14733985c8c6SSaleem Abdulrasool       for (int i = 0; i < width; ++i)
147444d93782SGreg Clayton         window.PutChar(ACS_HLINE);
147544d93782SGreg Clayton     }
147644d93782SGreg Clayton     window.PutChar(ACS_RTEE);
1477b9c1b51eSKate Stone   } else {
147844d93782SGreg Clayton     const int shortcut_key = m_key_value;
147944d93782SGreg Clayton     bool underlined_shortcut = false;
148044d93782SGreg Clayton     const attr_t hilgight_attr = A_REVERSE;
148144d93782SGreg Clayton     if (highlight)
148244d93782SGreg Clayton       window.AttributeOn(hilgight_attr);
1483b9c1b51eSKate Stone     if (isprint(shortcut_key)) {
148444d93782SGreg Clayton       size_t lower_pos = m_name.find(tolower(shortcut_key));
148544d93782SGreg Clayton       size_t upper_pos = m_name.find(toupper(shortcut_key));
148644d93782SGreg Clayton       const char *name = m_name.c_str();
148744d93782SGreg Clayton       size_t pos = std::min<size_t>(lower_pos, upper_pos);
1488b9c1b51eSKate Stone       if (pos != std::string::npos) {
148944d93782SGreg Clayton         underlined_shortcut = true;
1490b9c1b51eSKate Stone         if (pos > 0) {
149144d93782SGreg Clayton           window.PutCString(name, pos);
149244d93782SGreg Clayton           name += pos;
149344d93782SGreg Clayton         }
149444d93782SGreg Clayton         const attr_t shortcut_attr = A_UNDERLINE | A_BOLD;
149544d93782SGreg Clayton         window.AttributeOn(shortcut_attr);
149644d93782SGreg Clayton         window.PutChar(name[0]);
149744d93782SGreg Clayton         window.AttributeOff(shortcut_attr);
149844d93782SGreg Clayton         name++;
149944d93782SGreg Clayton         if (name[0])
150044d93782SGreg Clayton           window.PutCString(name);
150144d93782SGreg Clayton       }
150244d93782SGreg Clayton     }
150344d93782SGreg Clayton 
1504b9c1b51eSKate Stone     if (!underlined_shortcut) {
150544d93782SGreg Clayton       window.PutCString(m_name.c_str());
150644d93782SGreg Clayton     }
150744d93782SGreg Clayton 
150844d93782SGreg Clayton     if (highlight)
150944d93782SGreg Clayton       window.AttributeOff(hilgight_attr);
151044d93782SGreg Clayton 
1511b9c1b51eSKate Stone     if (m_key_name.empty()) {
1512b9c1b51eSKate Stone       if (!underlined_shortcut && isprint(m_key_value)) {
151344d93782SGreg Clayton         window.AttributeOn(COLOR_PAIR(3));
151444d93782SGreg Clayton         window.Printf(" (%c)", m_key_value);
151544d93782SGreg Clayton         window.AttributeOff(COLOR_PAIR(3));
151644d93782SGreg Clayton       }
1517b9c1b51eSKate Stone     } else {
151844d93782SGreg Clayton       window.AttributeOn(COLOR_PAIR(3));
151944d93782SGreg Clayton       window.Printf(" (%s)", m_key_name.c_str());
152044d93782SGreg Clayton       window.AttributeOff(COLOR_PAIR(3));
152144d93782SGreg Clayton     }
152244d93782SGreg Clayton   }
152344d93782SGreg Clayton }
152444d93782SGreg Clayton 
1525b9c1b51eSKate Stone bool Menu::WindowDelegateDraw(Window &window, bool force) {
152644d93782SGreg Clayton   Menus &submenus = GetSubmenus();
152744d93782SGreg Clayton   const size_t num_submenus = submenus.size();
152844d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
152944d93782SGreg Clayton   Menu::Type menu_type = GetType();
1530b9c1b51eSKate Stone   switch (menu_type) {
1531b9c1b51eSKate Stone   case Menu::Type::Bar: {
153244d93782SGreg Clayton     window.SetBackground(2);
153344d93782SGreg Clayton     window.MoveCursor(0, 0);
1534b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
153544d93782SGreg Clayton       Menu *menu = submenus[i].get();
153644d93782SGreg Clayton       if (i > 0)
153744d93782SGreg Clayton         window.PutChar(' ');
153844d93782SGreg Clayton       menu->SetStartingColumn(window.GetCursorX());
153944d93782SGreg Clayton       window.PutCString("| ");
154044d93782SGreg Clayton       menu->DrawMenuTitle(window, false);
154144d93782SGreg Clayton     }
154244d93782SGreg Clayton     window.PutCString(" |");
1543b9c1b51eSKate Stone   } break;
154444d93782SGreg Clayton 
1545b9c1b51eSKate Stone   case Menu::Type::Item: {
154644d93782SGreg Clayton     int y = 1;
154744d93782SGreg Clayton     int x = 3;
154844d93782SGreg Clayton     // Draw the menu
154944d93782SGreg Clayton     int cursor_x = 0;
155044d93782SGreg Clayton     int cursor_y = 0;
155144d93782SGreg Clayton     window.Erase();
155244d93782SGreg Clayton     window.SetBackground(2);
155344d93782SGreg Clayton     window.Box();
1554b9c1b51eSKate Stone     for (size_t i = 0; i < num_submenus; ++i) {
1555b9c1b51eSKate Stone       const bool is_selected = (i == static_cast<size_t>(selected_idx));
155644d93782SGreg Clayton       window.MoveCursor(x, y + i);
1557b9c1b51eSKate Stone       if (is_selected) {
155844d93782SGreg Clayton         // Remember where we want the cursor to be
155944d93782SGreg Clayton         cursor_x = x - 1;
156044d93782SGreg Clayton         cursor_y = y + i;
156144d93782SGreg Clayton       }
156244d93782SGreg Clayton       submenus[i]->DrawMenuTitle(window, is_selected);
156344d93782SGreg Clayton     }
156444d93782SGreg Clayton     window.MoveCursor(cursor_x, cursor_y);
1565b9c1b51eSKate Stone   } break;
156644d93782SGreg Clayton 
156744d93782SGreg Clayton   default:
156844d93782SGreg Clayton   case Menu::Type::Separator:
156944d93782SGreg Clayton     break;
157044d93782SGreg Clayton   }
157144d93782SGreg Clayton   return true; // Drawing handled...
157244d93782SGreg Clayton }
157344d93782SGreg Clayton 
1574b9c1b51eSKate Stone HandleCharResult Menu::WindowDelegateHandleChar(Window &window, int key) {
157544d93782SGreg Clayton   HandleCharResult result = eKeyNotHandled;
157644d93782SGreg Clayton 
157744d93782SGreg Clayton   Menus &submenus = GetSubmenus();
157844d93782SGreg Clayton   const size_t num_submenus = submenus.size();
157944d93782SGreg Clayton   const int selected_idx = GetSelectedSubmenuIndex();
158044d93782SGreg Clayton   Menu::Type menu_type = GetType();
1581b9c1b51eSKate Stone   if (menu_type == Menu::Type::Bar) {
158244d93782SGreg Clayton     MenuSP run_menu_sp;
1583b9c1b51eSKate Stone     switch (key) {
158444d93782SGreg Clayton     case KEY_DOWN:
158544d93782SGreg Clayton     case KEY_UP:
158644d93782SGreg Clayton       // Show last menu or first menu
15873985c8c6SSaleem Abdulrasool       if (selected_idx < static_cast<int>(num_submenus))
158844d93782SGreg Clayton         run_menu_sp = submenus[selected_idx];
158944d93782SGreg Clayton       else if (!submenus.empty())
159044d93782SGreg Clayton         run_menu_sp = submenus.front();
159144d93782SGreg Clayton       result = eKeyHandled;
159244d93782SGreg Clayton       break;
159344d93782SGreg Clayton 
159444d93782SGreg Clayton     case KEY_RIGHT:
159544d93782SGreg Clayton       ++m_selected;
15963985c8c6SSaleem Abdulrasool       if (m_selected >= static_cast<int>(num_submenus))
159744d93782SGreg Clayton         m_selected = 0;
15983985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
159944d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
160044d93782SGreg Clayton       else if (!submenus.empty())
160144d93782SGreg Clayton         run_menu_sp = submenus.front();
160244d93782SGreg Clayton       result = eKeyHandled;
160344d93782SGreg Clayton       break;
160444d93782SGreg Clayton 
160544d93782SGreg Clayton     case KEY_LEFT:
160644d93782SGreg Clayton       --m_selected;
160744d93782SGreg Clayton       if (m_selected < 0)
160844d93782SGreg Clayton         m_selected = num_submenus - 1;
16093985c8c6SSaleem Abdulrasool       if (m_selected < static_cast<int>(num_submenus))
161044d93782SGreg Clayton         run_menu_sp = submenus[m_selected];
161144d93782SGreg Clayton       else if (!submenus.empty())
161244d93782SGreg Clayton         run_menu_sp = submenus.front();
161344d93782SGreg Clayton       result = eKeyHandled;
161444d93782SGreg Clayton       break;
161544d93782SGreg Clayton 
161644d93782SGreg Clayton     default:
1617b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
1618b9c1b51eSKate Stone         if (submenus[i]->GetKeyValue() == key) {
161944d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
162044d93782SGreg Clayton           run_menu_sp = submenus[i];
162144d93782SGreg Clayton           result = eKeyHandled;
162244d93782SGreg Clayton           break;
162344d93782SGreg Clayton         }
162444d93782SGreg Clayton       }
162544d93782SGreg Clayton       break;
162644d93782SGreg Clayton     }
162744d93782SGreg Clayton 
1628b9c1b51eSKate Stone     if (run_menu_sp) {
162905097246SAdrian Prantl       // Run the action on this menu in case we need to populate the menu with
163005097246SAdrian Prantl       // dynamic content and also in case check marks, and any other menu
163105097246SAdrian Prantl       // decorations need to be calculated
163244d93782SGreg Clayton       if (run_menu_sp->Action() == MenuActionResult::Quit)
163344d93782SGreg Clayton         return eQuitApplication;
163444d93782SGreg Clayton 
163544d93782SGreg Clayton       Rect menu_bounds;
163644d93782SGreg Clayton       menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
163744d93782SGreg Clayton       menu_bounds.origin.y = 1;
163844d93782SGreg Clayton       menu_bounds.size.width = run_menu_sp->GetDrawWidth();
163944d93782SGreg Clayton       menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
164044d93782SGreg Clayton       if (m_menu_window_sp)
164144d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
164244d93782SGreg Clayton 
1643b9c1b51eSKate Stone       m_menu_window_sp = window.GetParent()->CreateSubWindow(
1644b9c1b51eSKate Stone           run_menu_sp->GetName().c_str(), menu_bounds, true);
164544d93782SGreg Clayton       m_menu_window_sp->SetDelegate(run_menu_sp);
164644d93782SGreg Clayton     }
1647b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Item) {
1648b9c1b51eSKate Stone     switch (key) {
164944d93782SGreg Clayton     case KEY_DOWN:
1650b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
165144d93782SGreg Clayton         const int start_select = m_selected;
1652b9c1b51eSKate Stone         while (++m_selected != start_select) {
16533985c8c6SSaleem Abdulrasool           if (static_cast<size_t>(m_selected) >= num_submenus)
165444d93782SGreg Clayton             m_selected = 0;
165544d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
165644d93782SGreg Clayton             continue;
165744d93782SGreg Clayton           else
165844d93782SGreg Clayton             break;
165944d93782SGreg Clayton         }
166044d93782SGreg Clayton         return eKeyHandled;
166144d93782SGreg Clayton       }
166244d93782SGreg Clayton       break;
166344d93782SGreg Clayton 
166444d93782SGreg Clayton     case KEY_UP:
1665b9c1b51eSKate Stone       if (m_submenus.size() > 1) {
166644d93782SGreg Clayton         const int start_select = m_selected;
1667b9c1b51eSKate Stone         while (--m_selected != start_select) {
16683985c8c6SSaleem Abdulrasool           if (m_selected < static_cast<int>(0))
166944d93782SGreg Clayton             m_selected = num_submenus - 1;
167044d93782SGreg Clayton           if (m_submenus[m_selected]->GetType() == Type::Separator)
167144d93782SGreg Clayton             continue;
167244d93782SGreg Clayton           else
167344d93782SGreg Clayton             break;
167444d93782SGreg Clayton         }
167544d93782SGreg Clayton         return eKeyHandled;
167644d93782SGreg Clayton       }
167744d93782SGreg Clayton       break;
167844d93782SGreg Clayton 
167944d93782SGreg Clayton     case KEY_RETURN:
1680b9c1b51eSKate Stone       if (static_cast<size_t>(selected_idx) < num_submenus) {
168144d93782SGreg Clayton         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
168244d93782SGreg Clayton           return eQuitApplication;
168344d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
168444d93782SGreg Clayton         return eKeyHandled;
168544d93782SGreg Clayton       }
168644d93782SGreg Clayton       break;
168744d93782SGreg Clayton 
1688b9c1b51eSKate Stone     case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in
1689b9c1b51eSKate Stone                      // case other chars are entered for escaped sequences
169044d93782SGreg Clayton       window.GetParent()->RemoveSubWindow(&window);
169144d93782SGreg Clayton       return eKeyHandled;
169244d93782SGreg Clayton 
169344d93782SGreg Clayton     default:
1694b9c1b51eSKate Stone       for (size_t i = 0; i < num_submenus; ++i) {
169544d93782SGreg Clayton         Menu *menu = submenus[i].get();
1696b9c1b51eSKate Stone         if (menu->GetKeyValue() == key) {
169744d93782SGreg Clayton           SetSelectedSubmenuIndex(i);
169844d93782SGreg Clayton           window.GetParent()->RemoveSubWindow(&window);
169944d93782SGreg Clayton           if (menu->Action() == MenuActionResult::Quit)
170044d93782SGreg Clayton             return eQuitApplication;
170144d93782SGreg Clayton           return eKeyHandled;
170244d93782SGreg Clayton         }
170344d93782SGreg Clayton       }
170444d93782SGreg Clayton       break;
170544d93782SGreg Clayton     }
1706b9c1b51eSKate Stone   } else if (menu_type == Menu::Type::Separator) {
170744d93782SGreg Clayton   }
170844d93782SGreg Clayton   return result;
170944d93782SGreg Clayton }
171044d93782SGreg Clayton 
1711b9c1b51eSKate Stone class Application {
171244d93782SGreg Clayton public:
1713b9c1b51eSKate Stone   Application(FILE *in, FILE *out)
1714b9c1b51eSKate Stone       : m_window_sp(), m_screen(nullptr), m_in(in), m_out(out) {}
171544d93782SGreg Clayton 
1716b9c1b51eSKate Stone   ~Application() {
171744d93782SGreg Clayton     m_window_delegates.clear();
171844d93782SGreg Clayton     m_window_sp.reset();
1719b9c1b51eSKate Stone     if (m_screen) {
172044d93782SGreg Clayton       ::delscreen(m_screen);
1721c5dac77aSEugene Zelenko       m_screen = nullptr;
172244d93782SGreg Clayton     }
172344d93782SGreg Clayton   }
172444d93782SGreg Clayton 
1725b9c1b51eSKate Stone   void Initialize() {
172644d93782SGreg Clayton     ::setlocale(LC_ALL, "");
172744d93782SGreg Clayton     ::setlocale(LC_CTYPE, "");
1728c5dac77aSEugene Zelenko     m_screen = ::newterm(nullptr, m_out, m_in);
172944d93782SGreg Clayton     ::start_color();
173044d93782SGreg Clayton     ::curs_set(0);
173144d93782SGreg Clayton     ::noecho();
173244d93782SGreg Clayton     ::keypad(stdscr, TRUE);
173344d93782SGreg Clayton   }
173444d93782SGreg Clayton 
1735b9c1b51eSKate Stone   void Terminate() { ::endwin(); }
173644d93782SGreg Clayton 
1737b9c1b51eSKate Stone   void Run(Debugger &debugger) {
173844d93782SGreg Clayton     bool done = false;
173944d93782SGreg Clayton     int delay_in_tenths_of_a_second = 1;
174044d93782SGreg Clayton 
174105097246SAdrian Prantl     // Alas the threading model in curses is a bit lame so we need to resort to
174205097246SAdrian Prantl     // polling every 0.5 seconds. We could poll for stdin ourselves and then
174305097246SAdrian Prantl     // pass the keys down but then we need to translate all of the escape
174405097246SAdrian Prantl     // sequences ourselves. So we resort to polling for input because we need
174505097246SAdrian Prantl     // to receive async process events while in this loop.
174644d93782SGreg Clayton 
1747b9c1b51eSKate Stone     halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths
1748b9c1b51eSKate Stone                                             // of seconds seconds when calling
1749b9c1b51eSKate Stone                                             // Window::GetChar()
175044d93782SGreg Clayton 
1751b9c1b51eSKate Stone     ListenerSP listener_sp(
1752b9c1b51eSKate Stone         Listener::MakeListener("lldb.IOHandler.curses.Application"));
175344d93782SGreg Clayton     ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
175444d93782SGreg Clayton     ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
175544d93782SGreg Clayton     ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
175644d93782SGreg Clayton     debugger.EnableForwardEvents(listener_sp);
175744d93782SGreg Clayton 
175844d93782SGreg Clayton     bool update = true;
175944d93782SGreg Clayton #if defined(__APPLE__)
176044d93782SGreg Clayton     std::deque<int> escape_chars;
176144d93782SGreg Clayton #endif
176244d93782SGreg Clayton 
1763b9c1b51eSKate Stone     while (!done) {
1764b9c1b51eSKate Stone       if (update) {
176544d93782SGreg Clayton         m_window_sp->Draw(false);
176605097246SAdrian Prantl         // All windows should be calling Window::DeferredRefresh() instead of
176705097246SAdrian Prantl         // Window::Refresh() so we can do a single update and avoid any screen
176805097246SAdrian Prantl         // blinking
176944d93782SGreg Clayton         update_panels();
177044d93782SGreg Clayton 
1771b9c1b51eSKate Stone         // Cursor hiding isn't working on MacOSX, so hide it in the top left
1772b9c1b51eSKate Stone         // corner
177344d93782SGreg Clayton         m_window_sp->MoveCursor(0, 0);
177444d93782SGreg Clayton 
177544d93782SGreg Clayton         doupdate();
177644d93782SGreg Clayton         update = false;
177744d93782SGreg Clayton       }
177844d93782SGreg Clayton 
177944d93782SGreg Clayton #if defined(__APPLE__)
178005097246SAdrian Prantl       // Terminal.app doesn't map its function keys correctly, F1-F4 default
178105097246SAdrian Prantl       // to: \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if
1782b9c1b51eSKate Stone       // possible
178344d93782SGreg Clayton       int ch;
178444d93782SGreg Clayton       if (escape_chars.empty())
178544d93782SGreg Clayton         ch = m_window_sp->GetChar();
1786b9c1b51eSKate Stone       else {
178744d93782SGreg Clayton         ch = escape_chars.front();
178844d93782SGreg Clayton         escape_chars.pop_front();
178944d93782SGreg Clayton       }
1790b9c1b51eSKate Stone       if (ch == KEY_ESCAPE) {
179144d93782SGreg Clayton         int ch2 = m_window_sp->GetChar();
1792b9c1b51eSKate Stone         if (ch2 == 'O') {
179344d93782SGreg Clayton           int ch3 = m_window_sp->GetChar();
1794b9c1b51eSKate Stone           switch (ch3) {
1795b9c1b51eSKate Stone           case 'P':
1796b9c1b51eSKate Stone             ch = KEY_F(1);
1797b9c1b51eSKate Stone             break;
1798b9c1b51eSKate Stone           case 'Q':
1799b9c1b51eSKate Stone             ch = KEY_F(2);
1800b9c1b51eSKate Stone             break;
1801b9c1b51eSKate Stone           case 'R':
1802b9c1b51eSKate Stone             ch = KEY_F(3);
1803b9c1b51eSKate Stone             break;
1804b9c1b51eSKate Stone           case 'S':
1805b9c1b51eSKate Stone             ch = KEY_F(4);
1806b9c1b51eSKate Stone             break;
180744d93782SGreg Clayton           default:
180844d93782SGreg Clayton             escape_chars.push_back(ch2);
180944d93782SGreg Clayton             if (ch3 != -1)
181044d93782SGreg Clayton               escape_chars.push_back(ch3);
181144d93782SGreg Clayton             break;
181244d93782SGreg Clayton           }
1813b9c1b51eSKate Stone         } else if (ch2 != -1)
181444d93782SGreg Clayton           escape_chars.push_back(ch2);
181544d93782SGreg Clayton       }
181644d93782SGreg Clayton #else
181744d93782SGreg Clayton       int ch = m_window_sp->GetChar();
181844d93782SGreg Clayton 
181944d93782SGreg Clayton #endif
1820b9c1b51eSKate Stone       if (ch == -1) {
1821b9c1b51eSKate Stone         if (feof(m_in) || ferror(m_in)) {
182244d93782SGreg Clayton           done = true;
1823b9c1b51eSKate Stone         } else {
182444d93782SGreg Clayton           // Just a timeout from using halfdelay(), check for events
182544d93782SGreg Clayton           EventSP event_sp;
1826b9c1b51eSKate Stone           while (listener_sp->PeekAtNextEvent()) {
1827d35031e1SPavel Labath             listener_sp->GetEvent(event_sp, std::chrono::seconds(0));
182844d93782SGreg Clayton 
1829b9c1b51eSKate Stone             if (event_sp) {
183044d93782SGreg Clayton               Broadcaster *broadcaster = event_sp->GetBroadcaster();
1831b9c1b51eSKate Stone               if (broadcaster) {
183244d93782SGreg Clayton                 // uint32_t event_type = event_sp->GetType();
1833b9c1b51eSKate Stone                 ConstString broadcaster_class(
1834b9c1b51eSKate Stone                     broadcaster->GetBroadcasterClass());
1835b9c1b51eSKate Stone                 if (broadcaster_class == broadcaster_class_process) {
1836b9c1b51eSKate Stone                   debugger.GetCommandInterpreter().UpdateExecutionContext(
1837b9c1b51eSKate Stone                       nullptr);
183844d93782SGreg Clayton                   update = true;
183944d93782SGreg Clayton                   continue; // Don't get any key, just update our view
184044d93782SGreg Clayton                 }
184144d93782SGreg Clayton               }
184244d93782SGreg Clayton             }
184344d93782SGreg Clayton           }
184444d93782SGreg Clayton         }
1845b9c1b51eSKate Stone       } else {
184644d93782SGreg Clayton         HandleCharResult key_result = m_window_sp->HandleChar(ch);
1847b9c1b51eSKate Stone         switch (key_result) {
184844d93782SGreg Clayton         case eKeyHandled:
1849c5dac77aSEugene Zelenko           debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
185044d93782SGreg Clayton           update = true;
185144d93782SGreg Clayton           break;
185244d93782SGreg Clayton         case eKeyNotHandled:
185344d93782SGreg Clayton           break;
185444d93782SGreg Clayton         case eQuitApplication:
185544d93782SGreg Clayton           done = true;
185644d93782SGreg Clayton           break;
185744d93782SGreg Clayton         }
185844d93782SGreg Clayton       }
185944d93782SGreg Clayton     }
186044d93782SGreg Clayton 
186144d93782SGreg Clayton     debugger.CancelForwardEvents(listener_sp);
186244d93782SGreg Clayton   }
186344d93782SGreg Clayton 
1864b9c1b51eSKate Stone   WindowSP &GetMainWindow() {
186544d93782SGreg Clayton     if (!m_window_sp)
1866796ac80bSJonas Devlieghere       m_window_sp = std::make_shared<Window>("main", stdscr, false);
186744d93782SGreg Clayton     return m_window_sp;
186844d93782SGreg Clayton   }
186944d93782SGreg Clayton 
187044d93782SGreg Clayton protected:
187144d93782SGreg Clayton   WindowSP m_window_sp;
187244d93782SGreg Clayton   WindowDelegates m_window_delegates;
187344d93782SGreg Clayton   SCREEN *m_screen;
187444d93782SGreg Clayton   FILE *m_in;
187544d93782SGreg Clayton   FILE *m_out;
187644d93782SGreg Clayton };
187744d93782SGreg Clayton 
187844d93782SGreg Clayton } // namespace curses
187944d93782SGreg Clayton 
188044d93782SGreg Clayton using namespace curses;
188144d93782SGreg Clayton 
1882b9c1b51eSKate Stone struct Row {
18838369b28dSGreg Clayton   ValueObjectManager value;
188444d93782SGreg Clayton   Row *parent;
18858369b28dSGreg Clayton   // The process stop ID when the children were calculated.
18868369b28dSGreg Clayton   uint32_t children_stop_id;
188744d93782SGreg Clayton   int row_idx;
188844d93782SGreg Clayton   int x;
188944d93782SGreg Clayton   int y;
189044d93782SGreg Clayton   bool might_have_children;
189144d93782SGreg Clayton   bool expanded;
189244d93782SGreg Clayton   bool calculated_children;
189344d93782SGreg Clayton   std::vector<Row> children;
189444d93782SGreg Clayton 
1895b9c1b51eSKate Stone   Row(const ValueObjectSP &v, Row *p)
18968369b28dSGreg Clayton       : value(v, lldb::eDynamicDontRunTarget, true), parent(p), row_idx(0),
18978369b28dSGreg Clayton         x(1), y(1), might_have_children(v ? v->MightHaveChildren() : false),
1898b9c1b51eSKate Stone         expanded(false), calculated_children(false), children() {}
189944d93782SGreg Clayton 
1900b9c1b51eSKate Stone   size_t GetDepth() const {
190144d93782SGreg Clayton     if (parent)
190244d93782SGreg Clayton       return 1 + parent->GetDepth();
190344d93782SGreg Clayton     return 0;
190444d93782SGreg Clayton   }
190544d93782SGreg Clayton 
1906171dd2e6SJonas Devlieghere   void Expand() { expanded = true; }
19078369b28dSGreg Clayton 
19088369b28dSGreg Clayton   std::vector<Row> &GetChildren() {
19098369b28dSGreg Clayton     ProcessSP process_sp = value.GetProcessSP();
19108369b28dSGreg Clayton     auto stop_id = process_sp->GetStopID();
19118369b28dSGreg Clayton     if (process_sp && stop_id != children_stop_id) {
19128369b28dSGreg Clayton       children_stop_id = stop_id;
19138369b28dSGreg Clayton       calculated_children = false;
19148369b28dSGreg Clayton     }
1915b9c1b51eSKate Stone     if (!calculated_children) {
19168369b28dSGreg Clayton       children.clear();
191744d93782SGreg Clayton       calculated_children = true;
19188369b28dSGreg Clayton       ValueObjectSP valobj = value.GetSP();
1919b9c1b51eSKate Stone       if (valobj) {
192044d93782SGreg Clayton         const size_t num_children = valobj->GetNumChildren();
1921b9c1b51eSKate Stone         for (size_t i = 0; i < num_children; ++i) {
192244d93782SGreg Clayton           children.push_back(Row(valobj->GetChildAtIndex(i, true), this));
192344d93782SGreg Clayton         }
192444d93782SGreg Clayton       }
192544d93782SGreg Clayton     }
19268369b28dSGreg Clayton     return children;
192744d93782SGreg Clayton   }
192844d93782SGreg Clayton 
19298369b28dSGreg Clayton   void Unexpand() {
19308369b28dSGreg Clayton     expanded = false;
19318369b28dSGreg Clayton     calculated_children = false;
19328369b28dSGreg Clayton     children.clear();
19338369b28dSGreg Clayton   }
193444d93782SGreg Clayton 
1935b9c1b51eSKate Stone   void DrawTree(Window &window) {
193644d93782SGreg Clayton     if (parent)
193744d93782SGreg Clayton       parent->DrawTreeForChild(window, this, 0);
193844d93782SGreg Clayton 
1939b9c1b51eSKate Stone     if (might_have_children) {
194005097246SAdrian Prantl       // It we can get UTF8 characters to work we should try to use the
194105097246SAdrian Prantl       // "symbol" UTF8 string below
194244d93782SGreg Clayton       //            const char *symbol = "";
194344d93782SGreg Clayton       //            if (row.expanded)
194444d93782SGreg Clayton       //                symbol = "\xe2\x96\xbd ";
194544d93782SGreg Clayton       //            else
194644d93782SGreg Clayton       //                symbol = "\xe2\x96\xb7 ";
194744d93782SGreg Clayton       //            window.PutCString (symbol);
194844d93782SGreg Clayton 
194905097246SAdrian Prantl       // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 'v'
195005097246SAdrian Prantl       // or '>' character...
195144d93782SGreg Clayton       //            if (expanded)
195244d93782SGreg Clayton       //                window.PutChar (ACS_DARROW);
195344d93782SGreg Clayton       //            else
195444d93782SGreg Clayton       //                window.PutChar (ACS_RARROW);
195505097246SAdrian Prantl       // Since we can't find any good looking right arrow/down arrow symbols,
195605097246SAdrian Prantl       // just use a diamond...
195744d93782SGreg Clayton       window.PutChar(ACS_DIAMOND);
195844d93782SGreg Clayton       window.PutChar(ACS_HLINE);
195944d93782SGreg Clayton     }
196044d93782SGreg Clayton   }
196144d93782SGreg Clayton 
1962b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, Row *child, uint32_t reverse_depth) {
196344d93782SGreg Clayton     if (parent)
196444d93782SGreg Clayton       parent->DrawTreeForChild(window, this, reverse_depth + 1);
196544d93782SGreg Clayton 
19668369b28dSGreg Clayton     if (&GetChildren().back() == child) {
196744d93782SGreg Clayton       // Last child
1968b9c1b51eSKate Stone       if (reverse_depth == 0) {
196944d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
197044d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1971b9c1b51eSKate Stone       } else {
197244d93782SGreg Clayton         window.PutChar(' ');
197344d93782SGreg Clayton         window.PutChar(' ');
197444d93782SGreg Clayton       }
1975b9c1b51eSKate Stone     } else {
1976b9c1b51eSKate Stone       if (reverse_depth == 0) {
197744d93782SGreg Clayton         window.PutChar(ACS_LTEE);
197844d93782SGreg Clayton         window.PutChar(ACS_HLINE);
1979b9c1b51eSKate Stone       } else {
198044d93782SGreg Clayton         window.PutChar(ACS_VLINE);
198144d93782SGreg Clayton         window.PutChar(' ');
198244d93782SGreg Clayton       }
198344d93782SGreg Clayton     }
198444d93782SGreg Clayton   }
198544d93782SGreg Clayton };
198644d93782SGreg Clayton 
1987b9c1b51eSKate Stone struct DisplayOptions {
198844d93782SGreg Clayton   bool show_types;
198944d93782SGreg Clayton };
199044d93782SGreg Clayton 
199144d93782SGreg Clayton class TreeItem;
199244d93782SGreg Clayton 
1993b9c1b51eSKate Stone class TreeDelegate {
199444d93782SGreg Clayton public:
1995c5dac77aSEugene Zelenko   TreeDelegate() = default;
1996315b6884SEugene Zelenko   virtual ~TreeDelegate() = default;
1997315b6884SEugene Zelenko 
199844d93782SGreg Clayton   virtual void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) = 0;
199944d93782SGreg Clayton   virtual void TreeDelegateGenerateChildren(TreeItem &item) = 0;
2000b9c1b51eSKate Stone   virtual bool TreeDelegateItemSelected(
2001b9c1b51eSKate Stone       TreeItem &item) = 0; // Return true if we need to update views
200244d93782SGreg Clayton };
2003315b6884SEugene Zelenko 
200444d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
200544d93782SGreg Clayton 
2006b9c1b51eSKate Stone class TreeItem {
200744d93782SGreg Clayton public:
2008b9c1b51eSKate Stone   TreeItem(TreeItem *parent, TreeDelegate &delegate, bool might_have_children)
2009b9c1b51eSKate Stone       : m_parent(parent), m_delegate(delegate), m_user_data(nullptr),
2010b9c1b51eSKate Stone         m_identifier(0), m_row_idx(-1), m_children(),
2011b9c1b51eSKate Stone         m_might_have_children(might_have_children), m_is_expanded(false) {}
201244d93782SGreg Clayton 
2013b9c1b51eSKate Stone   TreeItem &operator=(const TreeItem &rhs) {
2014b9c1b51eSKate Stone     if (this != &rhs) {
201544d93782SGreg Clayton       m_parent = rhs.m_parent;
201644d93782SGreg Clayton       m_delegate = rhs.m_delegate;
2017ec990867SGreg Clayton       m_user_data = rhs.m_user_data;
201844d93782SGreg Clayton       m_identifier = rhs.m_identifier;
201944d93782SGreg Clayton       m_row_idx = rhs.m_row_idx;
202044d93782SGreg Clayton       m_children = rhs.m_children;
202144d93782SGreg Clayton       m_might_have_children = rhs.m_might_have_children;
202244d93782SGreg Clayton       m_is_expanded = rhs.m_is_expanded;
202344d93782SGreg Clayton     }
202444d93782SGreg Clayton     return *this;
202544d93782SGreg Clayton   }
202644d93782SGreg Clayton 
2027b9c1b51eSKate Stone   size_t GetDepth() const {
202844d93782SGreg Clayton     if (m_parent)
202944d93782SGreg Clayton       return 1 + m_parent->GetDepth();
203044d93782SGreg Clayton     return 0;
203144d93782SGreg Clayton   }
203244d93782SGreg Clayton 
2033b9c1b51eSKate Stone   int GetRowIndex() const { return m_row_idx; }
203444d93782SGreg Clayton 
2035b9c1b51eSKate Stone   void ClearChildren() { m_children.clear(); }
203644d93782SGreg Clayton 
2037b9c1b51eSKate Stone   void Resize(size_t n, const TreeItem &t) { m_children.resize(n, t); }
203844d93782SGreg Clayton 
2039b9c1b51eSKate Stone   TreeItem &operator[](size_t i) { return m_children[i]; }
204044d93782SGreg Clayton 
2041b9c1b51eSKate Stone   void SetRowIndex(int row_idx) { m_row_idx = row_idx; }
204244d93782SGreg Clayton 
2043b9c1b51eSKate Stone   size_t GetNumChildren() {
204444d93782SGreg Clayton     m_delegate.TreeDelegateGenerateChildren(*this);
204544d93782SGreg Clayton     return m_children.size();
204644d93782SGreg Clayton   }
204744d93782SGreg Clayton 
2048b9c1b51eSKate Stone   void ItemWasSelected() { m_delegate.TreeDelegateItemSelected(*this); }
2049315b6884SEugene Zelenko 
2050b9c1b51eSKate Stone   void CalculateRowIndexes(int &row_idx) {
205144d93782SGreg Clayton     SetRowIndex(row_idx);
205244d93782SGreg Clayton     ++row_idx;
205344d93782SGreg Clayton 
2054ec990867SGreg Clayton     const bool expanded = IsExpanded();
2055ec990867SGreg Clayton 
205605097246SAdrian Prantl     // The root item must calculate its children, or we must calculate the
205705097246SAdrian Prantl     // number of children if the item is expanded
2058c5dac77aSEugene Zelenko     if (m_parent == nullptr || expanded)
205944d93782SGreg Clayton       GetNumChildren();
206044d93782SGreg Clayton 
2061b9c1b51eSKate Stone     for (auto &item : m_children) {
206244d93782SGreg Clayton       if (expanded)
206344d93782SGreg Clayton         item.CalculateRowIndexes(row_idx);
206444d93782SGreg Clayton       else
206544d93782SGreg Clayton         item.SetRowIndex(-1);
206644d93782SGreg Clayton     }
206744d93782SGreg Clayton   }
206844d93782SGreg Clayton 
2069b9c1b51eSKate Stone   TreeItem *GetParent() { return m_parent; }
207044d93782SGreg Clayton 
2071b9c1b51eSKate Stone   bool IsExpanded() const { return m_is_expanded; }
207244d93782SGreg Clayton 
2073b9c1b51eSKate Stone   void Expand() { m_is_expanded = true; }
207444d93782SGreg Clayton 
2075b9c1b51eSKate Stone   void Unexpand() { m_is_expanded = false; }
207644d93782SGreg Clayton 
2077b9c1b51eSKate Stone   bool Draw(Window &window, const int first_visible_row,
2078b9c1b51eSKate Stone             const uint32_t selected_row_idx, int &row_idx, int &num_rows_left) {
207944d93782SGreg Clayton     if (num_rows_left <= 0)
208044d93782SGreg Clayton       return false;
208144d93782SGreg Clayton 
2082b9c1b51eSKate Stone     if (m_row_idx >= first_visible_row) {
208344d93782SGreg Clayton       window.MoveCursor(2, row_idx + 1);
208444d93782SGreg Clayton 
208544d93782SGreg Clayton       if (m_parent)
208644d93782SGreg Clayton         m_parent->DrawTreeForChild(window, this, 0);
208744d93782SGreg Clayton 
2088b9c1b51eSKate Stone       if (m_might_have_children) {
2089b9c1b51eSKate Stone         // It we can get UTF8 characters to work we should try to use the
209005097246SAdrian Prantl         // "symbol" UTF8 string below
209144d93782SGreg Clayton         //            const char *symbol = "";
209244d93782SGreg Clayton         //            if (row.expanded)
209344d93782SGreg Clayton         //                symbol = "\xe2\x96\xbd ";
209444d93782SGreg Clayton         //            else
209544d93782SGreg Clayton         //                symbol = "\xe2\x96\xb7 ";
209644d93782SGreg Clayton         //            window.PutCString (symbol);
209744d93782SGreg Clayton 
209844d93782SGreg Clayton         // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
209944d93782SGreg Clayton         // 'v' or '>' character...
210044d93782SGreg Clayton         //            if (expanded)
210144d93782SGreg Clayton         //                window.PutChar (ACS_DARROW);
210244d93782SGreg Clayton         //            else
210344d93782SGreg Clayton         //                window.PutChar (ACS_RARROW);
210405097246SAdrian Prantl         // Since we can't find any good looking right arrow/down arrow symbols,
210505097246SAdrian Prantl         // just use a diamond...
210644d93782SGreg Clayton         window.PutChar(ACS_DIAMOND);
210744d93782SGreg Clayton         window.PutChar(ACS_HLINE);
210844d93782SGreg Clayton       }
2109b9c1b51eSKate Stone       bool highlight = (selected_row_idx == static_cast<size_t>(m_row_idx)) &&
2110b9c1b51eSKate Stone                        window.IsActive();
211144d93782SGreg Clayton 
211244d93782SGreg Clayton       if (highlight)
211344d93782SGreg Clayton         window.AttributeOn(A_REVERSE);
211444d93782SGreg Clayton 
211544d93782SGreg Clayton       m_delegate.TreeDelegateDrawTreeItem(*this, window);
211644d93782SGreg Clayton 
211744d93782SGreg Clayton       if (highlight)
211844d93782SGreg Clayton         window.AttributeOff(A_REVERSE);
211944d93782SGreg Clayton       ++row_idx;
212044d93782SGreg Clayton       --num_rows_left;
212144d93782SGreg Clayton     }
212244d93782SGreg Clayton 
212344d93782SGreg Clayton     if (num_rows_left <= 0)
212444d93782SGreg Clayton       return false; // We are done drawing...
212544d93782SGreg Clayton 
2126b9c1b51eSKate Stone     if (IsExpanded()) {
2127b9c1b51eSKate Stone       for (auto &item : m_children) {
212805097246SAdrian Prantl         // If we displayed all the rows and item.Draw() returns false we are
212905097246SAdrian Prantl         // done drawing and can exit this for loop
2130b9c1b51eSKate Stone         if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx,
2131b9c1b51eSKate Stone                        num_rows_left))
213244d93782SGreg Clayton           break;
213344d93782SGreg Clayton       }
213444d93782SGreg Clayton     }
213544d93782SGreg Clayton     return num_rows_left >= 0; // Return true if not done drawing yet
213644d93782SGreg Clayton   }
213744d93782SGreg Clayton 
2138b9c1b51eSKate Stone   void DrawTreeForChild(Window &window, TreeItem *child,
2139b9c1b51eSKate Stone                         uint32_t reverse_depth) {
214044d93782SGreg Clayton     if (m_parent)
214144d93782SGreg Clayton       m_parent->DrawTreeForChild(window, this, reverse_depth + 1);
214244d93782SGreg Clayton 
2143b9c1b51eSKate Stone     if (&m_children.back() == child) {
214444d93782SGreg Clayton       // Last child
2145b9c1b51eSKate Stone       if (reverse_depth == 0) {
214644d93782SGreg Clayton         window.PutChar(ACS_LLCORNER);
214744d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2148b9c1b51eSKate Stone       } else {
214944d93782SGreg Clayton         window.PutChar(' ');
215044d93782SGreg Clayton         window.PutChar(' ');
215144d93782SGreg Clayton       }
2152b9c1b51eSKate Stone     } else {
2153b9c1b51eSKate Stone       if (reverse_depth == 0) {
215444d93782SGreg Clayton         window.PutChar(ACS_LTEE);
215544d93782SGreg Clayton         window.PutChar(ACS_HLINE);
2156b9c1b51eSKate Stone       } else {
215744d93782SGreg Clayton         window.PutChar(ACS_VLINE);
215844d93782SGreg Clayton         window.PutChar(' ');
215944d93782SGreg Clayton       }
216044d93782SGreg Clayton     }
216144d93782SGreg Clayton   }
216244d93782SGreg Clayton 
2163b9c1b51eSKate Stone   TreeItem *GetItemForRowIndex(uint32_t row_idx) {
21643985c8c6SSaleem Abdulrasool     if (static_cast<uint32_t>(m_row_idx) == row_idx)
216544d93782SGreg Clayton       return this;
216644d93782SGreg Clayton     if (m_children.empty())
2167c5dac77aSEugene Zelenko       return nullptr;
2168b9c1b51eSKate Stone     if (IsExpanded()) {
2169b9c1b51eSKate Stone       for (auto &item : m_children) {
217044d93782SGreg Clayton         TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
217144d93782SGreg Clayton         if (selected_item_ptr)
217244d93782SGreg Clayton           return selected_item_ptr;
217344d93782SGreg Clayton       }
217444d93782SGreg Clayton     }
2175c5dac77aSEugene Zelenko     return nullptr;
217644d93782SGreg Clayton   }
217744d93782SGreg Clayton 
2178b9c1b51eSKate Stone   void *GetUserData() const { return m_user_data; }
2179ec990867SGreg Clayton 
2180b9c1b51eSKate Stone   void SetUserData(void *user_data) { m_user_data = user_data; }
2181ec990867SGreg Clayton 
2182b9c1b51eSKate Stone   uint64_t GetIdentifier() const { return m_identifier; }
218344d93782SGreg Clayton 
2184b9c1b51eSKate Stone   void SetIdentifier(uint64_t identifier) { m_identifier = identifier; }
218544d93782SGreg Clayton 
2186b9c1b51eSKate Stone   void SetMightHaveChildren(bool b) { m_might_have_children = b; }
2187ec990867SGreg Clayton 
218844d93782SGreg Clayton protected:
218944d93782SGreg Clayton   TreeItem *m_parent;
219044d93782SGreg Clayton   TreeDelegate &m_delegate;
2191ec990867SGreg Clayton   void *m_user_data;
219244d93782SGreg Clayton   uint64_t m_identifier;
2193b9c1b51eSKate Stone   int m_row_idx; // Zero based visible row index, -1 if not visible or for the
2194b9c1b51eSKate Stone                  // root item
219544d93782SGreg Clayton   std::vector<TreeItem> m_children;
219644d93782SGreg Clayton   bool m_might_have_children;
219744d93782SGreg Clayton   bool m_is_expanded;
219844d93782SGreg Clayton };
219944d93782SGreg Clayton 
2200b9c1b51eSKate Stone class TreeWindowDelegate : public WindowDelegate {
220144d93782SGreg Clayton public:
2202b9c1b51eSKate Stone   TreeWindowDelegate(Debugger &debugger, const TreeDelegateSP &delegate_sp)
2203b9c1b51eSKate Stone       : m_debugger(debugger), m_delegate_sp(delegate_sp),
2204b9c1b51eSKate Stone         m_root(nullptr, *delegate_sp, true), m_selected_item(nullptr),
2205b9c1b51eSKate Stone         m_num_rows(0), m_selected_row_idx(0), m_first_visible_row(0),
2206b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
220744d93782SGreg Clayton 
2208b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
220944d93782SGreg Clayton 
2210b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2211b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2212b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
221344d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
221444d93782SGreg Clayton 
221544d93782SGreg Clayton     bool display_content = false;
2216b9c1b51eSKate Stone     if (process) {
221744d93782SGreg Clayton       StateType state = process->GetState();
2218b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
221944d93782SGreg Clayton         // We are stopped, so it is ok to
222044d93782SGreg Clayton         display_content = true;
2221b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
222244d93782SGreg Clayton         return true; // Don't do any updating when we are running
222344d93782SGreg Clayton       }
222444d93782SGreg Clayton     }
222544d93782SGreg Clayton 
222644d93782SGreg Clayton     m_min_x = 2;
222744d93782SGreg Clayton     m_min_y = 1;
222844d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
222944d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
223044d93782SGreg Clayton 
223144d93782SGreg Clayton     window.Erase();
223244d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
223344d93782SGreg Clayton 
2234b9c1b51eSKate Stone     if (display_content) {
223544d93782SGreg Clayton       const int num_visible_rows = NumVisibleRows();
223644d93782SGreg Clayton       m_num_rows = 0;
223744d93782SGreg Clayton       m_root.CalculateRowIndexes(m_num_rows);
223844d93782SGreg Clayton 
223905097246SAdrian Prantl       // If we unexpanded while having something selected our total number of
224005097246SAdrian Prantl       // rows is less than the num visible rows, then make sure we show all the
224105097246SAdrian Prantl       // rows by setting the first visible row accordingly.
224244d93782SGreg Clayton       if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
224344d93782SGreg Clayton         m_first_visible_row = 0;
224444d93782SGreg Clayton 
224544d93782SGreg Clayton       // Make sure the selected row is always visible
224644d93782SGreg Clayton       if (m_selected_row_idx < m_first_visible_row)
224744d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx;
224844d93782SGreg Clayton       else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
224944d93782SGreg Clayton         m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
225044d93782SGreg Clayton 
225144d93782SGreg Clayton       int row_idx = 0;
225244d93782SGreg Clayton       int num_rows_left = num_visible_rows;
2253b9c1b51eSKate Stone       m_root.Draw(window, m_first_visible_row, m_selected_row_idx, row_idx,
2254b9c1b51eSKate Stone                   num_rows_left);
225544d93782SGreg Clayton       // Get the selected row
225644d93782SGreg Clayton       m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
2257b9c1b51eSKate Stone     } else {
2258c5dac77aSEugene Zelenko       m_selected_item = nullptr;
225944d93782SGreg Clayton     }
226044d93782SGreg Clayton 
226144d93782SGreg Clayton     return true; // Drawing handled
226244d93782SGreg Clayton   }
226344d93782SGreg Clayton 
2264b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
226544d93782SGreg Clayton     return "Thread window keyboard shortcuts:";
226644d93782SGreg Clayton   }
226744d93782SGreg Clayton 
2268b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
226944d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
227044d93782SGreg Clayton         {KEY_UP, "Select previous item"},
227144d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
227244d93782SGreg Clayton         {KEY_RIGHT, "Expand the selected item"},
2273b9c1b51eSKate Stone         {KEY_LEFT,
2274b9c1b51eSKate Stone          "Unexpand the selected item or select parent if not expanded"},
227544d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
227644d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
227744d93782SGreg Clayton         {'h', "Show help dialog"},
227844d93782SGreg Clayton         {' ', "Toggle item expansion"},
227944d93782SGreg Clayton         {',', "Page up"},
228044d93782SGreg Clayton         {'.', "Page down"},
2281b9c1b51eSKate Stone         {'\0', nullptr}};
228244d93782SGreg Clayton     return g_source_view_key_help;
228344d93782SGreg Clayton   }
228444d93782SGreg Clayton 
2285b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2286b9c1b51eSKate Stone     switch (c) {
228744d93782SGreg Clayton     case ',':
228844d93782SGreg Clayton     case KEY_PPAGE:
228944d93782SGreg Clayton       // Page up key
2290b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
229144d93782SGreg Clayton         if (m_first_visible_row > m_max_y)
229244d93782SGreg Clayton           m_first_visible_row -= m_max_y;
229344d93782SGreg Clayton         else
229444d93782SGreg Clayton           m_first_visible_row = 0;
229544d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
229644d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
229744d93782SGreg Clayton         if (m_selected_item)
229844d93782SGreg Clayton           m_selected_item->ItemWasSelected();
229944d93782SGreg Clayton       }
230044d93782SGreg Clayton       return eKeyHandled;
230144d93782SGreg Clayton 
230244d93782SGreg Clayton     case '.':
230344d93782SGreg Clayton     case KEY_NPAGE:
230444d93782SGreg Clayton       // Page down key
2305b9c1b51eSKate Stone       if (m_num_rows > m_max_y) {
2306b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
230744d93782SGreg Clayton           m_first_visible_row += m_max_y;
230844d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
230944d93782SGreg Clayton           m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
231044d93782SGreg Clayton           if (m_selected_item)
231144d93782SGreg Clayton             m_selected_item->ItemWasSelected();
231244d93782SGreg Clayton         }
231344d93782SGreg Clayton       }
231444d93782SGreg Clayton       return eKeyHandled;
231544d93782SGreg Clayton 
231644d93782SGreg Clayton     case KEY_UP:
2317b9c1b51eSKate Stone       if (m_selected_row_idx > 0) {
231844d93782SGreg Clayton         --m_selected_row_idx;
231944d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
232044d93782SGreg Clayton         if (m_selected_item)
232144d93782SGreg Clayton           m_selected_item->ItemWasSelected();
232244d93782SGreg Clayton       }
232344d93782SGreg Clayton       return eKeyHandled;
2324315b6884SEugene Zelenko 
232544d93782SGreg Clayton     case KEY_DOWN:
2326b9c1b51eSKate Stone       if (m_selected_row_idx + 1 < m_num_rows) {
232744d93782SGreg Clayton         ++m_selected_row_idx;
232844d93782SGreg Clayton         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
232944d93782SGreg Clayton         if (m_selected_item)
233044d93782SGreg Clayton           m_selected_item->ItemWasSelected();
233144d93782SGreg Clayton       }
233244d93782SGreg Clayton       return eKeyHandled;
233344d93782SGreg Clayton 
233444d93782SGreg Clayton     case KEY_RIGHT:
2335b9c1b51eSKate Stone       if (m_selected_item) {
233644d93782SGreg Clayton         if (!m_selected_item->IsExpanded())
233744d93782SGreg Clayton           m_selected_item->Expand();
233844d93782SGreg Clayton       }
233944d93782SGreg Clayton       return eKeyHandled;
234044d93782SGreg Clayton 
234144d93782SGreg Clayton     case KEY_LEFT:
2342b9c1b51eSKate Stone       if (m_selected_item) {
234344d93782SGreg Clayton         if (m_selected_item->IsExpanded())
234444d93782SGreg Clayton           m_selected_item->Unexpand();
2345b9c1b51eSKate Stone         else if (m_selected_item->GetParent()) {
234644d93782SGreg Clayton           m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
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       }
235244d93782SGreg Clayton       return eKeyHandled;
235344d93782SGreg Clayton 
235444d93782SGreg Clayton     case ' ':
235544d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2356b9c1b51eSKate Stone       if (m_selected_item) {
235744d93782SGreg Clayton         if (m_selected_item->IsExpanded())
235844d93782SGreg Clayton           m_selected_item->Unexpand();
235944d93782SGreg Clayton         else
236044d93782SGreg Clayton           m_selected_item->Expand();
236144d93782SGreg Clayton       }
236244d93782SGreg Clayton       return eKeyHandled;
236344d93782SGreg Clayton 
236444d93782SGreg Clayton     case 'h':
236544d93782SGreg Clayton       window.CreateHelpSubwindow();
236644d93782SGreg Clayton       return eKeyHandled;
236744d93782SGreg Clayton 
236844d93782SGreg Clayton     default:
236944d93782SGreg Clayton       break;
237044d93782SGreg Clayton     }
237144d93782SGreg Clayton     return eKeyNotHandled;
237244d93782SGreg Clayton   }
237344d93782SGreg Clayton 
237444d93782SGreg Clayton protected:
237544d93782SGreg Clayton   Debugger &m_debugger;
237644d93782SGreg Clayton   TreeDelegateSP m_delegate_sp;
237744d93782SGreg Clayton   TreeItem m_root;
237844d93782SGreg Clayton   TreeItem *m_selected_item;
237944d93782SGreg Clayton   int m_num_rows;
238044d93782SGreg Clayton   int m_selected_row_idx;
238144d93782SGreg Clayton   int m_first_visible_row;
238244d93782SGreg Clayton   int m_min_x;
238344d93782SGreg Clayton   int m_min_y;
238444d93782SGreg Clayton   int m_max_x;
238544d93782SGreg Clayton   int m_max_y;
238644d93782SGreg Clayton };
238744d93782SGreg Clayton 
2388b9c1b51eSKate Stone class FrameTreeDelegate : public TreeDelegate {
238944d93782SGreg Clayton public:
2390b9c1b51eSKate Stone   FrameTreeDelegate() : TreeDelegate() {
2391b9c1b51eSKate Stone     FormatEntity::Parse(
2392b9c1b51eSKate Stone         "frame #${frame.index}: {${function.name}${function.pc-offset}}}",
2393554f68d3SGreg Clayton         m_format);
239444d93782SGreg Clayton   }
239544d93782SGreg Clayton 
2396315b6884SEugene Zelenko   ~FrameTreeDelegate() override = default;
239744d93782SGreg Clayton 
2398b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2399ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2400b9c1b51eSKate Stone     if (thread) {
240144d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2402ec990867SGreg Clayton       StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
2403b9c1b51eSKate Stone       if (frame_sp) {
240444d93782SGreg Clayton         StreamString strm;
2405b9c1b51eSKate Stone         const SymbolContext &sc =
2406b9c1b51eSKate Stone             frame_sp->GetSymbolContext(eSymbolContextEverything);
240744d93782SGreg Clayton         ExecutionContext exe_ctx(frame_sp);
2408b9c1b51eSKate Stone         if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr,
2409b9c1b51eSKate Stone                                  nullptr, false, false)) {
241044d93782SGreg Clayton           int right_pad = 1;
2411c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
241244d93782SGreg Clayton         }
241344d93782SGreg Clayton       }
241444d93782SGreg Clayton     }
241544d93782SGreg Clayton   }
2416315b6884SEugene Zelenko 
2417b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
241844d93782SGreg Clayton     // No children for frames yet...
241944d93782SGreg Clayton   }
242044d93782SGreg Clayton 
2421b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2422ec990867SGreg Clayton     Thread *thread = (Thread *)item.GetUserData();
2423b9c1b51eSKate Stone     if (thread) {
2424b9c1b51eSKate Stone       thread->GetProcess()->GetThreadList().SetSelectedThreadByID(
2425b9c1b51eSKate Stone           thread->GetID());
242644d93782SGreg Clayton       const uint64_t frame_idx = item.GetIdentifier();
2427ec990867SGreg Clayton       thread->SetSelectedFrameByIndex(frame_idx);
242844d93782SGreg Clayton       return true;
242944d93782SGreg Clayton     }
243044d93782SGreg Clayton     return false;
243144d93782SGreg Clayton   }
2432315b6884SEugene Zelenko 
2433554f68d3SGreg Clayton protected:
2434554f68d3SGreg Clayton   FormatEntity::Entry m_format;
243544d93782SGreg Clayton };
243644d93782SGreg Clayton 
2437b9c1b51eSKate Stone class ThreadTreeDelegate : public TreeDelegate {
243844d93782SGreg Clayton public:
2439b9c1b51eSKate Stone   ThreadTreeDelegate(Debugger &debugger)
2440b9c1b51eSKate Stone       : TreeDelegate(), m_debugger(debugger), m_tid(LLDB_INVALID_THREAD_ID),
2441b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2442b9c1b51eSKate Stone     FormatEntity::Parse("thread #${thread.index}: tid = ${thread.id}{, stop "
2443b9c1b51eSKate Stone                         "reason = ${thread.stop-reason}}",
2444554f68d3SGreg Clayton                         m_format);
244544d93782SGreg Clayton   }
244644d93782SGreg Clayton 
2447315b6884SEugene Zelenko   ~ThreadTreeDelegate() override = default;
244844d93782SGreg Clayton 
2449b9c1b51eSKate Stone   ProcessSP GetProcess() {
2450b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2451b9c1b51eSKate Stone         .GetExecutionContext()
2452b9c1b51eSKate Stone         .GetProcessSP();
2453ec990867SGreg Clayton   }
2454ec990867SGreg Clayton 
2455b9c1b51eSKate Stone   ThreadSP GetThread(const TreeItem &item) {
2456ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2457ec990867SGreg Clayton     if (process_sp)
2458ec990867SGreg Clayton       return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
2459ec990867SGreg Clayton     return ThreadSP();
2460ec990867SGreg Clayton   }
2461ec990867SGreg Clayton 
2462b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2463ec990867SGreg Clayton     ThreadSP thread_sp = GetThread(item);
2464b9c1b51eSKate Stone     if (thread_sp) {
246544d93782SGreg Clayton       StreamString strm;
246644d93782SGreg Clayton       ExecutionContext exe_ctx(thread_sp);
2467b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2468b9c1b51eSKate Stone                                nullptr, false, false)) {
246944d93782SGreg Clayton         int right_pad = 1;
2470c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
247144d93782SGreg Clayton       }
247244d93782SGreg Clayton     }
247344d93782SGreg Clayton   }
2474315b6884SEugene Zelenko 
2475b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2476ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2477b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
247844d93782SGreg Clayton       StateType state = process_sp->GetState();
2479b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2480ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2481b9c1b51eSKate Stone         if (thread_sp) {
2482b9c1b51eSKate Stone           if (m_stop_id == process_sp->GetStopID() &&
2483b9c1b51eSKate Stone               thread_sp->GetID() == m_tid)
248444d93782SGreg Clayton             return; // Children are already up to date
2485b9c1b51eSKate Stone           if (!m_frame_delegate_sp) {
248644d93782SGreg Clayton             // Always expand the thread item the first time we show it
2487796ac80bSJonas Devlieghere             m_frame_delegate_sp = std::make_shared<FrameTreeDelegate>();
248844d93782SGreg Clayton           }
248944d93782SGreg Clayton 
249044d93782SGreg Clayton           m_stop_id = process_sp->GetStopID();
249144d93782SGreg Clayton           m_tid = thread_sp->GetID();
249244d93782SGreg Clayton 
249344d93782SGreg Clayton           TreeItem t(&item, *m_frame_delegate_sp, false);
249444d93782SGreg Clayton           size_t num_frames = thread_sp->GetStackFrameCount();
249544d93782SGreg Clayton           item.Resize(num_frames, t);
2496b9c1b51eSKate Stone           for (size_t i = 0; i < num_frames; ++i) {
2497ec990867SGreg Clayton             item[i].SetUserData(thread_sp.get());
249844d93782SGreg Clayton             item[i].SetIdentifier(i);
249944d93782SGreg Clayton           }
250044d93782SGreg Clayton         }
250144d93782SGreg Clayton         return;
250244d93782SGreg Clayton       }
250344d93782SGreg Clayton     }
250444d93782SGreg Clayton     item.ClearChildren();
250544d93782SGreg Clayton   }
250644d93782SGreg Clayton 
2507b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override {
2508ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2509b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2510ec990867SGreg Clayton       StateType state = process_sp->GetState();
2511b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2512ec990867SGreg Clayton         ThreadSP thread_sp = GetThread(item);
2513b9c1b51eSKate Stone         if (thread_sp) {
251444d93782SGreg Clayton           ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
2515bb19a13cSSaleem Abdulrasool           std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
251644d93782SGreg Clayton           ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
2517b9c1b51eSKate Stone           if (selected_thread_sp->GetID() != thread_sp->GetID()) {
251844d93782SGreg Clayton             thread_list.SetSelectedThreadByID(thread_sp->GetID());
251944d93782SGreg Clayton             return true;
252044d93782SGreg Clayton           }
252144d93782SGreg Clayton         }
2522ec990867SGreg Clayton       }
2523ec990867SGreg Clayton     }
252444d93782SGreg Clayton     return false;
252544d93782SGreg Clayton   }
252644d93782SGreg Clayton 
252744d93782SGreg Clayton protected:
252844d93782SGreg Clayton   Debugger &m_debugger;
252944d93782SGreg Clayton   std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
253044d93782SGreg Clayton   lldb::user_id_t m_tid;
253144d93782SGreg Clayton   uint32_t m_stop_id;
2532554f68d3SGreg Clayton   FormatEntity::Entry m_format;
253344d93782SGreg Clayton };
253444d93782SGreg Clayton 
2535b9c1b51eSKate Stone class ThreadsTreeDelegate : public TreeDelegate {
2536ec990867SGreg Clayton public:
2537b9c1b51eSKate Stone   ThreadsTreeDelegate(Debugger &debugger)
2538b9c1b51eSKate Stone       : TreeDelegate(), m_thread_delegate_sp(), m_debugger(debugger),
2539b9c1b51eSKate Stone         m_stop_id(UINT32_MAX) {
2540554f68d3SGreg Clayton     FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
2541554f68d3SGreg Clayton                         m_format);
2542ec990867SGreg Clayton   }
2543ec990867SGreg Clayton 
2544315b6884SEugene Zelenko   ~ThreadsTreeDelegate() override = default;
2545ec990867SGreg Clayton 
2546b9c1b51eSKate Stone   ProcessSP GetProcess() {
2547b9c1b51eSKate Stone     return m_debugger.GetCommandInterpreter()
2548b9c1b51eSKate Stone         .GetExecutionContext()
2549b9c1b51eSKate Stone         .GetProcessSP();
2550ec990867SGreg Clayton   }
2551ec990867SGreg Clayton 
2552b9c1b51eSKate Stone   void TreeDelegateDrawTreeItem(TreeItem &item, Window &window) override {
2553ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2554b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2555ec990867SGreg Clayton       StreamString strm;
2556ec990867SGreg Clayton       ExecutionContext exe_ctx(process_sp);
2557b9c1b51eSKate Stone       if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr,
2558b9c1b51eSKate Stone                                nullptr, false, false)) {
2559ec990867SGreg Clayton         int right_pad = 1;
2560c156427dSZachary Turner         window.PutCStringTruncated(strm.GetString().str().c_str(), right_pad);
2561ec990867SGreg Clayton       }
2562ec990867SGreg Clayton     }
2563ec990867SGreg Clayton   }
2564ec990867SGreg Clayton 
2565b9c1b51eSKate Stone   void TreeDelegateGenerateChildren(TreeItem &item) override {
2566ec990867SGreg Clayton     ProcessSP process_sp = GetProcess();
2567b9c1b51eSKate Stone     if (process_sp && process_sp->IsAlive()) {
2568ec990867SGreg Clayton       StateType state = process_sp->GetState();
2569b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
2570ec990867SGreg Clayton         const uint32_t stop_id = process_sp->GetStopID();
2571ec990867SGreg Clayton         if (m_stop_id == stop_id)
2572ec990867SGreg Clayton           return; // Children are already up to date
2573ec990867SGreg Clayton 
2574ec990867SGreg Clayton         m_stop_id = stop_id;
2575ec990867SGreg Clayton 
2576b9c1b51eSKate Stone         if (!m_thread_delegate_sp) {
2577ec990867SGreg Clayton           // Always expand the thread item the first time we show it
2578ec990867SGreg Clayton           // item.Expand();
2579796ac80bSJonas Devlieghere           m_thread_delegate_sp =
2580796ac80bSJonas Devlieghere               std::make_shared<ThreadTreeDelegate>(m_debugger);
2581ec990867SGreg Clayton         }
2582ec990867SGreg Clayton 
2583ec990867SGreg Clayton         TreeItem t(&item, *m_thread_delegate_sp, false);
2584ec990867SGreg Clayton         ThreadList &threads = process_sp->GetThreadList();
2585bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
2586ec990867SGreg Clayton         size_t num_threads = threads.GetSize();
2587ec990867SGreg Clayton         item.Resize(num_threads, t);
2588b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
2589ec990867SGreg Clayton           item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
2590ec990867SGreg Clayton           item[i].SetMightHaveChildren(true);
2591ec990867SGreg Clayton         }
2592ec990867SGreg Clayton         return;
2593ec990867SGreg Clayton       }
2594ec990867SGreg Clayton     }
2595ec990867SGreg Clayton     item.ClearChildren();
2596ec990867SGreg Clayton   }
2597ec990867SGreg Clayton 
2598b9c1b51eSKate Stone   bool TreeDelegateItemSelected(TreeItem &item) override { return false; }
2599ec990867SGreg Clayton 
2600ec990867SGreg Clayton protected:
2601ec990867SGreg Clayton   std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
2602ec990867SGreg Clayton   Debugger &m_debugger;
2603ec990867SGreg Clayton   uint32_t m_stop_id;
2604554f68d3SGreg Clayton   FormatEntity::Entry m_format;
2605ec990867SGreg Clayton };
2606ec990867SGreg Clayton 
2607b9c1b51eSKate Stone class ValueObjectListDelegate : public WindowDelegate {
260844d93782SGreg Clayton public:
2609b9c1b51eSKate Stone   ValueObjectListDelegate()
2610171dd2e6SJonas Devlieghere       : m_rows(), m_selected_row(nullptr), m_selected_row_idx(0),
2611171dd2e6SJonas Devlieghere         m_first_visible_row(0), m_num_rows(0), m_max_x(0), m_max_y(0) {}
261244d93782SGreg Clayton 
2613b9c1b51eSKate Stone   ValueObjectListDelegate(ValueObjectList &valobj_list)
2614171dd2e6SJonas Devlieghere       : m_rows(), m_selected_row(nullptr), m_selected_row_idx(0),
2615171dd2e6SJonas Devlieghere         m_first_visible_row(0), m_num_rows(0), m_max_x(0), m_max_y(0) {
261644d93782SGreg Clayton     SetValues(valobj_list);
261744d93782SGreg Clayton   }
261844d93782SGreg Clayton 
2619315b6884SEugene Zelenko   ~ValueObjectListDelegate() override = default;
262044d93782SGreg Clayton 
2621b9c1b51eSKate Stone   void SetValues(ValueObjectList &valobj_list) {
2622c5dac77aSEugene Zelenko     m_selected_row = nullptr;
262344d93782SGreg Clayton     m_selected_row_idx = 0;
262444d93782SGreg Clayton     m_first_visible_row = 0;
262544d93782SGreg Clayton     m_num_rows = 0;
262644d93782SGreg Clayton     m_rows.clear();
26278369b28dSGreg Clayton     for (auto &valobj_sp : valobj_list.GetObjects())
26288369b28dSGreg Clayton       m_rows.push_back(Row(valobj_sp, nullptr));
262944d93782SGreg Clayton   }
263044d93782SGreg Clayton 
2631b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
263244d93782SGreg Clayton     m_num_rows = 0;
263344d93782SGreg Clayton     m_min_x = 2;
263444d93782SGreg Clayton     m_min_y = 1;
263544d93782SGreg Clayton     m_max_x = window.GetWidth() - 1;
263644d93782SGreg Clayton     m_max_y = window.GetHeight() - 1;
263744d93782SGreg Clayton 
263844d93782SGreg Clayton     window.Erase();
263944d93782SGreg Clayton     window.DrawTitleBox(window.GetName());
264044d93782SGreg Clayton 
264144d93782SGreg Clayton     const int num_visible_rows = NumVisibleRows();
264244d93782SGreg Clayton     const int num_rows = CalculateTotalNumberRows(m_rows);
264344d93782SGreg Clayton 
264405097246SAdrian Prantl     // If we unexpanded while having something selected our total number of
264505097246SAdrian Prantl     // rows is less than the num visible rows, then make sure we show all the
264605097246SAdrian Prantl     // rows by setting the first visible row accordingly.
264744d93782SGreg Clayton     if (m_first_visible_row > 0 && num_rows < num_visible_rows)
264844d93782SGreg Clayton       m_first_visible_row = 0;
264944d93782SGreg Clayton 
265044d93782SGreg Clayton     // Make sure the selected row is always visible
265144d93782SGreg Clayton     if (m_selected_row_idx < m_first_visible_row)
265244d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx;
265344d93782SGreg Clayton     else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
265444d93782SGreg Clayton       m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
265544d93782SGreg Clayton 
265644d93782SGreg Clayton     DisplayRows(window, m_rows, g_options);
265744d93782SGreg Clayton 
265844d93782SGreg Clayton     // Get the selected row
265944d93782SGreg Clayton     m_selected_row = GetRowForRowIndex(m_selected_row_idx);
266005097246SAdrian Prantl     // Keep the cursor on the selected row so the highlight and the cursor are
266105097246SAdrian Prantl     // always on the same line
266244d93782SGreg Clayton     if (m_selected_row)
2663b9c1b51eSKate Stone       window.MoveCursor(m_selected_row->x, m_selected_row->y);
266444d93782SGreg Clayton 
266544d93782SGreg Clayton     return true; // Drawing handled
266644d93782SGreg Clayton   }
266744d93782SGreg Clayton 
2668b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
266944d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
267044d93782SGreg Clayton         {KEY_UP, "Select previous item"},
267144d93782SGreg Clayton         {KEY_DOWN, "Select next item"},
267244d93782SGreg Clayton         {KEY_RIGHT, "Expand selected item"},
267344d93782SGreg Clayton         {KEY_LEFT, "Unexpand selected item or select parent if not expanded"},
267444d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
267544d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
267644d93782SGreg Clayton         {'A', "Format as annotated address"},
267744d93782SGreg Clayton         {'b', "Format as binary"},
267844d93782SGreg Clayton         {'B', "Format as hex bytes with ASCII"},
267944d93782SGreg Clayton         {'c', "Format as character"},
268044d93782SGreg Clayton         {'d', "Format as a signed integer"},
268144d93782SGreg Clayton         {'D', "Format selected value using the default format for the type"},
268244d93782SGreg Clayton         {'f', "Format as float"},
268344d93782SGreg Clayton         {'h', "Show help dialog"},
268444d93782SGreg Clayton         {'i', "Format as instructions"},
268544d93782SGreg Clayton         {'o', "Format as octal"},
268644d93782SGreg Clayton         {'p', "Format as pointer"},
268744d93782SGreg Clayton         {'s', "Format as C string"},
268844d93782SGreg Clayton         {'t', "Toggle showing/hiding type names"},
268944d93782SGreg Clayton         {'u', "Format as an unsigned integer"},
269044d93782SGreg Clayton         {'x', "Format as hex"},
269144d93782SGreg Clayton         {'X', "Format as uppercase hex"},
269244d93782SGreg Clayton         {' ', "Toggle item expansion"},
269344d93782SGreg Clayton         {',', "Page up"},
269444d93782SGreg Clayton         {'.', "Page down"},
2695b9c1b51eSKate Stone         {'\0', nullptr}};
269644d93782SGreg Clayton     return g_source_view_key_help;
269744d93782SGreg Clayton   }
269844d93782SGreg Clayton 
2699b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
2700b9c1b51eSKate Stone     switch (c) {
270144d93782SGreg Clayton     case 'x':
270244d93782SGreg Clayton     case 'X':
270344d93782SGreg Clayton     case 'o':
270444d93782SGreg Clayton     case 's':
270544d93782SGreg Clayton     case 'u':
270644d93782SGreg Clayton     case 'd':
270744d93782SGreg Clayton     case 'D':
270844d93782SGreg Clayton     case 'i':
270944d93782SGreg Clayton     case 'A':
271044d93782SGreg Clayton     case 'p':
271144d93782SGreg Clayton     case 'c':
271244d93782SGreg Clayton     case 'b':
271344d93782SGreg Clayton     case 'B':
271444d93782SGreg Clayton     case 'f':
271544d93782SGreg Clayton       // Change the format for the currently selected item
27168369b28dSGreg Clayton       if (m_selected_row) {
27178369b28dSGreg Clayton         auto valobj_sp = m_selected_row->value.GetSP();
27188369b28dSGreg Clayton         if (valobj_sp)
27198369b28dSGreg Clayton           valobj_sp->SetFormat(FormatForChar(c));
27208369b28dSGreg Clayton       }
272144d93782SGreg Clayton       return eKeyHandled;
272244d93782SGreg Clayton 
272344d93782SGreg Clayton     case 't':
272444d93782SGreg Clayton       // Toggle showing type names
272544d93782SGreg Clayton       g_options.show_types = !g_options.show_types;
272644d93782SGreg Clayton       return eKeyHandled;
272744d93782SGreg Clayton 
272844d93782SGreg Clayton     case ',':
272944d93782SGreg Clayton     case KEY_PPAGE:
273044d93782SGreg Clayton       // Page up key
2731b9c1b51eSKate Stone       if (m_first_visible_row > 0) {
27323985c8c6SSaleem Abdulrasool         if (static_cast<int>(m_first_visible_row) > m_max_y)
273344d93782SGreg Clayton           m_first_visible_row -= m_max_y;
273444d93782SGreg Clayton         else
273544d93782SGreg Clayton           m_first_visible_row = 0;
273644d93782SGreg Clayton         m_selected_row_idx = m_first_visible_row;
273744d93782SGreg Clayton       }
273844d93782SGreg Clayton       return eKeyHandled;
273944d93782SGreg Clayton 
274044d93782SGreg Clayton     case '.':
274144d93782SGreg Clayton     case KEY_NPAGE:
274244d93782SGreg Clayton       // Page down key
2743b9c1b51eSKate Stone       if (m_num_rows > static_cast<size_t>(m_max_y)) {
2744b9c1b51eSKate Stone         if (m_first_visible_row + m_max_y < m_num_rows) {
274544d93782SGreg Clayton           m_first_visible_row += m_max_y;
274644d93782SGreg Clayton           m_selected_row_idx = m_first_visible_row;
274744d93782SGreg Clayton         }
274844d93782SGreg Clayton       }
274944d93782SGreg Clayton       return eKeyHandled;
275044d93782SGreg Clayton 
275144d93782SGreg Clayton     case KEY_UP:
275244d93782SGreg Clayton       if (m_selected_row_idx > 0)
275344d93782SGreg Clayton         --m_selected_row_idx;
275444d93782SGreg Clayton       return eKeyHandled;
2755315b6884SEugene Zelenko 
275644d93782SGreg Clayton     case KEY_DOWN:
275744d93782SGreg Clayton       if (m_selected_row_idx + 1 < m_num_rows)
275844d93782SGreg Clayton         ++m_selected_row_idx;
275944d93782SGreg Clayton       return eKeyHandled;
276044d93782SGreg Clayton 
276144d93782SGreg Clayton     case KEY_RIGHT:
2762b9c1b51eSKate Stone       if (m_selected_row) {
276344d93782SGreg Clayton         if (!m_selected_row->expanded)
276444d93782SGreg Clayton           m_selected_row->Expand();
276544d93782SGreg Clayton       }
276644d93782SGreg Clayton       return eKeyHandled;
276744d93782SGreg Clayton 
276844d93782SGreg Clayton     case KEY_LEFT:
2769b9c1b51eSKate Stone       if (m_selected_row) {
277044d93782SGreg Clayton         if (m_selected_row->expanded)
277144d93782SGreg Clayton           m_selected_row->Unexpand();
277244d93782SGreg Clayton         else if (m_selected_row->parent)
277344d93782SGreg Clayton           m_selected_row_idx = m_selected_row->parent->row_idx;
277444d93782SGreg Clayton       }
277544d93782SGreg Clayton       return eKeyHandled;
277644d93782SGreg Clayton 
277744d93782SGreg Clayton     case ' ':
277844d93782SGreg Clayton       // Toggle expansion state when SPACE is pressed
2779b9c1b51eSKate Stone       if (m_selected_row) {
278044d93782SGreg Clayton         if (m_selected_row->expanded)
278144d93782SGreg Clayton           m_selected_row->Unexpand();
278244d93782SGreg Clayton         else
278344d93782SGreg Clayton           m_selected_row->Expand();
278444d93782SGreg Clayton       }
278544d93782SGreg Clayton       return eKeyHandled;
278644d93782SGreg Clayton 
278744d93782SGreg Clayton     case 'h':
278844d93782SGreg Clayton       window.CreateHelpSubwindow();
278944d93782SGreg Clayton       return eKeyHandled;
279044d93782SGreg Clayton 
279144d93782SGreg Clayton     default:
279244d93782SGreg Clayton       break;
279344d93782SGreg Clayton     }
279444d93782SGreg Clayton     return eKeyNotHandled;
279544d93782SGreg Clayton   }
279644d93782SGreg Clayton 
279744d93782SGreg Clayton protected:
279844d93782SGreg Clayton   std::vector<Row> m_rows;
279944d93782SGreg Clayton   Row *m_selected_row;
280044d93782SGreg Clayton   uint32_t m_selected_row_idx;
280144d93782SGreg Clayton   uint32_t m_first_visible_row;
280244d93782SGreg Clayton   uint32_t m_num_rows;
280344d93782SGreg Clayton   int m_min_x;
280444d93782SGreg Clayton   int m_min_y;
280544d93782SGreg Clayton   int m_max_x;
280644d93782SGreg Clayton   int m_max_y;
280744d93782SGreg Clayton 
2808b9c1b51eSKate Stone   static Format FormatForChar(int c) {
2809b9c1b51eSKate Stone     switch (c) {
2810b9c1b51eSKate Stone     case 'x':
2811b9c1b51eSKate Stone       return eFormatHex;
2812b9c1b51eSKate Stone     case 'X':
2813b9c1b51eSKate Stone       return eFormatHexUppercase;
2814b9c1b51eSKate Stone     case 'o':
2815b9c1b51eSKate Stone       return eFormatOctal;
2816b9c1b51eSKate Stone     case 's':
2817b9c1b51eSKate Stone       return eFormatCString;
2818b9c1b51eSKate Stone     case 'u':
2819b9c1b51eSKate Stone       return eFormatUnsigned;
2820b9c1b51eSKate Stone     case 'd':
2821b9c1b51eSKate Stone       return eFormatDecimal;
2822b9c1b51eSKate Stone     case 'D':
2823b9c1b51eSKate Stone       return eFormatDefault;
2824b9c1b51eSKate Stone     case 'i':
2825b9c1b51eSKate Stone       return eFormatInstruction;
2826b9c1b51eSKate Stone     case 'A':
2827b9c1b51eSKate Stone       return eFormatAddressInfo;
2828b9c1b51eSKate Stone     case 'p':
2829b9c1b51eSKate Stone       return eFormatPointer;
2830b9c1b51eSKate Stone     case 'c':
2831b9c1b51eSKate Stone       return eFormatChar;
2832b9c1b51eSKate Stone     case 'b':
2833b9c1b51eSKate Stone       return eFormatBinary;
2834b9c1b51eSKate Stone     case 'B':
2835b9c1b51eSKate Stone       return eFormatBytesWithASCII;
2836b9c1b51eSKate Stone     case 'f':
2837b9c1b51eSKate Stone       return eFormatFloat;
283844d93782SGreg Clayton     }
283944d93782SGreg Clayton     return eFormatDefault;
284044d93782SGreg Clayton   }
284144d93782SGreg Clayton 
2842b9c1b51eSKate Stone   bool DisplayRowObject(Window &window, Row &row, DisplayOptions &options,
2843b9c1b51eSKate Stone                         bool highlight, bool last_child) {
28448369b28dSGreg Clayton     ValueObject *valobj = row.value.GetSP().get();
284544d93782SGreg Clayton 
2846c5dac77aSEugene Zelenko     if (valobj == nullptr)
284744d93782SGreg Clayton       return false;
284844d93782SGreg Clayton 
2849b9c1b51eSKate Stone     const char *type_name =
2850b9c1b51eSKate Stone         options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
285144d93782SGreg Clayton     const char *name = valobj->GetName().GetCString();
285244d93782SGreg Clayton     const char *value = valobj->GetValueAsCString();
285344d93782SGreg Clayton     const char *summary = valobj->GetSummaryAsCString();
285444d93782SGreg Clayton 
285544d93782SGreg Clayton     window.MoveCursor(row.x, row.y);
285644d93782SGreg Clayton 
285744d93782SGreg Clayton     row.DrawTree(window);
285844d93782SGreg Clayton 
285944d93782SGreg Clayton     if (highlight)
286044d93782SGreg Clayton       window.AttributeOn(A_REVERSE);
286144d93782SGreg Clayton 
286244d93782SGreg Clayton     if (type_name && type_name[0])
286344d93782SGreg Clayton       window.Printf("(%s) ", type_name);
286444d93782SGreg Clayton 
286544d93782SGreg Clayton     if (name && name[0])
286644d93782SGreg Clayton       window.PutCString(name);
286744d93782SGreg Clayton 
286844d93782SGreg Clayton     attr_t changd_attr = 0;
286944d93782SGreg Clayton     if (valobj->GetValueDidChange())
287044d93782SGreg Clayton       changd_attr = COLOR_PAIR(5) | A_BOLD;
287144d93782SGreg Clayton 
2872b9c1b51eSKate Stone     if (value && value[0]) {
287344d93782SGreg Clayton       window.PutCString(" = ");
287444d93782SGreg Clayton       if (changd_attr)
287544d93782SGreg Clayton         window.AttributeOn(changd_attr);
287644d93782SGreg Clayton       window.PutCString(value);
287744d93782SGreg Clayton       if (changd_attr)
287844d93782SGreg Clayton         window.AttributeOff(changd_attr);
287944d93782SGreg Clayton     }
288044d93782SGreg Clayton 
2881b9c1b51eSKate Stone     if (summary && summary[0]) {
288244d93782SGreg Clayton       window.PutChar(' ');
288344d93782SGreg Clayton       if (changd_attr)
288444d93782SGreg Clayton         window.AttributeOn(changd_attr);
288544d93782SGreg Clayton       window.PutCString(summary);
288644d93782SGreg Clayton       if (changd_attr)
288744d93782SGreg Clayton         window.AttributeOff(changd_attr);
288844d93782SGreg Clayton     }
288944d93782SGreg Clayton 
289044d93782SGreg Clayton     if (highlight)
289144d93782SGreg Clayton       window.AttributeOff(A_REVERSE);
289244d93782SGreg Clayton 
289344d93782SGreg Clayton     return true;
289444d93782SGreg Clayton   }
2895315b6884SEugene Zelenko 
2896b9c1b51eSKate Stone   void DisplayRows(Window &window, std::vector<Row> &rows,
2897b9c1b51eSKate Stone                    DisplayOptions &options) {
289844d93782SGreg Clayton     // >   0x25B7
289944d93782SGreg Clayton     // \/  0x25BD
290044d93782SGreg Clayton 
290144d93782SGreg Clayton     bool window_is_active = window.IsActive();
2902b9c1b51eSKate Stone     for (auto &row : rows) {
290344d93782SGreg Clayton       const bool last_child = row.parent && &rows[rows.size() - 1] == &row;
290444d93782SGreg Clayton       // Save the row index in each Row structure
290544d93782SGreg Clayton       row.row_idx = m_num_rows;
290644d93782SGreg Clayton       if ((m_num_rows >= m_first_visible_row) &&
2907b9c1b51eSKate Stone           ((m_num_rows - m_first_visible_row) <
2908b9c1b51eSKate Stone            static_cast<size_t>(NumVisibleRows()))) {
290944d93782SGreg Clayton         row.x = m_min_x;
291044d93782SGreg Clayton         row.y = m_num_rows - m_first_visible_row + 1;
2911b9c1b51eSKate Stone         if (DisplayRowObject(window, row, options,
2912b9c1b51eSKate Stone                              window_is_active &&
2913b9c1b51eSKate Stone                                  m_num_rows == m_selected_row_idx,
2914b9c1b51eSKate Stone                              last_child)) {
291544d93782SGreg Clayton           ++m_num_rows;
2916b9c1b51eSKate Stone         } else {
291744d93782SGreg Clayton           row.x = 0;
291844d93782SGreg Clayton           row.y = 0;
291944d93782SGreg Clayton         }
2920b9c1b51eSKate Stone       } else {
292144d93782SGreg Clayton         row.x = 0;
292244d93782SGreg Clayton         row.y = 0;
292344d93782SGreg Clayton         ++m_num_rows;
292444d93782SGreg Clayton       }
292544d93782SGreg Clayton 
29268369b28dSGreg Clayton       auto &children = row.GetChildren();
29278369b28dSGreg Clayton       if (row.expanded && !children.empty()) {
29288369b28dSGreg Clayton         DisplayRows(window, children, options);
292944d93782SGreg Clayton       }
293044d93782SGreg Clayton     }
293144d93782SGreg Clayton   }
293244d93782SGreg Clayton 
29338369b28dSGreg Clayton   int CalculateTotalNumberRows(std::vector<Row> &rows) {
293444d93782SGreg Clayton     int row_count = 0;
29358369b28dSGreg Clayton     for (auto &row : rows) {
293644d93782SGreg Clayton       ++row_count;
293744d93782SGreg Clayton       if (row.expanded)
29388369b28dSGreg Clayton         row_count += CalculateTotalNumberRows(row.GetChildren());
293944d93782SGreg Clayton     }
294044d93782SGreg Clayton     return row_count;
294144d93782SGreg Clayton   }
2942315b6884SEugene Zelenko 
2943b9c1b51eSKate Stone   static Row *GetRowForRowIndexImpl(std::vector<Row> &rows, size_t &row_index) {
2944b9c1b51eSKate Stone     for (auto &row : rows) {
294544d93782SGreg Clayton       if (row_index == 0)
294644d93782SGreg Clayton         return &row;
2947b9c1b51eSKate Stone       else {
294844d93782SGreg Clayton         --row_index;
29498369b28dSGreg Clayton         auto &children = row.GetChildren();
29508369b28dSGreg Clayton         if (row.expanded && !children.empty()) {
29518369b28dSGreg Clayton           Row *result = GetRowForRowIndexImpl(children, row_index);
295244d93782SGreg Clayton           if (result)
295344d93782SGreg Clayton             return result;
295444d93782SGreg Clayton         }
295544d93782SGreg Clayton       }
295644d93782SGreg Clayton     }
2957c5dac77aSEugene Zelenko     return nullptr;
295844d93782SGreg Clayton   }
295944d93782SGreg Clayton 
2960b9c1b51eSKate Stone   Row *GetRowForRowIndex(size_t row_index) {
296144d93782SGreg Clayton     return GetRowForRowIndexImpl(m_rows, row_index);
296244d93782SGreg Clayton   }
296344d93782SGreg Clayton 
2964b9c1b51eSKate Stone   int NumVisibleRows() const { return m_max_y - m_min_y; }
296544d93782SGreg Clayton 
296644d93782SGreg Clayton   static DisplayOptions g_options;
296744d93782SGreg Clayton };
296844d93782SGreg Clayton 
2969b9c1b51eSKate Stone class FrameVariablesWindowDelegate : public ValueObjectListDelegate {
297044d93782SGreg Clayton public:
2971b9c1b51eSKate Stone   FrameVariablesWindowDelegate(Debugger &debugger)
2972b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger),
2973b9c1b51eSKate Stone         m_frame_block(nullptr) {}
297444d93782SGreg Clayton 
2975315b6884SEugene Zelenko   ~FrameVariablesWindowDelegate() override = default;
297644d93782SGreg Clayton 
2977b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
297844d93782SGreg Clayton     return "Frame variable window keyboard shortcuts:";
297944d93782SGreg Clayton   }
298044d93782SGreg Clayton 
2981b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
2982b9c1b51eSKate Stone     ExecutionContext exe_ctx(
2983b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
298444d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
2985c5dac77aSEugene Zelenko     Block *frame_block = nullptr;
2986c5dac77aSEugene Zelenko     StackFrame *frame = nullptr;
298744d93782SGreg Clayton 
2988b9c1b51eSKate Stone     if (process) {
298944d93782SGreg Clayton       StateType state = process->GetState();
2990b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
299144d93782SGreg Clayton         frame = exe_ctx.GetFramePtr();
299244d93782SGreg Clayton         if (frame)
299344d93782SGreg Clayton           frame_block = frame->GetFrameBlock();
2994b9c1b51eSKate Stone       } else if (StateIsRunningState(state)) {
299544d93782SGreg Clayton         return true; // Don't do any updating when we are running
299644d93782SGreg Clayton       }
299744d93782SGreg Clayton     }
299844d93782SGreg Clayton 
299944d93782SGreg Clayton     ValueObjectList local_values;
3000b9c1b51eSKate Stone     if (frame_block) {
300144d93782SGreg Clayton       // Only update the variables if they have changed
3002b9c1b51eSKate Stone       if (m_frame_block != frame_block) {
300344d93782SGreg Clayton         m_frame_block = frame_block;
300444d93782SGreg Clayton 
300544d93782SGreg Clayton         VariableList *locals = frame->GetVariableList(true);
3006b9c1b51eSKate Stone         if (locals) {
300744d93782SGreg Clayton           const DynamicValueType use_dynamic = eDynamicDontRunTarget;
300844d93782SGreg Clayton           const size_t num_locals = locals->GetSize();
3009b9c1b51eSKate Stone           for (size_t i = 0; i < num_locals; ++i) {
3010b9c1b51eSKate Stone             ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable(
3011b9c1b51eSKate Stone                 locals->GetVariableAtIndex(i), use_dynamic);
3012b9c1b51eSKate Stone             if (value_sp) {
3013eb72dc7dSGreg Clayton               ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3014eb72dc7dSGreg Clayton               if (synthetic_value_sp)
3015eb72dc7dSGreg Clayton                 local_values.Append(synthetic_value_sp);
3016eb72dc7dSGreg Clayton               else
3017eb72dc7dSGreg Clayton                 local_values.Append(value_sp);
3018eb72dc7dSGreg Clayton             }
3019eb72dc7dSGreg Clayton           }
302044d93782SGreg Clayton           // Update the values
302144d93782SGreg Clayton           SetValues(local_values);
302244d93782SGreg Clayton         }
302344d93782SGreg Clayton       }
3024b9c1b51eSKate Stone     } else {
3025c5dac77aSEugene Zelenko       m_frame_block = nullptr;
302644d93782SGreg Clayton       // Update the values with an empty list if there is no frame
302744d93782SGreg Clayton       SetValues(local_values);
302844d93782SGreg Clayton     }
302944d93782SGreg Clayton 
303044d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
303144d93782SGreg Clayton   }
303244d93782SGreg Clayton 
303344d93782SGreg Clayton protected:
303444d93782SGreg Clayton   Debugger &m_debugger;
303544d93782SGreg Clayton   Block *m_frame_block;
303644d93782SGreg Clayton };
303744d93782SGreg Clayton 
3038b9c1b51eSKate Stone class RegistersWindowDelegate : public ValueObjectListDelegate {
303944d93782SGreg Clayton public:
3040b9c1b51eSKate Stone   RegistersWindowDelegate(Debugger &debugger)
3041b9c1b51eSKate Stone       : ValueObjectListDelegate(), m_debugger(debugger) {}
304244d93782SGreg Clayton 
3043315b6884SEugene Zelenko   ~RegistersWindowDelegate() override = default;
304444d93782SGreg Clayton 
3045b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
304644d93782SGreg Clayton     return "Register window keyboard shortcuts:";
304744d93782SGreg Clayton   }
304844d93782SGreg Clayton 
3049b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3050b9c1b51eSKate Stone     ExecutionContext exe_ctx(
3051b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext());
305244d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
305344d93782SGreg Clayton 
305444d93782SGreg Clayton     ValueObjectList value_list;
3055b9c1b51eSKate Stone     if (frame) {
3056b9c1b51eSKate Stone       if (frame->GetStackID() != m_stack_id) {
305744d93782SGreg Clayton         m_stack_id = frame->GetStackID();
305844d93782SGreg Clayton         RegisterContextSP reg_ctx(frame->GetRegisterContext());
3059b9c1b51eSKate Stone         if (reg_ctx) {
306044d93782SGreg Clayton           const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
3061b9c1b51eSKate Stone           for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) {
3062b9c1b51eSKate Stone             value_list.Append(
3063b9c1b51eSKate Stone                 ValueObjectRegisterSet::Create(frame, reg_ctx, set_idx));
306444d93782SGreg Clayton           }
306544d93782SGreg Clayton         }
306644d93782SGreg Clayton         SetValues(value_list);
306744d93782SGreg Clayton       }
3068b9c1b51eSKate Stone     } else {
306944d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
307044d93782SGreg Clayton       if (process && process->IsAlive())
307144d93782SGreg Clayton         return true; // Don't do any updating if we are running
3072b9c1b51eSKate Stone       else {
307305097246SAdrian Prantl         // Update the values with an empty list if there is no process or the
307405097246SAdrian Prantl         // process isn't alive anymore
307544d93782SGreg Clayton         SetValues(value_list);
307644d93782SGreg Clayton       }
307744d93782SGreg Clayton     }
307844d93782SGreg Clayton     return ValueObjectListDelegate::WindowDelegateDraw(window, force);
307944d93782SGreg Clayton   }
308044d93782SGreg Clayton 
308144d93782SGreg Clayton protected:
308244d93782SGreg Clayton   Debugger &m_debugger;
308344d93782SGreg Clayton   StackID m_stack_id;
308444d93782SGreg Clayton };
308544d93782SGreg Clayton 
3086b9c1b51eSKate Stone static const char *CursesKeyToCString(int ch) {
308744d93782SGreg Clayton   static char g_desc[32];
3088b9c1b51eSKate Stone   if (ch >= KEY_F0 && ch < KEY_F0 + 64) {
308944d93782SGreg Clayton     snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
309044d93782SGreg Clayton     return g_desc;
309144d93782SGreg Clayton   }
3092b9c1b51eSKate Stone   switch (ch) {
3093b9c1b51eSKate Stone   case KEY_DOWN:
3094b9c1b51eSKate Stone     return "down";
3095b9c1b51eSKate Stone   case KEY_UP:
3096b9c1b51eSKate Stone     return "up";
3097b9c1b51eSKate Stone   case KEY_LEFT:
3098b9c1b51eSKate Stone     return "left";
3099b9c1b51eSKate Stone   case KEY_RIGHT:
3100b9c1b51eSKate Stone     return "right";
3101b9c1b51eSKate Stone   case KEY_HOME:
3102b9c1b51eSKate Stone     return "home";
3103b9c1b51eSKate Stone   case KEY_BACKSPACE:
3104b9c1b51eSKate Stone     return "backspace";
3105b9c1b51eSKate Stone   case KEY_DL:
3106b9c1b51eSKate Stone     return "delete-line";
3107b9c1b51eSKate Stone   case KEY_IL:
3108b9c1b51eSKate Stone     return "insert-line";
3109b9c1b51eSKate Stone   case KEY_DC:
3110b9c1b51eSKate Stone     return "delete-char";
3111b9c1b51eSKate Stone   case KEY_IC:
3112b9c1b51eSKate Stone     return "insert-char";
3113b9c1b51eSKate Stone   case KEY_CLEAR:
3114b9c1b51eSKate Stone     return "clear";
3115b9c1b51eSKate Stone   case KEY_EOS:
3116b9c1b51eSKate Stone     return "clear-to-eos";
3117b9c1b51eSKate Stone   case KEY_EOL:
3118b9c1b51eSKate Stone     return "clear-to-eol";
3119b9c1b51eSKate Stone   case KEY_SF:
3120b9c1b51eSKate Stone     return "scroll-forward";
3121b9c1b51eSKate Stone   case KEY_SR:
3122b9c1b51eSKate Stone     return "scroll-backward";
3123b9c1b51eSKate Stone   case KEY_NPAGE:
3124b9c1b51eSKate Stone     return "page-down";
3125b9c1b51eSKate Stone   case KEY_PPAGE:
3126b9c1b51eSKate Stone     return "page-up";
3127b9c1b51eSKate Stone   case KEY_STAB:
3128b9c1b51eSKate Stone     return "set-tab";
3129b9c1b51eSKate Stone   case KEY_CTAB:
3130b9c1b51eSKate Stone     return "clear-tab";
3131b9c1b51eSKate Stone   case KEY_CATAB:
3132b9c1b51eSKate Stone     return "clear-all-tabs";
3133b9c1b51eSKate Stone   case KEY_ENTER:
3134b9c1b51eSKate Stone     return "enter";
3135b9c1b51eSKate Stone   case KEY_PRINT:
3136b9c1b51eSKate Stone     return "print";
3137b9c1b51eSKate Stone   case KEY_LL:
3138b9c1b51eSKate Stone     return "lower-left key";
3139b9c1b51eSKate Stone   case KEY_A1:
3140b9c1b51eSKate Stone     return "upper left of keypad";
3141b9c1b51eSKate Stone   case KEY_A3:
3142b9c1b51eSKate Stone     return "upper right of keypad";
3143b9c1b51eSKate Stone   case KEY_B2:
3144b9c1b51eSKate Stone     return "center of keypad";
3145b9c1b51eSKate Stone   case KEY_C1:
3146b9c1b51eSKate Stone     return "lower left of keypad";
3147b9c1b51eSKate Stone   case KEY_C3:
3148b9c1b51eSKate Stone     return "lower right of keypad";
3149b9c1b51eSKate Stone   case KEY_BTAB:
3150b9c1b51eSKate Stone     return "back-tab key";
3151b9c1b51eSKate Stone   case KEY_BEG:
3152b9c1b51eSKate Stone     return "begin key";
3153b9c1b51eSKate Stone   case KEY_CANCEL:
3154b9c1b51eSKate Stone     return "cancel key";
3155b9c1b51eSKate Stone   case KEY_CLOSE:
3156b9c1b51eSKate Stone     return "close key";
3157b9c1b51eSKate Stone   case KEY_COMMAND:
3158b9c1b51eSKate Stone     return "command key";
3159b9c1b51eSKate Stone   case KEY_COPY:
3160b9c1b51eSKate Stone     return "copy key";
3161b9c1b51eSKate Stone   case KEY_CREATE:
3162b9c1b51eSKate Stone     return "create key";
3163b9c1b51eSKate Stone   case KEY_END:
3164b9c1b51eSKate Stone     return "end key";
3165b9c1b51eSKate Stone   case KEY_EXIT:
3166b9c1b51eSKate Stone     return "exit key";
3167b9c1b51eSKate Stone   case KEY_FIND:
3168b9c1b51eSKate Stone     return "find key";
3169b9c1b51eSKate Stone   case KEY_HELP:
3170b9c1b51eSKate Stone     return "help key";
3171b9c1b51eSKate Stone   case KEY_MARK:
3172b9c1b51eSKate Stone     return "mark key";
3173b9c1b51eSKate Stone   case KEY_MESSAGE:
3174b9c1b51eSKate Stone     return "message key";
3175b9c1b51eSKate Stone   case KEY_MOVE:
3176b9c1b51eSKate Stone     return "move key";
3177b9c1b51eSKate Stone   case KEY_NEXT:
3178b9c1b51eSKate Stone     return "next key";
3179b9c1b51eSKate Stone   case KEY_OPEN:
3180b9c1b51eSKate Stone     return "open key";
3181b9c1b51eSKate Stone   case KEY_OPTIONS:
3182b9c1b51eSKate Stone     return "options key";
3183b9c1b51eSKate Stone   case KEY_PREVIOUS:
3184b9c1b51eSKate Stone     return "previous key";
3185b9c1b51eSKate Stone   case KEY_REDO:
3186b9c1b51eSKate Stone     return "redo key";
3187b9c1b51eSKate Stone   case KEY_REFERENCE:
3188b9c1b51eSKate Stone     return "reference key";
3189b9c1b51eSKate Stone   case KEY_REFRESH:
3190b9c1b51eSKate Stone     return "refresh key";
3191b9c1b51eSKate Stone   case KEY_REPLACE:
3192b9c1b51eSKate Stone     return "replace key";
3193b9c1b51eSKate Stone   case KEY_RESTART:
3194b9c1b51eSKate Stone     return "restart key";
3195b9c1b51eSKate Stone   case KEY_RESUME:
3196b9c1b51eSKate Stone     return "resume key";
3197b9c1b51eSKate Stone   case KEY_SAVE:
3198b9c1b51eSKate Stone     return "save key";
3199b9c1b51eSKate Stone   case KEY_SBEG:
3200b9c1b51eSKate Stone     return "shifted begin key";
3201b9c1b51eSKate Stone   case KEY_SCANCEL:
3202b9c1b51eSKate Stone     return "shifted cancel key";
3203b9c1b51eSKate Stone   case KEY_SCOMMAND:
3204b9c1b51eSKate Stone     return "shifted command key";
3205b9c1b51eSKate Stone   case KEY_SCOPY:
3206b9c1b51eSKate Stone     return "shifted copy key";
3207b9c1b51eSKate Stone   case KEY_SCREATE:
3208b9c1b51eSKate Stone     return "shifted create key";
3209b9c1b51eSKate Stone   case KEY_SDC:
3210b9c1b51eSKate Stone     return "shifted delete-character key";
3211b9c1b51eSKate Stone   case KEY_SDL:
3212b9c1b51eSKate Stone     return "shifted delete-line key";
3213b9c1b51eSKate Stone   case KEY_SELECT:
3214b9c1b51eSKate Stone     return "select key";
3215b9c1b51eSKate Stone   case KEY_SEND:
3216b9c1b51eSKate Stone     return "shifted end key";
3217b9c1b51eSKate Stone   case KEY_SEOL:
3218b9c1b51eSKate Stone     return "shifted clear-to-end-of-line key";
3219b9c1b51eSKate Stone   case KEY_SEXIT:
3220b9c1b51eSKate Stone     return "shifted exit key";
3221b9c1b51eSKate Stone   case KEY_SFIND:
3222b9c1b51eSKate Stone     return "shifted find key";
3223b9c1b51eSKate Stone   case KEY_SHELP:
3224b9c1b51eSKate Stone     return "shifted help key";
3225b9c1b51eSKate Stone   case KEY_SHOME:
3226b9c1b51eSKate Stone     return "shifted home key";
3227b9c1b51eSKate Stone   case KEY_SIC:
3228b9c1b51eSKate Stone     return "shifted insert-character key";
3229b9c1b51eSKate Stone   case KEY_SLEFT:
3230b9c1b51eSKate Stone     return "shifted left-arrow key";
3231b9c1b51eSKate Stone   case KEY_SMESSAGE:
3232b9c1b51eSKate Stone     return "shifted message key";
3233b9c1b51eSKate Stone   case KEY_SMOVE:
3234b9c1b51eSKate Stone     return "shifted move key";
3235b9c1b51eSKate Stone   case KEY_SNEXT:
3236b9c1b51eSKate Stone     return "shifted next key";
3237b9c1b51eSKate Stone   case KEY_SOPTIONS:
3238b9c1b51eSKate Stone     return "shifted options key";
3239b9c1b51eSKate Stone   case KEY_SPREVIOUS:
3240b9c1b51eSKate Stone     return "shifted previous key";
3241b9c1b51eSKate Stone   case KEY_SPRINT:
3242b9c1b51eSKate Stone     return "shifted print key";
3243b9c1b51eSKate Stone   case KEY_SREDO:
3244b9c1b51eSKate Stone     return "shifted redo key";
3245b9c1b51eSKate Stone   case KEY_SREPLACE:
3246b9c1b51eSKate Stone     return "shifted replace key";
3247b9c1b51eSKate Stone   case KEY_SRIGHT:
3248b9c1b51eSKate Stone     return "shifted right-arrow key";
3249b9c1b51eSKate Stone   case KEY_SRSUME:
3250b9c1b51eSKate Stone     return "shifted resume key";
3251b9c1b51eSKate Stone   case KEY_SSAVE:
3252b9c1b51eSKate Stone     return "shifted save key";
3253b9c1b51eSKate Stone   case KEY_SSUSPEND:
3254b9c1b51eSKate Stone     return "shifted suspend key";
3255b9c1b51eSKate Stone   case KEY_SUNDO:
3256b9c1b51eSKate Stone     return "shifted undo key";
3257b9c1b51eSKate Stone   case KEY_SUSPEND:
3258b9c1b51eSKate Stone     return "suspend key";
3259b9c1b51eSKate Stone   case KEY_UNDO:
3260b9c1b51eSKate Stone     return "undo key";
3261b9c1b51eSKate Stone   case KEY_MOUSE:
3262b9c1b51eSKate Stone     return "Mouse event has occurred";
3263b9c1b51eSKate Stone   case KEY_RESIZE:
3264b9c1b51eSKate Stone     return "Terminal resize event";
326527801f4fSBruce Mitchener #ifdef KEY_EVENT
3266b9c1b51eSKate Stone   case KEY_EVENT:
3267b9c1b51eSKate Stone     return "We were interrupted by an event";
326827801f4fSBruce Mitchener #endif
3269b9c1b51eSKate Stone   case KEY_RETURN:
3270b9c1b51eSKate Stone     return "return";
3271b9c1b51eSKate Stone   case ' ':
3272b9c1b51eSKate Stone     return "space";
3273b9c1b51eSKate Stone   case '\t':
3274b9c1b51eSKate Stone     return "tab";
3275b9c1b51eSKate Stone   case KEY_ESCAPE:
3276b9c1b51eSKate Stone     return "escape";
327744d93782SGreg Clayton   default:
327844d93782SGreg Clayton     if (isprint(ch))
327944d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "%c", ch);
328044d93782SGreg Clayton     else
328144d93782SGreg Clayton       snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
328244d93782SGreg Clayton     return g_desc;
328344d93782SGreg Clayton   }
3284c5dac77aSEugene Zelenko   return nullptr;
328544d93782SGreg Clayton }
328644d93782SGreg Clayton 
3287b9c1b51eSKate Stone HelpDialogDelegate::HelpDialogDelegate(const char *text,
3288b9c1b51eSKate Stone                                        KeyHelp *key_help_array)
3289b9c1b51eSKate Stone     : m_text(), m_first_visible_line(0) {
3290b9c1b51eSKate Stone   if (text && text[0]) {
329144d93782SGreg Clayton     m_text.SplitIntoLines(text);
329244d93782SGreg Clayton     m_text.AppendString("");
329344d93782SGreg Clayton   }
3294b9c1b51eSKate Stone   if (key_help_array) {
3295b9c1b51eSKate Stone     for (KeyHelp *key = key_help_array; key->ch; ++key) {
329644d93782SGreg Clayton       StreamString key_description;
3297b9c1b51eSKate Stone       key_description.Printf("%10s - %s", CursesKeyToCString(key->ch),
3298b9c1b51eSKate Stone                              key->description);
3299c156427dSZachary Turner       m_text.AppendString(key_description.GetString());
330044d93782SGreg Clayton     }
330144d93782SGreg Clayton   }
330244d93782SGreg Clayton }
330344d93782SGreg Clayton 
3304315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
330544d93782SGreg Clayton 
3306b9c1b51eSKate Stone bool HelpDialogDelegate::WindowDelegateDraw(Window &window, bool force) {
330744d93782SGreg Clayton   window.Erase();
330844d93782SGreg Clayton   const int window_height = window.GetHeight();
330944d93782SGreg Clayton   int x = 2;
331044d93782SGreg Clayton   int y = 1;
331144d93782SGreg Clayton   const int min_y = y;
331244d93782SGreg Clayton   const int max_y = window_height - 1 - y;
33133985c8c6SSaleem Abdulrasool   const size_t num_visible_lines = max_y - min_y + 1;
331444d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
331544d93782SGreg Clayton   const char *bottom_message;
331644d93782SGreg Clayton   if (num_lines <= num_visible_lines)
331744d93782SGreg Clayton     bottom_message = "Press any key to exit";
331844d93782SGreg Clayton   else
331944d93782SGreg Clayton     bottom_message = "Use arrows to scroll, any other key to exit";
332044d93782SGreg Clayton   window.DrawTitleBox(window.GetName(), bottom_message);
3321b9c1b51eSKate Stone   while (y <= max_y) {
332244d93782SGreg Clayton     window.MoveCursor(x, y);
3323b9c1b51eSKate Stone     window.PutCStringTruncated(
3324b9c1b51eSKate Stone         m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
332544d93782SGreg Clayton     ++y;
332644d93782SGreg Clayton   }
332744d93782SGreg Clayton   return true;
332844d93782SGreg Clayton }
332944d93782SGreg Clayton 
3330b9c1b51eSKate Stone HandleCharResult HelpDialogDelegate::WindowDelegateHandleChar(Window &window,
3331b9c1b51eSKate Stone                                                               int key) {
333244d93782SGreg Clayton   bool done = false;
333344d93782SGreg Clayton   const size_t num_lines = m_text.GetSize();
333444d93782SGreg Clayton   const size_t num_visible_lines = window.GetHeight() - 2;
333544d93782SGreg Clayton 
3336b9c1b51eSKate Stone   if (num_lines <= num_visible_lines) {
333744d93782SGreg Clayton     done = true;
333805097246SAdrian Prantl     // If we have all lines visible and don't need scrolling, then any key
333905097246SAdrian Prantl     // press will cause us to exit
3340b9c1b51eSKate Stone   } else {
3341b9c1b51eSKate Stone     switch (key) {
334244d93782SGreg Clayton     case KEY_UP:
334344d93782SGreg Clayton       if (m_first_visible_line > 0)
334444d93782SGreg Clayton         --m_first_visible_line;
334544d93782SGreg Clayton       break;
334644d93782SGreg Clayton 
334744d93782SGreg Clayton     case KEY_DOWN:
334844d93782SGreg Clayton       if (m_first_visible_line + num_visible_lines < num_lines)
334944d93782SGreg Clayton         ++m_first_visible_line;
335044d93782SGreg Clayton       break;
335144d93782SGreg Clayton 
335244d93782SGreg Clayton     case KEY_PPAGE:
335344d93782SGreg Clayton     case ',':
3354b9c1b51eSKate Stone       if (m_first_visible_line > 0) {
33553985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
335644d93782SGreg Clayton           m_first_visible_line -= num_visible_lines;
335744d93782SGreg Clayton         else
335844d93782SGreg Clayton           m_first_visible_line = 0;
335944d93782SGreg Clayton       }
336044d93782SGreg Clayton       break;
3361315b6884SEugene Zelenko 
336244d93782SGreg Clayton     case KEY_NPAGE:
336344d93782SGreg Clayton     case '.':
3364b9c1b51eSKate Stone       if (m_first_visible_line + num_visible_lines < num_lines) {
336544d93782SGreg Clayton         m_first_visible_line += num_visible_lines;
33663985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > num_lines)
336744d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
336844d93782SGreg Clayton       }
336944d93782SGreg Clayton       break;
3370315b6884SEugene Zelenko 
337144d93782SGreg Clayton     default:
337244d93782SGreg Clayton       done = true;
337344d93782SGreg Clayton       break;
337444d93782SGreg Clayton     }
337544d93782SGreg Clayton   }
337644d93782SGreg Clayton   if (done)
337744d93782SGreg Clayton     window.GetParent()->RemoveSubWindow(&window);
337844d93782SGreg Clayton   return eKeyHandled;
337944d93782SGreg Clayton }
338044d93782SGreg Clayton 
3381b9c1b51eSKate Stone class ApplicationDelegate : public WindowDelegate, public MenuDelegate {
338244d93782SGreg Clayton public:
338344d93782SGreg Clayton   enum {
338444d93782SGreg Clayton     eMenuID_LLDB = 1,
338544d93782SGreg Clayton     eMenuID_LLDBAbout,
338644d93782SGreg Clayton     eMenuID_LLDBExit,
338744d93782SGreg Clayton 
338844d93782SGreg Clayton     eMenuID_Target,
338944d93782SGreg Clayton     eMenuID_TargetCreate,
339044d93782SGreg Clayton     eMenuID_TargetDelete,
339144d93782SGreg Clayton 
339244d93782SGreg Clayton     eMenuID_Process,
339344d93782SGreg Clayton     eMenuID_ProcessAttach,
339444d93782SGreg Clayton     eMenuID_ProcessDetach,
339544d93782SGreg Clayton     eMenuID_ProcessLaunch,
339644d93782SGreg Clayton     eMenuID_ProcessContinue,
339744d93782SGreg Clayton     eMenuID_ProcessHalt,
339844d93782SGreg Clayton     eMenuID_ProcessKill,
339944d93782SGreg Clayton 
340044d93782SGreg Clayton     eMenuID_Thread,
340144d93782SGreg Clayton     eMenuID_ThreadStepIn,
340244d93782SGreg Clayton     eMenuID_ThreadStepOver,
340344d93782SGreg Clayton     eMenuID_ThreadStepOut,
340444d93782SGreg Clayton 
340544d93782SGreg Clayton     eMenuID_View,
340644d93782SGreg Clayton     eMenuID_ViewBacktrace,
340744d93782SGreg Clayton     eMenuID_ViewRegisters,
340844d93782SGreg Clayton     eMenuID_ViewSource,
340944d93782SGreg Clayton     eMenuID_ViewVariables,
341044d93782SGreg Clayton 
341144d93782SGreg Clayton     eMenuID_Help,
341244d93782SGreg Clayton     eMenuID_HelpGUIHelp
341344d93782SGreg Clayton   };
341444d93782SGreg Clayton 
3415b9c1b51eSKate Stone   ApplicationDelegate(Application &app, Debugger &debugger)
3416b9c1b51eSKate Stone       : WindowDelegate(), MenuDelegate(), m_app(app), m_debugger(debugger) {}
341744d93782SGreg Clayton 
3418315b6884SEugene Zelenko   ~ApplicationDelegate() override = default;
3419bd5ae6b4SGreg Clayton 
3420b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
342144d93782SGreg Clayton     return false; // Drawing not handled, let standard window drawing happen
342244d93782SGreg Clayton   }
342344d93782SGreg Clayton 
3424b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int key) override {
3425b9c1b51eSKate Stone     switch (key) {
34265fdb09bbSGreg Clayton     case '\t':
342744d93782SGreg Clayton       window.SelectNextWindowAsActive();
342844d93782SGreg Clayton       return eKeyHandled;
34295fdb09bbSGreg Clayton 
34305fdb09bbSGreg Clayton     case 'h':
34315fdb09bbSGreg Clayton       window.CreateHelpSubwindow();
34325fdb09bbSGreg Clayton       return eKeyHandled;
34335fdb09bbSGreg Clayton 
34345fdb09bbSGreg Clayton     case KEY_ESCAPE:
34355fdb09bbSGreg Clayton       return eQuitApplication;
34365fdb09bbSGreg Clayton 
34375fdb09bbSGreg Clayton     default:
34385fdb09bbSGreg Clayton       break;
343944d93782SGreg Clayton     }
344044d93782SGreg Clayton     return eKeyNotHandled;
344144d93782SGreg Clayton   }
344244d93782SGreg Clayton 
3443b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
34445fdb09bbSGreg Clayton     return "Welcome to the LLDB curses GUI.\n\n"
34455fdb09bbSGreg Clayton            "Press the TAB key to change the selected view.\n"
3446b9c1b51eSKate Stone            "Each view has its own keyboard shortcuts, press 'h' to open a "
3447b9c1b51eSKate Stone            "dialog to display them.\n\n"
34485fdb09bbSGreg Clayton            "Common key bindings for all views:";
34495fdb09bbSGreg Clayton   }
34505fdb09bbSGreg Clayton 
3451b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
34525fdb09bbSGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
34535fdb09bbSGreg Clayton         {'\t', "Select next view"},
34545fdb09bbSGreg Clayton         {'h', "Show help dialog with view specific key bindings"},
34555fdb09bbSGreg Clayton         {',', "Page up"},
34565fdb09bbSGreg Clayton         {'.', "Page down"},
34575fdb09bbSGreg Clayton         {KEY_UP, "Select previous"},
34585fdb09bbSGreg Clayton         {KEY_DOWN, "Select next"},
34595fdb09bbSGreg Clayton         {KEY_LEFT, "Unexpand or select parent"},
34605fdb09bbSGreg Clayton         {KEY_RIGHT, "Expand"},
34615fdb09bbSGreg Clayton         {KEY_PPAGE, "Page up"},
34625fdb09bbSGreg Clayton         {KEY_NPAGE, "Page down"},
3463b9c1b51eSKate Stone         {'\0', nullptr}};
34645fdb09bbSGreg Clayton     return g_source_view_key_help;
34655fdb09bbSGreg Clayton   }
34665fdb09bbSGreg Clayton 
3467b9c1b51eSKate Stone   MenuActionResult MenuDelegateAction(Menu &menu) override {
3468b9c1b51eSKate Stone     switch (menu.GetIdentifier()) {
3469b9c1b51eSKate Stone     case eMenuID_ThreadStepIn: {
3470b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3471b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3472b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
347344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3474b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3475b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
34764b4b2478SJim Ingham           exe_ctx.GetThreadRef().StepIn(true);
347744d93782SGreg Clayton       }
347844d93782SGreg Clayton     }
347944d93782SGreg Clayton       return MenuActionResult::Handled;
348044d93782SGreg Clayton 
3481b9c1b51eSKate Stone     case eMenuID_ThreadStepOut: {
3482b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3483b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3484b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
348544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3486b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3487b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
348844d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
348944d93782SGreg Clayton       }
349044d93782SGreg Clayton     }
349144d93782SGreg Clayton       return MenuActionResult::Handled;
349244d93782SGreg Clayton 
3493b9c1b51eSKate Stone     case eMenuID_ThreadStepOver: {
3494b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3495b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3496b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope()) {
349744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3498b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3499b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
350044d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOver(true);
350144d93782SGreg Clayton       }
350244d93782SGreg Clayton     }
350344d93782SGreg Clayton       return MenuActionResult::Handled;
350444d93782SGreg Clayton 
3505b9c1b51eSKate Stone     case eMenuID_ProcessContinue: {
3506b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3507b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3508b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
350944d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3510b9c1b51eSKate Stone         if (process && process->IsAlive() &&
3511b9c1b51eSKate Stone             StateIsStoppedState(process->GetState(), true))
351244d93782SGreg Clayton           process->Resume();
351344d93782SGreg Clayton       }
351444d93782SGreg Clayton     }
351544d93782SGreg Clayton       return MenuActionResult::Handled;
351644d93782SGreg Clayton 
3517b9c1b51eSKate Stone     case eMenuID_ProcessKill: {
3518b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3519b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3520b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
352144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
352244d93782SGreg Clayton         if (process && process->IsAlive())
3523ede3193bSJason Molenda           process->Destroy(false);
352444d93782SGreg Clayton       }
352544d93782SGreg Clayton     }
352644d93782SGreg Clayton       return MenuActionResult::Handled;
352744d93782SGreg Clayton 
3528b9c1b51eSKate Stone     case eMenuID_ProcessHalt: {
3529b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3530b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3531b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
353244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
353344d93782SGreg Clayton         if (process && process->IsAlive())
353444d93782SGreg Clayton           process->Halt();
353544d93782SGreg Clayton       }
353644d93782SGreg Clayton     }
353744d93782SGreg Clayton       return MenuActionResult::Handled;
353844d93782SGreg Clayton 
3539b9c1b51eSKate Stone     case eMenuID_ProcessDetach: {
3540b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3541b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
3542b9c1b51eSKate Stone       if (exe_ctx.HasProcessScope()) {
354344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
354444d93782SGreg Clayton         if (process && process->IsAlive())
354544d93782SGreg Clayton           process->Detach(false);
354644d93782SGreg Clayton       }
354744d93782SGreg Clayton     }
354844d93782SGreg Clayton       return MenuActionResult::Handled;
354944d93782SGreg Clayton 
3550b9c1b51eSKate Stone     case eMenuID_Process: {
3551b9c1b51eSKate Stone       // Populate the menu with all of the threads if the process is stopped
355205097246SAdrian Prantl       // when the Process menu gets selected and is about to display its
355305097246SAdrian Prantl       // submenu.
355444d93782SGreg Clayton       Menus &submenus = menu.GetSubmenus();
3555b9c1b51eSKate Stone       ExecutionContext exe_ctx =
3556b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
355744d93782SGreg Clayton       Process *process = exe_ctx.GetProcessPtr();
3558b9c1b51eSKate Stone       if (process && process->IsAlive() &&
3559b9c1b51eSKate Stone           StateIsStoppedState(process->GetState(), true)) {
356044d93782SGreg Clayton         if (submenus.size() == 7)
356144d93782SGreg Clayton           menu.AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
356244d93782SGreg Clayton         else if (submenus.size() > 8)
356344d93782SGreg Clayton           submenus.erase(submenus.begin() + 8, submenus.end());
356444d93782SGreg Clayton 
356544d93782SGreg Clayton         ThreadList &threads = process->GetThreadList();
3566bb19a13cSSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
356744d93782SGreg Clayton         size_t num_threads = threads.GetSize();
3568b9c1b51eSKate Stone         for (size_t i = 0; i < num_threads; ++i) {
356944d93782SGreg Clayton           ThreadSP thread_sp = threads.GetThreadAtIndex(i);
357044d93782SGreg Clayton           char menu_char = '\0';
357144d93782SGreg Clayton           if (i < 9)
357244d93782SGreg Clayton             menu_char = '1' + i;
357344d93782SGreg Clayton           StreamString thread_menu_title;
357444d93782SGreg Clayton           thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
357544d93782SGreg Clayton           const char *thread_name = thread_sp->GetName();
357644d93782SGreg Clayton           if (thread_name && thread_name[0])
357744d93782SGreg Clayton             thread_menu_title.Printf(" %s", thread_name);
3578b9c1b51eSKate Stone           else {
357944d93782SGreg Clayton             const char *queue_name = thread_sp->GetQueueName();
358044d93782SGreg Clayton             if (queue_name && queue_name[0])
358144d93782SGreg Clayton               thread_menu_title.Printf(" %s", queue_name);
358244d93782SGreg Clayton           }
3583b9c1b51eSKate Stone           menu.AddSubmenu(
3584c156427dSZachary Turner               MenuSP(new Menu(thread_menu_title.GetString().str().c_str(),
3585c156427dSZachary Turner                               nullptr, menu_char, thread_sp->GetID())));
358644d93782SGreg Clayton         }
3587b9c1b51eSKate Stone       } else if (submenus.size() > 7) {
358805097246SAdrian Prantl         // Remove the separator and any other thread submenu items that were
358905097246SAdrian Prantl         // previously added
359044d93782SGreg Clayton         submenus.erase(submenus.begin() + 7, submenus.end());
359144d93782SGreg Clayton       }
3592b9c1b51eSKate Stone       // Since we are adding and removing items we need to recalculate the name
3593b9c1b51eSKate Stone       // lengths
359444d93782SGreg Clayton       menu.RecalculateNameLengths();
359544d93782SGreg Clayton     }
359644d93782SGreg Clayton       return MenuActionResult::Handled;
359744d93782SGreg Clayton 
3598b9c1b51eSKate Stone     case eMenuID_ViewVariables: {
359944d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
360044d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
360144d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
360244d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
360344d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
360444d93782SGreg Clayton 
3605b9c1b51eSKate Stone       if (variables_window_sp) {
360644d93782SGreg Clayton         const Rect variables_bounds = variables_window_sp->GetBounds();
360744d93782SGreg Clayton 
360844d93782SGreg Clayton         main_window_sp->RemoveSubWindow(variables_window_sp.get());
360944d93782SGreg Clayton 
3610b9c1b51eSKate Stone         if (registers_window_sp) {
3611b9c1b51eSKate Stone           // We have a registers window, so give all the area back to the
3612b9c1b51eSKate Stone           // registers window
361344d93782SGreg Clayton           Rect registers_bounds = variables_bounds;
361444d93782SGreg Clayton           registers_bounds.size.width = source_bounds.size.width;
361544d93782SGreg Clayton           registers_window_sp->SetBounds(registers_bounds);
3616b9c1b51eSKate Stone         } else {
361705097246SAdrian Prantl           // We have no registers window showing so give the bottom area back
361805097246SAdrian Prantl           // to the source view
361944d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3620b9c1b51eSKate Stone                                    source_bounds.size.height +
3621b9c1b51eSKate Stone                                        variables_bounds.size.height);
362244d93782SGreg Clayton         }
3623b9c1b51eSKate Stone       } else {
362444d93782SGreg Clayton         Rect new_variables_rect;
3625b9c1b51eSKate Stone         if (registers_window_sp) {
362644d93782SGreg Clayton           // We have a registers window so split the area of the registers
362744d93782SGreg Clayton           // window into two columns where the left hand side will be the
362844d93782SGreg Clayton           // variables and the right hand side will be the registers
362944d93782SGreg Clayton           const Rect variables_bounds = registers_window_sp->GetBounds();
363044d93782SGreg Clayton           Rect new_registers_rect;
3631b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_variables_rect,
3632b9c1b51eSKate Stone                                                    new_registers_rect);
363344d93782SGreg Clayton           registers_window_sp->SetBounds(new_registers_rect);
3634b9c1b51eSKate Stone         } else {
363544d93782SGreg Clayton           // No variables window, grab the bottom part of the source window
363644d93782SGreg Clayton           Rect new_source_rect;
3637b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3638b9c1b51eSKate Stone                                                   new_variables_rect);
363944d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
364044d93782SGreg Clayton         }
3641b9c1b51eSKate Stone         WindowSP new_window_sp = main_window_sp->CreateSubWindow(
3642b9c1b51eSKate Stone             "Variables", new_variables_rect, false);
3643b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3644b9c1b51eSKate Stone             WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
364544d93782SGreg Clayton       }
364644d93782SGreg Clayton       touchwin(stdscr);
364744d93782SGreg Clayton     }
364844d93782SGreg Clayton       return MenuActionResult::Handled;
364944d93782SGreg Clayton 
3650b9c1b51eSKate Stone     case eMenuID_ViewRegisters: {
365144d93782SGreg Clayton       WindowSP main_window_sp = m_app.GetMainWindow();
365244d93782SGreg Clayton       WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
365344d93782SGreg Clayton       WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
365444d93782SGreg Clayton       WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
365544d93782SGreg Clayton       const Rect source_bounds = source_window_sp->GetBounds();
365644d93782SGreg Clayton 
3657b9c1b51eSKate Stone       if (registers_window_sp) {
3658b9c1b51eSKate Stone         if (variables_window_sp) {
365944d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
366044d93782SGreg Clayton 
3661b9c1b51eSKate Stone           // We have a variables window, so give all the area back to the
3662b9c1b51eSKate Stone           // variables window
3663b9c1b51eSKate Stone           variables_window_sp->Resize(variables_bounds.size.width +
3664b9c1b51eSKate Stone                                           registers_window_sp->GetWidth(),
366544d93782SGreg Clayton                                       variables_bounds.size.height);
3666b9c1b51eSKate Stone         } else {
366705097246SAdrian Prantl           // We have no variables window showing so give the bottom area back
366805097246SAdrian Prantl           // to the source view
366944d93782SGreg Clayton           source_window_sp->Resize(source_bounds.size.width,
3670b9c1b51eSKate Stone                                    source_bounds.size.height +
3671b9c1b51eSKate Stone                                        registers_window_sp->GetHeight());
367244d93782SGreg Clayton         }
367344d93782SGreg Clayton         main_window_sp->RemoveSubWindow(registers_window_sp.get());
3674b9c1b51eSKate Stone       } else {
367544d93782SGreg Clayton         Rect new_regs_rect;
3676b9c1b51eSKate Stone         if (variables_window_sp) {
367705097246SAdrian Prantl           // We have a variables window, split it into two columns where the
367805097246SAdrian Prantl           // left hand side will be the variables and the right hand side will
367905097246SAdrian Prantl           // be the registers
368044d93782SGreg Clayton           const Rect variables_bounds = variables_window_sp->GetBounds();
368144d93782SGreg Clayton           Rect new_vars_rect;
3682b9c1b51eSKate Stone           variables_bounds.VerticalSplitPercentage(0.50, new_vars_rect,
3683b9c1b51eSKate Stone                                                    new_regs_rect);
368444d93782SGreg Clayton           variables_window_sp->SetBounds(new_vars_rect);
3685b9c1b51eSKate Stone         } else {
368644d93782SGreg Clayton           // No registers window, grab the bottom part of the source window
368744d93782SGreg Clayton           Rect new_source_rect;
3688b9c1b51eSKate Stone           source_bounds.HorizontalSplitPercentage(0.70, new_source_rect,
3689b9c1b51eSKate Stone                                                   new_regs_rect);
369044d93782SGreg Clayton           source_window_sp->SetBounds(new_source_rect);
369144d93782SGreg Clayton         }
3692b9c1b51eSKate Stone         WindowSP new_window_sp =
3693b9c1b51eSKate Stone             main_window_sp->CreateSubWindow("Registers", new_regs_rect, false);
3694b9c1b51eSKate Stone         new_window_sp->SetDelegate(
3695b9c1b51eSKate Stone             WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
369644d93782SGreg Clayton       }
369744d93782SGreg Clayton       touchwin(stdscr);
369844d93782SGreg Clayton     }
369944d93782SGreg Clayton       return MenuActionResult::Handled;
370044d93782SGreg Clayton 
370144d93782SGreg Clayton     case eMenuID_HelpGUIHelp:
37025fdb09bbSGreg Clayton       m_app.GetMainWindow()->CreateHelpSubwindow();
370344d93782SGreg Clayton       return MenuActionResult::Handled;
370444d93782SGreg Clayton 
370544d93782SGreg Clayton     default:
370644d93782SGreg Clayton       break;
370744d93782SGreg Clayton     }
370844d93782SGreg Clayton 
370944d93782SGreg Clayton     return MenuActionResult::NotHandled;
371044d93782SGreg Clayton   }
3711b9c1b51eSKate Stone 
371244d93782SGreg Clayton protected:
371344d93782SGreg Clayton   Application &m_app;
371444d93782SGreg Clayton   Debugger &m_debugger;
371544d93782SGreg Clayton };
371644d93782SGreg Clayton 
3717b9c1b51eSKate Stone class StatusBarWindowDelegate : public WindowDelegate {
371844d93782SGreg Clayton public:
3719b9c1b51eSKate Stone   StatusBarWindowDelegate(Debugger &debugger) : m_debugger(debugger) {
3720b9c1b51eSKate Stone     FormatEntity::Parse("Thread: ${thread.id%tid}", m_format);
372144d93782SGreg Clayton   }
372244d93782SGreg Clayton 
3723315b6884SEugene Zelenko   ~StatusBarWindowDelegate() override = default;
3724bd5ae6b4SGreg Clayton 
3725b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3726b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3727b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
372844d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
372944d93782SGreg Clayton     Thread *thread = exe_ctx.GetThreadPtr();
373044d93782SGreg Clayton     StackFrame *frame = exe_ctx.GetFramePtr();
373144d93782SGreg Clayton     window.Erase();
373244d93782SGreg Clayton     window.SetBackground(2);
373344d93782SGreg Clayton     window.MoveCursor(0, 0);
3734b9c1b51eSKate Stone     if (process) {
373544d93782SGreg Clayton       const StateType state = process->GetState();
3736b9c1b51eSKate Stone       window.Printf("Process: %5" PRIu64 " %10s", process->GetID(),
3737b9c1b51eSKate Stone                     StateAsCString(state));
373844d93782SGreg Clayton 
3739b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
37405b031ebcSEd Maste         StreamString strm;
3741b9c1b51eSKate Stone         if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx,
3742b9c1b51eSKate Stone                                            nullptr, nullptr, false, false)) {
374344d93782SGreg Clayton           window.MoveCursor(40, 0);
3744c156427dSZachary Turner           window.PutCStringTruncated(strm.GetString().str().c_str(), 1);
37455b031ebcSEd Maste         }
374644d93782SGreg Clayton 
374744d93782SGreg Clayton         window.MoveCursor(60, 0);
374844d93782SGreg Clayton         if (frame)
3749b9c1b51eSKate Stone           window.Printf("Frame: %3u  PC = 0x%16.16" PRIx64,
3750b9c1b51eSKate Stone                         frame->GetFrameIndex(),
3751b9c1b51eSKate Stone                         frame->GetFrameCodeAddress().GetOpcodeLoadAddress(
3752b9c1b51eSKate Stone                             exe_ctx.GetTargetPtr()));
3753b9c1b51eSKate Stone       } else if (state == eStateExited) {
375444d93782SGreg Clayton         const char *exit_desc = process->GetExitDescription();
375544d93782SGreg Clayton         const int exit_status = process->GetExitStatus();
375644d93782SGreg Clayton         if (exit_desc && exit_desc[0])
375744d93782SGreg Clayton           window.Printf(" with status = %i (%s)", exit_status, exit_desc);
375844d93782SGreg Clayton         else
375944d93782SGreg Clayton           window.Printf(" with status = %i", exit_status);
376044d93782SGreg Clayton       }
376144d93782SGreg Clayton     }
376244d93782SGreg Clayton     return true;
376344d93782SGreg Clayton   }
376444d93782SGreg Clayton 
376544d93782SGreg Clayton protected:
376644d93782SGreg Clayton   Debugger &m_debugger;
3767554f68d3SGreg Clayton   FormatEntity::Entry m_format;
376844d93782SGreg Clayton };
376944d93782SGreg Clayton 
3770b9c1b51eSKate Stone class SourceFileWindowDelegate : public WindowDelegate {
377144d93782SGreg Clayton public:
3772b9c1b51eSKate Stone   SourceFileWindowDelegate(Debugger &debugger)
3773b9c1b51eSKate Stone       : WindowDelegate(), m_debugger(debugger), m_sc(), m_file_sp(),
3774b9c1b51eSKate Stone         m_disassembly_scope(nullptr), m_disassembly_sp(), m_disassembly_range(),
3775b9c1b51eSKate Stone         m_title(), m_line_width(4), m_selected_line(0), m_pc_line(0),
3776b9c1b51eSKate Stone         m_stop_id(0), m_frame_idx(UINT32_MAX), m_first_visible_line(0),
3777b9c1b51eSKate Stone         m_min_x(0), m_min_y(0), m_max_x(0), m_max_y(0) {}
377844d93782SGreg Clayton 
3779315b6884SEugene Zelenko   ~SourceFileWindowDelegate() override = default;
378044d93782SGreg Clayton 
3781b9c1b51eSKate Stone   void Update(const SymbolContext &sc) { m_sc = sc; }
378244d93782SGreg Clayton 
3783b9c1b51eSKate Stone   uint32_t NumVisibleLines() const { return m_max_y - m_min_y; }
378444d93782SGreg Clayton 
3785b9c1b51eSKate Stone   const char *WindowDelegateGetHelpText() override {
378644d93782SGreg Clayton     return "Source/Disassembly window keyboard shortcuts:";
378744d93782SGreg Clayton   }
378844d93782SGreg Clayton 
3789b9c1b51eSKate Stone   KeyHelp *WindowDelegateGetKeyHelp() override {
379044d93782SGreg Clayton     static curses::KeyHelp g_source_view_key_help[] = {
379144d93782SGreg Clayton         {KEY_RETURN, "Run to selected line with one shot breakpoint"},
379244d93782SGreg Clayton         {KEY_UP, "Select previous source line"},
379344d93782SGreg Clayton         {KEY_DOWN, "Select next source line"},
379444d93782SGreg Clayton         {KEY_PPAGE, "Page up"},
379544d93782SGreg Clayton         {KEY_NPAGE, "Page down"},
379644d93782SGreg Clayton         {'b', "Set breakpoint on selected source/disassembly line"},
379744d93782SGreg Clayton         {'c', "Continue process"},
379844d93782SGreg Clayton         {'d', "Detach and resume process"},
379944d93782SGreg Clayton         {'D', "Detach with process suspended"},
380044d93782SGreg Clayton         {'h', "Show help dialog"},
380144d93782SGreg Clayton         {'k', "Kill process"},
380244d93782SGreg Clayton         {'n', "Step over (source line)"},
380344d93782SGreg Clayton         {'N', "Step over (single instruction)"},
380444d93782SGreg Clayton         {'o', "Step out"},
380544d93782SGreg Clayton         {'s', "Step in (source line)"},
380644d93782SGreg Clayton         {'S', "Step in (single instruction)"},
380744d93782SGreg Clayton         {',', "Page up"},
380844d93782SGreg Clayton         {'.', "Page down"},
3809b9c1b51eSKate Stone         {'\0', nullptr}};
381044d93782SGreg Clayton     return g_source_view_key_help;
381144d93782SGreg Clayton   }
381244d93782SGreg Clayton 
3813b9c1b51eSKate Stone   bool WindowDelegateDraw(Window &window, bool force) override {
3814b9c1b51eSKate Stone     ExecutionContext exe_ctx =
3815b9c1b51eSKate Stone         m_debugger.GetCommandInterpreter().GetExecutionContext();
381644d93782SGreg Clayton     Process *process = exe_ctx.GetProcessPtr();
3817c5dac77aSEugene Zelenko     Thread *thread = nullptr;
381844d93782SGreg Clayton 
381944d93782SGreg Clayton     bool update_location = false;
3820b9c1b51eSKate Stone     if (process) {
382144d93782SGreg Clayton       StateType state = process->GetState();
3822b9c1b51eSKate Stone       if (StateIsStoppedState(state, true)) {
382344d93782SGreg Clayton         // We are stopped, so it is ok to
382444d93782SGreg Clayton         update_location = true;
382544d93782SGreg Clayton       }
382644d93782SGreg Clayton     }
382744d93782SGreg Clayton 
382844d93782SGreg Clayton     m_min_x = 1;
3829ec990867SGreg Clayton     m_min_y = 2;
383044d93782SGreg Clayton     m_max_x = window.GetMaxX() - 1;
383144d93782SGreg Clayton     m_max_y = window.GetMaxY() - 1;
383244d93782SGreg Clayton 
383344d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
383444d93782SGreg Clayton     StackFrameSP frame_sp;
383544d93782SGreg Clayton     bool set_selected_line_to_pc = false;
383644d93782SGreg Clayton 
3837b9c1b51eSKate Stone     if (update_location) {
383844d93782SGreg Clayton       const bool process_alive = process ? process->IsAlive() : false;
383944d93782SGreg Clayton       bool thread_changed = false;
3840b9c1b51eSKate Stone       if (process_alive) {
384144d93782SGreg Clayton         thread = exe_ctx.GetThreadPtr();
3842b9c1b51eSKate Stone         if (thread) {
384344d93782SGreg Clayton           frame_sp = thread->GetSelectedFrame();
384444d93782SGreg Clayton           auto tid = thread->GetID();
384544d93782SGreg Clayton           thread_changed = tid != m_tid;
384644d93782SGreg Clayton           m_tid = tid;
3847b9c1b51eSKate Stone         } else {
3848b9c1b51eSKate Stone           if (m_tid != LLDB_INVALID_THREAD_ID) {
384944d93782SGreg Clayton             thread_changed = true;
385044d93782SGreg Clayton             m_tid = LLDB_INVALID_THREAD_ID;
385144d93782SGreg Clayton           }
385244d93782SGreg Clayton         }
385344d93782SGreg Clayton       }
385444d93782SGreg Clayton       const uint32_t stop_id = process ? process->GetStopID() : 0;
385544d93782SGreg Clayton       const bool stop_id_changed = stop_id != m_stop_id;
385644d93782SGreg Clayton       bool frame_changed = false;
385744d93782SGreg Clayton       m_stop_id = stop_id;
3858ec990867SGreg Clayton       m_title.Clear();
3859b9c1b51eSKate Stone       if (frame_sp) {
386044d93782SGreg Clayton         m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
3861b9c1b51eSKate Stone         if (m_sc.module_sp) {
3862b9c1b51eSKate Stone           m_title.Printf(
3863b9c1b51eSKate Stone               "%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
3864ec990867SGreg Clayton           ConstString func_name = m_sc.GetFunctionName();
3865ec990867SGreg Clayton           if (func_name)
3866ec990867SGreg Clayton             m_title.Printf("`%s", func_name.GetCString());
3867ec990867SGreg Clayton         }
386844d93782SGreg Clayton         const uint32_t frame_idx = frame_sp->GetFrameIndex();
386944d93782SGreg Clayton         frame_changed = frame_idx != m_frame_idx;
387044d93782SGreg Clayton         m_frame_idx = frame_idx;
3871b9c1b51eSKate Stone       } else {
387244d93782SGreg Clayton         m_sc.Clear(true);
387344d93782SGreg Clayton         frame_changed = m_frame_idx != UINT32_MAX;
387444d93782SGreg Clayton         m_frame_idx = UINT32_MAX;
387544d93782SGreg Clayton       }
387644d93782SGreg Clayton 
3877b9c1b51eSKate Stone       const bool context_changed =
3878b9c1b51eSKate Stone           thread_changed || frame_changed || stop_id_changed;
387944d93782SGreg Clayton 
3880b9c1b51eSKate Stone       if (process_alive) {
3881b9c1b51eSKate Stone         if (m_sc.line_entry.IsValid()) {
388244d93782SGreg Clayton           m_pc_line = m_sc.line_entry.line;
388344d93782SGreg Clayton           if (m_pc_line != UINT32_MAX)
388444d93782SGreg Clayton             --m_pc_line; // Convert to zero based line number...
388544d93782SGreg Clayton           // Update the selected line if the stop ID changed...
388644d93782SGreg Clayton           if (context_changed)
388744d93782SGreg Clayton             m_selected_line = m_pc_line;
388844d93782SGreg Clayton 
3889b9c1b51eSKate Stone           if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) {
389005097246SAdrian Prantl             // Same file, nothing to do, we should either have the lines or not
389105097246SAdrian Prantl             // (source file missing)
3892b9c1b51eSKate Stone             if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) {
389344d93782SGreg Clayton               if (m_selected_line >= m_first_visible_line + num_visible_lines)
389444d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
3895b9c1b51eSKate Stone             } else {
389644d93782SGreg Clayton               if (m_selected_line > 10)
389744d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
389844d93782SGreg Clayton               else
389944d93782SGreg Clayton                 m_first_visible_line = 0;
390044d93782SGreg Clayton             }
3901b9c1b51eSKate Stone           } else {
390244d93782SGreg Clayton             // File changed, set selected line to the line with the PC
390344d93782SGreg Clayton             m_selected_line = m_pc_line;
3904b9c1b51eSKate Stone             m_file_sp =
3905b9c1b51eSKate Stone                 m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
3906b9c1b51eSKate Stone             if (m_file_sp) {
390744d93782SGreg Clayton               const size_t num_lines = m_file_sp->GetNumLines();
3908eebf32faSPavel Labath               m_line_width = 1;
390944d93782SGreg Clayton               for (size_t n = num_lines; n >= 10; n = n / 10)
391044d93782SGreg Clayton                 ++m_line_width;
391144d93782SGreg Clayton 
3912b9c1b51eSKate Stone               if (num_lines < num_visible_lines ||
3913b9c1b51eSKate Stone                   m_selected_line < num_visible_lines)
391444d93782SGreg Clayton                 m_first_visible_line = 0;
391544d93782SGreg Clayton               else
391644d93782SGreg Clayton                 m_first_visible_line = m_selected_line - 10;
391744d93782SGreg Clayton             }
391844d93782SGreg Clayton           }
3919b9c1b51eSKate Stone         } else {
392044d93782SGreg Clayton           m_file_sp.reset();
392144d93782SGreg Clayton         }
392244d93782SGreg Clayton 
3923b9c1b51eSKate Stone         if (!m_file_sp || m_file_sp->GetNumLines() == 0) {
392444d93782SGreg Clayton           // Show disassembly
392544d93782SGreg Clayton           bool prefer_file_cache = false;
3926b9c1b51eSKate Stone           if (m_sc.function) {
3927b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.function) {
392844d93782SGreg Clayton               m_disassembly_scope = m_sc.function;
3929b9c1b51eSKate Stone               m_disassembly_sp = m_sc.function->GetInstructions(
3930b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3931b9c1b51eSKate Stone               if (m_disassembly_sp) {
393244d93782SGreg Clayton                 set_selected_line_to_pc = true;
393344d93782SGreg Clayton                 m_disassembly_range = m_sc.function->GetAddressRange();
3934b9c1b51eSKate Stone               } else {
393544d93782SGreg Clayton                 m_disassembly_range.Clear();
393644d93782SGreg Clayton               }
3937b9c1b51eSKate Stone             } else {
393844d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
393944d93782SGreg Clayton             }
3940b9c1b51eSKate Stone           } else if (m_sc.symbol) {
3941b9c1b51eSKate Stone             if (m_disassembly_scope != m_sc.symbol) {
394244d93782SGreg Clayton               m_disassembly_scope = m_sc.symbol;
3943b9c1b51eSKate Stone               m_disassembly_sp = m_sc.symbol->GetInstructions(
3944b9c1b51eSKate Stone                   exe_ctx, nullptr, prefer_file_cache);
3945b9c1b51eSKate Stone               if (m_disassembly_sp) {
394644d93782SGreg Clayton                 set_selected_line_to_pc = true;
3947b9c1b51eSKate Stone                 m_disassembly_range.GetBaseAddress() =
3948b9c1b51eSKate Stone                     m_sc.symbol->GetAddress();
394944d93782SGreg Clayton                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
3950b9c1b51eSKate Stone               } else {
395144d93782SGreg Clayton                 m_disassembly_range.Clear();
395244d93782SGreg Clayton               }
3953b9c1b51eSKate Stone             } else {
395444d93782SGreg Clayton               set_selected_line_to_pc = context_changed;
395544d93782SGreg Clayton             }
395644d93782SGreg Clayton           }
395744d93782SGreg Clayton         }
3958b9c1b51eSKate Stone       } else {
395944d93782SGreg Clayton         m_pc_line = UINT32_MAX;
396044d93782SGreg Clayton       }
396144d93782SGreg Clayton     }
396244d93782SGreg Clayton 
3963ec990867SGreg Clayton     const int window_width = window.GetWidth();
396444d93782SGreg Clayton     window.Erase();
396544d93782SGreg Clayton     window.DrawTitleBox("Sources");
3966b9c1b51eSKate Stone     if (!m_title.GetString().empty()) {
3967ec990867SGreg Clayton       window.AttributeOn(A_REVERSE);
3968ec990867SGreg Clayton       window.MoveCursor(1, 1);
3969ec990867SGreg Clayton       window.PutChar(' ');
3970c156427dSZachary Turner       window.PutCStringTruncated(m_title.GetString().str().c_str(), 1);
3971ec990867SGreg Clayton       int x = window.GetCursorX();
3972b9c1b51eSKate Stone       if (x < window_width - 1) {
3973ec990867SGreg Clayton         window.Printf("%*s", window_width - x - 1, "");
3974ec990867SGreg Clayton       }
3975ec990867SGreg Clayton       window.AttributeOff(A_REVERSE);
3976ec990867SGreg Clayton     }
397744d93782SGreg Clayton 
397844d93782SGreg Clayton     Target *target = exe_ctx.GetTargetPtr();
397944d93782SGreg Clayton     const size_t num_source_lines = GetNumSourceLines();
3980b9c1b51eSKate Stone     if (num_source_lines > 0) {
398144d93782SGreg Clayton       // Display source
398244d93782SGreg Clayton       BreakpointLines bp_lines;
3983b9c1b51eSKate Stone       if (target) {
398444d93782SGreg Clayton         BreakpointList &bp_list = target->GetBreakpointList();
398544d93782SGreg Clayton         const size_t num_bps = bp_list.GetSize();
3986b9c1b51eSKate Stone         for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
398744d93782SGreg Clayton           BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
398844d93782SGreg Clayton           const size_t num_bps_locs = bp_sp->GetNumLocations();
3989b9c1b51eSKate Stone           for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs; ++bp_loc_idx) {
3990b9c1b51eSKate Stone             BreakpointLocationSP bp_loc_sp =
3991b9c1b51eSKate Stone                 bp_sp->GetLocationAtIndex(bp_loc_idx);
399244d93782SGreg Clayton             LineEntry bp_loc_line_entry;
3993b9c1b51eSKate Stone             if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry(
3994b9c1b51eSKate Stone                     bp_loc_line_entry)) {
3995b9c1b51eSKate Stone               if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) {
399644d93782SGreg Clayton                 bp_lines.insert(bp_loc_line_entry.line);
399744d93782SGreg Clayton               }
399844d93782SGreg Clayton             }
399944d93782SGreg Clayton           }
400044d93782SGreg Clayton         }
400144d93782SGreg Clayton       }
400244d93782SGreg Clayton 
400344d93782SGreg Clayton       const attr_t selected_highlight_attr = A_REVERSE;
400444d93782SGreg Clayton       const attr_t pc_highlight_attr = COLOR_PAIR(1);
400544d93782SGreg Clayton 
4006b9c1b51eSKate Stone       for (size_t i = 0; i < num_visible_lines; ++i) {
400744d93782SGreg Clayton         const uint32_t curr_line = m_first_visible_line + i;
4008b9c1b51eSKate Stone         if (curr_line < num_source_lines) {
4009ec990867SGreg Clayton           const int line_y = m_min_y + i;
401044d93782SGreg Clayton           window.MoveCursor(1, line_y);
401144d93782SGreg Clayton           const bool is_pc_line = curr_line == m_pc_line;
401244d93782SGreg Clayton           const bool line_is_selected = m_selected_line == curr_line;
401344d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
401444d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
401544d93782SGreg Clayton           attr_t highlight_attr = 0;
401644d93782SGreg Clayton           attr_t bp_attr = 0;
401744d93782SGreg Clayton           if (is_pc_line)
401844d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
401944d93782SGreg Clayton           else if (line_is_selected)
402044d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
402144d93782SGreg Clayton 
402244d93782SGreg Clayton           if (bp_lines.find(curr_line + 1) != bp_lines.end())
402344d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
402444d93782SGreg Clayton 
402544d93782SGreg Clayton           if (bp_attr)
402644d93782SGreg Clayton             window.AttributeOn(bp_attr);
402744d93782SGreg Clayton 
4028eebf32faSPavel Labath           window.Printf(" %*u ", m_line_width, curr_line + 1);
402944d93782SGreg Clayton 
403044d93782SGreg Clayton           if (bp_attr)
403144d93782SGreg Clayton             window.AttributeOff(bp_attr);
403244d93782SGreg Clayton 
403344d93782SGreg Clayton           window.PutChar(ACS_VLINE);
403444d93782SGreg Clayton           // Mark the line with the PC with a diamond
403544d93782SGreg Clayton           if (is_pc_line)
403644d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
403744d93782SGreg Clayton           else
403844d93782SGreg Clayton             window.PutChar(' ');
403944d93782SGreg Clayton 
404044d93782SGreg Clayton           if (highlight_attr)
404144d93782SGreg Clayton             window.AttributeOn(highlight_attr);
4042b9c1b51eSKate Stone           const uint32_t line_len =
4043b9c1b51eSKate Stone               m_file_sp->GetLineLength(curr_line + 1, false);
404444d93782SGreg Clayton           if (line_len > 0)
404544d93782SGreg Clayton             window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
404644d93782SGreg Clayton 
4047b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4048b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
404944d93782SGreg Clayton             StopInfoSP stop_info_sp;
405044d93782SGreg Clayton             if (thread)
405144d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4052b9c1b51eSKate Stone             if (stop_info_sp) {
405344d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4054b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
405544d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4056ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
405744d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4058b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4059b9c1b51eSKate Stone                 // line_y);
4060b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4061b9c1b51eSKate Stone                               stop_description);
406244d93782SGreg Clayton               }
4063b9c1b51eSKate Stone             } else {
4064ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
406544d93782SGreg Clayton             }
406644d93782SGreg Clayton           }
406744d93782SGreg Clayton           if (highlight_attr)
406844d93782SGreg Clayton             window.AttributeOff(highlight_attr);
4069b9c1b51eSKate Stone         } else {
407044d93782SGreg Clayton           break;
407144d93782SGreg Clayton         }
407244d93782SGreg Clayton       }
4073b9c1b51eSKate Stone     } else {
407444d93782SGreg Clayton       size_t num_disassembly_lines = GetNumDisassemblyLines();
4075b9c1b51eSKate Stone       if (num_disassembly_lines > 0) {
407644d93782SGreg Clayton         // Display disassembly
407744d93782SGreg Clayton         BreakpointAddrs bp_file_addrs;
407844d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
4079b9c1b51eSKate Stone         if (target) {
408044d93782SGreg Clayton           BreakpointList &bp_list = target->GetBreakpointList();
408144d93782SGreg Clayton           const size_t num_bps = bp_list.GetSize();
4082b9c1b51eSKate Stone           for (size_t bp_idx = 0; bp_idx < num_bps; ++bp_idx) {
408344d93782SGreg Clayton             BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
408444d93782SGreg Clayton             const size_t num_bps_locs = bp_sp->GetNumLocations();
4085b9c1b51eSKate Stone             for (size_t bp_loc_idx = 0; bp_loc_idx < num_bps_locs;
4086b9c1b51eSKate Stone                  ++bp_loc_idx) {
4087b9c1b51eSKate Stone               BreakpointLocationSP bp_loc_sp =
4088b9c1b51eSKate Stone                   bp_sp->GetLocationAtIndex(bp_loc_idx);
408944d93782SGreg Clayton               LineEntry bp_loc_line_entry;
4090b9c1b51eSKate Stone               const lldb::addr_t file_addr =
4091b9c1b51eSKate Stone                   bp_loc_sp->GetAddress().GetFileAddress();
4092b9c1b51eSKate Stone               if (file_addr != LLDB_INVALID_ADDRESS) {
409344d93782SGreg Clayton                 if (m_disassembly_range.ContainsFileAddress(file_addr))
409444d93782SGreg Clayton                   bp_file_addrs.insert(file_addr);
409544d93782SGreg Clayton               }
409644d93782SGreg Clayton             }
409744d93782SGreg Clayton           }
409844d93782SGreg Clayton         }
409944d93782SGreg Clayton 
410044d93782SGreg Clayton         const attr_t selected_highlight_attr = A_REVERSE;
410144d93782SGreg Clayton         const attr_t pc_highlight_attr = COLOR_PAIR(1);
410244d93782SGreg Clayton 
410344d93782SGreg Clayton         StreamString strm;
410444d93782SGreg Clayton 
410544d93782SGreg Clayton         InstructionList &insts = m_disassembly_sp->GetInstructionList();
410644d93782SGreg Clayton         Address pc_address;
410744d93782SGreg Clayton 
410844d93782SGreg Clayton         if (frame_sp)
410944d93782SGreg Clayton           pc_address = frame_sp->GetFrameCodeAddress();
4110b9c1b51eSKate Stone         const uint32_t pc_idx =
4111b9c1b51eSKate Stone             pc_address.IsValid()
4112b9c1b51eSKate Stone                 ? insts.GetIndexOfInstructionAtAddress(pc_address)
4113b9c1b51eSKate Stone                 : UINT32_MAX;
4114b9c1b51eSKate Stone         if (set_selected_line_to_pc) {
411544d93782SGreg Clayton           m_selected_line = pc_idx;
411644d93782SGreg Clayton         }
411744d93782SGreg Clayton 
411844d93782SGreg Clayton         const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
41193985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
412044d93782SGreg Clayton           m_first_visible_line = 0;
412144d93782SGreg Clayton 
4122b9c1b51eSKate Stone         if (pc_idx < num_disassembly_lines) {
41233985c8c6SSaleem Abdulrasool           if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
412444d93782SGreg Clayton               pc_idx >= m_first_visible_line + num_visible_lines)
412544d93782SGreg Clayton             m_first_visible_line = pc_idx - non_visible_pc_offset;
412644d93782SGreg Clayton         }
412744d93782SGreg Clayton 
4128b9c1b51eSKate Stone         for (size_t i = 0; i < num_visible_lines; ++i) {
412944d93782SGreg Clayton           const uint32_t inst_idx = m_first_visible_line + i;
413044d93782SGreg Clayton           Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
413144d93782SGreg Clayton           if (!inst)
413244d93782SGreg Clayton             break;
413344d93782SGreg Clayton 
4134ec990867SGreg Clayton           const int line_y = m_min_y + i;
4135ec990867SGreg Clayton           window.MoveCursor(1, line_y);
413644d93782SGreg Clayton           const bool is_pc_line = frame_sp && inst_idx == pc_idx;
413744d93782SGreg Clayton           const bool line_is_selected = m_selected_line == inst_idx;
413844d93782SGreg Clayton           // Highlight the line as the PC line first, then if the selected line
413944d93782SGreg Clayton           // isn't the same as the PC line, highlight it differently
414044d93782SGreg Clayton           attr_t highlight_attr = 0;
414144d93782SGreg Clayton           attr_t bp_attr = 0;
414244d93782SGreg Clayton           if (is_pc_line)
414344d93782SGreg Clayton             highlight_attr = pc_highlight_attr;
414444d93782SGreg Clayton           else if (line_is_selected)
414544d93782SGreg Clayton             highlight_attr = selected_highlight_attr;
414644d93782SGreg Clayton 
4147b9c1b51eSKate Stone           if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) !=
4148b9c1b51eSKate Stone               bp_file_addrs.end())
414944d93782SGreg Clayton             bp_attr = COLOR_PAIR(2);
415044d93782SGreg Clayton 
415144d93782SGreg Clayton           if (bp_attr)
415244d93782SGreg Clayton             window.AttributeOn(bp_attr);
415344d93782SGreg Clayton 
4154324a1036SSaleem Abdulrasool           window.Printf(" 0x%16.16llx ",
4155b9c1b51eSKate Stone                         static_cast<unsigned long long>(
4156b9c1b51eSKate Stone                             inst->GetAddress().GetLoadAddress(target)));
415744d93782SGreg Clayton 
415844d93782SGreg Clayton           if (bp_attr)
415944d93782SGreg Clayton             window.AttributeOff(bp_attr);
416044d93782SGreg Clayton 
416144d93782SGreg Clayton           window.PutChar(ACS_VLINE);
416244d93782SGreg Clayton           // Mark the line with the PC with a diamond
416344d93782SGreg Clayton           if (is_pc_line)
416444d93782SGreg Clayton             window.PutChar(ACS_DIAMOND);
416544d93782SGreg Clayton           else
416644d93782SGreg Clayton             window.PutChar(' ');
416744d93782SGreg Clayton 
416844d93782SGreg Clayton           if (highlight_attr)
416944d93782SGreg Clayton             window.AttributeOn(highlight_attr);
417044d93782SGreg Clayton 
417144d93782SGreg Clayton           const char *mnemonic = inst->GetMnemonic(&exe_ctx);
417244d93782SGreg Clayton           const char *operands = inst->GetOperands(&exe_ctx);
417344d93782SGreg Clayton           const char *comment = inst->GetComment(&exe_ctx);
417444d93782SGreg Clayton 
4175c5dac77aSEugene Zelenko           if (mnemonic != nullptr && mnemonic[0] == '\0')
4176c5dac77aSEugene Zelenko             mnemonic = nullptr;
4177c5dac77aSEugene Zelenko           if (operands != nullptr && operands[0] == '\0')
4178c5dac77aSEugene Zelenko             operands = nullptr;
4179c5dac77aSEugene Zelenko           if (comment != nullptr && comment[0] == '\0')
4180c5dac77aSEugene Zelenko             comment = nullptr;
418144d93782SGreg Clayton 
418244d93782SGreg Clayton           strm.Clear();
418344d93782SGreg Clayton 
4184c5dac77aSEugene Zelenko           if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
418544d93782SGreg Clayton             strm.Printf("%-8s %-25s ; %s", mnemonic, operands, comment);
4186c5dac77aSEugene Zelenko           else if (mnemonic != nullptr && operands != nullptr)
418744d93782SGreg Clayton             strm.Printf("%-8s %s", mnemonic, operands);
4188c5dac77aSEugene Zelenko           else if (mnemonic != nullptr)
418944d93782SGreg Clayton             strm.Printf("%s", mnemonic);
419044d93782SGreg Clayton 
419144d93782SGreg Clayton           int right_pad = 1;
4192c156427dSZachary Turner           window.PutCStringTruncated(strm.GetData(), right_pad);
419344d93782SGreg Clayton 
4194b9c1b51eSKate Stone           if (is_pc_line && frame_sp &&
4195b9c1b51eSKate Stone               frame_sp->GetConcreteFrameIndex() == 0) {
419644d93782SGreg Clayton             StopInfoSP stop_info_sp;
419744d93782SGreg Clayton             if (thread)
419844d93782SGreg Clayton               stop_info_sp = thread->GetStopInfo();
4199b9c1b51eSKate Stone             if (stop_info_sp) {
420044d93782SGreg Clayton               const char *stop_description = stop_info_sp->GetDescription();
4201b9c1b51eSKate Stone               if (stop_description && stop_description[0]) {
420244d93782SGreg Clayton                 size_t stop_description_len = strlen(stop_description);
4203ec990867SGreg Clayton                 int desc_x = window_width - stop_description_len - 16;
420444d93782SGreg Clayton                 window.Printf("%*s", desc_x - window.GetCursorX(), "");
4205b9c1b51eSKate Stone                 // window.MoveCursor(window_width - stop_description_len - 15,
4206b9c1b51eSKate Stone                 // line_y);
4207b9c1b51eSKate Stone                 window.Printf("<<< Thread %u: %s ", thread->GetIndexID(),
4208b9c1b51eSKate Stone                               stop_description);
420944d93782SGreg Clayton               }
4210b9c1b51eSKate Stone             } else {
4211ec990867SGreg Clayton               window.Printf("%*s", window_width - window.GetCursorX() - 1, "");
421244d93782SGreg Clayton             }
421344d93782SGreg Clayton           }
421444d93782SGreg Clayton           if (highlight_attr)
421544d93782SGreg Clayton             window.AttributeOff(highlight_attr);
421644d93782SGreg Clayton         }
421744d93782SGreg Clayton       }
421844d93782SGreg Clayton     }
421944d93782SGreg Clayton     return true; // Drawing handled
422044d93782SGreg Clayton   }
422144d93782SGreg Clayton 
4222b9c1b51eSKate Stone   size_t GetNumLines() {
422344d93782SGreg Clayton     size_t num_lines = GetNumSourceLines();
422444d93782SGreg Clayton     if (num_lines == 0)
422544d93782SGreg Clayton       num_lines = GetNumDisassemblyLines();
422644d93782SGreg Clayton     return num_lines;
422744d93782SGreg Clayton   }
422844d93782SGreg Clayton 
4229b9c1b51eSKate Stone   size_t GetNumSourceLines() const {
423044d93782SGreg Clayton     if (m_file_sp)
423144d93782SGreg Clayton       return m_file_sp->GetNumLines();
423244d93782SGreg Clayton     return 0;
423344d93782SGreg Clayton   }
4234315b6884SEugene Zelenko 
4235b9c1b51eSKate Stone   size_t GetNumDisassemblyLines() const {
423644d93782SGreg Clayton     if (m_disassembly_sp)
423744d93782SGreg Clayton       return m_disassembly_sp->GetInstructionList().GetSize();
423844d93782SGreg Clayton     return 0;
423944d93782SGreg Clayton   }
424044d93782SGreg Clayton 
4241b9c1b51eSKate Stone   HandleCharResult WindowDelegateHandleChar(Window &window, int c) override {
424244d93782SGreg Clayton     const uint32_t num_visible_lines = NumVisibleLines();
424344d93782SGreg Clayton     const size_t num_lines = GetNumLines();
424444d93782SGreg Clayton 
4245b9c1b51eSKate Stone     switch (c) {
424644d93782SGreg Clayton     case ',':
424744d93782SGreg Clayton     case KEY_PPAGE:
424844d93782SGreg Clayton       // Page up key
42493985c8c6SSaleem Abdulrasool       if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
425044d93782SGreg Clayton         m_first_visible_line -= num_visible_lines;
425144d93782SGreg Clayton       else
425244d93782SGreg Clayton         m_first_visible_line = 0;
425344d93782SGreg Clayton       m_selected_line = m_first_visible_line;
425444d93782SGreg Clayton       return eKeyHandled;
425544d93782SGreg Clayton 
425644d93782SGreg Clayton     case '.':
425744d93782SGreg Clayton     case KEY_NPAGE:
425844d93782SGreg Clayton       // Page down key
425944d93782SGreg Clayton       {
426044d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < num_lines)
426144d93782SGreg Clayton           m_first_visible_line += num_visible_lines;
426244d93782SGreg Clayton         else if (num_lines < num_visible_lines)
426344d93782SGreg Clayton           m_first_visible_line = 0;
426444d93782SGreg Clayton         else
426544d93782SGreg Clayton           m_first_visible_line = num_lines - num_visible_lines;
426644d93782SGreg Clayton         m_selected_line = m_first_visible_line;
426744d93782SGreg Clayton       }
426844d93782SGreg Clayton       return eKeyHandled;
426944d93782SGreg Clayton 
427044d93782SGreg Clayton     case KEY_UP:
4271b9c1b51eSKate Stone       if (m_selected_line > 0) {
427244d93782SGreg Clayton         m_selected_line--;
42733985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
427444d93782SGreg Clayton           m_first_visible_line = m_selected_line;
427544d93782SGreg Clayton       }
427644d93782SGreg Clayton       return eKeyHandled;
427744d93782SGreg Clayton 
427844d93782SGreg Clayton     case KEY_DOWN:
4279b9c1b51eSKate Stone       if (m_selected_line + 1 < num_lines) {
428044d93782SGreg Clayton         m_selected_line++;
428144d93782SGreg Clayton         if (m_first_visible_line + num_visible_lines < m_selected_line)
428244d93782SGreg Clayton           m_first_visible_line++;
428344d93782SGreg Clayton       }
428444d93782SGreg Clayton       return eKeyHandled;
428544d93782SGreg Clayton 
428644d93782SGreg Clayton     case '\r':
428744d93782SGreg Clayton     case '\n':
428844d93782SGreg Clayton     case KEY_ENTER:
428944d93782SGreg Clayton       // Set a breakpoint and run to the line using a one shot breakpoint
4290b9c1b51eSKate Stone       if (GetNumSourceLines() > 0) {
4291b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4292b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4293b9c1b51eSKate Stone         if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) {
4294b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4295b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
429644d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4297b9c1b51eSKate Stone               m_selected_line +
4298b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4299431b1584SAdrian Prantl               0,     // Unspecified column.
43002411167fSJim Ingham               0,     // No offset
430144d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
430244d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
430344d93782SGreg Clayton               false,               // internal
4304055ad9beSIlia K               false,               // request_hardware
4305055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
430644d93782SGreg Clayton           // Make breakpoint one shot
430744d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
430844d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
430944d93782SGreg Clayton         }
4310b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4311b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4312b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4313b9c1b51eSKate Stone                                       .get();
4314b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4315b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4316b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
431744d93782SGreg Clayton           Address addr = inst->GetAddress();
4318b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4319b9c1b51eSKate Stone               addr,   // lldb_private::Address
432044d93782SGreg Clayton               false,  // internal
432144d93782SGreg Clayton               false); // request_hardware
432244d93782SGreg Clayton           // Make breakpoint one shot
432344d93782SGreg Clayton           bp_sp->GetOptions()->SetOneShot(true);
432444d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
432544d93782SGreg Clayton         }
432644d93782SGreg Clayton       }
432744d93782SGreg Clayton       return eKeyHandled;
432844d93782SGreg Clayton 
432944d93782SGreg Clayton     case 'b': // 'b' == toggle breakpoint on currently selected line
4330b9c1b51eSKate Stone       if (m_selected_line < GetNumSourceLines()) {
4331b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4332b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4333b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
4334b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4335b9c1b51eSKate Stone               nullptr, // Don't limit the breakpoint to certain modules
433644d93782SGreg Clayton               m_file_sp->GetFileSpec(), // Source file
4337b9c1b51eSKate Stone               m_selected_line +
4338b9c1b51eSKate Stone                   1, // Source line number (m_selected_line is zero based)
4339431b1584SAdrian Prantl               0,     // No column specified.
43402411167fSJim Ingham               0,     // No offset
434144d93782SGreg Clayton               eLazyBoolCalculate,  // Check inlines using global setting
434244d93782SGreg Clayton               eLazyBoolCalculate,  // Skip prologue using global setting,
434344d93782SGreg Clayton               false,               // internal
4344055ad9beSIlia K               false,               // request_hardware
4345055ad9beSIlia K               eLazyBoolCalculate); // move_to_nearest_code
434644d93782SGreg Clayton         }
4347b9c1b51eSKate Stone       } else if (m_selected_line < GetNumDisassemblyLines()) {
4348b9c1b51eSKate Stone         const Instruction *inst = m_disassembly_sp->GetInstructionList()
4349b9c1b51eSKate Stone                                       .GetInstructionAtIndex(m_selected_line)
4350b9c1b51eSKate Stone                                       .get();
4351b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4352b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4353b9c1b51eSKate Stone         if (exe_ctx.HasTargetScope()) {
435444d93782SGreg Clayton           Address addr = inst->GetAddress();
4355b9c1b51eSKate Stone           BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(
4356b9c1b51eSKate Stone               addr,   // lldb_private::Address
435744d93782SGreg Clayton               false,  // internal
435844d93782SGreg Clayton               false); // request_hardware
435944d93782SGreg Clayton         }
436044d93782SGreg Clayton       }
436144d93782SGreg Clayton       return eKeyHandled;
436244d93782SGreg Clayton 
436344d93782SGreg Clayton     case 'd': // 'd' == detach and let run
436444d93782SGreg Clayton     case 'D': // 'D' == detach and keep stopped
436544d93782SGreg Clayton     {
4366b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4367b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
436844d93782SGreg Clayton       if (exe_ctx.HasProcessScope())
436944d93782SGreg Clayton         exe_ctx.GetProcessRef().Detach(c == 'D');
437044d93782SGreg Clayton     }
437144d93782SGreg Clayton       return eKeyHandled;
437244d93782SGreg Clayton 
437344d93782SGreg Clayton     case 'k':
437444d93782SGreg Clayton       // 'k' == kill
437544d93782SGreg Clayton       {
4376b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4377b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
437844d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
4379ede3193bSJason Molenda           exe_ctx.GetProcessRef().Destroy(false);
438044d93782SGreg Clayton       }
438144d93782SGreg Clayton       return eKeyHandled;
438244d93782SGreg Clayton 
438344d93782SGreg Clayton     case 'c':
438444d93782SGreg Clayton       // 'c' == continue
438544d93782SGreg Clayton       {
4386b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4387b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
438844d93782SGreg Clayton         if (exe_ctx.HasProcessScope())
438944d93782SGreg Clayton           exe_ctx.GetProcessRef().Resume();
439044d93782SGreg Clayton       }
439144d93782SGreg Clayton       return eKeyHandled;
439244d93782SGreg Clayton 
439344d93782SGreg Clayton     case 'o':
439444d93782SGreg Clayton       // 'o' == step out
439544d93782SGreg Clayton       {
4396b9c1b51eSKate Stone         ExecutionContext exe_ctx =
4397b9c1b51eSKate Stone             m_debugger.GetCommandInterpreter().GetExecutionContext();
4398b9c1b51eSKate Stone         if (exe_ctx.HasThreadScope() &&
4399b9c1b51eSKate Stone             StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
440044d93782SGreg Clayton           exe_ctx.GetThreadRef().StepOut();
440144d93782SGreg Clayton         }
440244d93782SGreg Clayton       }
440344d93782SGreg Clayton       return eKeyHandled;
4404315b6884SEugene Zelenko 
440544d93782SGreg Clayton     case 'n': // 'n' == step over
440644d93782SGreg Clayton     case 'N': // 'N' == step over instruction
440744d93782SGreg Clayton     {
4408b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4409b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4410b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4411b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
441244d93782SGreg Clayton         bool source_step = (c == 'n');
441344d93782SGreg Clayton         exe_ctx.GetThreadRef().StepOver(source_step);
441444d93782SGreg Clayton       }
441544d93782SGreg Clayton     }
441644d93782SGreg Clayton       return eKeyHandled;
4417315b6884SEugene Zelenko 
441844d93782SGreg Clayton     case 's': // 's' == step into
441944d93782SGreg Clayton     case 'S': // 'S' == step into instruction
442044d93782SGreg Clayton     {
4421b9c1b51eSKate Stone       ExecutionContext exe_ctx =
4422b9c1b51eSKate Stone           m_debugger.GetCommandInterpreter().GetExecutionContext();
4423b9c1b51eSKate Stone       if (exe_ctx.HasThreadScope() &&
4424b9c1b51eSKate Stone           StateIsStoppedState(exe_ctx.GetProcessRef().GetState(), true)) {
442544d93782SGreg Clayton         bool source_step = (c == 's');
44264b4b2478SJim Ingham         exe_ctx.GetThreadRef().StepIn(source_step);
442744d93782SGreg Clayton       }
442844d93782SGreg Clayton     }
442944d93782SGreg Clayton       return eKeyHandled;
443044d93782SGreg Clayton 
443144d93782SGreg Clayton     case 'h':
443244d93782SGreg Clayton       window.CreateHelpSubwindow();
443344d93782SGreg Clayton       return eKeyHandled;
443444d93782SGreg Clayton 
443544d93782SGreg Clayton     default:
443644d93782SGreg Clayton       break;
443744d93782SGreg Clayton     }
443844d93782SGreg Clayton     return eKeyNotHandled;
443944d93782SGreg Clayton   }
444044d93782SGreg Clayton 
444144d93782SGreg Clayton protected:
444244d93782SGreg Clayton   typedef std::set<uint32_t> BreakpointLines;
444344d93782SGreg Clayton   typedef std::set<lldb::addr_t> BreakpointAddrs;
444444d93782SGreg Clayton 
444544d93782SGreg Clayton   Debugger &m_debugger;
444644d93782SGreg Clayton   SymbolContext m_sc;
444744d93782SGreg Clayton   SourceManager::FileSP m_file_sp;
444844d93782SGreg Clayton   SymbolContextScope *m_disassembly_scope;
444944d93782SGreg Clayton   lldb::DisassemblerSP m_disassembly_sp;
445044d93782SGreg Clayton   AddressRange m_disassembly_range;
4451ec990867SGreg Clayton   StreamString m_title;
445244d93782SGreg Clayton   lldb::user_id_t m_tid;
445344d93782SGreg Clayton   int m_line_width;
445444d93782SGreg Clayton   uint32_t m_selected_line; // The selected line
445544d93782SGreg Clayton   uint32_t m_pc_line;       // The line with the PC
445644d93782SGreg Clayton   uint32_t m_stop_id;
445744d93782SGreg Clayton   uint32_t m_frame_idx;
445844d93782SGreg Clayton   int m_first_visible_line;
445944d93782SGreg Clayton   int m_min_x;
446044d93782SGreg Clayton   int m_min_y;
446144d93782SGreg Clayton   int m_max_x;
446244d93782SGreg Clayton   int m_max_y;
446344d93782SGreg Clayton };
446444d93782SGreg Clayton 
446544d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = {true};
446644d93782SGreg Clayton 
4467b9c1b51eSKate Stone IOHandlerCursesGUI::IOHandlerCursesGUI(Debugger &debugger)
4468b9c1b51eSKate Stone     : IOHandler(debugger, IOHandler::Type::Curses) {}
446944d93782SGreg Clayton 
4470b9c1b51eSKate Stone void IOHandlerCursesGUI::Activate() {
447144d93782SGreg Clayton   IOHandler::Activate();
4472b9c1b51eSKate Stone   if (!m_app_ap) {
447344d93782SGreg Clayton     m_app_ap.reset(new Application(GetInputFILE(), GetOutputFILE()));
447444d93782SGreg Clayton 
447544d93782SGreg Clayton     // This is both a window and a menu delegate
4476b9c1b51eSKate Stone     std::shared_ptr<ApplicationDelegate> app_delegate_sp(
4477b9c1b51eSKate Stone         new ApplicationDelegate(*m_app_ap, m_debugger));
447844d93782SGreg Clayton 
4479b9c1b51eSKate Stone     MenuDelegateSP app_menu_delegate_sp =
4480b9c1b51eSKate Stone         std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
4481b9c1b51eSKate Stone     MenuSP lldb_menu_sp(
4482b9c1b51eSKate Stone         new Menu("LLDB", "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
4483b9c1b51eSKate Stone     MenuSP exit_menuitem_sp(
4484b9c1b51eSKate Stone         new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
448544d93782SGreg Clayton     exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
4486b9c1b51eSKate Stone     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(
4487b9c1b51eSKate Stone         "About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
448844d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
448944d93782SGreg Clayton     lldb_menu_sp->AddSubmenu(exit_menuitem_sp);
449044d93782SGreg Clayton 
4491b9c1b51eSKate Stone     MenuSP target_menu_sp(new Menu("Target", "F2", KEY_F(2),
4492b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Target));
4493b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4494b9c1b51eSKate Stone         "Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
4495b9c1b51eSKate Stone     target_menu_sp->AddSubmenu(MenuSP(new Menu(
4496b9c1b51eSKate Stone         "Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
449744d93782SGreg Clayton 
4498b9c1b51eSKate Stone     MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3),
4499b9c1b51eSKate Stone                                     ApplicationDelegate::eMenuID_Process));
4500b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4501b9c1b51eSKate Stone         "Attach", nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
4502b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4503b9c1b51eSKate Stone         "Detach", nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
4504b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4505b9c1b51eSKate Stone         "Launch", nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
450644d93782SGreg Clayton     process_menu_sp->AddSubmenu(MenuSP(new Menu(Menu::Type::Separator)));
4507b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(
4508b9c1b51eSKate Stone         MenuSP(new Menu("Continue", nullptr, 'c',
4509b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ProcessContinue)));
4510b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4511b9c1b51eSKate Stone         "Halt", nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
4512b9c1b51eSKate Stone     process_menu_sp->AddSubmenu(MenuSP(new Menu(
4513b9c1b51eSKate Stone         "Kill", nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
451444d93782SGreg Clayton 
4515b9c1b51eSKate Stone     MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4),
4516b9c1b51eSKate Stone                                    ApplicationDelegate::eMenuID_Thread));
4517b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4518b9c1b51eSKate Stone         "Step In", nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
4519b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(
4520b9c1b51eSKate Stone         MenuSP(new Menu("Step Over", nullptr, 'v',
4521b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ThreadStepOver)));
4522b9c1b51eSKate Stone     thread_menu_sp->AddSubmenu(MenuSP(new Menu(
4523b9c1b51eSKate Stone         "Step Out", nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
452444d93782SGreg Clayton 
4525b9c1b51eSKate Stone     MenuSP view_menu_sp(
4526b9c1b51eSKate Stone         new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
4527b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4528b9c1b51eSKate Stone         MenuSP(new Menu("Backtrace", nullptr, 'b',
4529b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewBacktrace)));
4530b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4531b9c1b51eSKate Stone         MenuSP(new Menu("Registers", nullptr, 'r',
4532b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewRegisters)));
4533b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(MenuSP(new Menu(
4534b9c1b51eSKate Stone         "Source", nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
4535b9c1b51eSKate Stone     view_menu_sp->AddSubmenu(
4536b9c1b51eSKate Stone         MenuSP(new Menu("Variables", nullptr, 'v',
4537b9c1b51eSKate Stone                         ApplicationDelegate::eMenuID_ViewVariables)));
453844d93782SGreg Clayton 
4539b9c1b51eSKate Stone     MenuSP help_menu_sp(
4540b9c1b51eSKate Stone         new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
4541b9c1b51eSKate Stone     help_menu_sp->AddSubmenu(MenuSP(new Menu(
4542b9c1b51eSKate Stone         "GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
454344d93782SGreg Clayton 
454444d93782SGreg Clayton     m_app_ap->Initialize();
454544d93782SGreg Clayton     WindowSP &main_window_sp = m_app_ap->GetMainWindow();
454644d93782SGreg Clayton 
454744d93782SGreg Clayton     MenuSP menubar_sp(new Menu(Menu::Type::Bar));
454844d93782SGreg Clayton     menubar_sp->AddSubmenu(lldb_menu_sp);
454944d93782SGreg Clayton     menubar_sp->AddSubmenu(target_menu_sp);
455044d93782SGreg Clayton     menubar_sp->AddSubmenu(process_menu_sp);
455144d93782SGreg Clayton     menubar_sp->AddSubmenu(thread_menu_sp);
455244d93782SGreg Clayton     menubar_sp->AddSubmenu(view_menu_sp);
455344d93782SGreg Clayton     menubar_sp->AddSubmenu(help_menu_sp);
455444d93782SGreg Clayton     menubar_sp->SetDelegate(app_menu_delegate_sp);
455544d93782SGreg Clayton 
455644d93782SGreg Clayton     Rect content_bounds = main_window_sp->GetFrame();
455744d93782SGreg Clayton     Rect menubar_bounds = content_bounds.MakeMenuBar();
455844d93782SGreg Clayton     Rect status_bounds = content_bounds.MakeStatusBar();
455944d93782SGreg Clayton     Rect source_bounds;
456044d93782SGreg Clayton     Rect variables_bounds;
456144d93782SGreg Clayton     Rect threads_bounds;
456244d93782SGreg Clayton     Rect source_variables_bounds;
4563b9c1b51eSKate Stone     content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds,
4564b9c1b51eSKate Stone                                            threads_bounds);
4565b9c1b51eSKate Stone     source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds,
4566b9c1b51eSKate Stone                                                       variables_bounds);
456744d93782SGreg Clayton 
4568b9c1b51eSKate Stone     WindowSP menubar_window_sp =
4569b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
457005097246SAdrian Prantl     // Let the menubar get keys if the active window doesn't handle the keys
457105097246SAdrian Prantl     // that are typed so it can respond to menubar key presses.
4572b9c1b51eSKate Stone     menubar_window_sp->SetCanBeActive(
4573b9c1b51eSKate Stone         false); // Don't let the menubar become the active window
457444d93782SGreg Clayton     menubar_window_sp->SetDelegate(menubar_sp);
457544d93782SGreg Clayton 
4576b9c1b51eSKate Stone     WindowSP source_window_sp(
4577b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Source", source_bounds, true));
4578b9c1b51eSKate Stone     WindowSP variables_window_sp(
4579b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Variables", variables_bounds, false));
4580b9c1b51eSKate Stone     WindowSP threads_window_sp(
4581b9c1b51eSKate Stone         main_window_sp->CreateSubWindow("Threads", threads_bounds, false));
4582b9c1b51eSKate Stone     WindowSP status_window_sp(
45836bb7e21fSPavel Labath         main_window_sp->CreateSubWindow("Status", status_bounds, false));
4584b9c1b51eSKate Stone     status_window_sp->SetCanBeActive(
4585b9c1b51eSKate Stone         false); // Don't let the status bar become the active window
4586b9c1b51eSKate Stone     main_window_sp->SetDelegate(
4587b9c1b51eSKate Stone         std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
4588b9c1b51eSKate Stone     source_window_sp->SetDelegate(
4589b9c1b51eSKate Stone         WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
4590b9c1b51eSKate Stone     variables_window_sp->SetDelegate(
4591b9c1b51eSKate Stone         WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
4592ec990867SGreg Clayton     TreeDelegateSP thread_delegate_sp(new ThreadsTreeDelegate(m_debugger));
4593b9c1b51eSKate Stone     threads_window_sp->SetDelegate(WindowDelegateSP(
4594b9c1b51eSKate Stone         new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
4595b9c1b51eSKate Stone     status_window_sp->SetDelegate(
4596b9c1b51eSKate Stone         WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
459744d93782SGreg Clayton 
45985fdb09bbSGreg Clayton     // Show the main help window once the first time the curses GUI is launched
45995fdb09bbSGreg Clayton     static bool g_showed_help = false;
4600b9c1b51eSKate Stone     if (!g_showed_help) {
46015fdb09bbSGreg Clayton       g_showed_help = true;
46025fdb09bbSGreg Clayton       main_window_sp->CreateHelpSubwindow();
46035fdb09bbSGreg Clayton     }
46045fdb09bbSGreg Clayton 
460544d93782SGreg Clayton     init_pair(1, COLOR_WHITE, COLOR_BLUE);
460644d93782SGreg Clayton     init_pair(2, COLOR_BLACK, COLOR_WHITE);
460744d93782SGreg Clayton     init_pair(3, COLOR_MAGENTA, COLOR_WHITE);
460844d93782SGreg Clayton     init_pair(4, COLOR_MAGENTA, COLOR_BLACK);
460944d93782SGreg Clayton     init_pair(5, COLOR_RED, COLOR_BLACK);
461044d93782SGreg Clayton   }
461144d93782SGreg Clayton }
461244d93782SGreg Clayton 
4613b9c1b51eSKate Stone void IOHandlerCursesGUI::Deactivate() { m_app_ap->Terminate(); }
461444d93782SGreg Clayton 
4615b9c1b51eSKate Stone void IOHandlerCursesGUI::Run() {
461644d93782SGreg Clayton   m_app_ap->Run(m_debugger);
461744d93782SGreg Clayton   SetIsDone(true);
461844d93782SGreg Clayton }
461944d93782SGreg Clayton 
4620315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
462144d93782SGreg Clayton 
4622b9c1b51eSKate Stone void IOHandlerCursesGUI::Cancel() {}
462344d93782SGreg Clayton 
4624b9c1b51eSKate Stone bool IOHandlerCursesGUI::Interrupt() { return false; }
462544d93782SGreg Clayton 
4626b9c1b51eSKate Stone void IOHandlerCursesGUI::GotEOF() {}
462744d93782SGreg Clayton 
4628315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
4629