144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===//
244d93782SGreg Clayton //
344d93782SGreg Clayton //                     The LLVM Compiler Infrastructure
444d93782SGreg Clayton //
544d93782SGreg Clayton // This file is distributed under the University of Illinois Open Source
644d93782SGreg Clayton // License. See LICENSE.TXT for details.
744d93782SGreg Clayton //
844d93782SGreg Clayton //===----------------------------------------------------------------------===//
944d93782SGreg Clayton 
10315b6884SEugene Zelenko // C Includes
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
16315b6884SEugene Zelenko // C++ Includes
177c9aa073STodd Fiala #if defined(__APPLE__)
187c9aa073STodd Fiala #include <deque>
197c9aa073STodd Fiala #endif
2044d93782SGreg Clayton #include <string>
2144d93782SGreg Clayton 
22315b6884SEugene Zelenko // Other libraries and framework includes
23315b6884SEugene Zelenko // Project includes
2444d93782SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h"
2544d93782SGreg Clayton #include "lldb/Core/IOHandler.h"
2644d93782SGreg Clayton #include "lldb/Core/Debugger.h"
27ec990867SGreg Clayton #include "lldb/Core/Module.h"
2844d93782SGreg Clayton #include "lldb/Core/State.h"
2944d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
3044d93782SGreg Clayton #include "lldb/Core/ValueObjectRegister.h"
31cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3244d93782SGreg Clayton #include "lldb/Host/Editline.h"
33cacde7dfSTodd Fiala #endif
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3544d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
3644d93782SGreg Clayton #include "lldb/Symbol/Block.h"
3744d93782SGreg Clayton #include "lldb/Symbol/Function.h"
3844d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
3944d93782SGreg Clayton #include "lldb/Target/RegisterContext.h"
4044d93782SGreg Clayton #include "lldb/Target/ThreadPlan.h"
41c5dac77aSEugene Zelenko #ifndef LLDB_DISABLE_CURSES
42c5dac77aSEugene Zelenko #include "lldb/Core/ValueObject.h"
43c5dac77aSEugene Zelenko #include "lldb/Symbol/VariableList.h"
44c5dac77aSEugene Zelenko #include "lldb/Target/Target.h"
45c5dac77aSEugene Zelenko #include "lldb/Target/Process.h"
46c5dac77aSEugene Zelenko #include "lldb/Target/Thread.h"
47c5dac77aSEugene Zelenko #include "lldb/Target/StackFrame.h"
48c5dac77aSEugene Zelenko #endif
497c9aa073STodd Fiala 
50fab31220STed Woodward #ifdef _MSC_VER
51fab31220STed Woodward #include <Windows.h>
52fab31220STed Woodward #endif
53fab31220STed Woodward 
5444d93782SGreg Clayton using namespace lldb;
5544d93782SGreg Clayton using namespace lldb_private;
5644d93782SGreg Clayton 
57e30f11d9SKate Stone IOHandler::IOHandler (Debugger &debugger, IOHandler::Type type) :
5844d93782SGreg Clayton     IOHandler (debugger,
59e30f11d9SKate Stone                type,
6044d93782SGreg Clayton                StreamFileSP(),  // Adopt STDIN from top input reader
6144d93782SGreg Clayton                StreamFileSP(),  // Adopt STDOUT from top input reader
62340b0309SGreg Clayton                StreamFileSP(),  // Adopt STDERR from top input reader
63340b0309SGreg Clayton                0)               // Flags
6444d93782SGreg Clayton {
6544d93782SGreg Clayton }
6644d93782SGreg Clayton 
6744d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger,
68e30f11d9SKate Stone                       IOHandler::Type type,
6944d93782SGreg Clayton                       const lldb::StreamFileSP &input_sp,
7044d93782SGreg Clayton                       const lldb::StreamFileSP &output_sp,
71340b0309SGreg Clayton                       const lldb::StreamFileSP &error_sp,
72340b0309SGreg Clayton                       uint32_t flags) :
7344d93782SGreg Clayton     m_debugger (debugger),
7444d93782SGreg Clayton     m_input_sp (input_sp),
7544d93782SGreg Clayton     m_output_sp (output_sp),
7644d93782SGreg Clayton     m_error_sp (error_sp),
77e30f11d9SKate Stone     m_popped (false),
78340b0309SGreg Clayton     m_flags (flags),
79e30f11d9SKate Stone     m_type (type),
80c5dac77aSEugene Zelenko     m_user_data(nullptr),
8144d93782SGreg Clayton     m_done (false),
8244d93782SGreg Clayton     m_active (false)
8344d93782SGreg Clayton {
8444d93782SGreg Clayton     // If any files are not specified, then adopt them from the top input reader.
8544d93782SGreg Clayton     if (!m_input_sp || !m_output_sp || !m_error_sp)
8644d93782SGreg Clayton         debugger.AdoptTopIOHandlerFilesIfInvalid (m_input_sp,
8744d93782SGreg Clayton                                                   m_output_sp,
8844d93782SGreg Clayton                                                   m_error_sp);
8944d93782SGreg Clayton }
9044d93782SGreg Clayton 
91315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
9244d93782SGreg Clayton 
9344d93782SGreg Clayton int
9444d93782SGreg Clayton IOHandler::GetInputFD()
9544d93782SGreg Clayton {
96c5dac77aSEugene Zelenko     return (m_input_sp ? m_input_sp->GetFile().GetDescriptor() : -1);
9744d93782SGreg Clayton }
9844d93782SGreg Clayton 
9944d93782SGreg Clayton int
10044d93782SGreg Clayton IOHandler::GetOutputFD()
10144d93782SGreg Clayton {
102c5dac77aSEugene Zelenko     return (m_output_sp ? m_output_sp->GetFile().GetDescriptor() : -1);
10344d93782SGreg Clayton }
10444d93782SGreg Clayton 
10544d93782SGreg Clayton int
10644d93782SGreg Clayton IOHandler::GetErrorFD()
10744d93782SGreg Clayton {
108c5dac77aSEugene Zelenko     return (m_error_sp ? m_error_sp->GetFile().GetDescriptor() : -1);
10944d93782SGreg Clayton }
11044d93782SGreg Clayton 
11144d93782SGreg Clayton FILE *
11244d93782SGreg Clayton IOHandler::GetInputFILE()
11344d93782SGreg Clayton {
114c5dac77aSEugene Zelenko     return (m_input_sp ? m_input_sp->GetFile().GetStream() : nullptr);
11544d93782SGreg Clayton }
11644d93782SGreg Clayton 
11744d93782SGreg Clayton FILE *
11844d93782SGreg Clayton IOHandler::GetOutputFILE()
11944d93782SGreg Clayton {
120c5dac77aSEugene Zelenko     return (m_output_sp ? m_output_sp->GetFile().GetStream() : nullptr);
12144d93782SGreg Clayton }
12244d93782SGreg Clayton 
12344d93782SGreg Clayton FILE *
12444d93782SGreg Clayton IOHandler::GetErrorFILE()
12544d93782SGreg Clayton {
126c5dac77aSEugene Zelenko     return (m_error_sp ? m_error_sp->GetFile().GetStream() : nullptr);
12744d93782SGreg Clayton }
12844d93782SGreg Clayton 
12944d93782SGreg Clayton StreamFileSP &
13044d93782SGreg Clayton IOHandler::GetInputStreamFile()
13144d93782SGreg Clayton {
13244d93782SGreg Clayton     return m_input_sp;
13344d93782SGreg Clayton }
13444d93782SGreg Clayton 
13544d93782SGreg Clayton StreamFileSP &
13644d93782SGreg Clayton IOHandler::GetOutputStreamFile()
13744d93782SGreg Clayton {
13844d93782SGreg Clayton     return m_output_sp;
13944d93782SGreg Clayton }
14044d93782SGreg Clayton 
14144d93782SGreg Clayton StreamFileSP &
14244d93782SGreg Clayton IOHandler::GetErrorStreamFile()
14344d93782SGreg Clayton {
14444d93782SGreg Clayton     return m_error_sp;
14544d93782SGreg Clayton }
14644d93782SGreg Clayton 
147340b0309SGreg Clayton bool
148340b0309SGreg Clayton IOHandler::GetIsInteractive ()
149340b0309SGreg Clayton {
150340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsInteractive ();
151340b0309SGreg Clayton }
152340b0309SGreg Clayton 
153340b0309SGreg Clayton bool
154340b0309SGreg Clayton IOHandler::GetIsRealTerminal ()
155340b0309SGreg Clayton {
156340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsRealTerminal();
157340b0309SGreg Clayton }
15844d93782SGreg Clayton 
159e30f11d9SKate Stone void
160e30f11d9SKate Stone IOHandler::SetPopped (bool b)
161e30f11d9SKate Stone {
162e30f11d9SKate Stone     m_popped.SetValue(b, eBroadcastOnChange);
163e30f11d9SKate Stone }
164e30f11d9SKate Stone 
165e30f11d9SKate Stone void
166e30f11d9SKate Stone IOHandler::WaitForPop ()
167e30f11d9SKate Stone {
168e30f11d9SKate Stone     m_popped.WaitForValueEqualTo(true);
169e30f11d9SKate Stone }
170e30f11d9SKate Stone 
1714446487dSPavel Labath void
1724446487dSPavel Labath IOHandlerStack::PrintAsync(Stream *stream, const char *s, size_t len)
1734446487dSPavel Labath {
1744446487dSPavel Labath     if (stream)
1754446487dSPavel Labath     {
17616ff8604SSaleem Abdulrasool         std::lock_guard<std::recursive_mutex> guard(m_mutex);
1774446487dSPavel Labath         if (m_top)
1784446487dSPavel Labath             m_top->PrintAsync(stream, s, len);
1794446487dSPavel Labath     }
1804446487dSPavel Labath }
1814446487dSPavel Labath 
18244d93782SGreg Clayton IOHandlerConfirm::IOHandlerConfirm (Debugger &debugger,
18344d93782SGreg Clayton                                     const char *prompt,
18444d93782SGreg Clayton                                     bool default_response) :
18544d93782SGreg Clayton     IOHandlerEditline(debugger,
186e30f11d9SKate Stone                       IOHandler::Type::Confirm,
187c5dac77aSEugene Zelenko                       nullptr,  // nullptr editline_name means no history loaded/saved
188c5dac77aSEugene Zelenko                       nullptr,  // No prompt
189c5dac77aSEugene Zelenko                       nullptr,  // No continuation prompt
19044d93782SGreg Clayton                       false,    // Multi-line
191e30f11d9SKate Stone                       false,    // Don't colorize the prompt (i.e. the confirm message.)
192f6913cd7SGreg Clayton                       0,
19344d93782SGreg Clayton                       *this),
19444d93782SGreg Clayton     m_default_response (default_response),
19544d93782SGreg Clayton     m_user_response (default_response)
19644d93782SGreg Clayton {
19744d93782SGreg Clayton     StreamString prompt_stream;
19844d93782SGreg Clayton     prompt_stream.PutCString(prompt);
19944d93782SGreg Clayton     if (m_default_response)
20044d93782SGreg Clayton         prompt_stream.Printf(": [Y/n] ");
20144d93782SGreg Clayton     else
20244d93782SGreg Clayton         prompt_stream.Printf(": [y/N] ");
20344d93782SGreg Clayton 
20444d93782SGreg Clayton     SetPrompt (prompt_stream.GetString().c_str());
20544d93782SGreg Clayton }
20644d93782SGreg Clayton 
207315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
20844d93782SGreg Clayton 
20944d93782SGreg Clayton int
21044d93782SGreg Clayton IOHandlerConfirm::IOHandlerComplete (IOHandler &io_handler,
21144d93782SGreg Clayton                                      const char *current_line,
21244d93782SGreg Clayton                                      const char *cursor,
21344d93782SGreg Clayton                                      const char *last_char,
21444d93782SGreg Clayton                                      int skip_first_n_matches,
21544d93782SGreg Clayton                                      int max_matches,
21644d93782SGreg Clayton                                      StringList &matches)
21744d93782SGreg Clayton {
21844d93782SGreg Clayton     if (current_line == cursor)
21944d93782SGreg Clayton     {
22044d93782SGreg Clayton         if (m_default_response)
22144d93782SGreg Clayton         {
22244d93782SGreg Clayton             matches.AppendString("y");
22344d93782SGreg Clayton         }
22444d93782SGreg Clayton         else
22544d93782SGreg Clayton         {
22644d93782SGreg Clayton             matches.AppendString("n");
22744d93782SGreg Clayton         }
22844d93782SGreg Clayton     }
22944d93782SGreg Clayton     return matches.GetSize();
23044d93782SGreg Clayton }
23144d93782SGreg Clayton 
23244d93782SGreg Clayton void
23344d93782SGreg Clayton IOHandlerConfirm::IOHandlerInputComplete (IOHandler &io_handler, std::string &line)
23444d93782SGreg Clayton {
23544d93782SGreg Clayton     if (line.empty())
23644d93782SGreg Clayton     {
23744d93782SGreg Clayton         // User just hit enter, set the response to the default
23844d93782SGreg Clayton         m_user_response = m_default_response;
23944d93782SGreg Clayton         io_handler.SetIsDone(true);
24044d93782SGreg Clayton         return;
24144d93782SGreg Clayton     }
24244d93782SGreg Clayton 
24344d93782SGreg Clayton     if (line.size() == 1)
24444d93782SGreg Clayton     {
24544d93782SGreg Clayton         switch (line[0])
24644d93782SGreg Clayton         {
24744d93782SGreg Clayton             case 'y':
24844d93782SGreg Clayton             case 'Y':
24944d93782SGreg Clayton                 m_user_response = true;
25044d93782SGreg Clayton                 io_handler.SetIsDone(true);
25144d93782SGreg Clayton                 return;
25244d93782SGreg Clayton             case 'n':
25344d93782SGreg Clayton             case 'N':
25444d93782SGreg Clayton                 m_user_response = false;
25544d93782SGreg Clayton                 io_handler.SetIsDone(true);
25644d93782SGreg Clayton                 return;
25744d93782SGreg Clayton             default:
25844d93782SGreg Clayton                 break;
25944d93782SGreg Clayton         }
26044d93782SGreg Clayton     }
26144d93782SGreg Clayton 
26244d93782SGreg Clayton     if (line == "yes" || line == "YES" || line == "Yes")
26344d93782SGreg Clayton     {
26444d93782SGreg Clayton         m_user_response = true;
26544d93782SGreg Clayton         io_handler.SetIsDone(true);
26644d93782SGreg Clayton     }
26744d93782SGreg Clayton     else if (line == "no" || line == "NO" || line == "No")
26844d93782SGreg Clayton     {
26944d93782SGreg Clayton         m_user_response = false;
27044d93782SGreg Clayton         io_handler.SetIsDone(true);
27144d93782SGreg Clayton     }
27244d93782SGreg Clayton }
27344d93782SGreg Clayton 
27444d93782SGreg Clayton int
27544d93782SGreg Clayton IOHandlerDelegate::IOHandlerComplete (IOHandler &io_handler,
27644d93782SGreg Clayton                                       const char *current_line,
27744d93782SGreg Clayton                                       const char *cursor,
27844d93782SGreg Clayton                                       const char *last_char,
27944d93782SGreg Clayton                                       int skip_first_n_matches,
28044d93782SGreg Clayton                                       int max_matches,
28144d93782SGreg Clayton                                       StringList &matches)
28244d93782SGreg Clayton {
28344d93782SGreg Clayton     switch (m_completion)
28444d93782SGreg Clayton     {
28544d93782SGreg Clayton     case Completion::None:
28644d93782SGreg Clayton         break;
28744d93782SGreg Clayton 
28844d93782SGreg Clayton     case Completion::LLDBCommand:
28944d93782SGreg Clayton         return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion (current_line,
29044d93782SGreg Clayton                                                                                   cursor,
29144d93782SGreg Clayton                                                                                   last_char,
29244d93782SGreg Clayton                                                                                   skip_first_n_matches,
29344d93782SGreg Clayton                                                                                   max_matches,
29444d93782SGreg Clayton                                                                                   matches);
29544d93782SGreg Clayton 
29644d93782SGreg Clayton     case Completion::Expression:
29744d93782SGreg Clayton         {
29844d93782SGreg Clayton             bool word_complete = false;
29944d93782SGreg Clayton             const char *word_start = cursor;
30044d93782SGreg Clayton             if (cursor > current_line)
30144d93782SGreg Clayton                 --word_start;
30244d93782SGreg Clayton             while (word_start > current_line && !isspace(*word_start))
30344d93782SGreg Clayton                 --word_start;
30444d93782SGreg Clayton             CommandCompletions::InvokeCommonCompletionCallbacks(io_handler.GetDebugger().GetCommandInterpreter(),
30544d93782SGreg Clayton                                                                 CommandCompletions::eVariablePathCompletion,
30644d93782SGreg Clayton                                                                 word_start,
30744d93782SGreg Clayton                                                                 skip_first_n_matches,
30844d93782SGreg Clayton                                                                 max_matches,
309c5dac77aSEugene Zelenko                                                                 nullptr,
31044d93782SGreg Clayton                                                                 word_complete,
31144d93782SGreg Clayton                                                                 matches);
31244d93782SGreg Clayton 
31344d93782SGreg Clayton             size_t num_matches = matches.GetSize();
31444d93782SGreg Clayton             if (num_matches > 0)
31544d93782SGreg Clayton             {
31644d93782SGreg Clayton                 std::string common_prefix;
31744d93782SGreg Clayton                 matches.LongestCommonPrefix (common_prefix);
31844d93782SGreg Clayton                 const size_t partial_name_len = strlen(word_start);
31944d93782SGreg Clayton 
32044d93782SGreg Clayton                 // If we matched a unique single command, add a space...
32144d93782SGreg Clayton                 // Only do this if the completer told us this was a complete word, however...
32244d93782SGreg Clayton                 if (num_matches == 1 && word_complete)
32344d93782SGreg Clayton                 {
32444d93782SGreg Clayton                     common_prefix.push_back(' ');
32544d93782SGreg Clayton                 }
32644d93782SGreg Clayton                 common_prefix.erase (0, partial_name_len);
32744d93782SGreg Clayton                 matches.InsertStringAtIndex(0, std::move(common_prefix));
32844d93782SGreg Clayton             }
32944d93782SGreg Clayton             return num_matches;
33044d93782SGreg Clayton         }
33144d93782SGreg Clayton         break;
33244d93782SGreg Clayton     }
33344d93782SGreg Clayton 
33444d93782SGreg Clayton     return 0;
33544d93782SGreg Clayton }
33644d93782SGreg Clayton 
33744d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
338e30f11d9SKate Stone                                       IOHandler::Type type,
33944d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
34044d93782SGreg Clayton                                       const char *prompt,
341e30f11d9SKate Stone                                       const char *continuation_prompt,
34244d93782SGreg Clayton                                       bool multi_line,
343e30f11d9SKate Stone                                       bool color_prompts,
344f6913cd7SGreg Clayton                                       uint32_t line_number_start,
34544d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
34644d93782SGreg Clayton     IOHandlerEditline(debugger,
347e30f11d9SKate Stone                       type,
34844d93782SGreg Clayton                       StreamFileSP(), // Inherit input from top input reader
34944d93782SGreg Clayton                       StreamFileSP(), // Inherit output from top input reader
35044d93782SGreg Clayton                       StreamFileSP(), // Inherit error from top input reader
351340b0309SGreg Clayton                       0,              // Flags
35244d93782SGreg Clayton                       editline_name,  // Used for saving history files
35344d93782SGreg Clayton                       prompt,
354e30f11d9SKate Stone                       continuation_prompt,
35544d93782SGreg Clayton                       multi_line,
356e30f11d9SKate Stone                       color_prompts,
357f6913cd7SGreg Clayton                       line_number_start,
35844d93782SGreg Clayton                       delegate)
35944d93782SGreg Clayton {
36044d93782SGreg Clayton }
36144d93782SGreg Clayton 
36244d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
363e30f11d9SKate Stone                                       IOHandler::Type type,
36444d93782SGreg Clayton                                       const lldb::StreamFileSP &input_sp,
36544d93782SGreg Clayton                                       const lldb::StreamFileSP &output_sp,
36644d93782SGreg Clayton                                       const lldb::StreamFileSP &error_sp,
367340b0309SGreg Clayton                                       uint32_t flags,
36844d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
36944d93782SGreg Clayton                                       const char *prompt,
370e30f11d9SKate Stone                                       const char *continuation_prompt,
37144d93782SGreg Clayton                                       bool multi_line,
372e30f11d9SKate Stone                                       bool color_prompts,
373f6913cd7SGreg Clayton                                       uint32_t line_number_start,
37444d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
375e30f11d9SKate Stone     IOHandler (debugger, type, input_sp, output_sp, error_sp, flags),
376cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
37744d93782SGreg Clayton     m_editline_ap (),
378cacde7dfSTodd Fiala #endif
37944d93782SGreg Clayton     m_delegate (delegate),
38044d93782SGreg Clayton     m_prompt (),
381e30f11d9SKate Stone     m_continuation_prompt(),
382c5dac77aSEugene Zelenko     m_current_lines_ptr(nullptr),
383f6913cd7SGreg Clayton     m_base_line_number (line_number_start),
384e30f11d9SKate Stone     m_curr_line_idx (UINT32_MAX),
385e30f11d9SKate Stone     m_multi_line (multi_line),
386e30f11d9SKate Stone     m_color_prompts (color_prompts),
387e034a04eSGreg Clayton     m_interrupt_exits (true),
388e034a04eSGreg Clayton     m_editing (false)
38944d93782SGreg Clayton {
39044d93782SGreg Clayton     SetPrompt(prompt);
39144d93782SGreg Clayton 
392cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
393914b8d98SDeepak Panickal     bool use_editline = false;
394340b0309SGreg Clayton 
395340b0309SGreg Clayton     use_editline = m_input_sp->GetFile().GetIsRealTerminal();
39644d93782SGreg Clayton 
39744d93782SGreg Clayton     if (use_editline)
39844d93782SGreg Clayton     {
39944d93782SGreg Clayton         m_editline_ap.reset(new Editline (editline_name,
40044d93782SGreg Clayton                                           GetInputFILE (),
40144d93782SGreg Clayton                                           GetOutputFILE (),
402e30f11d9SKate Stone                                           GetErrorFILE (),
403e30f11d9SKate Stone                                           m_color_prompts));
404e30f11d9SKate Stone         m_editline_ap->SetIsInputCompleteCallback (IsInputCompleteCallback, this);
40544d93782SGreg Clayton         m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this);
406e30f11d9SKate Stone         // See if the delegate supports fixing indentation
407e30f11d9SKate Stone         const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
408e30f11d9SKate Stone         if (indent_chars)
409e30f11d9SKate Stone         {
410e30f11d9SKate Stone             // The delegate does support indentation, hook it up so when any indentation
411e30f11d9SKate Stone             // character is typed, the delegate gets a chance to fix it
412e30f11d9SKate Stone             m_editline_ap->SetFixIndentationCallback (FixIndentationCallback, this, indent_chars);
413e30f11d9SKate Stone         }
41444d93782SGreg Clayton     }
415cacde7dfSTodd Fiala #endif
416e30f11d9SKate Stone     SetBaseLineNumber (m_base_line_number);
417e30f11d9SKate Stone     SetPrompt(prompt ? prompt : "");
418e30f11d9SKate Stone     SetContinuationPrompt(continuation_prompt);
41944d93782SGreg Clayton }
42044d93782SGreg Clayton 
42144d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline ()
42244d93782SGreg Clayton {
423cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
42444d93782SGreg Clayton     m_editline_ap.reset();
425cacde7dfSTodd Fiala #endif
42644d93782SGreg Clayton }
42744d93782SGreg Clayton 
428e30f11d9SKate Stone void
429e30f11d9SKate Stone IOHandlerEditline::Activate ()
430e30f11d9SKate Stone {
431e30f11d9SKate Stone     IOHandler::Activate();
432e30f11d9SKate Stone     m_delegate.IOHandlerActivated(*this);
433e30f11d9SKate Stone }
434e30f11d9SKate Stone 
435e30f11d9SKate Stone void
436e30f11d9SKate Stone IOHandlerEditline::Deactivate ()
437e30f11d9SKate Stone {
438e30f11d9SKate Stone     IOHandler::Deactivate();
439e30f11d9SKate Stone     m_delegate.IOHandlerDeactivated(*this);
440e30f11d9SKate Stone }
441e30f11d9SKate Stone 
44244d93782SGreg Clayton bool
443f0066ad0SGreg Clayton IOHandlerEditline::GetLine (std::string &line, bool &interrupted)
44444d93782SGreg Clayton {
445cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
44644d93782SGreg Clayton     if (m_editline_ap)
44744d93782SGreg Clayton     {
448e30f11d9SKate Stone         return m_editline_ap->GetLine (line, interrupted);
44944d93782SGreg Clayton     }
45044d93782SGreg Clayton     else
45144d93782SGreg Clayton     {
452cacde7dfSTodd Fiala #endif
45344d93782SGreg Clayton         line.clear();
45444d93782SGreg Clayton 
45544d93782SGreg Clayton         FILE *in = GetInputFILE();
45644d93782SGreg Clayton         if (in)
45744d93782SGreg Clayton         {
458340b0309SGreg Clayton             if (GetIsInteractive())
45944d93782SGreg Clayton             {
460c5dac77aSEugene Zelenko                 const char *prompt = nullptr;
461e30f11d9SKate Stone 
462e30f11d9SKate Stone                 if (m_multi_line && m_curr_line_idx > 0)
463e30f11d9SKate Stone                     prompt = GetContinuationPrompt();
464e30f11d9SKate Stone 
465c5dac77aSEugene Zelenko                 if (prompt == nullptr)
466e30f11d9SKate Stone                     prompt = GetPrompt();
467e30f11d9SKate Stone 
46844d93782SGreg Clayton                 if (prompt && prompt[0])
46944d93782SGreg Clayton                 {
47044d93782SGreg Clayton                     FILE *out = GetOutputFILE();
47144d93782SGreg Clayton                     if (out)
47244d93782SGreg Clayton                     {
47344d93782SGreg Clayton                         ::fprintf(out, "%s", prompt);
47444d93782SGreg Clayton                         ::fflush(out);
47544d93782SGreg Clayton                     }
47644d93782SGreg Clayton                 }
47744d93782SGreg Clayton             }
47844d93782SGreg Clayton             char buffer[256];
47944d93782SGreg Clayton             bool done = false;
4800f86e6e7SGreg Clayton             bool got_line = false;
481e034a04eSGreg Clayton             m_editing = true;
48244d93782SGreg Clayton             while (!done)
48344d93782SGreg Clayton             {
484c5dac77aSEugene Zelenko                 if (fgets(buffer, sizeof(buffer), in) == nullptr)
485c9cf5798SGreg Clayton                 {
486c7797accSGreg Clayton                     const int saved_errno = errno;
487c9cf5798SGreg Clayton                     if (feof(in))
48844d93782SGreg Clayton                         done = true;
489c7797accSGreg Clayton                     else if (ferror(in))
490c7797accSGreg Clayton                     {
491c7797accSGreg Clayton                         if (saved_errno != EINTR)
492c7797accSGreg Clayton                             done = true;
493c7797accSGreg Clayton                     }
494c9cf5798SGreg Clayton                 }
49544d93782SGreg Clayton                 else
49644d93782SGreg Clayton                 {
4970f86e6e7SGreg Clayton                     got_line = true;
49844d93782SGreg Clayton                     size_t buffer_len = strlen(buffer);
49944d93782SGreg Clayton                     assert (buffer[buffer_len] == '\0');
50044d93782SGreg Clayton                     char last_char = buffer[buffer_len-1];
50144d93782SGreg Clayton                     if (last_char == '\r' || last_char == '\n')
50244d93782SGreg Clayton                     {
50344d93782SGreg Clayton                         done = true;
50444d93782SGreg Clayton                         // Strip trailing newlines
50544d93782SGreg Clayton                         while (last_char == '\r' || last_char == '\n')
50644d93782SGreg Clayton                         {
50744d93782SGreg Clayton                             --buffer_len;
50844d93782SGreg Clayton                             if (buffer_len == 0)
50944d93782SGreg Clayton                                 break;
51044d93782SGreg Clayton                             last_char = buffer[buffer_len-1];
51144d93782SGreg Clayton                         }
51244d93782SGreg Clayton                     }
51344d93782SGreg Clayton                     line.append(buffer, buffer_len);
51444d93782SGreg Clayton                 }
51544d93782SGreg Clayton             }
516e034a04eSGreg Clayton             m_editing = false;
5170f86e6e7SGreg Clayton             // We might have gotten a newline on a line by itself
5180f86e6e7SGreg Clayton             // make sure to return true in this case.
5190f86e6e7SGreg Clayton             return got_line;
52044d93782SGreg Clayton         }
52144d93782SGreg Clayton         else
52244d93782SGreg Clayton         {
52344d93782SGreg Clayton             // No more input file, we are done...
52444d93782SGreg Clayton             SetIsDone(true);
52544d93782SGreg Clayton         }
526340b0309SGreg Clayton         return false;
527cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
52844d93782SGreg Clayton     }
529cacde7dfSTodd Fiala #endif
53044d93782SGreg Clayton }
53144d93782SGreg Clayton 
532cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
533e30f11d9SKate Stone bool
534e30f11d9SKate Stone IOHandlerEditline::IsInputCompleteCallback (Editline *editline,
53544d93782SGreg Clayton                                           StringList &lines,
53644d93782SGreg Clayton                                           void *baton)
53744d93782SGreg Clayton {
53844d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
539e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader, lines);
540e30f11d9SKate Stone }
541e30f11d9SKate Stone 
542e30f11d9SKate Stone int
543e30f11d9SKate Stone IOHandlerEditline::FixIndentationCallback (Editline *editline,
544e30f11d9SKate Stone                                            const StringList &lines,
545e30f11d9SKate Stone                                            int cursor_position,
546e30f11d9SKate Stone                                            void *baton)
547e30f11d9SKate Stone {
548e30f11d9SKate Stone     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
549e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerFixIndentation(*editline_reader, lines, cursor_position);
55044d93782SGreg Clayton }
55144d93782SGreg Clayton 
55244d93782SGreg Clayton int
55344d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line,
55444d93782SGreg Clayton                                          const char *cursor,
55544d93782SGreg Clayton                                          const char *last_char,
55644d93782SGreg Clayton                                          int skip_first_n_matches,
55744d93782SGreg Clayton                                          int max_matches,
55844d93782SGreg Clayton                                          StringList &matches,
55944d93782SGreg Clayton                                          void *baton)
56044d93782SGreg Clayton {
56144d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
56244d93782SGreg Clayton     if (editline_reader)
56344d93782SGreg Clayton         return editline_reader->m_delegate.IOHandlerComplete (*editline_reader,
56444d93782SGreg Clayton                                                               current_line,
56544d93782SGreg Clayton                                                               cursor,
56644d93782SGreg Clayton                                                               last_char,
56744d93782SGreg Clayton                                                               skip_first_n_matches,
56844d93782SGreg Clayton                                                               max_matches,
56944d93782SGreg Clayton                                                               matches);
57044d93782SGreg Clayton     return 0;
57144d93782SGreg Clayton }
572cacde7dfSTodd Fiala #endif
57344d93782SGreg Clayton 
57444d93782SGreg Clayton const char *
57544d93782SGreg Clayton IOHandlerEditline::GetPrompt ()
57644d93782SGreg Clayton {
577cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
57844d93782SGreg Clayton     if (m_editline_ap)
579cacde7dfSTodd Fiala     {
58044d93782SGreg Clayton         return m_editline_ap->GetPrompt ();
581cacde7dfSTodd Fiala     }
582cacde7dfSTodd Fiala     else
583cacde7dfSTodd Fiala     {
584cacde7dfSTodd Fiala #endif
585cacde7dfSTodd Fiala         if (m_prompt.empty())
586c5dac77aSEugene Zelenko             return nullptr;
587cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
588cacde7dfSTodd Fiala     }
589cacde7dfSTodd Fiala #endif
59044d93782SGreg Clayton     return m_prompt.c_str();
59144d93782SGreg Clayton }
59244d93782SGreg Clayton 
59344d93782SGreg Clayton bool
59444d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p)
59544d93782SGreg Clayton {
59644d93782SGreg Clayton     if (p && p[0])
59744d93782SGreg Clayton         m_prompt = p;
59844d93782SGreg Clayton     else
59944d93782SGreg Clayton         m_prompt.clear();
600cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
60144d93782SGreg Clayton     if (m_editline_ap)
602c5dac77aSEugene Zelenko         m_editline_ap->SetPrompt(m_prompt.empty() ? nullptr : m_prompt.c_str());
603cacde7dfSTodd Fiala #endif
60444d93782SGreg Clayton     return true;
60544d93782SGreg Clayton }
60644d93782SGreg Clayton 
607e30f11d9SKate Stone const char *
608e30f11d9SKate Stone IOHandlerEditline::GetContinuationPrompt ()
609e30f11d9SKate Stone {
610c5dac77aSEugene Zelenko     return (m_continuation_prompt.empty() ? nullptr : m_continuation_prompt.c_str());
611e30f11d9SKate Stone }
612e30f11d9SKate Stone 
613e30f11d9SKate Stone void
614e30f11d9SKate Stone IOHandlerEditline::SetContinuationPrompt (const char *p)
615e30f11d9SKate Stone {
616e30f11d9SKate Stone     if (p && p[0])
617e30f11d9SKate Stone         m_continuation_prompt = p;
618e30f11d9SKate Stone     else
619e30f11d9SKate Stone         m_continuation_prompt.clear();
620e30f11d9SKate Stone 
621d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
622e30f11d9SKate Stone     if (m_editline_ap)
623c5dac77aSEugene Zelenko         m_editline_ap->SetContinuationPrompt(m_continuation_prompt.empty() ? nullptr : m_continuation_prompt.c_str());
624d553d00cSZachary Turner #endif
625e30f11d9SKate Stone }
626e30f11d9SKate Stone 
627f6913cd7SGreg Clayton void
628f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line)
629f6913cd7SGreg Clayton {
630f6913cd7SGreg Clayton     m_base_line_number = line;
631f6913cd7SGreg Clayton }
632e30f11d9SKate Stone 
633e30f11d9SKate Stone uint32_t
634e30f11d9SKate Stone IOHandlerEditline::GetCurrentLineIndex () const
635e30f11d9SKate Stone {
636d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
637e30f11d9SKate Stone     if (m_editline_ap)
638e30f11d9SKate Stone         return m_editline_ap->GetCurrentLine();
639e30f11d9SKate Stone #endif
640e30f11d9SKate Stone     return m_curr_line_idx;
641e30f11d9SKate Stone }
642e30f11d9SKate Stone 
64344d93782SGreg Clayton bool
644f0066ad0SGreg Clayton IOHandlerEditline::GetLines (StringList &lines, bool &interrupted)
64544d93782SGreg Clayton {
646e30f11d9SKate Stone     m_current_lines_ptr = &lines;
647e30f11d9SKate Stone 
64844d93782SGreg Clayton     bool success = false;
649cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
65044d93782SGreg Clayton     if (m_editline_ap)
65144d93782SGreg Clayton     {
652e30f11d9SKate Stone         return m_editline_ap->GetLines (m_base_line_number, lines, interrupted);
65344d93782SGreg Clayton     }
65444d93782SGreg Clayton     else
65544d93782SGreg Clayton     {
656cacde7dfSTodd Fiala #endif
657e30f11d9SKate Stone         bool done = false;
658c3d874a5SGreg Clayton         Error error;
65944d93782SGreg Clayton 
660e30f11d9SKate Stone         while (!done)
66144d93782SGreg Clayton         {
662f6913cd7SGreg Clayton             // Show line numbers if we are asked to
66344d93782SGreg Clayton             std::string line;
664f6913cd7SGreg Clayton             if (m_base_line_number > 0 && GetIsInteractive())
665f6913cd7SGreg Clayton             {
666f6913cd7SGreg Clayton                 FILE *out = GetOutputFILE();
667f6913cd7SGreg Clayton                 if (out)
668c5dac77aSEugene Zelenko                     ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(), GetPrompt() == nullptr ? " " : "");
669f6913cd7SGreg Clayton             }
670f6913cd7SGreg Clayton 
671e30f11d9SKate Stone             m_curr_line_idx = lines.GetSize();
672e30f11d9SKate Stone 
673f0066ad0SGreg Clayton             bool interrupted = false;
674e30f11d9SKate Stone             if (GetLine(line, interrupted) && !interrupted)
67544d93782SGreg Clayton             {
67644d93782SGreg Clayton                 lines.AppendString(line);
677e30f11d9SKate Stone                 done = m_delegate.IOHandlerIsInputComplete(*this, lines);
678f0066ad0SGreg Clayton             }
67944d93782SGreg Clayton             else
68044d93782SGreg Clayton             {
681e30f11d9SKate Stone                 done = true;
68244d93782SGreg Clayton             }
68344d93782SGreg Clayton         }
68444d93782SGreg Clayton         success = lines.GetSize() > 0;
685cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
68644d93782SGreg Clayton     }
687cacde7dfSTodd Fiala #endif
68844d93782SGreg Clayton     return success;
68944d93782SGreg Clayton }
69044d93782SGreg Clayton 
69144d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data
69244d93782SGreg Clayton // from the "in" and place output into "out" and "err and return
69344d93782SGreg Clayton // when done.
69444d93782SGreg Clayton void
69544d93782SGreg Clayton IOHandlerEditline::Run ()
69644d93782SGreg Clayton {
69744d93782SGreg Clayton     std::string line;
69844d93782SGreg Clayton     while (IsActive())
69944d93782SGreg Clayton     {
700f0066ad0SGreg Clayton         bool interrupted = false;
70144d93782SGreg Clayton         if (m_multi_line)
70244d93782SGreg Clayton         {
70344d93782SGreg Clayton             StringList lines;
704f0066ad0SGreg Clayton             if (GetLines (lines, interrupted))
705f0066ad0SGreg Clayton             {
706f0066ad0SGreg Clayton                 if (interrupted)
707f0066ad0SGreg Clayton                 {
708e30f11d9SKate Stone                     m_done = m_interrupt_exits;
709e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
710e30f11d9SKate Stone 
711f0066ad0SGreg Clayton                 }
712f0066ad0SGreg Clayton                 else
71344d93782SGreg Clayton                 {
71444d93782SGreg Clayton                     line = lines.CopyList();
71544d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
71644d93782SGreg Clayton                 }
717f0066ad0SGreg Clayton             }
71844d93782SGreg Clayton             else
71944d93782SGreg Clayton             {
72044d93782SGreg Clayton                 m_done = true;
72144d93782SGreg Clayton             }
72244d93782SGreg Clayton         }
72344d93782SGreg Clayton         else
72444d93782SGreg Clayton         {
725f0066ad0SGreg Clayton             if (GetLine(line, interrupted))
72644d93782SGreg Clayton             {
727e30f11d9SKate Stone                 if (interrupted)
728e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
729e30f11d9SKate Stone                 else
73044d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
73144d93782SGreg Clayton             }
73244d93782SGreg Clayton             else
73344d93782SGreg Clayton             {
73444d93782SGreg Clayton                 m_done = true;
73544d93782SGreg Clayton             }
73644d93782SGreg Clayton         }
73744d93782SGreg Clayton     }
73844d93782SGreg Clayton }
73944d93782SGreg Clayton 
74044d93782SGreg Clayton void
741e68f5d6bSGreg Clayton IOHandlerEditline::Cancel ()
742e68f5d6bSGreg Clayton {
743cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
744e68f5d6bSGreg Clayton     if (m_editline_ap)
7454446487dSPavel Labath         m_editline_ap->Cancel ();
746cacde7dfSTodd Fiala #endif
747e68f5d6bSGreg Clayton }
748e68f5d6bSGreg Clayton 
749f0066ad0SGreg Clayton bool
75044d93782SGreg Clayton IOHandlerEditline::Interrupt ()
75144d93782SGreg Clayton {
752f0066ad0SGreg Clayton     // Let the delgate handle it first
753f0066ad0SGreg Clayton     if (m_delegate.IOHandlerInterrupt(*this))
754f0066ad0SGreg Clayton         return true;
755f0066ad0SGreg Clayton 
756cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
75744d93782SGreg Clayton     if (m_editline_ap)
758f0066ad0SGreg Clayton         return m_editline_ap->Interrupt();
759cacde7dfSTodd Fiala #endif
760f0066ad0SGreg Clayton     return false;
76144d93782SGreg Clayton }
76244d93782SGreg Clayton 
76344d93782SGreg Clayton void
76444d93782SGreg Clayton IOHandlerEditline::GotEOF()
76544d93782SGreg Clayton {
766cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
76744d93782SGreg Clayton     if (m_editline_ap)
76844d93782SGreg Clayton         m_editline_ap->Interrupt();
769cacde7dfSTodd Fiala #endif
77044d93782SGreg Clayton }
77144d93782SGreg Clayton 
7724446487dSPavel Labath void
7734446487dSPavel Labath IOHandlerEditline::PrintAsync (Stream *stream, const char *s, size_t len)
7744446487dSPavel Labath {
7754446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
7764446487dSPavel Labath     if (m_editline_ap)
7774446487dSPavel Labath         m_editline_ap->PrintAsync(stream, s, len);
7784446487dSPavel Labath     else
7794446487dSPavel Labath #endif
780fab31220STed Woodward     {
781fab31220STed Woodward         const char *prompt = GetPrompt();
782fab31220STed Woodward #ifdef _MSC_VER
783fab31220STed Woodward         if (prompt)
784fab31220STed Woodward         {
785fab31220STed Woodward             // Back up over previous prompt using Windows API
786fab31220STed Woodward             CONSOLE_SCREEN_BUFFER_INFO screen_buffer_info;
787fab31220STed Woodward             HANDLE console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
788fab31220STed Woodward             GetConsoleScreenBufferInfo(console_handle, &screen_buffer_info);
789fab31220STed Woodward             COORD coord = screen_buffer_info.dwCursorPosition;
790fab31220STed Woodward             coord.X -= strlen(prompt);
791fab31220STed Woodward             if (coord.X < 0)
792fab31220STed Woodward                 coord.X = 0;
793fab31220STed Woodward             SetConsoleCursorPosition(console_handle, coord);
794fab31220STed Woodward         }
795fab31220STed Woodward #endif
7964446487dSPavel Labath         IOHandler::PrintAsync(stream, s, len);
797fab31220STed Woodward         if (prompt)
798fab31220STed Woodward             IOHandler::PrintAsync(GetOutputStreamFile().get(), prompt, strlen(prompt));
799fab31220STed Woodward     }
8004446487dSPavel Labath }
8014446487dSPavel Labath 
802914b8d98SDeepak Panickal // we may want curses to be disabled for some builds
803914b8d98SDeepak Panickal // for instance, windows
804914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
805914b8d98SDeepak Panickal 
80644d93782SGreg Clayton #define KEY_RETURN   10
80744d93782SGreg Clayton #define KEY_ESCAPE  27
80844d93782SGreg Clayton 
80944d93782SGreg Clayton namespace curses
81044d93782SGreg Clayton {
81144d93782SGreg Clayton     class Menu;
81244d93782SGreg Clayton     class MenuDelegate;
81344d93782SGreg Clayton     class Window;
81444d93782SGreg Clayton     class WindowDelegate;
81544d93782SGreg Clayton     typedef std::shared_ptr<Menu> MenuSP;
81644d93782SGreg Clayton     typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
81744d93782SGreg Clayton     typedef std::shared_ptr<Window> WindowSP;
81844d93782SGreg Clayton     typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
81944d93782SGreg Clayton     typedef std::vector<MenuSP> Menus;
82044d93782SGreg Clayton     typedef std::vector<WindowSP> Windows;
82144d93782SGreg Clayton     typedef std::vector<WindowDelegateSP> WindowDelegates;
82244d93782SGreg Clayton 
82344d93782SGreg Clayton #if 0
82444d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
82544d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
82644d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
82744d93782SGreg Clayton #endif
828315b6884SEugene Zelenko 
82944d93782SGreg Clayton     struct Point
83044d93782SGreg Clayton     {
83144d93782SGreg Clayton         int x;
83244d93782SGreg Clayton         int y;
83344d93782SGreg Clayton 
83444d93782SGreg Clayton         Point (int _x = 0, int _y = 0) :
83544d93782SGreg Clayton             x(_x),
83644d93782SGreg Clayton             y(_y)
83744d93782SGreg Clayton         {
83844d93782SGreg Clayton         }
83944d93782SGreg Clayton 
84044d93782SGreg Clayton         void
84144d93782SGreg Clayton         Clear ()
84244d93782SGreg Clayton         {
84344d93782SGreg Clayton             x = 0;
84444d93782SGreg Clayton             y = 0;
84544d93782SGreg Clayton         }
84644d93782SGreg Clayton 
84744d93782SGreg Clayton         Point &
84844d93782SGreg Clayton         operator += (const Point &rhs)
84944d93782SGreg Clayton         {
85044d93782SGreg Clayton             x += rhs.x;
85144d93782SGreg Clayton             y += rhs.y;
85244d93782SGreg Clayton             return *this;
85344d93782SGreg Clayton         }
85444d93782SGreg Clayton 
85544d93782SGreg Clayton         void
85644d93782SGreg Clayton         Dump ()
85744d93782SGreg Clayton         {
85844d93782SGreg Clayton             printf ("(x=%i, y=%i)\n", x, y);
85944d93782SGreg Clayton         }
86044d93782SGreg Clayton     };
86144d93782SGreg Clayton 
86244d93782SGreg Clayton     bool operator == (const Point &lhs, const Point &rhs)
86344d93782SGreg Clayton     {
86444d93782SGreg Clayton         return lhs.x == rhs.x && lhs.y == rhs.y;
86544d93782SGreg Clayton     }
866315b6884SEugene Zelenko 
86744d93782SGreg Clayton     bool operator != (const Point &lhs, const Point &rhs)
86844d93782SGreg Clayton     {
86944d93782SGreg Clayton         return lhs.x != rhs.x || lhs.y != rhs.y;
87044d93782SGreg Clayton     }
87144d93782SGreg Clayton 
87244d93782SGreg Clayton     struct Size
87344d93782SGreg Clayton     {
87444d93782SGreg Clayton         int width;
87544d93782SGreg Clayton         int height;
87644d93782SGreg Clayton         Size (int w = 0, int h = 0) :
87744d93782SGreg Clayton             width (w),
87844d93782SGreg Clayton             height (h)
87944d93782SGreg Clayton         {
88044d93782SGreg Clayton         }
88144d93782SGreg Clayton 
88244d93782SGreg Clayton         void
88344d93782SGreg Clayton         Clear ()
88444d93782SGreg Clayton         {
88544d93782SGreg Clayton             width = 0;
88644d93782SGreg Clayton             height = 0;
88744d93782SGreg Clayton         }
88844d93782SGreg Clayton 
88944d93782SGreg Clayton         void
89044d93782SGreg Clayton         Dump ()
89144d93782SGreg Clayton         {
89244d93782SGreg Clayton             printf ("(w=%i, h=%i)\n", width, height);
89344d93782SGreg Clayton         }
89444d93782SGreg Clayton     };
89544d93782SGreg Clayton 
89644d93782SGreg Clayton     bool operator == (const Size &lhs, const Size &rhs)
89744d93782SGreg Clayton     {
89844d93782SGreg Clayton         return lhs.width == rhs.width && lhs.height == rhs.height;
89944d93782SGreg Clayton     }
900315b6884SEugene Zelenko 
90144d93782SGreg Clayton     bool operator != (const Size &lhs, const Size &rhs)
90244d93782SGreg Clayton     {
90344d93782SGreg Clayton         return lhs.width != rhs.width || lhs.height != rhs.height;
90444d93782SGreg Clayton     }
90544d93782SGreg Clayton 
90644d93782SGreg Clayton     struct Rect
90744d93782SGreg Clayton     {
90844d93782SGreg Clayton         Point origin;
90944d93782SGreg Clayton         Size size;
91044d93782SGreg Clayton 
91144d93782SGreg Clayton         Rect () :
91244d93782SGreg Clayton             origin(),
91344d93782SGreg Clayton             size()
91444d93782SGreg Clayton         {
91544d93782SGreg Clayton         }
91644d93782SGreg Clayton 
91744d93782SGreg Clayton         Rect (const Point &p, const Size &s) :
91844d93782SGreg Clayton             origin (p),
91944d93782SGreg Clayton             size (s)
92044d93782SGreg Clayton         {
92144d93782SGreg Clayton         }
92244d93782SGreg Clayton 
92344d93782SGreg Clayton         void
92444d93782SGreg Clayton         Clear ()
92544d93782SGreg Clayton         {
92644d93782SGreg Clayton             origin.Clear();
92744d93782SGreg Clayton             size.Clear();
92844d93782SGreg Clayton         }
92944d93782SGreg Clayton 
93044d93782SGreg Clayton         void
93144d93782SGreg Clayton         Dump ()
93244d93782SGreg Clayton         {
93344d93782SGreg Clayton             printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height);
93444d93782SGreg Clayton         }
93544d93782SGreg Clayton 
93644d93782SGreg Clayton         void
93744d93782SGreg Clayton         Inset (int w, int h)
93844d93782SGreg Clayton         {
93944d93782SGreg Clayton             if (size.width > w*2)
94044d93782SGreg Clayton                 size.width -= w*2;
94144d93782SGreg Clayton             origin.x += w;
94244d93782SGreg Clayton 
94344d93782SGreg Clayton             if (size.height > h*2)
94444d93782SGreg Clayton                 size.height -= h*2;
94544d93782SGreg Clayton             origin.y += h;
94644d93782SGreg Clayton         }
947315b6884SEugene Zelenko 
94844d93782SGreg Clayton         // Return a status bar rectangle which is the last line of
94944d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
95044d93782SGreg Clayton         // include the status bar area.
95144d93782SGreg Clayton         Rect
95244d93782SGreg Clayton         MakeStatusBar ()
95344d93782SGreg Clayton         {
95444d93782SGreg Clayton             Rect status_bar;
95544d93782SGreg Clayton             if (size.height > 1)
95644d93782SGreg Clayton             {
95744d93782SGreg Clayton                 status_bar.origin.x = origin.x;
95844d93782SGreg Clayton                 status_bar.origin.y = size.height;
95944d93782SGreg Clayton                 status_bar.size.width = size.width;
96044d93782SGreg Clayton                 status_bar.size.height = 1;
96144d93782SGreg Clayton                 --size.height;
96244d93782SGreg Clayton             }
96344d93782SGreg Clayton             return status_bar;
96444d93782SGreg Clayton         }
96544d93782SGreg Clayton 
96644d93782SGreg Clayton         // Return a menubar rectangle which is the first line of
96744d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
96844d93782SGreg Clayton         // include the menubar area.
96944d93782SGreg Clayton         Rect
97044d93782SGreg Clayton         MakeMenuBar ()
97144d93782SGreg Clayton         {
97244d93782SGreg Clayton             Rect menubar;
97344d93782SGreg Clayton             if (size.height > 1)
97444d93782SGreg Clayton             {
97544d93782SGreg Clayton                 menubar.origin.x = origin.x;
97644d93782SGreg Clayton                 menubar.origin.y = origin.y;
97744d93782SGreg Clayton                 menubar.size.width = size.width;
97844d93782SGreg Clayton                 menubar.size.height = 1;
97944d93782SGreg Clayton                 ++origin.y;
98044d93782SGreg Clayton                 --size.height;
98144d93782SGreg Clayton             }
98244d93782SGreg Clayton             return menubar;
98344d93782SGreg Clayton         }
98444d93782SGreg Clayton 
98544d93782SGreg Clayton         void
98644d93782SGreg Clayton         HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const
98744d93782SGreg Clayton         {
98844d93782SGreg Clayton             float top_height = top_percentage * size.height;
98944d93782SGreg Clayton             HorizontalSplit (top_height, top, bottom);
99044d93782SGreg Clayton         }
99144d93782SGreg Clayton 
99244d93782SGreg Clayton         void
99344d93782SGreg Clayton         HorizontalSplit (int top_height, Rect &top, Rect &bottom) const
99444d93782SGreg Clayton         {
99544d93782SGreg Clayton             top = *this;
99644d93782SGreg Clayton             if (top_height < size.height)
99744d93782SGreg Clayton             {
99844d93782SGreg Clayton                 top.size.height = top_height;
99944d93782SGreg Clayton                 bottom.origin.x = origin.x;
100044d93782SGreg Clayton                 bottom.origin.y = origin.y + top.size.height;
100144d93782SGreg Clayton                 bottom.size.width = size.width;
100244d93782SGreg Clayton                 bottom.size.height = size.height - top.size.height;
100344d93782SGreg Clayton             }
100444d93782SGreg Clayton             else
100544d93782SGreg Clayton             {
100644d93782SGreg Clayton                 bottom.Clear();
100744d93782SGreg Clayton             }
100844d93782SGreg Clayton         }
100944d93782SGreg Clayton 
101044d93782SGreg Clayton         void
101144d93782SGreg Clayton         VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const
101244d93782SGreg Clayton         {
101344d93782SGreg Clayton             float left_width = left_percentage * size.width;
101444d93782SGreg Clayton             VerticalSplit (left_width, left, right);
101544d93782SGreg Clayton         }
101644d93782SGreg Clayton 
101744d93782SGreg Clayton         void
101844d93782SGreg Clayton         VerticalSplit (int left_width, Rect &left, Rect &right) const
101944d93782SGreg Clayton         {
102044d93782SGreg Clayton             left = *this;
102144d93782SGreg Clayton             if (left_width < size.width)
102244d93782SGreg Clayton             {
102344d93782SGreg Clayton                 left.size.width = left_width;
102444d93782SGreg Clayton                 right.origin.x = origin.x + left.size.width;
102544d93782SGreg Clayton                 right.origin.y = origin.y;
102644d93782SGreg Clayton                 right.size.width = size.width - left.size.width;
102744d93782SGreg Clayton                 right.size.height = size.height;
102844d93782SGreg Clayton             }
102944d93782SGreg Clayton             else
103044d93782SGreg Clayton             {
103144d93782SGreg Clayton                 right.Clear();
103244d93782SGreg Clayton             }
103344d93782SGreg Clayton         }
103444d93782SGreg Clayton     };
103544d93782SGreg Clayton 
103644d93782SGreg Clayton     bool operator == (const Rect &lhs, const Rect &rhs)
103744d93782SGreg Clayton     {
103844d93782SGreg Clayton         return lhs.origin == rhs.origin && lhs.size == rhs.size;
103944d93782SGreg Clayton     }
1040315b6884SEugene Zelenko 
104144d93782SGreg Clayton     bool operator != (const Rect &lhs, const Rect &rhs)
104244d93782SGreg Clayton     {
104344d93782SGreg Clayton         return lhs.origin != rhs.origin || lhs.size != rhs.size;
104444d93782SGreg Clayton     }
104544d93782SGreg Clayton 
104644d93782SGreg Clayton     enum HandleCharResult
104744d93782SGreg Clayton     {
104844d93782SGreg Clayton         eKeyNotHandled      = 0,
104944d93782SGreg Clayton         eKeyHandled         = 1,
105044d93782SGreg Clayton         eQuitApplication    = 2
105144d93782SGreg Clayton     };
105244d93782SGreg Clayton 
105344d93782SGreg Clayton     enum class MenuActionResult
105444d93782SGreg Clayton     {
105544d93782SGreg Clayton         Handled,
105644d93782SGreg Clayton         NotHandled,
105744d93782SGreg Clayton         Quit    // Exit all menus and quit
105844d93782SGreg Clayton     };
105944d93782SGreg Clayton 
106044d93782SGreg Clayton     struct KeyHelp
106144d93782SGreg Clayton     {
106244d93782SGreg Clayton         int ch;
106344d93782SGreg Clayton         const char *description;
106444d93782SGreg Clayton     };
106544d93782SGreg Clayton 
106644d93782SGreg Clayton     class WindowDelegate
106744d93782SGreg Clayton     {
106844d93782SGreg Clayton     public:
106944d93782SGreg Clayton         virtual
1070315b6884SEugene Zelenko         ~WindowDelegate() = default;
107144d93782SGreg Clayton 
107244d93782SGreg Clayton         virtual bool
107344d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force)
107444d93782SGreg Clayton         {
107544d93782SGreg Clayton             return false; // Drawing not handled
107644d93782SGreg Clayton         }
107744d93782SGreg Clayton 
107844d93782SGreg Clayton         virtual HandleCharResult
107944d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key)
108044d93782SGreg Clayton         {
108144d93782SGreg Clayton             return eKeyNotHandled;
108244d93782SGreg Clayton         }
108344d93782SGreg Clayton 
108444d93782SGreg Clayton         virtual const char *
108544d93782SGreg Clayton         WindowDelegateGetHelpText ()
108644d93782SGreg Clayton         {
1087c5dac77aSEugene Zelenko             return nullptr;
108844d93782SGreg Clayton         }
108944d93782SGreg Clayton 
109044d93782SGreg Clayton         virtual KeyHelp *
109144d93782SGreg Clayton         WindowDelegateGetKeyHelp ()
109244d93782SGreg Clayton         {
1093c5dac77aSEugene Zelenko             return nullptr;
109444d93782SGreg Clayton         }
109544d93782SGreg Clayton     };
109644d93782SGreg Clayton 
109744d93782SGreg Clayton     class HelpDialogDelegate :
109844d93782SGreg Clayton         public WindowDelegate
109944d93782SGreg Clayton     {
110044d93782SGreg Clayton     public:
110144d93782SGreg Clayton         HelpDialogDelegate (const char *text, KeyHelp *key_help_array);
110244d93782SGreg Clayton 
1103bd5ae6b4SGreg Clayton         ~HelpDialogDelegate() override;
110444d93782SGreg Clayton 
1105bd5ae6b4SGreg Clayton         bool
1106bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
110744d93782SGreg Clayton 
1108bd5ae6b4SGreg Clayton         HandleCharResult
1109bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
111044d93782SGreg Clayton 
111144d93782SGreg Clayton         size_t
111244d93782SGreg Clayton         GetNumLines() const
111344d93782SGreg Clayton         {
111444d93782SGreg Clayton             return m_text.GetSize();
111544d93782SGreg Clayton         }
111644d93782SGreg Clayton 
111744d93782SGreg Clayton         size_t
111844d93782SGreg Clayton         GetMaxLineLength () const
111944d93782SGreg Clayton         {
112044d93782SGreg Clayton             return m_text.GetMaxStringLength();
112144d93782SGreg Clayton         }
112244d93782SGreg Clayton 
112344d93782SGreg Clayton     protected:
112444d93782SGreg Clayton         StringList m_text;
112544d93782SGreg Clayton         int m_first_visible_line;
112644d93782SGreg Clayton     };
112744d93782SGreg Clayton 
112844d93782SGreg Clayton     class Window
112944d93782SGreg Clayton     {
113044d93782SGreg Clayton     public:
113144d93782SGreg Clayton         Window (const char *name) :
113244d93782SGreg Clayton             m_name (name),
1133c5dac77aSEugene Zelenko             m_window(nullptr),
1134c5dac77aSEugene Zelenko             m_panel(nullptr),
1135c5dac77aSEugene Zelenko             m_parent(nullptr),
113644d93782SGreg Clayton             m_subwindows (),
113744d93782SGreg Clayton             m_delegate_sp (),
113844d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
113944d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
114044d93782SGreg Clayton             m_delete (false),
114144d93782SGreg Clayton             m_needs_update (true),
114244d93782SGreg Clayton             m_can_activate (true),
114344d93782SGreg Clayton             m_is_subwin (false)
114444d93782SGreg Clayton         {
114544d93782SGreg Clayton         }
114644d93782SGreg Clayton 
114744d93782SGreg Clayton         Window (const char *name, WINDOW *w, bool del = true) :
114844d93782SGreg Clayton             m_name (name),
1149c5dac77aSEugene Zelenko             m_window(nullptr),
1150c5dac77aSEugene Zelenko             m_panel(nullptr),
1151c5dac77aSEugene Zelenko             m_parent(nullptr),
115244d93782SGreg Clayton             m_subwindows (),
115344d93782SGreg Clayton             m_delegate_sp (),
115444d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
115544d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
115644d93782SGreg Clayton             m_delete (del),
115744d93782SGreg Clayton             m_needs_update (true),
115844d93782SGreg Clayton             m_can_activate (true),
115944d93782SGreg Clayton             m_is_subwin (false)
116044d93782SGreg Clayton         {
116144d93782SGreg Clayton             if (w)
116244d93782SGreg Clayton                 Reset(w);
116344d93782SGreg Clayton         }
116444d93782SGreg Clayton 
116544d93782SGreg Clayton         Window (const char *name, const Rect &bounds) :
116644d93782SGreg Clayton             m_name (name),
1167c5dac77aSEugene Zelenko             m_window(nullptr),
1168c5dac77aSEugene Zelenko             m_parent(nullptr),
116944d93782SGreg Clayton             m_subwindows (),
117044d93782SGreg Clayton             m_delegate_sp (),
117144d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
117244d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
117344d93782SGreg Clayton             m_delete (true),
117444d93782SGreg Clayton             m_needs_update (true),
117544d93782SGreg Clayton             m_can_activate (true),
117644d93782SGreg Clayton             m_is_subwin (false)
117744d93782SGreg Clayton         {
117844d93782SGreg Clayton             Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y));
117944d93782SGreg Clayton         }
118044d93782SGreg Clayton 
118144d93782SGreg Clayton         virtual
118244d93782SGreg Clayton         ~Window ()
118344d93782SGreg Clayton         {
118444d93782SGreg Clayton             RemoveSubWindows ();
118544d93782SGreg Clayton             Reset ();
118644d93782SGreg Clayton         }
118744d93782SGreg Clayton 
118844d93782SGreg Clayton         void
1189c5dac77aSEugene Zelenko         Reset(WINDOW *w = nullptr, bool del = true)
119044d93782SGreg Clayton         {
119144d93782SGreg Clayton             if (m_window == w)
119244d93782SGreg Clayton                 return;
119344d93782SGreg Clayton 
119444d93782SGreg Clayton             if (m_panel)
119544d93782SGreg Clayton             {
119644d93782SGreg Clayton                 ::del_panel (m_panel);
1197c5dac77aSEugene Zelenko                 m_panel = nullptr;
119844d93782SGreg Clayton             }
119944d93782SGreg Clayton             if (m_window && m_delete)
120044d93782SGreg Clayton             {
120144d93782SGreg Clayton                 ::delwin (m_window);
1202c5dac77aSEugene Zelenko                 m_window = nullptr;
120344d93782SGreg Clayton                 m_delete = false;
120444d93782SGreg Clayton             }
120544d93782SGreg Clayton             if (w)
120644d93782SGreg Clayton             {
120744d93782SGreg Clayton                 m_window = w;
120844d93782SGreg Clayton                 m_panel = ::new_panel (m_window);
120944d93782SGreg Clayton                 m_delete = del;
121044d93782SGreg Clayton             }
121144d93782SGreg Clayton         }
121244d93782SGreg Clayton 
121344d93782SGreg Clayton         void    AttributeOn (attr_t attr)   { ::wattron (m_window, attr); }
121444d93782SGreg Clayton         void    AttributeOff (attr_t attr)  { ::wattroff (m_window, attr); }
121544d93782SGreg Clayton         void    Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); }
121644d93782SGreg Clayton         void    Clear ()    { ::wclear (m_window); }
121744d93782SGreg Clayton         void    Erase ()    { ::werase (m_window); }
121844d93782SGreg Clayton         Rect    GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window
121944d93782SGreg Clayton         int     GetChar ()  { return ::wgetch (m_window); }
122044d93782SGreg Clayton         int     GetCursorX ()     { return getcurx (m_window); }
122144d93782SGreg Clayton         int     GetCursorY ()     { return getcury (m_window); }
122244d93782SGreg Clayton         Rect    GetFrame ()    { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system
122344d93782SGreg Clayton         Point   GetParentOrigin() { return Point (GetParentX(), GetParentY()); }
122444d93782SGreg Clayton         Size    GetSize()         { return Size (GetWidth(), GetHeight()); }
122544d93782SGreg Clayton         int     GetParentX ()     { return getparx (m_window); }
122644d93782SGreg Clayton         int     GetParentY ()     { return getpary (m_window); }
122744d93782SGreg Clayton         int     GetMaxX()   { return getmaxx (m_window); }
122844d93782SGreg Clayton         int     GetMaxY()   { return getmaxy (m_window); }
122944d93782SGreg Clayton         int     GetWidth()  { return GetMaxX(); }
123044d93782SGreg Clayton         int     GetHeight() { return GetMaxY(); }
123144d93782SGreg Clayton         void    MoveCursor (int x, int y) {  ::wmove (m_window, y, x); }
123244d93782SGreg Clayton         void    MoveWindow (int x, int y) {  MoveWindow(Point(x,y)); }
123344d93782SGreg Clayton         void    Resize (int w, int h) { ::wresize(m_window, h, w); }
123444d93782SGreg Clayton         void    Resize (const Size &size) { ::wresize(m_window, size.height, size.width); }
123544d93782SGreg Clayton         void    PutChar (int ch)    { ::waddch (m_window, ch); }
123644d93782SGreg Clayton         void    PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); }
123744d93782SGreg Clayton         void    Refresh ()  { ::wrefresh (m_window); }
123844d93782SGreg Clayton         void    DeferredRefresh ()
123944d93782SGreg Clayton         {
124044d93782SGreg Clayton             // We are using panels, so we don't need to call this...
124144d93782SGreg Clayton             //::wnoutrefresh(m_window);
124244d93782SGreg Clayton         }
124344d93782SGreg Clayton         void    SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); }
124444d93782SGreg Clayton         void    UnderlineOn ()  { AttributeOn(A_UNDERLINE); }
124544d93782SGreg Clayton         void    UnderlineOff () { AttributeOff(A_UNDERLINE); }
124644d93782SGreg Clayton 
124744d93782SGreg Clayton         void    PutCStringTruncated (const char *s, int right_pad)
124844d93782SGreg Clayton         {
124944d93782SGreg Clayton             int bytes_left = GetWidth() - GetCursorX();
125044d93782SGreg Clayton             if (bytes_left > right_pad)
125144d93782SGreg Clayton             {
125244d93782SGreg Clayton                 bytes_left -= right_pad;
125344d93782SGreg Clayton                 ::waddnstr (m_window, s, bytes_left);
125444d93782SGreg Clayton             }
125544d93782SGreg Clayton         }
125644d93782SGreg Clayton 
125744d93782SGreg Clayton         void
125844d93782SGreg Clayton         MoveWindow (const Point &origin)
125944d93782SGreg Clayton         {
126044d93782SGreg Clayton             const bool moving_window = origin != GetParentOrigin();
126144d93782SGreg Clayton             if (m_is_subwin && moving_window)
126244d93782SGreg Clayton             {
126344d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
126444d93782SGreg Clayton                 Size size = GetSize();
126544d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
126644d93782SGreg Clayton                                  size.height,
126744d93782SGreg Clayton                                  size.width,
126844d93782SGreg Clayton                                  origin.y,
126944d93782SGreg Clayton                                  origin.x), true);
127044d93782SGreg Clayton             }
127144d93782SGreg Clayton             else
127244d93782SGreg Clayton             {
127344d93782SGreg Clayton                 ::mvwin (m_window, origin.y, origin.x);
127444d93782SGreg Clayton             }
127544d93782SGreg Clayton         }
127644d93782SGreg Clayton 
127744d93782SGreg Clayton         void
127844d93782SGreg Clayton         SetBounds (const Rect &bounds)
127944d93782SGreg Clayton         {
128044d93782SGreg Clayton             const bool moving_window = bounds.origin != GetParentOrigin();
128144d93782SGreg Clayton             if (m_is_subwin && moving_window)
128244d93782SGreg Clayton             {
128344d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
128444d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
128544d93782SGreg Clayton                                  bounds.size.height,
128644d93782SGreg Clayton                                  bounds.size.width,
128744d93782SGreg Clayton                                  bounds.origin.y,
128844d93782SGreg Clayton                                  bounds.origin.x), true);
128944d93782SGreg Clayton             }
129044d93782SGreg Clayton             else
129144d93782SGreg Clayton             {
129244d93782SGreg Clayton                 if (moving_window)
129344d93782SGreg Clayton                     MoveWindow(bounds.origin);
129444d93782SGreg Clayton                 Resize (bounds.size);
129544d93782SGreg Clayton             }
129644d93782SGreg Clayton         }
129744d93782SGreg Clayton 
129844d93782SGreg Clayton         void
129944d93782SGreg Clayton         Printf (const char *format, ...)  __attribute__ ((format (printf, 2, 3)))
130044d93782SGreg Clayton         {
130144d93782SGreg Clayton             va_list args;
130244d93782SGreg Clayton             va_start (args, format);
130344d93782SGreg Clayton             vwprintw(m_window, format, args);
130444d93782SGreg Clayton             va_end (args);
130544d93782SGreg Clayton         }
130644d93782SGreg Clayton 
130744d93782SGreg Clayton         void
130844d93782SGreg Clayton         Touch ()
130944d93782SGreg Clayton         {
131044d93782SGreg Clayton             ::touchwin (m_window);
131144d93782SGreg Clayton             if (m_parent)
131244d93782SGreg Clayton                 m_parent->Touch();
131344d93782SGreg Clayton         }
131444d93782SGreg Clayton 
131544d93782SGreg Clayton         WindowSP
131644d93782SGreg Clayton         CreateSubWindow (const char *name, const Rect &bounds, bool make_active)
131744d93782SGreg Clayton         {
131844d93782SGreg Clayton             WindowSP subwindow_sp;
131944d93782SGreg Clayton             if (m_window)
132044d93782SGreg Clayton             {
132144d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::subwin (m_window,
132244d93782SGreg Clayton                                                               bounds.size.height,
132344d93782SGreg Clayton                                                               bounds.size.width,
132444d93782SGreg Clayton                                                               bounds.origin.y,
132544d93782SGreg Clayton                                                               bounds.origin.x), true));
132644d93782SGreg Clayton                 subwindow_sp->m_is_subwin = true;
132744d93782SGreg Clayton             }
132844d93782SGreg Clayton             else
132944d93782SGreg Clayton             {
133044d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height,
133144d93782SGreg Clayton                                                               bounds.size.width,
133244d93782SGreg Clayton                                                               bounds.origin.y,
133344d93782SGreg Clayton                                                               bounds.origin.x), true));
133444d93782SGreg Clayton                 subwindow_sp->m_is_subwin = false;
133544d93782SGreg Clayton             }
133644d93782SGreg Clayton             subwindow_sp->m_parent = this;
133744d93782SGreg Clayton             if (make_active)
133844d93782SGreg Clayton             {
133944d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
134044d93782SGreg Clayton                 m_curr_active_window_idx = m_subwindows.size();
134144d93782SGreg Clayton             }
134244d93782SGreg Clayton             m_subwindows.push_back(subwindow_sp);
134344d93782SGreg Clayton             ::top_panel (subwindow_sp->m_panel);
134444d93782SGreg Clayton             m_needs_update = true;
134544d93782SGreg Clayton             return subwindow_sp;
134644d93782SGreg Clayton         }
134744d93782SGreg Clayton 
134844d93782SGreg Clayton         bool
134944d93782SGreg Clayton         RemoveSubWindow (Window *window)
135044d93782SGreg Clayton         {
135144d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
135244d93782SGreg Clayton             size_t i = 0;
135344d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
135444d93782SGreg Clayton             {
135544d93782SGreg Clayton                 if ((*pos).get() == window)
135644d93782SGreg Clayton                 {
135744d93782SGreg Clayton                     if (m_prev_active_window_idx == i)
135844d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
135944d93782SGreg Clayton                     else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i)
136044d93782SGreg Clayton                         --m_prev_active_window_idx;
136144d93782SGreg Clayton 
136244d93782SGreg Clayton                     if (m_curr_active_window_idx == i)
136344d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
136444d93782SGreg Clayton                     else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i)
136544d93782SGreg Clayton                         --m_curr_active_window_idx;
136644d93782SGreg Clayton                     window->Erase();
136744d93782SGreg Clayton                     m_subwindows.erase(pos);
136844d93782SGreg Clayton                     m_needs_update = true;
136944d93782SGreg Clayton                     if (m_parent)
137044d93782SGreg Clayton                         m_parent->Touch();
137144d93782SGreg Clayton                     else
137244d93782SGreg Clayton                         ::touchwin (stdscr);
137344d93782SGreg Clayton                     return true;
137444d93782SGreg Clayton                 }
137544d93782SGreg Clayton             }
137644d93782SGreg Clayton             return false;
137744d93782SGreg Clayton         }
137844d93782SGreg Clayton 
137944d93782SGreg Clayton         WindowSP
138044d93782SGreg Clayton         FindSubWindow (const char *name)
138144d93782SGreg Clayton         {
138244d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
138344d93782SGreg Clayton             size_t i = 0;
138444d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
138544d93782SGreg Clayton             {
138644d93782SGreg Clayton                 if ((*pos)->m_name.compare(name) == 0)
138744d93782SGreg Clayton                     return *pos;
138844d93782SGreg Clayton             }
138944d93782SGreg Clayton             return WindowSP();
139044d93782SGreg Clayton         }
139144d93782SGreg Clayton 
139244d93782SGreg Clayton         void
139344d93782SGreg Clayton         RemoveSubWindows ()
139444d93782SGreg Clayton         {
139544d93782SGreg Clayton             m_curr_active_window_idx = UINT32_MAX;
139644d93782SGreg Clayton             m_prev_active_window_idx = UINT32_MAX;
139744d93782SGreg Clayton             for (Windows::iterator pos = m_subwindows.begin();
139844d93782SGreg Clayton                  pos != m_subwindows.end();
139944d93782SGreg Clayton                  pos = m_subwindows.erase(pos))
140044d93782SGreg Clayton             {
140144d93782SGreg Clayton                 (*pos)->Erase();
140244d93782SGreg Clayton             }
140344d93782SGreg Clayton             if (m_parent)
140444d93782SGreg Clayton                 m_parent->Touch();
140544d93782SGreg Clayton             else
140644d93782SGreg Clayton                 ::touchwin (stdscr);
140744d93782SGreg Clayton         }
140844d93782SGreg Clayton 
140944d93782SGreg Clayton         WINDOW *
141044d93782SGreg Clayton         get()
141144d93782SGreg Clayton         {
141244d93782SGreg Clayton             return m_window;
141344d93782SGreg Clayton         }
141444d93782SGreg Clayton 
141544d93782SGreg Clayton         operator WINDOW *()
141644d93782SGreg Clayton         {
141744d93782SGreg Clayton             return m_window;
141844d93782SGreg Clayton         }
141944d93782SGreg Clayton 
142044d93782SGreg Clayton         //----------------------------------------------------------------------
142144d93782SGreg Clayton         // Window drawing utilities
142244d93782SGreg Clayton         //----------------------------------------------------------------------
142344d93782SGreg Clayton         void
1424c5dac77aSEugene Zelenko         DrawTitleBox(const char *title, const char *bottom_message = nullptr)
142544d93782SGreg Clayton         {
142644d93782SGreg Clayton             attr_t attr = 0;
142744d93782SGreg Clayton             if (IsActive())
142844d93782SGreg Clayton                 attr = A_BOLD | COLOR_PAIR(2);
142944d93782SGreg Clayton             else
143044d93782SGreg Clayton                 attr = 0;
143144d93782SGreg Clayton             if (attr)
143244d93782SGreg Clayton                 AttributeOn(attr);
143344d93782SGreg Clayton 
143444d93782SGreg Clayton             Box();
143544d93782SGreg Clayton             MoveCursor(3, 0);
143644d93782SGreg Clayton 
143744d93782SGreg Clayton             if (title && title[0])
143844d93782SGreg Clayton             {
143944d93782SGreg Clayton                 PutChar ('<');
144044d93782SGreg Clayton                 PutCString (title);
144144d93782SGreg Clayton                 PutChar ('>');
144244d93782SGreg Clayton             }
144344d93782SGreg Clayton 
144444d93782SGreg Clayton             if (bottom_message && bottom_message[0])
144544d93782SGreg Clayton             {
144644d93782SGreg Clayton                 int bottom_message_length = strlen(bottom_message);
144744d93782SGreg Clayton                 int x = GetWidth() - 3 - (bottom_message_length + 2);
144844d93782SGreg Clayton 
144944d93782SGreg Clayton                 if (x > 0)
145044d93782SGreg Clayton                 {
145144d93782SGreg Clayton                     MoveCursor (x, GetHeight() - 1);
145244d93782SGreg Clayton                     PutChar ('[');
145344d93782SGreg Clayton                     PutCString(bottom_message);
145444d93782SGreg Clayton                     PutChar (']');
145544d93782SGreg Clayton                 }
145644d93782SGreg Clayton                 else
145744d93782SGreg Clayton                 {
145844d93782SGreg Clayton                     MoveCursor (1, GetHeight() - 1);
145944d93782SGreg Clayton                     PutChar ('[');
146044d93782SGreg Clayton                     PutCStringTruncated (bottom_message, 1);
146144d93782SGreg Clayton                 }
146244d93782SGreg Clayton             }
146344d93782SGreg Clayton             if (attr)
146444d93782SGreg Clayton                 AttributeOff(attr);
146544d93782SGreg Clayton         }
146644d93782SGreg Clayton 
146744d93782SGreg Clayton         virtual void
146844d93782SGreg Clayton         Draw (bool force)
146944d93782SGreg Clayton         {
147044d93782SGreg Clayton             if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force))
147144d93782SGreg Clayton                 return;
147244d93782SGreg Clayton 
147344d93782SGreg Clayton             for (auto &subwindow_sp : m_subwindows)
147444d93782SGreg Clayton                 subwindow_sp->Draw(force);
147544d93782SGreg Clayton         }
147644d93782SGreg Clayton 
147744d93782SGreg Clayton         bool
147844d93782SGreg Clayton         CreateHelpSubwindow ()
147944d93782SGreg Clayton         {
148044d93782SGreg Clayton             if (m_delegate_sp)
148144d93782SGreg Clayton             {
148244d93782SGreg Clayton                 const char *text = m_delegate_sp->WindowDelegateGetHelpText ();
148344d93782SGreg Clayton                 KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp ();
148444d93782SGreg Clayton                 if ((text && text[0]) || key_help)
148544d93782SGreg Clayton                 {
148644d93782SGreg Clayton                     std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help));
148744d93782SGreg Clayton                     const size_t num_lines = help_delegate_ap->GetNumLines();
148844d93782SGreg Clayton                     const size_t max_length = help_delegate_ap->GetMaxLineLength();
148944d93782SGreg Clayton                     Rect bounds = GetBounds();
149044d93782SGreg Clayton                     bounds.Inset(1, 1);
14913985c8c6SSaleem Abdulrasool                     if (max_length + 4 < static_cast<size_t>(bounds.size.width))
149244d93782SGreg Clayton                     {
149344d93782SGreg Clayton                         bounds.origin.x += (bounds.size.width - max_length + 4)/2;
149444d93782SGreg Clayton                         bounds.size.width = max_length + 4;
149544d93782SGreg Clayton                     }
149644d93782SGreg Clayton                     else
149744d93782SGreg Clayton                     {
149844d93782SGreg Clayton                         if (bounds.size.width > 100)
149944d93782SGreg Clayton                         {
150044d93782SGreg Clayton                             const int inset_w = bounds.size.width / 4;
150144d93782SGreg Clayton                             bounds.origin.x += inset_w;
150244d93782SGreg Clayton                             bounds.size.width -= 2*inset_w;
150344d93782SGreg Clayton                         }
150444d93782SGreg Clayton                     }
150544d93782SGreg Clayton 
15063985c8c6SSaleem Abdulrasool                     if (num_lines + 2 < static_cast<size_t>(bounds.size.height))
150744d93782SGreg Clayton                     {
150844d93782SGreg Clayton                         bounds.origin.y += (bounds.size.height - num_lines + 2)/2;
150944d93782SGreg Clayton                         bounds.size.height = num_lines + 2;
151044d93782SGreg Clayton                     }
151144d93782SGreg Clayton                     else
151244d93782SGreg Clayton                     {
151344d93782SGreg Clayton                         if (bounds.size.height > 100)
151444d93782SGreg Clayton                         {
151544d93782SGreg Clayton                             const int inset_h = bounds.size.height / 4;
151644d93782SGreg Clayton                             bounds.origin.y += inset_h;
151744d93782SGreg Clayton                             bounds.size.height -= 2*inset_h;
151844d93782SGreg Clayton                         }
151944d93782SGreg Clayton                     }
15205fdb09bbSGreg Clayton                     WindowSP help_window_sp;
15215fdb09bbSGreg Clayton                     Window *parent_window = GetParent();
15225fdb09bbSGreg Clayton                     if (parent_window)
15235fdb09bbSGreg Clayton                         help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
15245fdb09bbSGreg Clayton                     else
15255fdb09bbSGreg Clayton                         help_window_sp = CreateSubWindow("Help", bounds, true);
152644d93782SGreg Clayton                     help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release()));
152744d93782SGreg Clayton                     return true;
152844d93782SGreg Clayton                 }
152944d93782SGreg Clayton             }
153044d93782SGreg Clayton             return false;
153144d93782SGreg Clayton         }
153244d93782SGreg Clayton 
153344d93782SGreg Clayton         virtual HandleCharResult
153444d93782SGreg Clayton         HandleChar (int key)
153544d93782SGreg Clayton         {
153644d93782SGreg Clayton             // Always check the active window first
153744d93782SGreg Clayton             HandleCharResult result = eKeyNotHandled;
153844d93782SGreg Clayton             WindowSP active_window_sp = GetActiveWindow ();
153944d93782SGreg Clayton             if (active_window_sp)
154044d93782SGreg Clayton             {
154144d93782SGreg Clayton                 result = active_window_sp->HandleChar (key);
154244d93782SGreg Clayton                 if (result != eKeyNotHandled)
154344d93782SGreg Clayton                     return result;
154444d93782SGreg Clayton             }
154544d93782SGreg Clayton 
154644d93782SGreg Clayton             if (m_delegate_sp)
154744d93782SGreg Clayton             {
154844d93782SGreg Clayton                 result = m_delegate_sp->WindowDelegateHandleChar (*this, key);
154944d93782SGreg Clayton                 if (result != eKeyNotHandled)
155044d93782SGreg Clayton                     return result;
155144d93782SGreg Clayton             }
155244d93782SGreg Clayton 
155344d93782SGreg Clayton             // Then check for any windows that want any keys
155444d93782SGreg Clayton             // that weren't handled. This is typically only
155544d93782SGreg Clayton             // for a menubar.
155644d93782SGreg Clayton             // Make a copy of the subwindows in case any HandleChar()
155744d93782SGreg Clayton             // functions muck with the subwindows. If we don't do this,
155844d93782SGreg Clayton             // we can crash when iterating over the subwindows.
155944d93782SGreg Clayton             Windows subwindows (m_subwindows);
156044d93782SGreg Clayton             for (auto subwindow_sp : subwindows)
156144d93782SGreg Clayton             {
1562c5dac77aSEugene Zelenko                 if (!subwindow_sp->m_can_activate)
156344d93782SGreg Clayton                 {
156444d93782SGreg Clayton                     HandleCharResult result = subwindow_sp->HandleChar(key);
156544d93782SGreg Clayton                     if (result != eKeyNotHandled)
156644d93782SGreg Clayton                         return result;
156744d93782SGreg Clayton                 }
156844d93782SGreg Clayton             }
156944d93782SGreg Clayton 
157044d93782SGreg Clayton             return eKeyNotHandled;
157144d93782SGreg Clayton         }
157244d93782SGreg Clayton 
157344d93782SGreg Clayton         bool
157444d93782SGreg Clayton         SetActiveWindow (Window *window)
157544d93782SGreg Clayton         {
157644d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
157744d93782SGreg Clayton             for (size_t i = 0; i < num_subwindows; ++i)
157844d93782SGreg Clayton             {
157944d93782SGreg Clayton                 if (m_subwindows[i].get() == window)
158044d93782SGreg Clayton                 {
158144d93782SGreg Clayton                     m_prev_active_window_idx = m_curr_active_window_idx;
158244d93782SGreg Clayton                     ::top_panel (window->m_panel);
158344d93782SGreg Clayton                     m_curr_active_window_idx = i;
158444d93782SGreg Clayton                     return true;
158544d93782SGreg Clayton                 }
158644d93782SGreg Clayton             }
158744d93782SGreg Clayton             return false;
158844d93782SGreg Clayton         }
158944d93782SGreg Clayton 
159044d93782SGreg Clayton         WindowSP
159144d93782SGreg Clayton         GetActiveWindow ()
159244d93782SGreg Clayton         {
159344d93782SGreg Clayton             if (!m_subwindows.empty())
159444d93782SGreg Clayton             {
159544d93782SGreg Clayton                 if (m_curr_active_window_idx >= m_subwindows.size())
159644d93782SGreg Clayton                 {
159744d93782SGreg Clayton                     if (m_prev_active_window_idx < m_subwindows.size())
159844d93782SGreg Clayton                     {
159944d93782SGreg Clayton                         m_curr_active_window_idx = m_prev_active_window_idx;
160044d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
160144d93782SGreg Clayton                     }
160244d93782SGreg Clayton                     else if (IsActive())
160344d93782SGreg Clayton                     {
160444d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
160544d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
160644d93782SGreg Clayton 
160744d93782SGreg Clayton                         // Find first window that wants to be active if this window is active
160844d93782SGreg Clayton                         const size_t num_subwindows = m_subwindows.size();
160944d93782SGreg Clayton                         for (size_t i = 0; i < num_subwindows; ++i)
161044d93782SGreg Clayton                         {
161144d93782SGreg Clayton                             if (m_subwindows[i]->GetCanBeActive())
161244d93782SGreg Clayton                             {
161344d93782SGreg Clayton                                 m_curr_active_window_idx = i;
161444d93782SGreg Clayton                                 break;
161544d93782SGreg Clayton                             }
161644d93782SGreg Clayton                         }
161744d93782SGreg Clayton                     }
161844d93782SGreg Clayton                 }
161944d93782SGreg Clayton 
162044d93782SGreg Clayton                 if (m_curr_active_window_idx < m_subwindows.size())
162144d93782SGreg Clayton                     return m_subwindows[m_curr_active_window_idx];
162244d93782SGreg Clayton             }
162344d93782SGreg Clayton             return WindowSP();
162444d93782SGreg Clayton         }
162544d93782SGreg Clayton 
162644d93782SGreg Clayton         bool
162744d93782SGreg Clayton         GetCanBeActive () const
162844d93782SGreg Clayton         {
162944d93782SGreg Clayton             return m_can_activate;
163044d93782SGreg Clayton         }
163144d93782SGreg Clayton 
163244d93782SGreg Clayton         void
163344d93782SGreg Clayton         SetCanBeActive (bool b)
163444d93782SGreg Clayton         {
163544d93782SGreg Clayton             m_can_activate = b;
163644d93782SGreg Clayton         }
163744d93782SGreg Clayton 
163844d93782SGreg Clayton         const WindowDelegateSP &
163944d93782SGreg Clayton         GetDelegate () const
164044d93782SGreg Clayton         {
164144d93782SGreg Clayton             return m_delegate_sp;
164244d93782SGreg Clayton         }
164344d93782SGreg Clayton 
164444d93782SGreg Clayton         void
164544d93782SGreg Clayton         SetDelegate (const WindowDelegateSP &delegate_sp)
164644d93782SGreg Clayton         {
164744d93782SGreg Clayton             m_delegate_sp = delegate_sp;
164844d93782SGreg Clayton         }
164944d93782SGreg Clayton 
165044d93782SGreg Clayton         Window *
165144d93782SGreg Clayton         GetParent () const
165244d93782SGreg Clayton         {
165344d93782SGreg Clayton             return m_parent;
165444d93782SGreg Clayton         }
165544d93782SGreg Clayton 
165644d93782SGreg Clayton         bool
165744d93782SGreg Clayton         IsActive () const
165844d93782SGreg Clayton         {
165944d93782SGreg Clayton             if (m_parent)
166044d93782SGreg Clayton                 return m_parent->GetActiveWindow().get() == this;
166144d93782SGreg Clayton             else
166244d93782SGreg Clayton                 return true; // Top level window is always active
166344d93782SGreg Clayton         }
166444d93782SGreg Clayton 
166544d93782SGreg Clayton         void
166644d93782SGreg Clayton         SelectNextWindowAsActive ()
166744d93782SGreg Clayton         {
166844d93782SGreg Clayton             // Move active focus to next window
166944d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
167044d93782SGreg Clayton             if (m_curr_active_window_idx == UINT32_MAX)
167144d93782SGreg Clayton             {
167244d93782SGreg Clayton                 uint32_t idx = 0;
167344d93782SGreg Clayton                 for (auto subwindow_sp : m_subwindows)
167444d93782SGreg Clayton                 {
167544d93782SGreg Clayton                     if (subwindow_sp->GetCanBeActive())
167644d93782SGreg Clayton                     {
167744d93782SGreg Clayton                         m_curr_active_window_idx = idx;
167844d93782SGreg Clayton                         break;
167944d93782SGreg Clayton                     }
168044d93782SGreg Clayton                     ++idx;
168144d93782SGreg Clayton                 }
168244d93782SGreg Clayton             }
168344d93782SGreg Clayton             else if (m_curr_active_window_idx + 1 < num_subwindows)
168444d93782SGreg Clayton             {
168544d93782SGreg Clayton                 bool handled = false;
168644d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
168744d93782SGreg Clayton                 for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx)
168844d93782SGreg Clayton                 {
168944d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
169044d93782SGreg Clayton                     {
169144d93782SGreg Clayton                         m_curr_active_window_idx = idx;
169244d93782SGreg Clayton                         handled = true;
169344d93782SGreg Clayton                         break;
169444d93782SGreg Clayton                     }
169544d93782SGreg Clayton                 }
169644d93782SGreg Clayton                 if (!handled)
169744d93782SGreg Clayton                 {
169844d93782SGreg Clayton                     for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx)
169944d93782SGreg Clayton                     {
170044d93782SGreg Clayton                         if (m_subwindows[idx]->GetCanBeActive())
170144d93782SGreg Clayton                         {
170244d93782SGreg Clayton                             m_curr_active_window_idx = idx;
170344d93782SGreg Clayton                             break;
170444d93782SGreg Clayton                         }
170544d93782SGreg Clayton                     }
170644d93782SGreg Clayton                 }
170744d93782SGreg Clayton             }
170844d93782SGreg Clayton             else
170944d93782SGreg Clayton             {
171044d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
171144d93782SGreg Clayton                 for (size_t idx=0; idx<num_subwindows; ++idx)
171244d93782SGreg Clayton                 {
171344d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
171444d93782SGreg Clayton                     {
171544d93782SGreg Clayton                         m_curr_active_window_idx = idx;
171644d93782SGreg Clayton                         break;
171744d93782SGreg Clayton                     }
171844d93782SGreg Clayton                 }
171944d93782SGreg Clayton             }
172044d93782SGreg Clayton         }
172144d93782SGreg Clayton 
172244d93782SGreg Clayton         const char *
172344d93782SGreg Clayton         GetName () const
172444d93782SGreg Clayton         {
172544d93782SGreg Clayton             return m_name.c_str();
172644d93782SGreg Clayton         }
1727315b6884SEugene Zelenko 
172844d93782SGreg Clayton     protected:
172944d93782SGreg Clayton         std::string m_name;
173044d93782SGreg Clayton         WINDOW *m_window;
173144d93782SGreg Clayton         PANEL *m_panel;
173244d93782SGreg Clayton         Window *m_parent;
173344d93782SGreg Clayton         Windows m_subwindows;
173444d93782SGreg Clayton         WindowDelegateSP m_delegate_sp;
173544d93782SGreg Clayton         uint32_t m_curr_active_window_idx;
173644d93782SGreg Clayton         uint32_t m_prev_active_window_idx;
173744d93782SGreg Clayton         bool m_delete;
173844d93782SGreg Clayton         bool m_needs_update;
173944d93782SGreg Clayton         bool m_can_activate;
174044d93782SGreg Clayton         bool m_is_subwin;
174144d93782SGreg Clayton 
174244d93782SGreg Clayton     private:
174344d93782SGreg Clayton         DISALLOW_COPY_AND_ASSIGN(Window);
174444d93782SGreg Clayton     };
174544d93782SGreg Clayton 
174644d93782SGreg Clayton     class MenuDelegate
174744d93782SGreg Clayton     {
174844d93782SGreg Clayton     public:
1749315b6884SEugene Zelenko         virtual ~MenuDelegate() = default;
175044d93782SGreg Clayton 
175144d93782SGreg Clayton         virtual MenuActionResult
175244d93782SGreg Clayton         MenuDelegateAction (Menu &menu) = 0;
175344d93782SGreg Clayton     };
175444d93782SGreg Clayton 
175544d93782SGreg Clayton     class Menu : public WindowDelegate
175644d93782SGreg Clayton     {
175744d93782SGreg Clayton     public:
175844d93782SGreg Clayton         enum class Type
175944d93782SGreg Clayton         {
176044d93782SGreg Clayton             Invalid,
176144d93782SGreg Clayton             Bar,
176244d93782SGreg Clayton             Item,
176344d93782SGreg Clayton             Separator
176444d93782SGreg Clayton         };
176544d93782SGreg Clayton 
176644d93782SGreg Clayton         // Menubar or separator constructor
176744d93782SGreg Clayton         Menu (Type type);
176844d93782SGreg Clayton 
176944d93782SGreg Clayton         // Menuitem constructor
177044d93782SGreg Clayton         Menu (const char *name,
177144d93782SGreg Clayton               const char *key_name,
177244d93782SGreg Clayton               int key_value,
177344d93782SGreg Clayton               uint64_t identifier);
177444d93782SGreg Clayton 
1775315b6884SEugene Zelenko         ~Menu() override = default;
177644d93782SGreg Clayton 
177744d93782SGreg Clayton         const MenuDelegateSP &
177844d93782SGreg Clayton         GetDelegate () const
177944d93782SGreg Clayton         {
178044d93782SGreg Clayton             return m_delegate_sp;
178144d93782SGreg Clayton         }
178244d93782SGreg Clayton 
178344d93782SGreg Clayton         void
178444d93782SGreg Clayton         SetDelegate (const MenuDelegateSP &delegate_sp)
178544d93782SGreg Clayton         {
178644d93782SGreg Clayton             m_delegate_sp = delegate_sp;
178744d93782SGreg Clayton         }
178844d93782SGreg Clayton 
178944d93782SGreg Clayton         void
179044d93782SGreg Clayton         RecalculateNameLengths();
179144d93782SGreg Clayton 
179244d93782SGreg Clayton         void
179344d93782SGreg Clayton         AddSubmenu (const MenuSP &menu_sp);
179444d93782SGreg Clayton 
179544d93782SGreg Clayton         int
179644d93782SGreg Clayton         DrawAndRunMenu (Window &window);
179744d93782SGreg Clayton 
179844d93782SGreg Clayton         void
179944d93782SGreg Clayton         DrawMenuTitle (Window &window, bool highlight);
180044d93782SGreg Clayton 
1801bd5ae6b4SGreg Clayton         bool
1802bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
180344d93782SGreg Clayton 
1804bd5ae6b4SGreg Clayton         HandleCharResult
1805bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
180644d93782SGreg Clayton 
180744d93782SGreg Clayton         MenuActionResult
180844d93782SGreg Clayton         ActionPrivate (Menu &menu)
180944d93782SGreg Clayton         {
181044d93782SGreg Clayton             MenuActionResult result = MenuActionResult::NotHandled;
181144d93782SGreg Clayton             if (m_delegate_sp)
181244d93782SGreg Clayton             {
181344d93782SGreg Clayton                 result = m_delegate_sp->MenuDelegateAction (menu);
181444d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
181544d93782SGreg Clayton                     return result;
181644d93782SGreg Clayton             }
181744d93782SGreg Clayton             else if (m_parent)
181844d93782SGreg Clayton             {
181944d93782SGreg Clayton                 result = m_parent->ActionPrivate(menu);
182044d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
182144d93782SGreg Clayton                     return result;
182244d93782SGreg Clayton             }
182344d93782SGreg Clayton             return m_canned_result;
182444d93782SGreg Clayton         }
182544d93782SGreg Clayton 
182644d93782SGreg Clayton         MenuActionResult
182744d93782SGreg Clayton         Action ()
182844d93782SGreg Clayton         {
182944d93782SGreg Clayton             // Call the recursive action so it can try to handle it
183044d93782SGreg Clayton             // with the menu delegate, and if not, try our parent menu
183144d93782SGreg Clayton             return ActionPrivate (*this);
183244d93782SGreg Clayton         }
183344d93782SGreg Clayton 
183444d93782SGreg Clayton         void
183544d93782SGreg Clayton         SetCannedResult (MenuActionResult result)
183644d93782SGreg Clayton         {
183744d93782SGreg Clayton             m_canned_result = result;
183844d93782SGreg Clayton         }
183944d93782SGreg Clayton 
184044d93782SGreg Clayton         Menus &
184144d93782SGreg Clayton         GetSubmenus()
184244d93782SGreg Clayton         {
184344d93782SGreg Clayton             return m_submenus;
184444d93782SGreg Clayton         }
184544d93782SGreg Clayton 
184644d93782SGreg Clayton         const Menus &
184744d93782SGreg Clayton         GetSubmenus() const
184844d93782SGreg Clayton         {
184944d93782SGreg Clayton             return m_submenus;
185044d93782SGreg Clayton         }
185144d93782SGreg Clayton 
185244d93782SGreg Clayton         int
185344d93782SGreg Clayton         GetSelectedSubmenuIndex () const
185444d93782SGreg Clayton         {
185544d93782SGreg Clayton             return m_selected;
185644d93782SGreg Clayton         }
185744d93782SGreg Clayton 
185844d93782SGreg Clayton         void
185944d93782SGreg Clayton         SetSelectedSubmenuIndex (int idx)
186044d93782SGreg Clayton         {
186144d93782SGreg Clayton             m_selected = idx;
186244d93782SGreg Clayton         }
186344d93782SGreg Clayton 
186444d93782SGreg Clayton         Type
186544d93782SGreg Clayton         GetType () const
186644d93782SGreg Clayton         {
186744d93782SGreg Clayton             return m_type;
186844d93782SGreg Clayton         }
186944d93782SGreg Clayton 
187044d93782SGreg Clayton         int
187144d93782SGreg Clayton         GetStartingColumn() const
187244d93782SGreg Clayton         {
187344d93782SGreg Clayton             return m_start_col;
187444d93782SGreg Clayton         }
187544d93782SGreg Clayton 
187644d93782SGreg Clayton         void
187744d93782SGreg Clayton         SetStartingColumn(int col)
187844d93782SGreg Clayton         {
187944d93782SGreg Clayton             m_start_col = col;
188044d93782SGreg Clayton         }
188144d93782SGreg Clayton 
188244d93782SGreg Clayton         int
188344d93782SGreg Clayton         GetKeyValue() const
188444d93782SGreg Clayton         {
188544d93782SGreg Clayton             return m_key_value;
188644d93782SGreg Clayton         }
188744d93782SGreg Clayton 
188844d93782SGreg Clayton         void
188944d93782SGreg Clayton         SetKeyValue(int key_value)
189044d93782SGreg Clayton         {
189144d93782SGreg Clayton             m_key_value = key_value;
189244d93782SGreg Clayton         }
189344d93782SGreg Clayton 
189444d93782SGreg Clayton         std::string &
189544d93782SGreg Clayton         GetName()
189644d93782SGreg Clayton         {
189744d93782SGreg Clayton             return m_name;
189844d93782SGreg Clayton         }
189944d93782SGreg Clayton 
190044d93782SGreg Clayton         std::string &
190144d93782SGreg Clayton         GetKeyName()
190244d93782SGreg Clayton         {
190344d93782SGreg Clayton             return m_key_name;
190444d93782SGreg Clayton         }
190544d93782SGreg Clayton 
190644d93782SGreg Clayton         int
190744d93782SGreg Clayton         GetDrawWidth () const
190844d93782SGreg Clayton         {
190944d93782SGreg Clayton             return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
191044d93782SGreg Clayton         }
191144d93782SGreg Clayton 
191244d93782SGreg Clayton         uint64_t
191344d93782SGreg Clayton         GetIdentifier() const
191444d93782SGreg Clayton         {
191544d93782SGreg Clayton             return m_identifier;
191644d93782SGreg Clayton         }
191744d93782SGreg Clayton 
191844d93782SGreg Clayton         void
191944d93782SGreg Clayton         SetIdentifier (uint64_t identifier)
192044d93782SGreg Clayton         {
192144d93782SGreg Clayton             m_identifier = identifier;
192244d93782SGreg Clayton         }
192344d93782SGreg Clayton 
192444d93782SGreg Clayton     protected:
192544d93782SGreg Clayton         std::string m_name;
192644d93782SGreg Clayton         std::string m_key_name;
192744d93782SGreg Clayton         uint64_t m_identifier;
192844d93782SGreg Clayton         Type m_type;
192944d93782SGreg Clayton         int m_key_value;
193044d93782SGreg Clayton         int m_start_col;
193144d93782SGreg Clayton         int m_max_submenu_name_length;
193244d93782SGreg Clayton         int m_max_submenu_key_name_length;
193344d93782SGreg Clayton         int m_selected;
193444d93782SGreg Clayton         Menu *m_parent;
193544d93782SGreg Clayton         Menus m_submenus;
193644d93782SGreg Clayton         WindowSP m_menu_window_sp;
193744d93782SGreg Clayton         MenuActionResult m_canned_result;
193844d93782SGreg Clayton         MenuDelegateSP m_delegate_sp;
193944d93782SGreg Clayton     };
194044d93782SGreg Clayton 
194144d93782SGreg Clayton     // Menubar or separator constructor
194244d93782SGreg Clayton     Menu::Menu (Type type) :
194344d93782SGreg Clayton         m_name (),
194444d93782SGreg Clayton         m_key_name (),
194544d93782SGreg Clayton         m_identifier (0),
194644d93782SGreg Clayton         m_type (type),
194744d93782SGreg Clayton         m_key_value (0),
194844d93782SGreg Clayton         m_start_col (0),
194944d93782SGreg Clayton         m_max_submenu_name_length (0),
195044d93782SGreg Clayton         m_max_submenu_key_name_length (0),
195144d93782SGreg Clayton         m_selected (0),
1952c5dac77aSEugene Zelenko         m_parent(nullptr),
195344d93782SGreg Clayton         m_submenus (),
195444d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
195544d93782SGreg Clayton         m_delegate_sp()
195644d93782SGreg Clayton     {
195744d93782SGreg Clayton     }
195844d93782SGreg Clayton 
195944d93782SGreg Clayton     // Menuitem constructor
196044d93782SGreg Clayton     Menu::Menu (const char *name,
196144d93782SGreg Clayton                 const char *key_name,
196244d93782SGreg Clayton                 int key_value,
196344d93782SGreg Clayton                 uint64_t identifier) :
196444d93782SGreg Clayton         m_name (),
196544d93782SGreg Clayton         m_key_name (),
196644d93782SGreg Clayton         m_identifier (identifier),
196744d93782SGreg Clayton         m_type (Type::Invalid),
196844d93782SGreg Clayton         m_key_value (key_value),
196944d93782SGreg Clayton         m_start_col (0),
197044d93782SGreg Clayton         m_max_submenu_name_length (0),
197144d93782SGreg Clayton         m_max_submenu_key_name_length (0),
197244d93782SGreg Clayton         m_selected (0),
1973c5dac77aSEugene Zelenko         m_parent(nullptr),
197444d93782SGreg Clayton         m_submenus (),
197544d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
197644d93782SGreg Clayton         m_delegate_sp()
197744d93782SGreg Clayton     {
197844d93782SGreg Clayton         if (name && name[0])
197944d93782SGreg Clayton         {
198044d93782SGreg Clayton             m_name = name;
198144d93782SGreg Clayton             m_type = Type::Item;
198244d93782SGreg Clayton             if (key_name && key_name[0])
198344d93782SGreg Clayton                 m_key_name = key_name;
198444d93782SGreg Clayton         }
198544d93782SGreg Clayton         else
198644d93782SGreg Clayton         {
198744d93782SGreg Clayton             m_type = Type::Separator;
198844d93782SGreg Clayton         }
198944d93782SGreg Clayton     }
199044d93782SGreg Clayton 
199144d93782SGreg Clayton     void
199244d93782SGreg Clayton     Menu::RecalculateNameLengths()
199344d93782SGreg Clayton     {
199444d93782SGreg Clayton         m_max_submenu_name_length = 0;
199544d93782SGreg Clayton         m_max_submenu_key_name_length = 0;
199644d93782SGreg Clayton         Menus &submenus = GetSubmenus();
199744d93782SGreg Clayton         const size_t num_submenus = submenus.size();
199844d93782SGreg Clayton         for (size_t i = 0; i < num_submenus; ++i)
199944d93782SGreg Clayton         {
200044d93782SGreg Clayton             Menu *submenu = submenus[i].get();
20013985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
200244d93782SGreg Clayton                 m_max_submenu_name_length = submenu->m_name.size();
20033985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size())
200444d93782SGreg Clayton                 m_max_submenu_key_name_length = submenu->m_key_name.size();
200544d93782SGreg Clayton         }
200644d93782SGreg Clayton     }
200744d93782SGreg Clayton 
200844d93782SGreg Clayton     void
200944d93782SGreg Clayton     Menu::AddSubmenu (const MenuSP &menu_sp)
201044d93782SGreg Clayton     {
201144d93782SGreg Clayton         menu_sp->m_parent = this;
20123985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
201344d93782SGreg Clayton             m_max_submenu_name_length = menu_sp->m_name.size();
20143985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size())
201544d93782SGreg Clayton             m_max_submenu_key_name_length = menu_sp->m_key_name.size();
201644d93782SGreg Clayton         m_submenus.push_back(menu_sp);
201744d93782SGreg Clayton     }
201844d93782SGreg Clayton 
201944d93782SGreg Clayton     void
202044d93782SGreg Clayton     Menu::DrawMenuTitle (Window &window, bool highlight)
202144d93782SGreg Clayton     {
202244d93782SGreg Clayton         if (m_type == Type::Separator)
202344d93782SGreg Clayton         {
202444d93782SGreg Clayton             window.MoveCursor(0, window.GetCursorY());
202544d93782SGreg Clayton             window.PutChar(ACS_LTEE);
202644d93782SGreg Clayton             int width = window.GetWidth();
202744d93782SGreg Clayton             if (width > 2)
202844d93782SGreg Clayton             {
202944d93782SGreg Clayton                 width -= 2;
20303985c8c6SSaleem Abdulrasool                 for (int i = 0; i < width; ++i)
203144d93782SGreg Clayton                     window.PutChar(ACS_HLINE);
203244d93782SGreg Clayton             }
203344d93782SGreg Clayton             window.PutChar(ACS_RTEE);
203444d93782SGreg Clayton         }
203544d93782SGreg Clayton         else
203644d93782SGreg Clayton         {
203744d93782SGreg Clayton             const int shortcut_key = m_key_value;
203844d93782SGreg Clayton             bool underlined_shortcut = false;
203944d93782SGreg Clayton             const attr_t hilgight_attr = A_REVERSE;
204044d93782SGreg Clayton             if (highlight)
204144d93782SGreg Clayton                 window.AttributeOn(hilgight_attr);
204244d93782SGreg Clayton             if (isprint(shortcut_key))
204344d93782SGreg Clayton             {
204444d93782SGreg Clayton                 size_t lower_pos = m_name.find(tolower(shortcut_key));
204544d93782SGreg Clayton                 size_t upper_pos = m_name.find(toupper(shortcut_key));
204644d93782SGreg Clayton                 const char *name = m_name.c_str();
204744d93782SGreg Clayton                 size_t pos = std::min<size_t>(lower_pos, upper_pos);
204844d93782SGreg Clayton                 if (pos != std::string::npos)
204944d93782SGreg Clayton                 {
205044d93782SGreg Clayton                     underlined_shortcut = true;
205144d93782SGreg Clayton                     if (pos > 0)
205244d93782SGreg Clayton                     {
205344d93782SGreg Clayton                         window.PutCString(name, pos);
205444d93782SGreg Clayton                         name += pos;
205544d93782SGreg Clayton                     }
205644d93782SGreg Clayton                     const attr_t shortcut_attr = A_UNDERLINE|A_BOLD;
205744d93782SGreg Clayton                     window.AttributeOn (shortcut_attr);
205844d93782SGreg Clayton                     window.PutChar(name[0]);
205944d93782SGreg Clayton                     window.AttributeOff(shortcut_attr);
206044d93782SGreg Clayton                     name++;
206144d93782SGreg Clayton                     if (name[0])
206244d93782SGreg Clayton                         window.PutCString(name);
206344d93782SGreg Clayton                 }
206444d93782SGreg Clayton             }
206544d93782SGreg Clayton 
206644d93782SGreg Clayton             if (!underlined_shortcut)
206744d93782SGreg Clayton             {
206844d93782SGreg Clayton                 window.PutCString(m_name.c_str());
206944d93782SGreg Clayton             }
207044d93782SGreg Clayton 
207144d93782SGreg Clayton             if (highlight)
207244d93782SGreg Clayton                 window.AttributeOff(hilgight_attr);
207344d93782SGreg Clayton 
207444d93782SGreg Clayton             if (m_key_name.empty())
207544d93782SGreg Clayton             {
207644d93782SGreg Clayton                 if (!underlined_shortcut && isprint(m_key_value))
207744d93782SGreg Clayton                 {
207844d93782SGreg Clayton                     window.AttributeOn (COLOR_PAIR(3));
207944d93782SGreg Clayton                     window.Printf (" (%c)", m_key_value);
208044d93782SGreg Clayton                     window.AttributeOff (COLOR_PAIR(3));
208144d93782SGreg Clayton                 }
208244d93782SGreg Clayton             }
208344d93782SGreg Clayton             else
208444d93782SGreg Clayton             {
208544d93782SGreg Clayton                 window.AttributeOn (COLOR_PAIR(3));
208644d93782SGreg Clayton                 window.Printf (" (%s)", m_key_name.c_str());
208744d93782SGreg Clayton                 window.AttributeOff (COLOR_PAIR(3));
208844d93782SGreg Clayton             }
208944d93782SGreg Clayton         }
209044d93782SGreg Clayton     }
209144d93782SGreg Clayton 
209244d93782SGreg Clayton     bool
209344d93782SGreg Clayton     Menu::WindowDelegateDraw (Window &window, bool force)
209444d93782SGreg Clayton     {
209544d93782SGreg Clayton         Menus &submenus = GetSubmenus();
209644d93782SGreg Clayton         const size_t num_submenus = submenus.size();
209744d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
209844d93782SGreg Clayton         Menu::Type menu_type = GetType ();
209944d93782SGreg Clayton         switch (menu_type)
210044d93782SGreg Clayton         {
210144d93782SGreg Clayton         case  Menu::Type::Bar:
210244d93782SGreg Clayton             {
210344d93782SGreg Clayton                 window.SetBackground(2);
210444d93782SGreg Clayton                 window.MoveCursor(0, 0);
210544d93782SGreg Clayton                 for (size_t i = 0; i < num_submenus; ++i)
210644d93782SGreg Clayton                 {
210744d93782SGreg Clayton                     Menu *menu = submenus[i].get();
210844d93782SGreg Clayton                     if (i > 0)
210944d93782SGreg Clayton                         window.PutChar(' ');
211044d93782SGreg Clayton                     menu->SetStartingColumn (window.GetCursorX());
211144d93782SGreg Clayton                     window.PutCString("| ");
211244d93782SGreg Clayton                     menu->DrawMenuTitle (window, false);
211344d93782SGreg Clayton                 }
211444d93782SGreg Clayton                 window.PutCString(" |");
211544d93782SGreg Clayton                 window.DeferredRefresh();
211644d93782SGreg Clayton             }
211744d93782SGreg Clayton             break;
211844d93782SGreg Clayton 
211944d93782SGreg Clayton         case Menu::Type::Item:
212044d93782SGreg Clayton             {
212144d93782SGreg Clayton                 int y = 1;
212244d93782SGreg Clayton                 int x = 3;
212344d93782SGreg Clayton                 // Draw the menu
212444d93782SGreg Clayton                 int cursor_x = 0;
212544d93782SGreg Clayton                 int cursor_y = 0;
212644d93782SGreg Clayton                 window.Erase();
212744d93782SGreg Clayton                 window.SetBackground(2);
212844d93782SGreg Clayton                 window.Box();
212944d93782SGreg Clayton                 for (size_t i = 0; i < num_submenus; ++i)
213044d93782SGreg Clayton                 {
21313985c8c6SSaleem Abdulrasool                     const bool is_selected =
21323985c8c6SSaleem Abdulrasool                       (i == static_cast<size_t>(selected_idx));
213344d93782SGreg Clayton                     window.MoveCursor(x, y + i);
213444d93782SGreg Clayton                     if (is_selected)
213544d93782SGreg Clayton                     {
213644d93782SGreg Clayton                         // Remember where we want the cursor to be
213744d93782SGreg Clayton                         cursor_x = x-1;
213844d93782SGreg Clayton                         cursor_y = y+i;
213944d93782SGreg Clayton                     }
214044d93782SGreg Clayton                     submenus[i]->DrawMenuTitle (window, is_selected);
214144d93782SGreg Clayton                 }
214244d93782SGreg Clayton                 window.MoveCursor(cursor_x, cursor_y);
214344d93782SGreg Clayton                 window.DeferredRefresh();
214444d93782SGreg Clayton             }
214544d93782SGreg Clayton             break;
214644d93782SGreg Clayton 
214744d93782SGreg Clayton         default:
214844d93782SGreg Clayton         case Menu::Type::Separator:
214944d93782SGreg Clayton             break;
215044d93782SGreg Clayton         }
215144d93782SGreg Clayton         return true; // Drawing handled...
215244d93782SGreg Clayton     }
215344d93782SGreg Clayton 
215444d93782SGreg Clayton     HandleCharResult
215544d93782SGreg Clayton     Menu::WindowDelegateHandleChar (Window &window, int key)
215644d93782SGreg Clayton     {
215744d93782SGreg Clayton         HandleCharResult result = eKeyNotHandled;
215844d93782SGreg Clayton 
215944d93782SGreg Clayton         Menus &submenus = GetSubmenus();
216044d93782SGreg Clayton         const size_t num_submenus = submenus.size();
216144d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
216244d93782SGreg Clayton         Menu::Type menu_type = GetType ();
216344d93782SGreg Clayton         if (menu_type == Menu::Type::Bar)
216444d93782SGreg Clayton         {
216544d93782SGreg Clayton             MenuSP run_menu_sp;
216644d93782SGreg Clayton             switch (key)
216744d93782SGreg Clayton             {
216844d93782SGreg Clayton                 case KEY_DOWN:
216944d93782SGreg Clayton                 case KEY_UP:
217044d93782SGreg Clayton                     // Show last menu or first menu
21713985c8c6SSaleem Abdulrasool                     if (selected_idx < static_cast<int>(num_submenus))
217244d93782SGreg Clayton                         run_menu_sp = submenus[selected_idx];
217344d93782SGreg Clayton                     else if (!submenus.empty())
217444d93782SGreg Clayton                         run_menu_sp = submenus.front();
217544d93782SGreg Clayton                     result = eKeyHandled;
217644d93782SGreg Clayton                     break;
217744d93782SGreg Clayton 
217844d93782SGreg Clayton                 case KEY_RIGHT:
217944d93782SGreg Clayton                     ++m_selected;
21803985c8c6SSaleem Abdulrasool                     if (m_selected >= static_cast<int>(num_submenus))
218144d93782SGreg Clayton                         m_selected = 0;
21823985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
218344d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
218444d93782SGreg Clayton                     else if (!submenus.empty())
218544d93782SGreg Clayton                         run_menu_sp = submenus.front();
218644d93782SGreg Clayton                     result = eKeyHandled;
218744d93782SGreg Clayton                     break;
218844d93782SGreg Clayton 
218944d93782SGreg Clayton                 case KEY_LEFT:
219044d93782SGreg Clayton                     --m_selected;
219144d93782SGreg Clayton                     if (m_selected < 0)
219244d93782SGreg Clayton                         m_selected = num_submenus - 1;
21933985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
219444d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
219544d93782SGreg Clayton                     else if (!submenus.empty())
219644d93782SGreg Clayton                         run_menu_sp = submenus.front();
219744d93782SGreg Clayton                     result = eKeyHandled;
219844d93782SGreg Clayton                     break;
219944d93782SGreg Clayton 
220044d93782SGreg Clayton                 default:
220144d93782SGreg Clayton                     for (size_t i = 0; i < num_submenus; ++i)
220244d93782SGreg Clayton                     {
220344d93782SGreg Clayton                         if (submenus[i]->GetKeyValue() == key)
220444d93782SGreg Clayton                         {
220544d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
220644d93782SGreg Clayton                             run_menu_sp = submenus[i];
220744d93782SGreg Clayton                             result = eKeyHandled;
220844d93782SGreg Clayton                             break;
220944d93782SGreg Clayton                         }
221044d93782SGreg Clayton                     }
221144d93782SGreg Clayton                     break;
221244d93782SGreg Clayton             }
221344d93782SGreg Clayton 
221444d93782SGreg Clayton             if (run_menu_sp)
221544d93782SGreg Clayton             {
221644d93782SGreg Clayton                 // Run the action on this menu in case we need to populate the
221744d93782SGreg Clayton                 // menu with dynamic content and also in case check marks, and
221885025451SKamil Rytarowski                 // any other menu decorations need to be calculated
221944d93782SGreg Clayton                 if (run_menu_sp->Action() == MenuActionResult::Quit)
222044d93782SGreg Clayton                     return eQuitApplication;
222144d93782SGreg Clayton 
222244d93782SGreg Clayton                 Rect menu_bounds;
222344d93782SGreg Clayton                 menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
222444d93782SGreg Clayton                 menu_bounds.origin.y = 1;
222544d93782SGreg Clayton                 menu_bounds.size.width = run_menu_sp->GetDrawWidth();
222644d93782SGreg Clayton                 menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
222744d93782SGreg Clayton                 if (m_menu_window_sp)
222844d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
222944d93782SGreg Clayton 
223044d93782SGreg Clayton                 m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(),
223144d93782SGreg Clayton                                                                         menu_bounds,
223244d93782SGreg Clayton                                                                         true);
223344d93782SGreg Clayton                 m_menu_window_sp->SetDelegate (run_menu_sp);
223444d93782SGreg Clayton             }
223544d93782SGreg Clayton         }
223644d93782SGreg Clayton         else if (menu_type == Menu::Type::Item)
223744d93782SGreg Clayton         {
223844d93782SGreg Clayton             switch (key)
223944d93782SGreg Clayton             {
224044d93782SGreg Clayton                 case KEY_DOWN:
224144d93782SGreg Clayton                     if (m_submenus.size() > 1)
224244d93782SGreg Clayton                     {
224344d93782SGreg Clayton                         const int start_select = m_selected;
224444d93782SGreg Clayton                         while (++m_selected != start_select)
224544d93782SGreg Clayton                         {
22463985c8c6SSaleem Abdulrasool                             if (static_cast<size_t>(m_selected) >= num_submenus)
224744d93782SGreg Clayton                                 m_selected = 0;
224844d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
224944d93782SGreg Clayton                                 continue;
225044d93782SGreg Clayton                             else
225144d93782SGreg Clayton                                 break;
225244d93782SGreg Clayton                         }
225344d93782SGreg Clayton                         return eKeyHandled;
225444d93782SGreg Clayton                     }
225544d93782SGreg Clayton                     break;
225644d93782SGreg Clayton 
225744d93782SGreg Clayton                 case KEY_UP:
225844d93782SGreg Clayton                     if (m_submenus.size() > 1)
225944d93782SGreg Clayton                     {
226044d93782SGreg Clayton                         const int start_select = m_selected;
226144d93782SGreg Clayton                         while (--m_selected != start_select)
226244d93782SGreg Clayton                         {
22633985c8c6SSaleem Abdulrasool                             if (m_selected < static_cast<int>(0))
226444d93782SGreg Clayton                                 m_selected = num_submenus - 1;
226544d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
226644d93782SGreg Clayton                                 continue;
226744d93782SGreg Clayton                             else
226844d93782SGreg Clayton                                 break;
226944d93782SGreg Clayton                         }
227044d93782SGreg Clayton                         return eKeyHandled;
227144d93782SGreg Clayton                     }
227244d93782SGreg Clayton                     break;
227344d93782SGreg Clayton 
227444d93782SGreg Clayton                 case KEY_RETURN:
22753985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(selected_idx) < num_submenus)
227644d93782SGreg Clayton                     {
227744d93782SGreg Clayton                         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
227844d93782SGreg Clayton                             return eQuitApplication;
227944d93782SGreg Clayton                         window.GetParent()->RemoveSubWindow(&window);
228044d93782SGreg Clayton                         return eKeyHandled;
228144d93782SGreg Clayton                     }
228244d93782SGreg Clayton                     break;
228344d93782SGreg Clayton 
228444d93782SGreg Clayton                 case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences
228544d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(&window);
228644d93782SGreg Clayton                     return eKeyHandled;
228744d93782SGreg Clayton 
228844d93782SGreg Clayton                 default:
228944d93782SGreg Clayton                     for (size_t i = 0; i < num_submenus; ++i)
229044d93782SGreg Clayton                     {
229144d93782SGreg Clayton                         Menu *menu = submenus[i].get();
229244d93782SGreg Clayton                         if (menu->GetKeyValue() == key)
229344d93782SGreg Clayton                         {
229444d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
229544d93782SGreg Clayton                             window.GetParent()->RemoveSubWindow(&window);
229644d93782SGreg Clayton                             if (menu->Action() == MenuActionResult::Quit)
229744d93782SGreg Clayton                                 return eQuitApplication;
229844d93782SGreg Clayton                             return eKeyHandled;
229944d93782SGreg Clayton                         }
230044d93782SGreg Clayton                     }
230144d93782SGreg Clayton                     break;
230244d93782SGreg Clayton             }
230344d93782SGreg Clayton         }
230444d93782SGreg Clayton         else if (menu_type == Menu::Type::Separator)
230544d93782SGreg Clayton         {
230644d93782SGreg Clayton         }
230744d93782SGreg Clayton         return result;
230844d93782SGreg Clayton     }
230944d93782SGreg Clayton 
231044d93782SGreg Clayton     class Application
231144d93782SGreg Clayton     {
231244d93782SGreg Clayton     public:
231344d93782SGreg Clayton         Application (FILE *in, FILE *out) :
231444d93782SGreg Clayton             m_window_sp(),
2315c5dac77aSEugene Zelenko             m_screen(nullptr),
231644d93782SGreg Clayton             m_in (in),
231744d93782SGreg Clayton             m_out (out)
231844d93782SGreg Clayton         {
231944d93782SGreg Clayton         }
232044d93782SGreg Clayton 
232144d93782SGreg Clayton         ~Application ()
232244d93782SGreg Clayton         {
232344d93782SGreg Clayton             m_window_delegates.clear();
232444d93782SGreg Clayton             m_window_sp.reset();
232544d93782SGreg Clayton             if (m_screen)
232644d93782SGreg Clayton             {
232744d93782SGreg Clayton                 ::delscreen(m_screen);
2328c5dac77aSEugene Zelenko                 m_screen = nullptr;
232944d93782SGreg Clayton             }
233044d93782SGreg Clayton         }
233144d93782SGreg Clayton 
233244d93782SGreg Clayton         void
233344d93782SGreg Clayton         Initialize ()
233444d93782SGreg Clayton         {
233544d93782SGreg Clayton             ::setlocale(LC_ALL, "");
233644d93782SGreg Clayton             ::setlocale(LC_CTYPE, "");
233744d93782SGreg Clayton #if 0
233844d93782SGreg Clayton             ::initscr();
233944d93782SGreg Clayton #else
2340c5dac77aSEugene Zelenko             m_screen = ::newterm(nullptr, m_out, m_in);
234144d93782SGreg Clayton #endif
234244d93782SGreg Clayton             ::start_color();
234344d93782SGreg Clayton             ::curs_set(0);
234444d93782SGreg Clayton             ::noecho();
234544d93782SGreg Clayton             ::keypad(stdscr,TRUE);
234644d93782SGreg Clayton         }
234744d93782SGreg Clayton 
234844d93782SGreg Clayton         void
234944d93782SGreg Clayton         Terminate ()
235044d93782SGreg Clayton         {
235144d93782SGreg Clayton             ::endwin();
235244d93782SGreg Clayton         }
235344d93782SGreg Clayton 
235444d93782SGreg Clayton         void
235544d93782SGreg Clayton         Run (Debugger &debugger)
235644d93782SGreg Clayton         {
235744d93782SGreg Clayton             bool done = false;
235844d93782SGreg Clayton             int delay_in_tenths_of_a_second = 1;
235944d93782SGreg Clayton 
236044d93782SGreg Clayton             // Alas the threading model in curses is a bit lame so we need to
236144d93782SGreg Clayton             // resort to polling every 0.5 seconds. We could poll for stdin
236244d93782SGreg Clayton             // ourselves and then pass the keys down but then we need to
236344d93782SGreg Clayton             // translate all of the escape sequences ourselves. So we resort to
236444d93782SGreg Clayton             // polling for input because we need to receive async process events
236544d93782SGreg Clayton             // while in this loop.
236644d93782SGreg Clayton 
236744d93782SGreg Clayton             halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar()
236844d93782SGreg Clayton 
2369583bbb1dSJim Ingham             ListenerSP listener_sp (Listener::MakeListener("lldb.IOHandler.curses.Application"));
237044d93782SGreg Clayton             ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
237144d93782SGreg Clayton             ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
237244d93782SGreg Clayton             ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
237344d93782SGreg Clayton             debugger.EnableForwardEvents (listener_sp);
237444d93782SGreg Clayton 
237544d93782SGreg Clayton             bool update = true;
237644d93782SGreg Clayton #if defined(__APPLE__)
237744d93782SGreg Clayton             std::deque<int> escape_chars;
237844d93782SGreg Clayton #endif
237944d93782SGreg Clayton 
238044d93782SGreg Clayton             while (!done)
238144d93782SGreg Clayton             {
238244d93782SGreg Clayton                 if (update)
238344d93782SGreg Clayton                 {
238444d93782SGreg Clayton                     m_window_sp->Draw(false);
238544d93782SGreg Clayton                     // All windows should be calling Window::DeferredRefresh() instead
238644d93782SGreg Clayton                     // of Window::Refresh() so we can do a single update and avoid
238744d93782SGreg Clayton                     // any screen blinking
238844d93782SGreg Clayton                     update_panels();
238944d93782SGreg Clayton 
239044d93782SGreg Clayton                     // Cursor hiding isn't working on MacOSX, so hide it in the top left corner
239144d93782SGreg Clayton                     m_window_sp->MoveCursor(0, 0);
239244d93782SGreg Clayton 
239344d93782SGreg Clayton                     doupdate();
239444d93782SGreg Clayton                     update = false;
239544d93782SGreg Clayton                 }
239644d93782SGreg Clayton 
239744d93782SGreg Clayton #if defined(__APPLE__)
239844d93782SGreg Clayton                 // Terminal.app doesn't map its function keys correctly, F1-F4 default to:
239944d93782SGreg Clayton                 // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible
240044d93782SGreg Clayton                 int ch;
240144d93782SGreg Clayton                 if (escape_chars.empty())
240244d93782SGreg Clayton                     ch = m_window_sp->GetChar();
240344d93782SGreg Clayton                 else
240444d93782SGreg Clayton                 {
240544d93782SGreg Clayton                     ch = escape_chars.front();
240644d93782SGreg Clayton                     escape_chars.pop_front();
240744d93782SGreg Clayton                 }
240844d93782SGreg Clayton                 if (ch == KEY_ESCAPE)
240944d93782SGreg Clayton                 {
241044d93782SGreg Clayton                     int ch2 = m_window_sp->GetChar();
241144d93782SGreg Clayton                     if (ch2 == 'O')
241244d93782SGreg Clayton                     {
241344d93782SGreg Clayton                         int ch3 = m_window_sp->GetChar();
241444d93782SGreg Clayton                         switch (ch3)
241544d93782SGreg Clayton                         {
241644d93782SGreg Clayton                             case 'P': ch = KEY_F(1); break;
241744d93782SGreg Clayton                             case 'Q': ch = KEY_F(2); break;
241844d93782SGreg Clayton                             case 'R': ch = KEY_F(3); break;
241944d93782SGreg Clayton                             case 'S': ch = KEY_F(4); break;
242044d93782SGreg Clayton                             default:
242144d93782SGreg Clayton                                 escape_chars.push_back(ch2);
242244d93782SGreg Clayton                                 if (ch3 != -1)
242344d93782SGreg Clayton                                     escape_chars.push_back(ch3);
242444d93782SGreg Clayton                                 break;
242544d93782SGreg Clayton                         }
242644d93782SGreg Clayton                     }
242744d93782SGreg Clayton                     else if (ch2 != -1)
242844d93782SGreg Clayton                         escape_chars.push_back(ch2);
242944d93782SGreg Clayton                 }
243044d93782SGreg Clayton #else
243144d93782SGreg Clayton                 int ch = m_window_sp->GetChar();
243244d93782SGreg Clayton 
243344d93782SGreg Clayton #endif
243444d93782SGreg Clayton                 if (ch == -1)
243544d93782SGreg Clayton                 {
243644d93782SGreg Clayton                     if (feof(m_in) || ferror(m_in))
243744d93782SGreg Clayton                     {
243844d93782SGreg Clayton                         done = true;
243944d93782SGreg Clayton                     }
244044d93782SGreg Clayton                     else
244144d93782SGreg Clayton                     {
244244d93782SGreg Clayton                         // Just a timeout from using halfdelay(), check for events
244344d93782SGreg Clayton                         EventSP event_sp;
244444d93782SGreg Clayton                         while (listener_sp->PeekAtNextEvent())
244544d93782SGreg Clayton                         {
244644d93782SGreg Clayton                             listener_sp->GetNextEvent(event_sp);
244744d93782SGreg Clayton 
244844d93782SGreg Clayton                             if (event_sp)
244944d93782SGreg Clayton                             {
245044d93782SGreg Clayton                                 Broadcaster *broadcaster = event_sp->GetBroadcaster();
245144d93782SGreg Clayton                                 if (broadcaster)
245244d93782SGreg Clayton                                 {
245344d93782SGreg Clayton                                     //uint32_t event_type = event_sp->GetType();
245444d93782SGreg Clayton                                     ConstString broadcaster_class (broadcaster->GetBroadcasterClass());
245544d93782SGreg Clayton                                     if (broadcaster_class == broadcaster_class_process)
245644d93782SGreg Clayton                                     {
2457c5dac77aSEugene Zelenko                                         debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
245844d93782SGreg Clayton                                         update = true;
245944d93782SGreg Clayton                                         continue; // Don't get any key, just update our view
246044d93782SGreg Clayton                                     }
246144d93782SGreg Clayton                                 }
246244d93782SGreg Clayton                             }
246344d93782SGreg Clayton                         }
246444d93782SGreg Clayton                     }
246544d93782SGreg Clayton                 }
246644d93782SGreg Clayton                 else
246744d93782SGreg Clayton                 {
246844d93782SGreg Clayton                     HandleCharResult key_result = m_window_sp->HandleChar(ch);
246944d93782SGreg Clayton                     switch (key_result)
247044d93782SGreg Clayton                     {
247144d93782SGreg Clayton                         case eKeyHandled:
2472c5dac77aSEugene Zelenko                             debugger.GetCommandInterpreter().UpdateExecutionContext(nullptr);
247344d93782SGreg Clayton                             update = true;
247444d93782SGreg Clayton                             break;
247544d93782SGreg Clayton                         case eKeyNotHandled:
247644d93782SGreg Clayton                             break;
247744d93782SGreg Clayton                         case eQuitApplication:
247844d93782SGreg Clayton                             done = true;
247944d93782SGreg Clayton                             break;
248044d93782SGreg Clayton                     }
248144d93782SGreg Clayton                 }
248244d93782SGreg Clayton             }
248344d93782SGreg Clayton 
248444d93782SGreg Clayton             debugger.CancelForwardEvents (listener_sp);
248544d93782SGreg Clayton         }
248644d93782SGreg Clayton 
248744d93782SGreg Clayton         WindowSP &
248844d93782SGreg Clayton         GetMainWindow ()
248944d93782SGreg Clayton         {
249044d93782SGreg Clayton             if (!m_window_sp)
249144d93782SGreg Clayton                 m_window_sp.reset (new Window ("main", stdscr, false));
249244d93782SGreg Clayton             return m_window_sp;
249344d93782SGreg Clayton         }
249444d93782SGreg Clayton 
249544d93782SGreg Clayton         WindowDelegates &
249644d93782SGreg Clayton         GetWindowDelegates ()
249744d93782SGreg Clayton         {
249844d93782SGreg Clayton             return m_window_delegates;
249944d93782SGreg Clayton         }
250044d93782SGreg Clayton 
250144d93782SGreg Clayton     protected:
250244d93782SGreg Clayton         WindowSP m_window_sp;
250344d93782SGreg Clayton         WindowDelegates m_window_delegates;
250444d93782SGreg Clayton         SCREEN *m_screen;
250544d93782SGreg Clayton         FILE *m_in;
250644d93782SGreg Clayton         FILE *m_out;
250744d93782SGreg Clayton     };
250844d93782SGreg Clayton 
250944d93782SGreg Clayton } // namespace curses
251044d93782SGreg Clayton 
251144d93782SGreg Clayton using namespace curses;
251244d93782SGreg Clayton 
251344d93782SGreg Clayton struct Row
251444d93782SGreg Clayton {
251544d93782SGreg Clayton     ValueObjectSP valobj;
251644d93782SGreg Clayton     Row *parent;
251744d93782SGreg Clayton     int row_idx;
251844d93782SGreg Clayton     int x;
251944d93782SGreg Clayton     int y;
252044d93782SGreg Clayton     bool might_have_children;
252144d93782SGreg Clayton     bool expanded;
252244d93782SGreg Clayton     bool calculated_children;
252344d93782SGreg Clayton     std::vector<Row> children;
252444d93782SGreg Clayton 
252544d93782SGreg Clayton     Row (const ValueObjectSP &v, Row *p) :
252644d93782SGreg Clayton     valobj (v),
252744d93782SGreg Clayton     parent (p),
252844d93782SGreg Clayton     row_idx(0),
252944d93782SGreg Clayton     x(1),
253044d93782SGreg Clayton     y(1),
253144d93782SGreg Clayton     might_have_children (v ? v->MightHaveChildren() : false),
253244d93782SGreg Clayton     expanded (false),
253344d93782SGreg Clayton     calculated_children (false),
253444d93782SGreg Clayton     children()
253544d93782SGreg Clayton     {
253644d93782SGreg Clayton     }
253744d93782SGreg Clayton 
253844d93782SGreg Clayton     size_t
253944d93782SGreg Clayton     GetDepth () const
254044d93782SGreg Clayton     {
254144d93782SGreg Clayton         if (parent)
254244d93782SGreg Clayton             return 1 + parent->GetDepth();
254344d93782SGreg Clayton         return 0;
254444d93782SGreg Clayton     }
254544d93782SGreg Clayton 
254644d93782SGreg Clayton     void
254744d93782SGreg Clayton     Expand()
254844d93782SGreg Clayton     {
254944d93782SGreg Clayton         expanded = true;
255044d93782SGreg Clayton         if (!calculated_children)
255144d93782SGreg Clayton         {
255244d93782SGreg Clayton             calculated_children = true;
255344d93782SGreg Clayton             if (valobj)
255444d93782SGreg Clayton             {
255544d93782SGreg Clayton                 const size_t num_children = valobj->GetNumChildren();
255644d93782SGreg Clayton                 for (size_t i = 0; i < num_children; ++i)
255744d93782SGreg Clayton                 {
255844d93782SGreg Clayton                     children.push_back(Row (valobj->GetChildAtIndex(i, true), this));
255944d93782SGreg Clayton                 }
256044d93782SGreg Clayton             }
256144d93782SGreg Clayton         }
256244d93782SGreg Clayton     }
256344d93782SGreg Clayton 
256444d93782SGreg Clayton     void
256544d93782SGreg Clayton     Unexpand ()
256644d93782SGreg Clayton     {
256744d93782SGreg Clayton         expanded = false;
256844d93782SGreg Clayton     }
256944d93782SGreg Clayton 
257044d93782SGreg Clayton     void
257144d93782SGreg Clayton     DrawTree (Window &window)
257244d93782SGreg Clayton     {
257344d93782SGreg Clayton         if (parent)
257444d93782SGreg Clayton             parent->DrawTreeForChild (window, this, 0);
257544d93782SGreg Clayton 
257644d93782SGreg Clayton         if (might_have_children)
257744d93782SGreg Clayton         {
257844d93782SGreg Clayton             // It we can get UTF8 characters to work we should try to use the "symbol"
257944d93782SGreg Clayton             // UTF8 string below
258044d93782SGreg Clayton //            const char *symbol = "";
258144d93782SGreg Clayton //            if (row.expanded)
258244d93782SGreg Clayton //                symbol = "\xe2\x96\xbd ";
258344d93782SGreg Clayton //            else
258444d93782SGreg Clayton //                symbol = "\xe2\x96\xb7 ";
258544d93782SGreg Clayton //            window.PutCString (symbol);
258644d93782SGreg Clayton 
258744d93782SGreg Clayton             // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
258844d93782SGreg Clayton             // 'v' or '>' character...
258944d93782SGreg Clayton //            if (expanded)
259044d93782SGreg Clayton //                window.PutChar (ACS_DARROW);
259144d93782SGreg Clayton //            else
259244d93782SGreg Clayton //                window.PutChar (ACS_RARROW);
259344d93782SGreg Clayton             // Since we can't find any good looking right arrow/down arrow
259444d93782SGreg Clayton             // symbols, just use a diamond...
259544d93782SGreg Clayton             window.PutChar (ACS_DIAMOND);
259644d93782SGreg Clayton             window.PutChar (ACS_HLINE);
259744d93782SGreg Clayton         }
259844d93782SGreg Clayton     }
259944d93782SGreg Clayton 
260044d93782SGreg Clayton     void
260144d93782SGreg Clayton     DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth)
260244d93782SGreg Clayton     {
260344d93782SGreg Clayton         if (parent)
260444d93782SGreg Clayton             parent->DrawTreeForChild (window, this, reverse_depth + 1);
260544d93782SGreg Clayton 
260644d93782SGreg Clayton         if (&children.back() == child)
260744d93782SGreg Clayton         {
260844d93782SGreg Clayton             // Last child
260944d93782SGreg Clayton             if (reverse_depth == 0)
261044d93782SGreg Clayton             {
261144d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
261244d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
261344d93782SGreg Clayton             }
261444d93782SGreg Clayton             else
261544d93782SGreg Clayton             {
261644d93782SGreg Clayton                 window.PutChar (' ');
261744d93782SGreg Clayton                 window.PutChar (' ');
261844d93782SGreg Clayton             }
261944d93782SGreg Clayton         }
262044d93782SGreg Clayton         else
262144d93782SGreg Clayton         {
262244d93782SGreg Clayton             if (reverse_depth == 0)
262344d93782SGreg Clayton             {
262444d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
262544d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
262644d93782SGreg Clayton             }
262744d93782SGreg Clayton             else
262844d93782SGreg Clayton             {
262944d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
263044d93782SGreg Clayton                 window.PutChar (' ');
263144d93782SGreg Clayton             }
263244d93782SGreg Clayton         }
263344d93782SGreg Clayton     }
263444d93782SGreg Clayton };
263544d93782SGreg Clayton 
263644d93782SGreg Clayton struct DisplayOptions
263744d93782SGreg Clayton {
263844d93782SGreg Clayton     bool show_types;
263944d93782SGreg Clayton };
264044d93782SGreg Clayton 
264144d93782SGreg Clayton class TreeItem;
264244d93782SGreg Clayton 
264344d93782SGreg Clayton class TreeDelegate
264444d93782SGreg Clayton {
264544d93782SGreg Clayton public:
2646c5dac77aSEugene Zelenko     TreeDelegate() = default;
2647315b6884SEugene Zelenko     virtual ~TreeDelegate() = default;
2648315b6884SEugene Zelenko 
264944d93782SGreg Clayton     virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0;
265044d93782SGreg Clayton     virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0;
265144d93782SGreg Clayton     virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views
265244d93782SGreg Clayton };
2653315b6884SEugene Zelenko 
265444d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
265544d93782SGreg Clayton 
265644d93782SGreg Clayton class TreeItem
265744d93782SGreg Clayton {
265844d93782SGreg Clayton public:
265944d93782SGreg Clayton     TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) :
266044d93782SGreg Clayton         m_parent (parent),
266144d93782SGreg Clayton         m_delegate (delegate),
2662c5dac77aSEugene Zelenko         m_user_data(nullptr),
266344d93782SGreg Clayton         m_identifier (0),
266444d93782SGreg Clayton         m_row_idx (-1),
266544d93782SGreg Clayton         m_children (),
266644d93782SGreg Clayton         m_might_have_children (might_have_children),
266744d93782SGreg Clayton         m_is_expanded (false)
266844d93782SGreg Clayton     {
266944d93782SGreg Clayton     }
267044d93782SGreg Clayton 
267144d93782SGreg Clayton     TreeItem &
267244d93782SGreg Clayton     operator=(const TreeItem &rhs)
267344d93782SGreg Clayton     {
267444d93782SGreg Clayton         if (this != &rhs)
267544d93782SGreg Clayton         {
267644d93782SGreg Clayton             m_parent = rhs.m_parent;
267744d93782SGreg Clayton             m_delegate = rhs.m_delegate;
2678ec990867SGreg Clayton             m_user_data = rhs.m_user_data;
267944d93782SGreg Clayton             m_identifier = rhs.m_identifier;
268044d93782SGreg Clayton             m_row_idx = rhs.m_row_idx;
268144d93782SGreg Clayton             m_children = rhs.m_children;
268244d93782SGreg Clayton             m_might_have_children = rhs.m_might_have_children;
268344d93782SGreg Clayton             m_is_expanded = rhs.m_is_expanded;
268444d93782SGreg Clayton         }
268544d93782SGreg Clayton         return *this;
268644d93782SGreg Clayton     }
268744d93782SGreg Clayton 
268844d93782SGreg Clayton     size_t
268944d93782SGreg Clayton     GetDepth () const
269044d93782SGreg Clayton     {
269144d93782SGreg Clayton         if (m_parent)
269244d93782SGreg Clayton             return 1 + m_parent->GetDepth();
269344d93782SGreg Clayton         return 0;
269444d93782SGreg Clayton     }
269544d93782SGreg Clayton 
269644d93782SGreg Clayton     int
269744d93782SGreg Clayton     GetRowIndex () const
269844d93782SGreg Clayton     {
269944d93782SGreg Clayton         return m_row_idx;
270044d93782SGreg Clayton     }
270144d93782SGreg Clayton 
270244d93782SGreg Clayton     void
270344d93782SGreg Clayton     ClearChildren ()
270444d93782SGreg Clayton     {
270544d93782SGreg Clayton         m_children.clear();
270644d93782SGreg Clayton     }
270744d93782SGreg Clayton 
270844d93782SGreg Clayton     void
270944d93782SGreg Clayton     Resize (size_t n, const TreeItem &t)
271044d93782SGreg Clayton     {
271144d93782SGreg Clayton         m_children.resize(n, t);
271244d93782SGreg Clayton     }
271344d93782SGreg Clayton 
271444d93782SGreg Clayton     TreeItem &
271544d93782SGreg Clayton     operator [](size_t i)
271644d93782SGreg Clayton     {
271744d93782SGreg Clayton         return m_children[i];
271844d93782SGreg Clayton     }
271944d93782SGreg Clayton 
272044d93782SGreg Clayton     void
272144d93782SGreg Clayton     SetRowIndex (int row_idx)
272244d93782SGreg Clayton     {
272344d93782SGreg Clayton         m_row_idx = row_idx;
272444d93782SGreg Clayton     }
272544d93782SGreg Clayton 
272644d93782SGreg Clayton     size_t
272744d93782SGreg Clayton     GetNumChildren ()
272844d93782SGreg Clayton     {
272944d93782SGreg Clayton         m_delegate.TreeDelegateGenerateChildren (*this);
273044d93782SGreg Clayton         return m_children.size();
273144d93782SGreg Clayton     }
273244d93782SGreg Clayton 
273344d93782SGreg Clayton     void
273444d93782SGreg Clayton     ItemWasSelected ()
273544d93782SGreg Clayton     {
273644d93782SGreg Clayton         m_delegate.TreeDelegateItemSelected(*this);
273744d93782SGreg Clayton     }
2738315b6884SEugene Zelenko 
273944d93782SGreg Clayton     void
274044d93782SGreg Clayton     CalculateRowIndexes (int &row_idx)
274144d93782SGreg Clayton     {
274244d93782SGreg Clayton         SetRowIndex(row_idx);
274344d93782SGreg Clayton         ++row_idx;
274444d93782SGreg Clayton 
2745ec990867SGreg Clayton         const bool expanded = IsExpanded();
2746ec990867SGreg Clayton 
2747ec990867SGreg Clayton         // The root item must calculate its children,
2748ec990867SGreg Clayton         // or we must calculate the number of children
2749ec990867SGreg Clayton         // if the item is expanded
2750c5dac77aSEugene Zelenko         if (m_parent == nullptr || expanded)
275144d93782SGreg Clayton             GetNumChildren();
275244d93782SGreg Clayton 
275344d93782SGreg Clayton         for (auto &item : m_children)
275444d93782SGreg Clayton         {
275544d93782SGreg Clayton             if (expanded)
275644d93782SGreg Clayton                 item.CalculateRowIndexes(row_idx);
275744d93782SGreg Clayton             else
275844d93782SGreg Clayton                 item.SetRowIndex(-1);
275944d93782SGreg Clayton         }
276044d93782SGreg Clayton     }
276144d93782SGreg Clayton 
276244d93782SGreg Clayton     TreeItem *
276344d93782SGreg Clayton     GetParent ()
276444d93782SGreg Clayton     {
276544d93782SGreg Clayton         return m_parent;
276644d93782SGreg Clayton     }
276744d93782SGreg Clayton 
276844d93782SGreg Clayton     bool
276944d93782SGreg Clayton     IsExpanded () const
277044d93782SGreg Clayton     {
277144d93782SGreg Clayton         return m_is_expanded;
277244d93782SGreg Clayton     }
277344d93782SGreg Clayton 
277444d93782SGreg Clayton     void
277544d93782SGreg Clayton     Expand()
277644d93782SGreg Clayton     {
277744d93782SGreg Clayton         m_is_expanded = true;
277844d93782SGreg Clayton     }
277944d93782SGreg Clayton 
278044d93782SGreg Clayton     void
278144d93782SGreg Clayton     Unexpand ()
278244d93782SGreg Clayton     {
278344d93782SGreg Clayton         m_is_expanded = false;
278444d93782SGreg Clayton     }
278544d93782SGreg Clayton 
278644d93782SGreg Clayton     bool
278744d93782SGreg Clayton     Draw (Window &window,
278844d93782SGreg Clayton           const int first_visible_row,
278944d93782SGreg Clayton           const uint32_t selected_row_idx,
279044d93782SGreg Clayton           int &row_idx,
279144d93782SGreg Clayton           int &num_rows_left)
279244d93782SGreg Clayton     {
279344d93782SGreg Clayton         if (num_rows_left <= 0)
279444d93782SGreg Clayton             return false;
279544d93782SGreg Clayton 
279644d93782SGreg Clayton         if (m_row_idx >= first_visible_row)
279744d93782SGreg Clayton         {
279844d93782SGreg Clayton             window.MoveCursor(2, row_idx + 1);
279944d93782SGreg Clayton 
280044d93782SGreg Clayton             if (m_parent)
280144d93782SGreg Clayton                 m_parent->DrawTreeForChild (window, this, 0);
280244d93782SGreg Clayton 
280344d93782SGreg Clayton             if (m_might_have_children)
280444d93782SGreg Clayton             {
280544d93782SGreg Clayton                 // It we can get UTF8 characters to work we should try to use the "symbol"
280644d93782SGreg Clayton                 // UTF8 string below
280744d93782SGreg Clayton                 //            const char *symbol = "";
280844d93782SGreg Clayton                 //            if (row.expanded)
280944d93782SGreg Clayton                 //                symbol = "\xe2\x96\xbd ";
281044d93782SGreg Clayton                 //            else
281144d93782SGreg Clayton                 //                symbol = "\xe2\x96\xb7 ";
281244d93782SGreg Clayton                 //            window.PutCString (symbol);
281344d93782SGreg Clayton 
281444d93782SGreg Clayton                 // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
281544d93782SGreg Clayton                 // 'v' or '>' character...
281644d93782SGreg Clayton                 //            if (expanded)
281744d93782SGreg Clayton                 //                window.PutChar (ACS_DARROW);
281844d93782SGreg Clayton                 //            else
281944d93782SGreg Clayton                 //                window.PutChar (ACS_RARROW);
282044d93782SGreg Clayton                 // Since we can't find any good looking right arrow/down arrow
282144d93782SGreg Clayton                 // symbols, just use a diamond...
282244d93782SGreg Clayton                 window.PutChar (ACS_DIAMOND);
282344d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
282444d93782SGreg Clayton             }
28253985c8c6SSaleem Abdulrasool             bool highlight =
28263985c8c6SSaleem Abdulrasool               (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive();
282744d93782SGreg Clayton 
282844d93782SGreg Clayton             if (highlight)
282944d93782SGreg Clayton                 window.AttributeOn(A_REVERSE);
283044d93782SGreg Clayton 
283144d93782SGreg Clayton             m_delegate.TreeDelegateDrawTreeItem(*this, window);
283244d93782SGreg Clayton 
283344d93782SGreg Clayton             if (highlight)
283444d93782SGreg Clayton                 window.AttributeOff(A_REVERSE);
283544d93782SGreg Clayton             ++row_idx;
283644d93782SGreg Clayton             --num_rows_left;
283744d93782SGreg Clayton         }
283844d93782SGreg Clayton 
283944d93782SGreg Clayton         if (num_rows_left <= 0)
284044d93782SGreg Clayton             return false; // We are done drawing...
284144d93782SGreg Clayton 
284244d93782SGreg Clayton         if (IsExpanded())
284344d93782SGreg Clayton         {
284444d93782SGreg Clayton             for (auto &item : m_children)
284544d93782SGreg Clayton             {
284644d93782SGreg Clayton                 // If we displayed all the rows and item.Draw() returns
284744d93782SGreg Clayton                 // false we are done drawing and can exit this for loop
2848c5dac77aSEugene Zelenko                 if (!item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left))
284944d93782SGreg Clayton                     break;
285044d93782SGreg Clayton             }
285144d93782SGreg Clayton         }
285244d93782SGreg Clayton         return num_rows_left >= 0; // Return true if not done drawing yet
285344d93782SGreg Clayton     }
285444d93782SGreg Clayton 
285544d93782SGreg Clayton     void
285644d93782SGreg Clayton     DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth)
285744d93782SGreg Clayton     {
285844d93782SGreg Clayton         if (m_parent)
285944d93782SGreg Clayton             m_parent->DrawTreeForChild (window, this, reverse_depth + 1);
286044d93782SGreg Clayton 
286144d93782SGreg Clayton         if (&m_children.back() == child)
286244d93782SGreg Clayton         {
286344d93782SGreg Clayton             // Last child
286444d93782SGreg Clayton             if (reverse_depth == 0)
286544d93782SGreg Clayton             {
286644d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
286744d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
286844d93782SGreg Clayton             }
286944d93782SGreg Clayton             else
287044d93782SGreg Clayton             {
287144d93782SGreg Clayton                 window.PutChar (' ');
287244d93782SGreg Clayton                 window.PutChar (' ');
287344d93782SGreg Clayton             }
287444d93782SGreg Clayton         }
287544d93782SGreg Clayton         else
287644d93782SGreg Clayton         {
287744d93782SGreg Clayton             if (reverse_depth == 0)
287844d93782SGreg Clayton             {
287944d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
288044d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
288144d93782SGreg Clayton             }
288244d93782SGreg Clayton             else
288344d93782SGreg Clayton             {
288444d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
288544d93782SGreg Clayton                 window.PutChar (' ');
288644d93782SGreg Clayton             }
288744d93782SGreg Clayton         }
288844d93782SGreg Clayton     }
288944d93782SGreg Clayton 
289044d93782SGreg Clayton     TreeItem *
289144d93782SGreg Clayton     GetItemForRowIndex (uint32_t row_idx)
289244d93782SGreg Clayton     {
28933985c8c6SSaleem Abdulrasool         if (static_cast<uint32_t>(m_row_idx) == row_idx)
289444d93782SGreg Clayton             return this;
289544d93782SGreg Clayton         if (m_children.empty())
2896c5dac77aSEugene Zelenko             return nullptr;
289744d93782SGreg Clayton         if (IsExpanded())
289844d93782SGreg Clayton         {
289944d93782SGreg Clayton             for (auto &item : m_children)
290044d93782SGreg Clayton             {
290144d93782SGreg Clayton                 TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
290244d93782SGreg Clayton                 if (selected_item_ptr)
290344d93782SGreg Clayton                     return selected_item_ptr;
290444d93782SGreg Clayton             }
290544d93782SGreg Clayton         }
2906c5dac77aSEugene Zelenko         return nullptr;
290744d93782SGreg Clayton     }
290844d93782SGreg Clayton 
2909ec990867SGreg Clayton     void *
2910ec990867SGreg Clayton     GetUserData() const
2911ec990867SGreg Clayton     {
2912ec990867SGreg Clayton         return m_user_data;
2913ec990867SGreg Clayton     }
2914ec990867SGreg Clayton 
2915ec990867SGreg Clayton     void
2916ec990867SGreg Clayton     SetUserData (void *user_data)
2917ec990867SGreg Clayton     {
2918ec990867SGreg Clayton         m_user_data = user_data;
2919ec990867SGreg Clayton     }
2920ec990867SGreg Clayton 
292144d93782SGreg Clayton     uint64_t
292244d93782SGreg Clayton     GetIdentifier() const
292344d93782SGreg Clayton     {
292444d93782SGreg Clayton         return m_identifier;
292544d93782SGreg Clayton     }
292644d93782SGreg Clayton 
292744d93782SGreg Clayton     void
292844d93782SGreg Clayton     SetIdentifier (uint64_t identifier)
292944d93782SGreg Clayton     {
293044d93782SGreg Clayton         m_identifier = identifier;
293144d93782SGreg Clayton     }
293244d93782SGreg Clayton 
2933ec990867SGreg Clayton     void
2934ec990867SGreg Clayton     SetMightHaveChildren (bool b)
2935ec990867SGreg Clayton     {
2936ec990867SGreg Clayton         m_might_have_children = b;
2937ec990867SGreg Clayton     }
2938ec990867SGreg Clayton 
293944d93782SGreg Clayton protected:
294044d93782SGreg Clayton     TreeItem *m_parent;
294144d93782SGreg Clayton     TreeDelegate &m_delegate;
2942ec990867SGreg Clayton     void *m_user_data;
294344d93782SGreg Clayton     uint64_t m_identifier;
294444d93782SGreg Clayton     int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item
294544d93782SGreg Clayton     std::vector<TreeItem> m_children;
294644d93782SGreg Clayton     bool m_might_have_children;
294744d93782SGreg Clayton     bool m_is_expanded;
294844d93782SGreg Clayton };
294944d93782SGreg Clayton 
295044d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate
295144d93782SGreg Clayton {
295244d93782SGreg Clayton public:
295344d93782SGreg Clayton     TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) :
295444d93782SGreg Clayton         m_debugger (debugger),
295544d93782SGreg Clayton         m_delegate_sp (delegate_sp),
2956c5dac77aSEugene Zelenko         m_root(nullptr, *delegate_sp, true),
2957c5dac77aSEugene Zelenko         m_selected_item(nullptr),
295844d93782SGreg Clayton         m_num_rows (0),
295944d93782SGreg Clayton         m_selected_row_idx (0),
296044d93782SGreg Clayton         m_first_visible_row (0),
296144d93782SGreg Clayton         m_min_x (0),
296244d93782SGreg Clayton         m_min_y (0),
296344d93782SGreg Clayton         m_max_x (0),
296444d93782SGreg Clayton         m_max_y (0)
296544d93782SGreg Clayton     {
296644d93782SGreg Clayton     }
296744d93782SGreg Clayton 
296844d93782SGreg Clayton     int
296944d93782SGreg Clayton     NumVisibleRows () const
297044d93782SGreg Clayton     {
297144d93782SGreg Clayton         return m_max_y - m_min_y;
297244d93782SGreg Clayton     }
297344d93782SGreg Clayton 
2974bd5ae6b4SGreg Clayton     bool
2975bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
297644d93782SGreg Clayton     {
297744d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
297844d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
297944d93782SGreg Clayton 
298044d93782SGreg Clayton         bool display_content = false;
298144d93782SGreg Clayton         if (process)
298244d93782SGreg Clayton         {
298344d93782SGreg Clayton             StateType state = process->GetState();
298444d93782SGreg Clayton             if (StateIsStoppedState(state, true))
298544d93782SGreg Clayton             {
298644d93782SGreg Clayton                 // We are stopped, so it is ok to
298744d93782SGreg Clayton                 display_content = true;
298844d93782SGreg Clayton             }
298944d93782SGreg Clayton             else if (StateIsRunningState(state))
299044d93782SGreg Clayton             {
299144d93782SGreg Clayton                 return true; // Don't do any updating when we are running
299244d93782SGreg Clayton             }
299344d93782SGreg Clayton         }
299444d93782SGreg Clayton 
299544d93782SGreg Clayton         m_min_x = 2;
299644d93782SGreg Clayton         m_min_y = 1;
299744d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
299844d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
299944d93782SGreg Clayton 
300044d93782SGreg Clayton         window.Erase();
300144d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
300244d93782SGreg Clayton 
300344d93782SGreg Clayton         if (display_content)
300444d93782SGreg Clayton         {
300544d93782SGreg Clayton             const int num_visible_rows = NumVisibleRows();
300644d93782SGreg Clayton             m_num_rows = 0;
300744d93782SGreg Clayton             m_root.CalculateRowIndexes(m_num_rows);
300844d93782SGreg Clayton 
300944d93782SGreg Clayton             // If we unexpanded while having something selected our
301044d93782SGreg Clayton             // total number of rows is less than the num visible rows,
301144d93782SGreg Clayton             // then make sure we show all the rows by setting the first
301244d93782SGreg Clayton             // visible row accordingly.
301344d93782SGreg Clayton             if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
301444d93782SGreg Clayton                 m_first_visible_row = 0;
301544d93782SGreg Clayton 
301644d93782SGreg Clayton             // Make sure the selected row is always visible
301744d93782SGreg Clayton             if (m_selected_row_idx < m_first_visible_row)
301844d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx;
301944d93782SGreg Clayton             else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
302044d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
302144d93782SGreg Clayton 
302244d93782SGreg Clayton             int row_idx = 0;
302344d93782SGreg Clayton             int num_rows_left = num_visible_rows;
302444d93782SGreg Clayton             m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left);
302544d93782SGreg Clayton             // Get the selected row
302644d93782SGreg Clayton             m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx);
302744d93782SGreg Clayton         }
302844d93782SGreg Clayton         else
302944d93782SGreg Clayton         {
3030c5dac77aSEugene Zelenko             m_selected_item = nullptr;
303144d93782SGreg Clayton         }
303244d93782SGreg Clayton 
303344d93782SGreg Clayton         window.DeferredRefresh();
303444d93782SGreg Clayton 
303544d93782SGreg Clayton         return true; // Drawing handled
303644d93782SGreg Clayton     }
303744d93782SGreg Clayton 
3038bd5ae6b4SGreg Clayton     const char *
3039bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
304044d93782SGreg Clayton     {
304144d93782SGreg Clayton         return "Thread window keyboard shortcuts:";
304244d93782SGreg Clayton     }
304344d93782SGreg Clayton 
3044bd5ae6b4SGreg Clayton     KeyHelp *
3045bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
304644d93782SGreg Clayton     {
304744d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
304844d93782SGreg Clayton             { KEY_UP, "Select previous item" },
304944d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
305044d93782SGreg Clayton             { KEY_RIGHT, "Expand the selected item" },
305144d93782SGreg Clayton             { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" },
305244d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
305344d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
305444d93782SGreg Clayton             { 'h', "Show help dialog" },
305544d93782SGreg Clayton             { ' ', "Toggle item expansion" },
305644d93782SGreg Clayton             { ',', "Page up" },
305744d93782SGreg Clayton             { '.', "Page down" },
3058c5dac77aSEugene Zelenko             { '\0', nullptr }
305944d93782SGreg Clayton         };
306044d93782SGreg Clayton         return g_source_view_key_help;
306144d93782SGreg Clayton     }
306244d93782SGreg Clayton 
3063bd5ae6b4SGreg Clayton     HandleCharResult
3064bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
306544d93782SGreg Clayton     {
306644d93782SGreg Clayton         switch(c)
306744d93782SGreg Clayton         {
306844d93782SGreg Clayton             case ',':
306944d93782SGreg Clayton             case KEY_PPAGE:
307044d93782SGreg Clayton                 // Page up key
307144d93782SGreg Clayton                 if (m_first_visible_row > 0)
307244d93782SGreg Clayton                 {
307344d93782SGreg Clayton                     if (m_first_visible_row > m_max_y)
307444d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
307544d93782SGreg Clayton                     else
307644d93782SGreg Clayton                         m_first_visible_row = 0;
307744d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
307844d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
307944d93782SGreg Clayton                     if (m_selected_item)
308044d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
308144d93782SGreg Clayton                 }
308244d93782SGreg Clayton                 return eKeyHandled;
308344d93782SGreg Clayton 
308444d93782SGreg Clayton             case '.':
308544d93782SGreg Clayton             case KEY_NPAGE:
308644d93782SGreg Clayton                 // Page down key
308744d93782SGreg Clayton                 if (m_num_rows > m_max_y)
308844d93782SGreg Clayton                 {
308944d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
309044d93782SGreg Clayton                     {
309144d93782SGreg Clayton                         m_first_visible_row += m_max_y;
309244d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
309344d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
309444d93782SGreg Clayton                         if (m_selected_item)
309544d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
309644d93782SGreg Clayton                     }
309744d93782SGreg Clayton                 }
309844d93782SGreg Clayton                 return eKeyHandled;
309944d93782SGreg Clayton 
310044d93782SGreg Clayton             case KEY_UP:
310144d93782SGreg Clayton                 if (m_selected_row_idx > 0)
310244d93782SGreg Clayton                 {
310344d93782SGreg Clayton                     --m_selected_row_idx;
310444d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
310544d93782SGreg Clayton                     if (m_selected_item)
310644d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
310744d93782SGreg Clayton                 }
310844d93782SGreg Clayton                 return eKeyHandled;
3109315b6884SEugene Zelenko 
311044d93782SGreg Clayton             case KEY_DOWN:
311144d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
311244d93782SGreg Clayton                 {
311344d93782SGreg Clayton                     ++m_selected_row_idx;
311444d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
311544d93782SGreg Clayton                     if (m_selected_item)
311644d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
311744d93782SGreg Clayton                 }
311844d93782SGreg Clayton                 return eKeyHandled;
311944d93782SGreg Clayton 
312044d93782SGreg Clayton             case KEY_RIGHT:
312144d93782SGreg Clayton                 if (m_selected_item)
312244d93782SGreg Clayton                 {
312344d93782SGreg Clayton                     if (!m_selected_item->IsExpanded())
312444d93782SGreg Clayton                         m_selected_item->Expand();
312544d93782SGreg Clayton                 }
312644d93782SGreg Clayton                 return eKeyHandled;
312744d93782SGreg Clayton 
312844d93782SGreg Clayton             case KEY_LEFT:
312944d93782SGreg Clayton                 if (m_selected_item)
313044d93782SGreg Clayton                 {
313144d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
313244d93782SGreg Clayton                         m_selected_item->Unexpand();
313344d93782SGreg Clayton                     else if (m_selected_item->GetParent())
313444d93782SGreg Clayton                     {
313544d93782SGreg Clayton                         m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
313644d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
313744d93782SGreg Clayton                         if (m_selected_item)
313844d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
313944d93782SGreg Clayton                     }
314044d93782SGreg Clayton                 }
314144d93782SGreg Clayton                 return eKeyHandled;
314244d93782SGreg Clayton 
314344d93782SGreg Clayton             case ' ':
314444d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
314544d93782SGreg Clayton                 if (m_selected_item)
314644d93782SGreg Clayton                 {
314744d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
314844d93782SGreg Clayton                         m_selected_item->Unexpand();
314944d93782SGreg Clayton                     else
315044d93782SGreg Clayton                         m_selected_item->Expand();
315144d93782SGreg Clayton                 }
315244d93782SGreg Clayton                 return eKeyHandled;
315344d93782SGreg Clayton 
315444d93782SGreg Clayton             case 'h':
315544d93782SGreg Clayton                 window.CreateHelpSubwindow ();
315644d93782SGreg Clayton                 return eKeyHandled;
315744d93782SGreg Clayton 
315844d93782SGreg Clayton             default:
315944d93782SGreg Clayton                 break;
316044d93782SGreg Clayton         }
316144d93782SGreg Clayton         return eKeyNotHandled;
316244d93782SGreg Clayton     }
316344d93782SGreg Clayton 
316444d93782SGreg Clayton protected:
316544d93782SGreg Clayton     Debugger &m_debugger;
316644d93782SGreg Clayton     TreeDelegateSP m_delegate_sp;
316744d93782SGreg Clayton     TreeItem m_root;
316844d93782SGreg Clayton     TreeItem *m_selected_item;
316944d93782SGreg Clayton     int m_num_rows;
317044d93782SGreg Clayton     int m_selected_row_idx;
317144d93782SGreg Clayton     int m_first_visible_row;
317244d93782SGreg Clayton     int m_min_x;
317344d93782SGreg Clayton     int m_min_y;
317444d93782SGreg Clayton     int m_max_x;
317544d93782SGreg Clayton     int m_max_y;
317644d93782SGreg Clayton };
317744d93782SGreg Clayton 
317844d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate
317944d93782SGreg Clayton {
318044d93782SGreg Clayton public:
3181ec990867SGreg Clayton     FrameTreeDelegate () :
3182ec990867SGreg Clayton         TreeDelegate()
318344d93782SGreg Clayton     {
3184554f68d3SGreg Clayton         FormatEntity::Parse ("frame #${frame.index}: {${function.name}${function.pc-offset}}}",
3185554f68d3SGreg Clayton                              m_format);
318644d93782SGreg Clayton     }
318744d93782SGreg Clayton 
3188315b6884SEugene Zelenko     ~FrameTreeDelegate() override = default;
318944d93782SGreg Clayton 
3190bd5ae6b4SGreg Clayton     void
3191bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
319244d93782SGreg Clayton     {
3193ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3194ec990867SGreg Clayton         if (thread)
319544d93782SGreg Clayton         {
319644d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3197ec990867SGreg Clayton             StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
319844d93782SGreg Clayton             if (frame_sp)
319944d93782SGreg Clayton             {
320044d93782SGreg Clayton                 StreamString strm;
320144d93782SGreg Clayton                 const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
320244d93782SGreg Clayton                 ExecutionContext exe_ctx (frame_sp);
3203c5dac77aSEugene Zelenko                 if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, nullptr, nullptr, false, false))
320444d93782SGreg Clayton                 {
320544d93782SGreg Clayton                     int right_pad = 1;
320644d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
320744d93782SGreg Clayton                 }
320844d93782SGreg Clayton             }
320944d93782SGreg Clayton         }
321044d93782SGreg Clayton     }
3211315b6884SEugene Zelenko 
3212bd5ae6b4SGreg Clayton     void
3213bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)  override
321444d93782SGreg Clayton     {
321544d93782SGreg Clayton         // No children for frames yet...
321644d93782SGreg Clayton     }
321744d93782SGreg Clayton 
3218bd5ae6b4SGreg Clayton     bool
3219bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
322044d93782SGreg Clayton     {
3221ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3222ec990867SGreg Clayton         if (thread)
322344d93782SGreg Clayton         {
3224ec990867SGreg Clayton             thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID());
322544d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3226ec990867SGreg Clayton             thread->SetSelectedFrameByIndex(frame_idx);
322744d93782SGreg Clayton             return true;
322844d93782SGreg Clayton         }
322944d93782SGreg Clayton         return false;
323044d93782SGreg Clayton     }
3231315b6884SEugene Zelenko 
3232554f68d3SGreg Clayton protected:
3233554f68d3SGreg Clayton     FormatEntity::Entry m_format;
323444d93782SGreg Clayton };
323544d93782SGreg Clayton 
323644d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate
323744d93782SGreg Clayton {
323844d93782SGreg Clayton public:
323944d93782SGreg Clayton     ThreadTreeDelegate (Debugger &debugger) :
324044d93782SGreg Clayton         TreeDelegate(),
324144d93782SGreg Clayton         m_debugger (debugger),
324244d93782SGreg Clayton         m_tid (LLDB_INVALID_THREAD_ID),
324344d93782SGreg Clayton         m_stop_id (UINT32_MAX)
324444d93782SGreg Clayton     {
3245554f68d3SGreg Clayton         FormatEntity::Parse ("thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}",
3246554f68d3SGreg Clayton                              m_format);
324744d93782SGreg Clayton     }
324844d93782SGreg Clayton 
3249315b6884SEugene Zelenko     ~ThreadTreeDelegate() override = default;
325044d93782SGreg Clayton 
3251ec990867SGreg Clayton     ProcessSP
3252ec990867SGreg Clayton     GetProcess ()
3253ec990867SGreg Clayton     {
3254ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3255ec990867SGreg Clayton     }
3256ec990867SGreg Clayton 
3257ec990867SGreg Clayton     ThreadSP
3258ec990867SGreg Clayton     GetThread (const TreeItem &item)
3259ec990867SGreg Clayton     {
3260ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3261ec990867SGreg Clayton         if (process_sp)
3262ec990867SGreg Clayton             return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
3263ec990867SGreg Clayton         return ThreadSP();
3264ec990867SGreg Clayton     }
3265ec990867SGreg Clayton 
3266bd5ae6b4SGreg Clayton     void
3267bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
326844d93782SGreg Clayton     {
3269ec990867SGreg Clayton         ThreadSP thread_sp = GetThread (item);
327044d93782SGreg Clayton         if (thread_sp)
327144d93782SGreg Clayton         {
327244d93782SGreg Clayton             StreamString strm;
327344d93782SGreg Clayton             ExecutionContext exe_ctx (thread_sp);
3274c5dac77aSEugene Zelenko             if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr, nullptr, false, false))
327544d93782SGreg Clayton             {
327644d93782SGreg Clayton                 int right_pad = 1;
327744d93782SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
327844d93782SGreg Clayton             }
327944d93782SGreg Clayton         }
328044d93782SGreg Clayton     }
3281315b6884SEugene Zelenko 
3282bd5ae6b4SGreg Clayton     void
3283bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
328444d93782SGreg Clayton     {
3285ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
328644d93782SGreg Clayton         if (process_sp && process_sp->IsAlive())
328744d93782SGreg Clayton         {
328844d93782SGreg Clayton             StateType state = process_sp->GetState();
328944d93782SGreg Clayton             if (StateIsStoppedState(state, true))
329044d93782SGreg Clayton             {
3291ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
329244d93782SGreg Clayton                 if (thread_sp)
329344d93782SGreg Clayton                 {
329444d93782SGreg Clayton                     if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid)
329544d93782SGreg Clayton                         return; // Children are already up to date
3296ec990867SGreg Clayton                     if (!m_frame_delegate_sp)
329744d93782SGreg Clayton                     {
329844d93782SGreg Clayton                         // Always expand the thread item the first time we show it
3299ec990867SGreg Clayton                         m_frame_delegate_sp.reset (new FrameTreeDelegate());
330044d93782SGreg Clayton                     }
330144d93782SGreg Clayton 
330244d93782SGreg Clayton                     m_stop_id = process_sp->GetStopID();
330344d93782SGreg Clayton                     m_tid = thread_sp->GetID();
330444d93782SGreg Clayton 
330544d93782SGreg Clayton                     TreeItem t (&item, *m_frame_delegate_sp, false);
330644d93782SGreg Clayton                     size_t num_frames = thread_sp->GetStackFrameCount();
330744d93782SGreg Clayton                     item.Resize (num_frames, t);
330844d93782SGreg Clayton                     for (size_t i = 0; i < num_frames; ++i)
330944d93782SGreg Clayton                     {
3310ec990867SGreg Clayton                         item[i].SetUserData(thread_sp.get());
331144d93782SGreg Clayton                         item[i].SetIdentifier(i);
331244d93782SGreg Clayton                     }
331344d93782SGreg Clayton                 }
331444d93782SGreg Clayton                 return;
331544d93782SGreg Clayton             }
331644d93782SGreg Clayton         }
331744d93782SGreg Clayton         item.ClearChildren();
331844d93782SGreg Clayton     }
331944d93782SGreg Clayton 
3320bd5ae6b4SGreg Clayton     bool
3321bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
332244d93782SGreg Clayton     {
3323ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3324ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3325ec990867SGreg Clayton         {
3326ec990867SGreg Clayton             StateType state = process_sp->GetState();
3327ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3328ec990867SGreg Clayton             {
3329ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
333044d93782SGreg Clayton                 if (thread_sp)
333144d93782SGreg Clayton                 {
333244d93782SGreg Clayton                     ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
3333*bb19a13cSSaleem Abdulrasool                     std::lock_guard<std::recursive_mutex> guard(thread_list.GetMutex());
333444d93782SGreg Clayton                     ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
333544d93782SGreg Clayton                     if (selected_thread_sp->GetID() != thread_sp->GetID())
333644d93782SGreg Clayton                     {
333744d93782SGreg Clayton                         thread_list.SetSelectedThreadByID(thread_sp->GetID());
333844d93782SGreg Clayton                         return true;
333944d93782SGreg Clayton                     }
334044d93782SGreg Clayton                 }
3341ec990867SGreg Clayton             }
3342ec990867SGreg Clayton         }
334344d93782SGreg Clayton         return false;
334444d93782SGreg Clayton     }
334544d93782SGreg Clayton 
334644d93782SGreg Clayton protected:
334744d93782SGreg Clayton     Debugger &m_debugger;
334844d93782SGreg Clayton     std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
334944d93782SGreg Clayton     lldb::user_id_t m_tid;
335044d93782SGreg Clayton     uint32_t m_stop_id;
3351554f68d3SGreg Clayton     FormatEntity::Entry m_format;
335244d93782SGreg Clayton };
335344d93782SGreg Clayton 
3354ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate
3355ec990867SGreg Clayton {
3356ec990867SGreg Clayton public:
3357ec990867SGreg Clayton     ThreadsTreeDelegate (Debugger &debugger) :
3358ec990867SGreg Clayton         TreeDelegate(),
3359ec990867SGreg Clayton         m_thread_delegate_sp (),
3360ec990867SGreg Clayton         m_debugger (debugger),
3361ec990867SGreg Clayton         m_stop_id (UINT32_MAX)
3362ec990867SGreg Clayton     {
3363554f68d3SGreg Clayton         FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
3364554f68d3SGreg Clayton                             m_format);
3365ec990867SGreg Clayton     }
3366ec990867SGreg Clayton 
3367315b6884SEugene Zelenko     ~ThreadsTreeDelegate() override = default;
3368ec990867SGreg Clayton 
3369ec990867SGreg Clayton     ProcessSP
3370ec990867SGreg Clayton     GetProcess ()
3371ec990867SGreg Clayton     {
3372ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3373ec990867SGreg Clayton     }
3374ec990867SGreg Clayton 
3375bd5ae6b4SGreg Clayton     void
3376bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
3377ec990867SGreg Clayton     {
3378ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3379ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3380ec990867SGreg Clayton         {
3381ec990867SGreg Clayton             StreamString strm;
3382ec990867SGreg Clayton             ExecutionContext exe_ctx (process_sp);
3383c5dac77aSEugene Zelenko             if (FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr, nullptr, false, false))
3384ec990867SGreg Clayton             {
3385ec990867SGreg Clayton                 int right_pad = 1;
3386ec990867SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
3387ec990867SGreg Clayton             }
3388ec990867SGreg Clayton         }
3389ec990867SGreg Clayton     }
3390ec990867SGreg Clayton 
3391bd5ae6b4SGreg Clayton     void
3392bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
3393ec990867SGreg Clayton     {
3394ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3395ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3396ec990867SGreg Clayton         {
3397ec990867SGreg Clayton             StateType state = process_sp->GetState();
3398ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3399ec990867SGreg Clayton             {
3400ec990867SGreg Clayton                 const uint32_t stop_id = process_sp->GetStopID();
3401ec990867SGreg Clayton                 if (m_stop_id == stop_id)
3402ec990867SGreg Clayton                     return; // Children are already up to date
3403ec990867SGreg Clayton 
3404ec990867SGreg Clayton                 m_stop_id = stop_id;
3405ec990867SGreg Clayton 
3406ec990867SGreg Clayton                 if (!m_thread_delegate_sp)
3407ec990867SGreg Clayton                 {
3408ec990867SGreg Clayton                     // Always expand the thread item the first time we show it
3409ec990867SGreg Clayton                     //item.Expand();
3410ec990867SGreg Clayton                     m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger));
3411ec990867SGreg Clayton                 }
3412ec990867SGreg Clayton 
3413ec990867SGreg Clayton                 TreeItem t (&item, *m_thread_delegate_sp, false);
3414ec990867SGreg Clayton                 ThreadList &threads = process_sp->GetThreadList();
3415*bb19a13cSSaleem Abdulrasool                 std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
3416ec990867SGreg Clayton                 size_t num_threads = threads.GetSize();
3417ec990867SGreg Clayton                 item.Resize (num_threads, t);
3418ec990867SGreg Clayton                 for (size_t i = 0; i < num_threads; ++i)
3419ec990867SGreg Clayton                 {
3420ec990867SGreg Clayton                     item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
3421ec990867SGreg Clayton                     item[i].SetMightHaveChildren(true);
3422ec990867SGreg Clayton                 }
3423ec990867SGreg Clayton                 return;
3424ec990867SGreg Clayton             }
3425ec990867SGreg Clayton         }
3426ec990867SGreg Clayton         item.ClearChildren();
3427ec990867SGreg Clayton     }
3428ec990867SGreg Clayton 
3429bd5ae6b4SGreg Clayton     bool
3430bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
3431ec990867SGreg Clayton     {
3432ec990867SGreg Clayton         return false;
3433ec990867SGreg Clayton     }
3434ec990867SGreg Clayton 
3435ec990867SGreg Clayton protected:
3436ec990867SGreg Clayton     std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
3437ec990867SGreg Clayton     Debugger &m_debugger;
3438ec990867SGreg Clayton     uint32_t m_stop_id;
3439554f68d3SGreg Clayton     FormatEntity::Entry m_format;
3440ec990867SGreg Clayton };
3441ec990867SGreg Clayton 
344244d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate
344344d93782SGreg Clayton {
344444d93782SGreg Clayton public:
344544d93782SGreg Clayton     ValueObjectListDelegate () :
344644d93782SGreg Clayton         m_valobj_list (),
344744d93782SGreg Clayton         m_rows (),
3448c5dac77aSEugene Zelenko         m_selected_row(nullptr),
344944d93782SGreg Clayton         m_selected_row_idx (0),
345044d93782SGreg Clayton         m_first_visible_row (0),
345144d93782SGreg Clayton         m_num_rows (0),
345244d93782SGreg Clayton         m_max_x (0),
345344d93782SGreg Clayton         m_max_y (0)
345444d93782SGreg Clayton     {
345544d93782SGreg Clayton     }
345644d93782SGreg Clayton 
345744d93782SGreg Clayton     ValueObjectListDelegate (ValueObjectList &valobj_list) :
345844d93782SGreg Clayton         m_valobj_list (valobj_list),
345944d93782SGreg Clayton         m_rows (),
3460c5dac77aSEugene Zelenko         m_selected_row(nullptr),
346144d93782SGreg Clayton         m_selected_row_idx (0),
346244d93782SGreg Clayton         m_first_visible_row (0),
346344d93782SGreg Clayton         m_num_rows (0),
346444d93782SGreg Clayton         m_max_x (0),
346544d93782SGreg Clayton         m_max_y (0)
346644d93782SGreg Clayton     {
346744d93782SGreg Clayton         SetValues (valobj_list);
346844d93782SGreg Clayton     }
346944d93782SGreg Clayton 
3470315b6884SEugene Zelenko     ~ValueObjectListDelegate() override = default;
347144d93782SGreg Clayton 
347244d93782SGreg Clayton     void
347344d93782SGreg Clayton     SetValues (ValueObjectList &valobj_list)
347444d93782SGreg Clayton     {
3475c5dac77aSEugene Zelenko         m_selected_row = nullptr;
347644d93782SGreg Clayton         m_selected_row_idx = 0;
347744d93782SGreg Clayton         m_first_visible_row = 0;
347844d93782SGreg Clayton         m_num_rows = 0;
347944d93782SGreg Clayton         m_rows.clear();
348044d93782SGreg Clayton         m_valobj_list = valobj_list;
348144d93782SGreg Clayton         const size_t num_values = m_valobj_list.GetSize();
348244d93782SGreg Clayton         for (size_t i = 0; i < num_values; ++i)
3483c5dac77aSEugene Zelenko             m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), nullptr));
348444d93782SGreg Clayton     }
348544d93782SGreg Clayton 
3486bd5ae6b4SGreg Clayton     bool
3487bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
348844d93782SGreg Clayton     {
348944d93782SGreg Clayton         m_num_rows = 0;
349044d93782SGreg Clayton         m_min_x = 2;
349144d93782SGreg Clayton         m_min_y = 1;
349244d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
349344d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
349444d93782SGreg Clayton 
349544d93782SGreg Clayton         window.Erase();
349644d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
349744d93782SGreg Clayton 
349844d93782SGreg Clayton         const int num_visible_rows = NumVisibleRows();
349944d93782SGreg Clayton         const int num_rows = CalculateTotalNumberRows (m_rows);
350044d93782SGreg Clayton 
350144d93782SGreg Clayton         // If we unexpanded while having something selected our
350244d93782SGreg Clayton         // total number of rows is less than the num visible rows,
350344d93782SGreg Clayton         // then make sure we show all the rows by setting the first
350444d93782SGreg Clayton         // visible row accordingly.
350544d93782SGreg Clayton         if (m_first_visible_row > 0 && num_rows < num_visible_rows)
350644d93782SGreg Clayton             m_first_visible_row = 0;
350744d93782SGreg Clayton 
350844d93782SGreg Clayton         // Make sure the selected row is always visible
350944d93782SGreg Clayton         if (m_selected_row_idx < m_first_visible_row)
351044d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx;
351144d93782SGreg Clayton         else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
351244d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
351344d93782SGreg Clayton 
351444d93782SGreg Clayton         DisplayRows (window, m_rows, g_options);
351544d93782SGreg Clayton 
351644d93782SGreg Clayton         window.DeferredRefresh();
351744d93782SGreg Clayton 
351844d93782SGreg Clayton         // Get the selected row
351944d93782SGreg Clayton         m_selected_row = GetRowForRowIndex (m_selected_row_idx);
352044d93782SGreg Clayton         // Keep the cursor on the selected row so the highlight and the cursor
352144d93782SGreg Clayton         // are always on the same line
352244d93782SGreg Clayton         if (m_selected_row)
352344d93782SGreg Clayton             window.MoveCursor (m_selected_row->x,
352444d93782SGreg Clayton                                m_selected_row->y);
352544d93782SGreg Clayton 
352644d93782SGreg Clayton         return true; // Drawing handled
352744d93782SGreg Clayton     }
352844d93782SGreg Clayton 
3529bd5ae6b4SGreg Clayton     KeyHelp *
3530bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
353144d93782SGreg Clayton     {
353244d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
353344d93782SGreg Clayton             { KEY_UP, "Select previous item" },
353444d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
353544d93782SGreg Clayton             { KEY_RIGHT, "Expand selected item" },
353644d93782SGreg Clayton             { KEY_LEFT, "Unexpand selected item or select parent if not expanded" },
353744d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
353844d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
353944d93782SGreg Clayton             { 'A', "Format as annotated address" },
354044d93782SGreg Clayton             { 'b', "Format as binary" },
354144d93782SGreg Clayton             { 'B', "Format as hex bytes with ASCII" },
354244d93782SGreg Clayton             { 'c', "Format as character" },
354344d93782SGreg Clayton             { 'd', "Format as a signed integer" },
354444d93782SGreg Clayton             { 'D', "Format selected value using the default format for the type" },
354544d93782SGreg Clayton             { 'f', "Format as float" },
354644d93782SGreg Clayton             { 'h', "Show help dialog" },
354744d93782SGreg Clayton             { 'i', "Format as instructions" },
354844d93782SGreg Clayton             { 'o', "Format as octal" },
354944d93782SGreg Clayton             { 'p', "Format as pointer" },
355044d93782SGreg Clayton             { 's', "Format as C string" },
355144d93782SGreg Clayton             { 't', "Toggle showing/hiding type names" },
355244d93782SGreg Clayton             { 'u', "Format as an unsigned integer" },
355344d93782SGreg Clayton             { 'x', "Format as hex" },
355444d93782SGreg Clayton             { 'X', "Format as uppercase hex" },
355544d93782SGreg Clayton             { ' ', "Toggle item expansion" },
355644d93782SGreg Clayton             { ',', "Page up" },
355744d93782SGreg Clayton             { '.', "Page down" },
3558c5dac77aSEugene Zelenko             { '\0', nullptr }
355944d93782SGreg Clayton         };
356044d93782SGreg Clayton         return g_source_view_key_help;
356144d93782SGreg Clayton     }
356244d93782SGreg Clayton 
3563bd5ae6b4SGreg Clayton     HandleCharResult
3564bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
356544d93782SGreg Clayton     {
356644d93782SGreg Clayton         switch(c)
356744d93782SGreg Clayton         {
356844d93782SGreg Clayton             case 'x':
356944d93782SGreg Clayton             case 'X':
357044d93782SGreg Clayton             case 'o':
357144d93782SGreg Clayton             case 's':
357244d93782SGreg Clayton             case 'u':
357344d93782SGreg Clayton             case 'd':
357444d93782SGreg Clayton             case 'D':
357544d93782SGreg Clayton             case 'i':
357644d93782SGreg Clayton             case 'A':
357744d93782SGreg Clayton             case 'p':
357844d93782SGreg Clayton             case 'c':
357944d93782SGreg Clayton             case 'b':
358044d93782SGreg Clayton             case 'B':
358144d93782SGreg Clayton             case 'f':
358244d93782SGreg Clayton                 // Change the format for the currently selected item
358344d93782SGreg Clayton                 if (m_selected_row)
358444d93782SGreg Clayton                     m_selected_row->valobj->SetFormat (FormatForChar (c));
358544d93782SGreg Clayton                 return eKeyHandled;
358644d93782SGreg Clayton 
358744d93782SGreg Clayton             case 't':
358844d93782SGreg Clayton                 // Toggle showing type names
358944d93782SGreg Clayton                 g_options.show_types = !g_options.show_types;
359044d93782SGreg Clayton                 return eKeyHandled;
359144d93782SGreg Clayton 
359244d93782SGreg Clayton             case ',':
359344d93782SGreg Clayton             case KEY_PPAGE:
359444d93782SGreg Clayton                 // Page up key
359544d93782SGreg Clayton                 if (m_first_visible_row > 0)
359644d93782SGreg Clayton                 {
35973985c8c6SSaleem Abdulrasool                     if (static_cast<int>(m_first_visible_row) > m_max_y)
359844d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
359944d93782SGreg Clayton                     else
360044d93782SGreg Clayton                         m_first_visible_row = 0;
360144d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
360244d93782SGreg Clayton                 }
360344d93782SGreg Clayton                 return eKeyHandled;
360444d93782SGreg Clayton 
360544d93782SGreg Clayton             case '.':
360644d93782SGreg Clayton             case KEY_NPAGE:
360744d93782SGreg Clayton                 // Page down key
36083985c8c6SSaleem Abdulrasool                 if (m_num_rows > static_cast<size_t>(m_max_y))
360944d93782SGreg Clayton                 {
361044d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
361144d93782SGreg Clayton                     {
361244d93782SGreg Clayton                         m_first_visible_row += m_max_y;
361344d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
361444d93782SGreg Clayton                     }
361544d93782SGreg Clayton                 }
361644d93782SGreg Clayton                 return eKeyHandled;
361744d93782SGreg Clayton 
361844d93782SGreg Clayton             case KEY_UP:
361944d93782SGreg Clayton                 if (m_selected_row_idx > 0)
362044d93782SGreg Clayton                     --m_selected_row_idx;
362144d93782SGreg Clayton                 return eKeyHandled;
3622315b6884SEugene Zelenko 
362344d93782SGreg Clayton             case KEY_DOWN:
362444d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
362544d93782SGreg Clayton                     ++m_selected_row_idx;
362644d93782SGreg Clayton                 return eKeyHandled;
362744d93782SGreg Clayton 
362844d93782SGreg Clayton             case KEY_RIGHT:
362944d93782SGreg Clayton                 if (m_selected_row)
363044d93782SGreg Clayton                 {
363144d93782SGreg Clayton                     if (!m_selected_row->expanded)
363244d93782SGreg Clayton                         m_selected_row->Expand();
363344d93782SGreg Clayton                 }
363444d93782SGreg Clayton                 return eKeyHandled;
363544d93782SGreg Clayton 
363644d93782SGreg Clayton             case KEY_LEFT:
363744d93782SGreg Clayton                 if (m_selected_row)
363844d93782SGreg Clayton                 {
363944d93782SGreg Clayton                     if (m_selected_row->expanded)
364044d93782SGreg Clayton                         m_selected_row->Unexpand();
364144d93782SGreg Clayton                     else if (m_selected_row->parent)
364244d93782SGreg Clayton                         m_selected_row_idx = m_selected_row->parent->row_idx;
364344d93782SGreg Clayton                 }
364444d93782SGreg Clayton                 return eKeyHandled;
364544d93782SGreg Clayton 
364644d93782SGreg Clayton             case ' ':
364744d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
364844d93782SGreg Clayton                 if (m_selected_row)
364944d93782SGreg Clayton                 {
365044d93782SGreg Clayton                     if (m_selected_row->expanded)
365144d93782SGreg Clayton                         m_selected_row->Unexpand();
365244d93782SGreg Clayton                     else
365344d93782SGreg Clayton                         m_selected_row->Expand();
365444d93782SGreg Clayton                 }
365544d93782SGreg Clayton                 return eKeyHandled;
365644d93782SGreg Clayton 
365744d93782SGreg Clayton             case 'h':
365844d93782SGreg Clayton                 window.CreateHelpSubwindow ();
365944d93782SGreg Clayton                 return eKeyHandled;
366044d93782SGreg Clayton 
366144d93782SGreg Clayton             default:
366244d93782SGreg Clayton                 break;
366344d93782SGreg Clayton         }
366444d93782SGreg Clayton         return eKeyNotHandled;
366544d93782SGreg Clayton     }
366644d93782SGreg Clayton 
366744d93782SGreg Clayton protected:
366844d93782SGreg Clayton     ValueObjectList m_valobj_list;
366944d93782SGreg Clayton     std::vector<Row> m_rows;
367044d93782SGreg Clayton     Row *m_selected_row;
367144d93782SGreg Clayton     uint32_t m_selected_row_idx;
367244d93782SGreg Clayton     uint32_t m_first_visible_row;
367344d93782SGreg Clayton     uint32_t m_num_rows;
367444d93782SGreg Clayton     int m_min_x;
367544d93782SGreg Clayton     int m_min_y;
367644d93782SGreg Clayton     int m_max_x;
367744d93782SGreg Clayton     int m_max_y;
367844d93782SGreg Clayton 
367944d93782SGreg Clayton     static Format
368044d93782SGreg Clayton     FormatForChar (int c)
368144d93782SGreg Clayton     {
368244d93782SGreg Clayton         switch (c)
368344d93782SGreg Clayton         {
368444d93782SGreg Clayton             case 'x': return eFormatHex;
368544d93782SGreg Clayton             case 'X': return eFormatHexUppercase;
368644d93782SGreg Clayton             case 'o': return eFormatOctal;
368744d93782SGreg Clayton             case 's': return eFormatCString;
368844d93782SGreg Clayton             case 'u': return eFormatUnsigned;
368944d93782SGreg Clayton             case 'd': return eFormatDecimal;
369044d93782SGreg Clayton             case 'D': return eFormatDefault;
369144d93782SGreg Clayton             case 'i': return eFormatInstruction;
369244d93782SGreg Clayton             case 'A': return eFormatAddressInfo;
369344d93782SGreg Clayton             case 'p': return eFormatPointer;
369444d93782SGreg Clayton             case 'c': return eFormatChar;
369544d93782SGreg Clayton             case 'b': return eFormatBinary;
369644d93782SGreg Clayton             case 'B': return eFormatBytesWithASCII;
369744d93782SGreg Clayton             case 'f': return eFormatFloat;
369844d93782SGreg Clayton         }
369944d93782SGreg Clayton         return eFormatDefault;
370044d93782SGreg Clayton     }
370144d93782SGreg Clayton 
370244d93782SGreg Clayton     bool
370344d93782SGreg Clayton     DisplayRowObject (Window &window,
370444d93782SGreg Clayton                       Row &row,
370544d93782SGreg Clayton                       DisplayOptions &options,
370644d93782SGreg Clayton                       bool highlight,
370744d93782SGreg Clayton                       bool last_child)
370844d93782SGreg Clayton     {
370944d93782SGreg Clayton         ValueObject *valobj = row.valobj.get();
371044d93782SGreg Clayton 
3711c5dac77aSEugene Zelenko         if (valobj == nullptr)
371244d93782SGreg Clayton             return false;
371344d93782SGreg Clayton 
3714c5dac77aSEugene Zelenko         const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : nullptr;
371544d93782SGreg Clayton         const char *name = valobj->GetName().GetCString();
371644d93782SGreg Clayton         const char *value = valobj->GetValueAsCString ();
371744d93782SGreg Clayton         const char *summary = valobj->GetSummaryAsCString ();
371844d93782SGreg Clayton 
371944d93782SGreg Clayton         window.MoveCursor (row.x, row.y);
372044d93782SGreg Clayton 
372144d93782SGreg Clayton         row.DrawTree (window);
372244d93782SGreg Clayton 
372344d93782SGreg Clayton         if (highlight)
372444d93782SGreg Clayton             window.AttributeOn(A_REVERSE);
372544d93782SGreg Clayton 
372644d93782SGreg Clayton         if (type_name && type_name[0])
372744d93782SGreg Clayton             window.Printf ("(%s) ", type_name);
372844d93782SGreg Clayton 
372944d93782SGreg Clayton         if (name && name[0])
373044d93782SGreg Clayton             window.PutCString(name);
373144d93782SGreg Clayton 
373244d93782SGreg Clayton         attr_t changd_attr = 0;
373344d93782SGreg Clayton         if (valobj->GetValueDidChange())
373444d93782SGreg Clayton             changd_attr = COLOR_PAIR(5) | A_BOLD;
373544d93782SGreg Clayton 
373644d93782SGreg Clayton         if (value && value[0])
373744d93782SGreg Clayton         {
373844d93782SGreg Clayton             window.PutCString(" = ");
373944d93782SGreg Clayton             if (changd_attr)
374044d93782SGreg Clayton                 window.AttributeOn(changd_attr);
374144d93782SGreg Clayton             window.PutCString (value);
374244d93782SGreg Clayton             if (changd_attr)
374344d93782SGreg Clayton                 window.AttributeOff(changd_attr);
374444d93782SGreg Clayton         }
374544d93782SGreg Clayton 
374644d93782SGreg Clayton         if (summary && summary[0])
374744d93782SGreg Clayton         {
374844d93782SGreg Clayton             window.PutChar(' ');
374944d93782SGreg Clayton             if (changd_attr)
375044d93782SGreg Clayton                 window.AttributeOn(changd_attr);
375144d93782SGreg Clayton             window.PutCString(summary);
375244d93782SGreg Clayton             if (changd_attr)
375344d93782SGreg Clayton                 window.AttributeOff(changd_attr);
375444d93782SGreg Clayton         }
375544d93782SGreg Clayton 
375644d93782SGreg Clayton         if (highlight)
375744d93782SGreg Clayton             window.AttributeOff (A_REVERSE);
375844d93782SGreg Clayton 
375944d93782SGreg Clayton         return true;
376044d93782SGreg Clayton     }
3761315b6884SEugene Zelenko 
376244d93782SGreg Clayton     void
376344d93782SGreg Clayton     DisplayRows (Window &window,
376444d93782SGreg Clayton                  std::vector<Row> &rows,
376544d93782SGreg Clayton                  DisplayOptions &options)
376644d93782SGreg Clayton     {
376744d93782SGreg Clayton         // >   0x25B7
376844d93782SGreg Clayton         // \/  0x25BD
376944d93782SGreg Clayton 
377044d93782SGreg Clayton         bool window_is_active = window.IsActive();
377144d93782SGreg Clayton         for (auto &row : rows)
377244d93782SGreg Clayton         {
377344d93782SGreg Clayton             const bool last_child = row.parent && &rows[rows.size()-1] == &row;
377444d93782SGreg Clayton             // Save the row index in each Row structure
377544d93782SGreg Clayton             row.row_idx = m_num_rows;
377644d93782SGreg Clayton             if ((m_num_rows >= m_first_visible_row) &&
37773985c8c6SSaleem Abdulrasool                 ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows())))
377844d93782SGreg Clayton             {
377944d93782SGreg Clayton                 row.x = m_min_x;
378044d93782SGreg Clayton                 row.y = m_num_rows - m_first_visible_row + 1;
378144d93782SGreg Clayton                 if (DisplayRowObject (window,
378244d93782SGreg Clayton                                       row,
378344d93782SGreg Clayton                                       options,
378444d93782SGreg Clayton                                       window_is_active && m_num_rows == m_selected_row_idx,
378544d93782SGreg Clayton                                       last_child))
378644d93782SGreg Clayton                 {
378744d93782SGreg Clayton                     ++m_num_rows;
378844d93782SGreg Clayton                 }
378944d93782SGreg Clayton                 else
379044d93782SGreg Clayton                 {
379144d93782SGreg Clayton                     row.x = 0;
379244d93782SGreg Clayton                     row.y = 0;
379344d93782SGreg Clayton                 }
379444d93782SGreg Clayton             }
379544d93782SGreg Clayton             else
379644d93782SGreg Clayton             {
379744d93782SGreg Clayton                 row.x = 0;
379844d93782SGreg Clayton                 row.y = 0;
379944d93782SGreg Clayton                 ++m_num_rows;
380044d93782SGreg Clayton             }
380144d93782SGreg Clayton 
380244d93782SGreg Clayton             if (row.expanded && !row.children.empty())
380344d93782SGreg Clayton             {
380444d93782SGreg Clayton                 DisplayRows (window,
380544d93782SGreg Clayton                              row.children,
380644d93782SGreg Clayton                              options);
380744d93782SGreg Clayton             }
380844d93782SGreg Clayton         }
380944d93782SGreg Clayton     }
381044d93782SGreg Clayton 
381144d93782SGreg Clayton     int
381244d93782SGreg Clayton     CalculateTotalNumberRows (const std::vector<Row> &rows)
381344d93782SGreg Clayton     {
381444d93782SGreg Clayton         int row_count = 0;
381544d93782SGreg Clayton         for (const auto &row : rows)
381644d93782SGreg Clayton         {
381744d93782SGreg Clayton             ++row_count;
381844d93782SGreg Clayton             if (row.expanded)
381944d93782SGreg Clayton                 row_count += CalculateTotalNumberRows(row.children);
382044d93782SGreg Clayton         }
382144d93782SGreg Clayton         return row_count;
382244d93782SGreg Clayton     }
3823315b6884SEugene Zelenko 
382444d93782SGreg Clayton     static Row *
382544d93782SGreg Clayton     GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index)
382644d93782SGreg Clayton     {
382744d93782SGreg Clayton         for (auto &row : rows)
382844d93782SGreg Clayton         {
382944d93782SGreg Clayton             if (row_index == 0)
383044d93782SGreg Clayton                 return &row;
383144d93782SGreg Clayton             else
383244d93782SGreg Clayton             {
383344d93782SGreg Clayton                 --row_index;
383444d93782SGreg Clayton                 if (row.expanded && !row.children.empty())
383544d93782SGreg Clayton                 {
383644d93782SGreg Clayton                     Row *result = GetRowForRowIndexImpl (row.children, row_index);
383744d93782SGreg Clayton                     if (result)
383844d93782SGreg Clayton                         return result;
383944d93782SGreg Clayton                 }
384044d93782SGreg Clayton             }
384144d93782SGreg Clayton         }
3842c5dac77aSEugene Zelenko         return nullptr;
384344d93782SGreg Clayton     }
384444d93782SGreg Clayton 
384544d93782SGreg Clayton     Row *
384644d93782SGreg Clayton     GetRowForRowIndex (size_t row_index)
384744d93782SGreg Clayton     {
384844d93782SGreg Clayton         return GetRowForRowIndexImpl (m_rows, row_index);
384944d93782SGreg Clayton     }
385044d93782SGreg Clayton 
385144d93782SGreg Clayton     int
385244d93782SGreg Clayton     NumVisibleRows () const
385344d93782SGreg Clayton     {
385444d93782SGreg Clayton         return m_max_y - m_min_y;
385544d93782SGreg Clayton     }
385644d93782SGreg Clayton 
385744d93782SGreg Clayton     static DisplayOptions g_options;
385844d93782SGreg Clayton };
385944d93782SGreg Clayton 
386044d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate
386144d93782SGreg Clayton {
386244d93782SGreg Clayton public:
386344d93782SGreg Clayton     FrameVariablesWindowDelegate (Debugger &debugger) :
386444d93782SGreg Clayton         ValueObjectListDelegate (),
386544d93782SGreg Clayton         m_debugger (debugger),
3866c5dac77aSEugene Zelenko         m_frame_block(nullptr)
386744d93782SGreg Clayton     {
386844d93782SGreg Clayton     }
386944d93782SGreg Clayton 
3870315b6884SEugene Zelenko     ~FrameVariablesWindowDelegate() override = default;
387144d93782SGreg Clayton 
3872bd5ae6b4SGreg Clayton     const char *
3873bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
387444d93782SGreg Clayton     {
387544d93782SGreg Clayton         return "Frame variable window keyboard shortcuts:";
387644d93782SGreg Clayton     }
387744d93782SGreg Clayton 
3878bd5ae6b4SGreg Clayton     bool
3879bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
388044d93782SGreg Clayton     {
388144d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
388244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
3883c5dac77aSEugene Zelenko         Block *frame_block = nullptr;
3884c5dac77aSEugene Zelenko         StackFrame *frame = nullptr;
388544d93782SGreg Clayton 
388644d93782SGreg Clayton         if (process)
388744d93782SGreg Clayton         {
388844d93782SGreg Clayton             StateType state = process->GetState();
388944d93782SGreg Clayton             if (StateIsStoppedState(state, true))
389044d93782SGreg Clayton             {
389144d93782SGreg Clayton                 frame = exe_ctx.GetFramePtr();
389244d93782SGreg Clayton                 if (frame)
389344d93782SGreg Clayton                     frame_block = frame->GetFrameBlock ();
389444d93782SGreg Clayton             }
389544d93782SGreg Clayton             else if (StateIsRunningState(state))
389644d93782SGreg Clayton             {
389744d93782SGreg Clayton                 return true; // Don't do any updating when we are running
389844d93782SGreg Clayton             }
389944d93782SGreg Clayton         }
390044d93782SGreg Clayton 
390144d93782SGreg Clayton         ValueObjectList local_values;
390244d93782SGreg Clayton         if (frame_block)
390344d93782SGreg Clayton         {
390444d93782SGreg Clayton             // Only update the variables if they have changed
390544d93782SGreg Clayton             if (m_frame_block != frame_block)
390644d93782SGreg Clayton             {
390744d93782SGreg Clayton                 m_frame_block = frame_block;
390844d93782SGreg Clayton 
390944d93782SGreg Clayton                 VariableList *locals = frame->GetVariableList(true);
391044d93782SGreg Clayton                 if (locals)
391144d93782SGreg Clayton                 {
391244d93782SGreg Clayton                     const DynamicValueType use_dynamic = eDynamicDontRunTarget;
391344d93782SGreg Clayton                     const size_t num_locals = locals->GetSize();
391444d93782SGreg Clayton                     for (size_t i = 0; i < num_locals; ++i)
3915eb72dc7dSGreg Clayton                     {
3916eb72dc7dSGreg Clayton                         ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic);
3917eb72dc7dSGreg Clayton                         if (value_sp)
3918eb72dc7dSGreg Clayton                         {
3919eb72dc7dSGreg Clayton                             ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3920eb72dc7dSGreg Clayton                             if (synthetic_value_sp)
3921eb72dc7dSGreg Clayton                                 local_values.Append(synthetic_value_sp);
3922eb72dc7dSGreg Clayton                             else
3923eb72dc7dSGreg Clayton                                 local_values.Append(value_sp);
3924eb72dc7dSGreg Clayton                         }
3925eb72dc7dSGreg Clayton                     }
392644d93782SGreg Clayton                     // Update the values
392744d93782SGreg Clayton                     SetValues(local_values);
392844d93782SGreg Clayton                 }
392944d93782SGreg Clayton             }
393044d93782SGreg Clayton         }
393144d93782SGreg Clayton         else
393244d93782SGreg Clayton         {
3933c5dac77aSEugene Zelenko             m_frame_block = nullptr;
393444d93782SGreg Clayton             // Update the values with an empty list if there is no frame
393544d93782SGreg Clayton             SetValues(local_values);
393644d93782SGreg Clayton         }
393744d93782SGreg Clayton 
393844d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
393944d93782SGreg Clayton     }
394044d93782SGreg Clayton 
394144d93782SGreg Clayton protected:
394244d93782SGreg Clayton     Debugger &m_debugger;
394344d93782SGreg Clayton     Block *m_frame_block;
394444d93782SGreg Clayton };
394544d93782SGreg Clayton 
394644d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate
394744d93782SGreg Clayton {
394844d93782SGreg Clayton public:
394944d93782SGreg Clayton     RegistersWindowDelegate (Debugger &debugger) :
395044d93782SGreg Clayton         ValueObjectListDelegate (),
395144d93782SGreg Clayton         m_debugger (debugger)
395244d93782SGreg Clayton     {
395344d93782SGreg Clayton     }
395444d93782SGreg Clayton 
3955315b6884SEugene Zelenko     ~RegistersWindowDelegate() override = default;
395644d93782SGreg Clayton 
3957bd5ae6b4SGreg Clayton     const char *
3958bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
395944d93782SGreg Clayton     {
396044d93782SGreg Clayton         return "Register window keyboard shortcuts:";
396144d93782SGreg Clayton     }
396244d93782SGreg Clayton 
3963bd5ae6b4SGreg Clayton     bool
3964bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
396544d93782SGreg Clayton     {
396644d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
396744d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
396844d93782SGreg Clayton 
396944d93782SGreg Clayton         ValueObjectList value_list;
397044d93782SGreg Clayton         if (frame)
397144d93782SGreg Clayton         {
397244d93782SGreg Clayton             if (frame->GetStackID() != m_stack_id)
397344d93782SGreg Clayton             {
397444d93782SGreg Clayton                 m_stack_id = frame->GetStackID();
397544d93782SGreg Clayton                 RegisterContextSP reg_ctx (frame->GetRegisterContext());
397644d93782SGreg Clayton                 if (reg_ctx)
397744d93782SGreg Clayton                 {
397844d93782SGreg Clayton                     const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
397944d93782SGreg Clayton                     for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx)
398044d93782SGreg Clayton                     {
398144d93782SGreg Clayton                         value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx));
398244d93782SGreg Clayton                     }
398344d93782SGreg Clayton                 }
398444d93782SGreg Clayton                 SetValues(value_list);
398544d93782SGreg Clayton             }
398644d93782SGreg Clayton         }
398744d93782SGreg Clayton         else
398844d93782SGreg Clayton         {
398944d93782SGreg Clayton             Process *process = exe_ctx.GetProcessPtr();
399044d93782SGreg Clayton             if (process && process->IsAlive())
399144d93782SGreg Clayton                 return true; // Don't do any updating if we are running
399244d93782SGreg Clayton             else
399344d93782SGreg Clayton             {
399444d93782SGreg Clayton                 // Update the values with an empty list if there
399544d93782SGreg Clayton                 // is no process or the process isn't alive anymore
399644d93782SGreg Clayton                 SetValues(value_list);
399744d93782SGreg Clayton             }
399844d93782SGreg Clayton         }
399944d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
400044d93782SGreg Clayton     }
400144d93782SGreg Clayton 
400244d93782SGreg Clayton protected:
400344d93782SGreg Clayton     Debugger &m_debugger;
400444d93782SGreg Clayton     StackID m_stack_id;
400544d93782SGreg Clayton };
400644d93782SGreg Clayton 
400744d93782SGreg Clayton static const char *
400844d93782SGreg Clayton CursesKeyToCString (int ch)
400944d93782SGreg Clayton {
401044d93782SGreg Clayton     static char g_desc[32];
401144d93782SGreg Clayton     if (ch >= KEY_F0 && ch < KEY_F0 + 64)
401244d93782SGreg Clayton     {
401344d93782SGreg Clayton         snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
401444d93782SGreg Clayton         return g_desc;
401544d93782SGreg Clayton     }
401644d93782SGreg Clayton     switch (ch)
401744d93782SGreg Clayton     {
401844d93782SGreg Clayton         case KEY_DOWN:  return "down";
401944d93782SGreg Clayton         case KEY_UP:    return "up";
402044d93782SGreg Clayton         case KEY_LEFT:  return "left";
402144d93782SGreg Clayton         case KEY_RIGHT: return "right";
402244d93782SGreg Clayton         case KEY_HOME:  return "home";
402344d93782SGreg Clayton         case KEY_BACKSPACE: return "backspace";
402444d93782SGreg Clayton         case KEY_DL:        return "delete-line";
402544d93782SGreg Clayton         case KEY_IL:        return "insert-line";
402644d93782SGreg Clayton         case KEY_DC:        return "delete-char";
402744d93782SGreg Clayton         case KEY_IC:        return "insert-char";
402844d93782SGreg Clayton         case KEY_CLEAR:     return "clear";
402944d93782SGreg Clayton         case KEY_EOS:       return "clear-to-eos";
403044d93782SGreg Clayton         case KEY_EOL:       return "clear-to-eol";
403144d93782SGreg Clayton         case KEY_SF:        return "scroll-forward";
403244d93782SGreg Clayton         case KEY_SR:        return "scroll-backward";
403344d93782SGreg Clayton         case KEY_NPAGE:     return "page-down";
403444d93782SGreg Clayton         case KEY_PPAGE:     return "page-up";
403544d93782SGreg Clayton         case KEY_STAB:      return "set-tab";
403644d93782SGreg Clayton         case KEY_CTAB:      return "clear-tab";
403744d93782SGreg Clayton         case KEY_CATAB:     return "clear-all-tabs";
403844d93782SGreg Clayton         case KEY_ENTER:     return "enter";
403944d93782SGreg Clayton         case KEY_PRINT:     return "print";
404044d93782SGreg Clayton         case KEY_LL:        return "lower-left key";
404144d93782SGreg Clayton         case KEY_A1:        return "upper left of keypad";
404244d93782SGreg Clayton         case KEY_A3:        return "upper right of keypad";
404344d93782SGreg Clayton         case KEY_B2:        return "center of keypad";
404444d93782SGreg Clayton         case KEY_C1:        return "lower left of keypad";
404544d93782SGreg Clayton         case KEY_C3:        return "lower right of keypad";
404644d93782SGreg Clayton         case KEY_BTAB:      return "back-tab key";
404744d93782SGreg Clayton         case KEY_BEG:       return "begin key";
404844d93782SGreg Clayton         case KEY_CANCEL:    return "cancel key";
404944d93782SGreg Clayton         case KEY_CLOSE:     return "close key";
405044d93782SGreg Clayton         case KEY_COMMAND:   return "command key";
405144d93782SGreg Clayton         case KEY_COPY:      return "copy key";
405244d93782SGreg Clayton         case KEY_CREATE:    return "create key";
405344d93782SGreg Clayton         case KEY_END:       return "end key";
405444d93782SGreg Clayton         case KEY_EXIT:      return "exit key";
405544d93782SGreg Clayton         case KEY_FIND:      return "find key";
405644d93782SGreg Clayton         case KEY_HELP:      return "help key";
405744d93782SGreg Clayton         case KEY_MARK:      return "mark key";
405844d93782SGreg Clayton         case KEY_MESSAGE:   return "message key";
405944d93782SGreg Clayton         case KEY_MOVE:      return "move key";
406044d93782SGreg Clayton         case KEY_NEXT:      return "next key";
406144d93782SGreg Clayton         case KEY_OPEN:      return "open key";
406244d93782SGreg Clayton         case KEY_OPTIONS:   return "options key";
406344d93782SGreg Clayton         case KEY_PREVIOUS:  return "previous key";
406444d93782SGreg Clayton         case KEY_REDO:      return "redo key";
406544d93782SGreg Clayton         case KEY_REFERENCE: return "reference key";
406644d93782SGreg Clayton         case KEY_REFRESH:   return "refresh key";
406744d93782SGreg Clayton         case KEY_REPLACE:   return "replace key";
406844d93782SGreg Clayton         case KEY_RESTART:   return "restart key";
406944d93782SGreg Clayton         case KEY_RESUME:    return "resume key";
407044d93782SGreg Clayton         case KEY_SAVE:      return "save key";
407144d93782SGreg Clayton         case KEY_SBEG:      return "shifted begin key";
407244d93782SGreg Clayton         case KEY_SCANCEL:   return "shifted cancel key";
407344d93782SGreg Clayton         case KEY_SCOMMAND:  return "shifted command key";
407444d93782SGreg Clayton         case KEY_SCOPY:     return "shifted copy key";
407544d93782SGreg Clayton         case KEY_SCREATE:   return "shifted create key";
407644d93782SGreg Clayton         case KEY_SDC:       return "shifted delete-character key";
407744d93782SGreg Clayton         case KEY_SDL:       return "shifted delete-line key";
407844d93782SGreg Clayton         case KEY_SELECT:    return "select key";
407944d93782SGreg Clayton         case KEY_SEND:      return "shifted end key";
408044d93782SGreg Clayton         case KEY_SEOL:      return "shifted clear-to-end-of-line key";
408144d93782SGreg Clayton         case KEY_SEXIT:     return "shifted exit key";
408244d93782SGreg Clayton         case KEY_SFIND:     return "shifted find key";
408344d93782SGreg Clayton         case KEY_SHELP:     return "shifted help key";
408444d93782SGreg Clayton         case KEY_SHOME:     return "shifted home key";
408544d93782SGreg Clayton         case KEY_SIC:       return "shifted insert-character key";
408644d93782SGreg Clayton         case KEY_SLEFT:     return "shifted left-arrow key";
408744d93782SGreg Clayton         case KEY_SMESSAGE:  return "shifted message key";
408844d93782SGreg Clayton         case KEY_SMOVE:     return "shifted move key";
408944d93782SGreg Clayton         case KEY_SNEXT:     return "shifted next key";
409044d93782SGreg Clayton         case KEY_SOPTIONS:  return "shifted options key";
409144d93782SGreg Clayton         case KEY_SPREVIOUS: return "shifted previous key";
409244d93782SGreg Clayton         case KEY_SPRINT:    return "shifted print key";
409344d93782SGreg Clayton         case KEY_SREDO:     return "shifted redo key";
409444d93782SGreg Clayton         case KEY_SREPLACE:  return "shifted replace key";
409544d93782SGreg Clayton         case KEY_SRIGHT:    return "shifted right-arrow key";
409644d93782SGreg Clayton         case KEY_SRSUME:    return "shifted resume key";
409744d93782SGreg Clayton         case KEY_SSAVE:     return "shifted save key";
409844d93782SGreg Clayton         case KEY_SSUSPEND:  return "shifted suspend key";
409944d93782SGreg Clayton         case KEY_SUNDO:     return "shifted undo key";
410044d93782SGreg Clayton         case KEY_SUSPEND:   return "suspend key";
410144d93782SGreg Clayton         case KEY_UNDO:      return "undo key";
410244d93782SGreg Clayton         case KEY_MOUSE:     return "Mouse event has occurred";
410344d93782SGreg Clayton         case KEY_RESIZE:    return "Terminal resize event";
410427801f4fSBruce Mitchener #ifdef KEY_EVENT
410544d93782SGreg Clayton         case KEY_EVENT:     return "We were interrupted by an event";
410627801f4fSBruce Mitchener #endif
410744d93782SGreg Clayton         case KEY_RETURN:    return "return";
410844d93782SGreg Clayton         case ' ':           return "space";
41095fdb09bbSGreg Clayton         case '\t':          return "tab";
411044d93782SGreg Clayton         case KEY_ESCAPE:    return "escape";
411144d93782SGreg Clayton         default:
411244d93782SGreg Clayton             if (isprint(ch))
411344d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "%c", ch);
411444d93782SGreg Clayton             else
411544d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
411644d93782SGreg Clayton             return g_desc;
411744d93782SGreg Clayton     }
4118c5dac77aSEugene Zelenko     return nullptr;
411944d93782SGreg Clayton }
412044d93782SGreg Clayton 
412144d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) :
412244d93782SGreg Clayton     m_text (),
412344d93782SGreg Clayton     m_first_visible_line (0)
412444d93782SGreg Clayton {
412544d93782SGreg Clayton     if (text && text[0])
412644d93782SGreg Clayton     {
412744d93782SGreg Clayton         m_text.SplitIntoLines(text);
412844d93782SGreg Clayton         m_text.AppendString("");
412944d93782SGreg Clayton     }
413044d93782SGreg Clayton     if (key_help_array)
413144d93782SGreg Clayton     {
413244d93782SGreg Clayton         for (KeyHelp *key = key_help_array; key->ch; ++key)
413344d93782SGreg Clayton         {
413444d93782SGreg Clayton             StreamString key_description;
413544d93782SGreg Clayton             key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description);
413644d93782SGreg Clayton             m_text.AppendString(std::move(key_description.GetString()));
413744d93782SGreg Clayton         }
413844d93782SGreg Clayton     }
413944d93782SGreg Clayton }
414044d93782SGreg Clayton 
4141315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
414244d93782SGreg Clayton 
414344d93782SGreg Clayton bool
414444d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force)
414544d93782SGreg Clayton {
414644d93782SGreg Clayton     window.Erase();
414744d93782SGreg Clayton     const int window_height = window.GetHeight();
414844d93782SGreg Clayton     int x = 2;
414944d93782SGreg Clayton     int y = 1;
415044d93782SGreg Clayton     const int min_y = y;
415144d93782SGreg Clayton     const int max_y = window_height - 1 - y;
41523985c8c6SSaleem Abdulrasool     const size_t num_visible_lines = max_y - min_y + 1;
415344d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
415444d93782SGreg Clayton     const char *bottom_message;
415544d93782SGreg Clayton     if (num_lines <= num_visible_lines)
415644d93782SGreg Clayton         bottom_message = "Press any key to exit";
415744d93782SGreg Clayton     else
415844d93782SGreg Clayton         bottom_message = "Use arrows to scroll, any other key to exit";
415944d93782SGreg Clayton     window.DrawTitleBox(window.GetName(), bottom_message);
416044d93782SGreg Clayton     while (y <= max_y)
416144d93782SGreg Clayton     {
416244d93782SGreg Clayton         window.MoveCursor(x, y);
416344d93782SGreg Clayton         window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
416444d93782SGreg Clayton         ++y;
416544d93782SGreg Clayton     }
416644d93782SGreg Clayton     return true;
416744d93782SGreg Clayton }
416844d93782SGreg Clayton 
416944d93782SGreg Clayton HandleCharResult
417044d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key)
417144d93782SGreg Clayton {
417244d93782SGreg Clayton     bool done = false;
417344d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
417444d93782SGreg Clayton     const size_t num_visible_lines = window.GetHeight() - 2;
417544d93782SGreg Clayton 
417644d93782SGreg Clayton     if (num_lines <= num_visible_lines)
417744d93782SGreg Clayton     {
417844d93782SGreg Clayton         done = true;
417944d93782SGreg Clayton         // If we have all lines visible and don't need scrolling, then any
418044d93782SGreg Clayton         // key press will cause us to exit
418144d93782SGreg Clayton     }
418244d93782SGreg Clayton     else
418344d93782SGreg Clayton     {
418444d93782SGreg Clayton         switch (key)
418544d93782SGreg Clayton         {
418644d93782SGreg Clayton             case KEY_UP:
418744d93782SGreg Clayton                 if (m_first_visible_line > 0)
418844d93782SGreg Clayton                     --m_first_visible_line;
418944d93782SGreg Clayton                 break;
419044d93782SGreg Clayton 
419144d93782SGreg Clayton             case KEY_DOWN:
419244d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
419344d93782SGreg Clayton                     ++m_first_visible_line;
419444d93782SGreg Clayton                 break;
419544d93782SGreg Clayton 
419644d93782SGreg Clayton             case KEY_PPAGE:
419744d93782SGreg Clayton             case ',':
419844d93782SGreg Clayton                 if (m_first_visible_line > 0)
419944d93782SGreg Clayton                 {
42003985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
420144d93782SGreg Clayton                         m_first_visible_line -= num_visible_lines;
420244d93782SGreg Clayton                     else
420344d93782SGreg Clayton                         m_first_visible_line = 0;
420444d93782SGreg Clayton                 }
420544d93782SGreg Clayton                 break;
4206315b6884SEugene Zelenko 
420744d93782SGreg Clayton             case KEY_NPAGE:
420844d93782SGreg Clayton             case '.':
420944d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
421044d93782SGreg Clayton                 {
421144d93782SGreg Clayton                     m_first_visible_line += num_visible_lines;
42123985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > num_lines)
421344d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
421444d93782SGreg Clayton                 }
421544d93782SGreg Clayton                 break;
4216315b6884SEugene Zelenko 
421744d93782SGreg Clayton             default:
421844d93782SGreg Clayton                 done = true;
421944d93782SGreg Clayton                 break;
422044d93782SGreg Clayton         }
422144d93782SGreg Clayton     }
422244d93782SGreg Clayton     if (done)
422344d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
422444d93782SGreg Clayton     return eKeyHandled;
422544d93782SGreg Clayton }
422644d93782SGreg Clayton 
422744d93782SGreg Clayton class ApplicationDelegate :
422844d93782SGreg Clayton     public WindowDelegate,
422944d93782SGreg Clayton     public MenuDelegate
423044d93782SGreg Clayton {
423144d93782SGreg Clayton public:
423244d93782SGreg Clayton     enum {
423344d93782SGreg Clayton         eMenuID_LLDB = 1,
423444d93782SGreg Clayton         eMenuID_LLDBAbout,
423544d93782SGreg Clayton         eMenuID_LLDBExit,
423644d93782SGreg Clayton 
423744d93782SGreg Clayton         eMenuID_Target,
423844d93782SGreg Clayton         eMenuID_TargetCreate,
423944d93782SGreg Clayton         eMenuID_TargetDelete,
424044d93782SGreg Clayton 
424144d93782SGreg Clayton         eMenuID_Process,
424244d93782SGreg Clayton         eMenuID_ProcessAttach,
424344d93782SGreg Clayton         eMenuID_ProcessDetach,
424444d93782SGreg Clayton         eMenuID_ProcessLaunch,
424544d93782SGreg Clayton         eMenuID_ProcessContinue,
424644d93782SGreg Clayton         eMenuID_ProcessHalt,
424744d93782SGreg Clayton         eMenuID_ProcessKill,
424844d93782SGreg Clayton 
424944d93782SGreg Clayton         eMenuID_Thread,
425044d93782SGreg Clayton         eMenuID_ThreadStepIn,
425144d93782SGreg Clayton         eMenuID_ThreadStepOver,
425244d93782SGreg Clayton         eMenuID_ThreadStepOut,
425344d93782SGreg Clayton 
425444d93782SGreg Clayton         eMenuID_View,
425544d93782SGreg Clayton         eMenuID_ViewBacktrace,
425644d93782SGreg Clayton         eMenuID_ViewRegisters,
425744d93782SGreg Clayton         eMenuID_ViewSource,
425844d93782SGreg Clayton         eMenuID_ViewVariables,
425944d93782SGreg Clayton 
426044d93782SGreg Clayton         eMenuID_Help,
426144d93782SGreg Clayton         eMenuID_HelpGUIHelp
426244d93782SGreg Clayton     };
426344d93782SGreg Clayton 
426444d93782SGreg Clayton     ApplicationDelegate (Application &app, Debugger &debugger) :
426544d93782SGreg Clayton         WindowDelegate (),
426644d93782SGreg Clayton         MenuDelegate (),
426744d93782SGreg Clayton         m_app (app),
426844d93782SGreg Clayton         m_debugger (debugger)
426944d93782SGreg Clayton     {
427044d93782SGreg Clayton     }
427144d93782SGreg Clayton 
4272315b6884SEugene Zelenko     ~ApplicationDelegate() override = default;
4273bd5ae6b4SGreg Clayton 
4274bd5ae6b4SGreg Clayton     bool
4275bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
427644d93782SGreg Clayton     {
427744d93782SGreg Clayton         return false; // Drawing not handled, let standard window drawing happen
427844d93782SGreg Clayton     }
427944d93782SGreg Clayton 
4280bd5ae6b4SGreg Clayton     HandleCharResult
4281bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int key) override
428244d93782SGreg Clayton     {
42835fdb09bbSGreg Clayton         switch (key)
428444d93782SGreg Clayton         {
42855fdb09bbSGreg Clayton             case '\t':
428644d93782SGreg Clayton                 window.SelectNextWindowAsActive();
428744d93782SGreg Clayton                 return eKeyHandled;
42885fdb09bbSGreg Clayton 
42895fdb09bbSGreg Clayton             case 'h':
42905fdb09bbSGreg Clayton                 window.CreateHelpSubwindow();
42915fdb09bbSGreg Clayton                 return eKeyHandled;
42925fdb09bbSGreg Clayton 
42935fdb09bbSGreg Clayton             case KEY_ESCAPE:
42945fdb09bbSGreg Clayton                 return eQuitApplication;
42955fdb09bbSGreg Clayton 
42965fdb09bbSGreg Clayton             default:
42975fdb09bbSGreg Clayton                 break;
429844d93782SGreg Clayton         }
429944d93782SGreg Clayton         return eKeyNotHandled;
430044d93782SGreg Clayton     }
430144d93782SGreg Clayton 
4302bd5ae6b4SGreg Clayton     const char *
4303bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
43045fdb09bbSGreg Clayton     {
43055fdb09bbSGreg Clayton         return "Welcome to the LLDB curses GUI.\n\n"
43065fdb09bbSGreg Clayton         "Press the TAB key to change the selected view.\n"
43075fdb09bbSGreg Clayton         "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n"
43085fdb09bbSGreg Clayton         "Common key bindings for all views:";
43095fdb09bbSGreg Clayton     }
43105fdb09bbSGreg Clayton 
4311bd5ae6b4SGreg Clayton     KeyHelp *
4312bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
43135fdb09bbSGreg Clayton     {
43145fdb09bbSGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
43155fdb09bbSGreg Clayton             { '\t', "Select next view" },
43165fdb09bbSGreg Clayton             { 'h', "Show help dialog with view specific key bindings" },
43175fdb09bbSGreg Clayton             { ',', "Page up" },
43185fdb09bbSGreg Clayton             { '.', "Page down" },
43195fdb09bbSGreg Clayton             { KEY_UP, "Select previous" },
43205fdb09bbSGreg Clayton             { KEY_DOWN, "Select next" },
43215fdb09bbSGreg Clayton             { KEY_LEFT, "Unexpand or select parent" },
43225fdb09bbSGreg Clayton             { KEY_RIGHT, "Expand" },
43235fdb09bbSGreg Clayton             { KEY_PPAGE, "Page up" },
43245fdb09bbSGreg Clayton             { KEY_NPAGE, "Page down" },
4325c5dac77aSEugene Zelenko             { '\0', nullptr }
43265fdb09bbSGreg Clayton         };
43275fdb09bbSGreg Clayton         return g_source_view_key_help;
43285fdb09bbSGreg Clayton     }
43295fdb09bbSGreg Clayton 
4330bd5ae6b4SGreg Clayton     MenuActionResult
4331bd5ae6b4SGreg Clayton     MenuDelegateAction (Menu &menu) override
433244d93782SGreg Clayton     {
433344d93782SGreg Clayton         switch (menu.GetIdentifier())
433444d93782SGreg Clayton         {
433544d93782SGreg Clayton             case eMenuID_ThreadStepIn:
433644d93782SGreg Clayton                 {
433744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
433844d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
433944d93782SGreg Clayton                     {
434044d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
434144d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
43424b4b2478SJim Ingham                             exe_ctx.GetThreadRef().StepIn(true);
434344d93782SGreg Clayton                     }
434444d93782SGreg Clayton                 }
434544d93782SGreg Clayton                 return MenuActionResult::Handled;
434644d93782SGreg Clayton 
434744d93782SGreg Clayton             case eMenuID_ThreadStepOut:
434844d93782SGreg Clayton                 {
434944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
435044d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
435144d93782SGreg Clayton                     {
435244d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
435344d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
435444d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOut();
435544d93782SGreg Clayton                     }
435644d93782SGreg Clayton                 }
435744d93782SGreg Clayton                 return MenuActionResult::Handled;
435844d93782SGreg Clayton 
435944d93782SGreg Clayton             case eMenuID_ThreadStepOver:
436044d93782SGreg Clayton                 {
436144d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
436244d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
436344d93782SGreg Clayton                     {
436444d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
436544d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
436644d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOver(true);
436744d93782SGreg Clayton                     }
436844d93782SGreg Clayton                 }
436944d93782SGreg Clayton                 return MenuActionResult::Handled;
437044d93782SGreg Clayton 
437144d93782SGreg Clayton             case eMenuID_ProcessContinue:
437244d93782SGreg Clayton                 {
437344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
437444d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
437544d93782SGreg Clayton                     {
437644d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
437744d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
437844d93782SGreg Clayton                             process->Resume();
437944d93782SGreg Clayton                     }
438044d93782SGreg Clayton                 }
438144d93782SGreg Clayton                 return MenuActionResult::Handled;
438244d93782SGreg Clayton 
438344d93782SGreg Clayton             case eMenuID_ProcessKill:
438444d93782SGreg Clayton                 {
438544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
438644d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
438744d93782SGreg Clayton                     {
438844d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
438944d93782SGreg Clayton                         if (process && process->IsAlive())
4390ede3193bSJason Molenda                             process->Destroy(false);
439144d93782SGreg Clayton                     }
439244d93782SGreg Clayton                 }
439344d93782SGreg Clayton                 return MenuActionResult::Handled;
439444d93782SGreg Clayton 
439544d93782SGreg Clayton             case eMenuID_ProcessHalt:
439644d93782SGreg Clayton                 {
439744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
439844d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
439944d93782SGreg Clayton                     {
440044d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
440144d93782SGreg Clayton                         if (process && process->IsAlive())
440244d93782SGreg Clayton                             process->Halt();
440344d93782SGreg Clayton                     }
440444d93782SGreg Clayton                 }
440544d93782SGreg Clayton                 return MenuActionResult::Handled;
440644d93782SGreg Clayton 
440744d93782SGreg Clayton             case eMenuID_ProcessDetach:
440844d93782SGreg Clayton                 {
440944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
441044d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
441144d93782SGreg Clayton                     {
441244d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
441344d93782SGreg Clayton                         if (process && process->IsAlive())
441444d93782SGreg Clayton                             process->Detach(false);
441544d93782SGreg Clayton                     }
441644d93782SGreg Clayton                 }
441744d93782SGreg Clayton                 return MenuActionResult::Handled;
441844d93782SGreg Clayton 
441944d93782SGreg Clayton             case eMenuID_Process:
442044d93782SGreg Clayton                 {
442144d93782SGreg Clayton                     // Populate the menu with all of the threads if the process is stopped when
442244d93782SGreg Clayton                     // the Process menu gets selected and is about to display its submenu.
442344d93782SGreg Clayton                     Menus &submenus = menu.GetSubmenus();
442444d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
442544d93782SGreg Clayton                     Process *process = exe_ctx.GetProcessPtr();
442644d93782SGreg Clayton                     if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
442744d93782SGreg Clayton                     {
442844d93782SGreg Clayton                         if (submenus.size() == 7)
442944d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
443044d93782SGreg Clayton                         else if (submenus.size() > 8)
443144d93782SGreg Clayton                             submenus.erase (submenus.begin() + 8, submenus.end());
443244d93782SGreg Clayton 
443344d93782SGreg Clayton                         ThreadList &threads = process->GetThreadList();
4434*bb19a13cSSaleem Abdulrasool                         std::lock_guard<std::recursive_mutex> guard(threads.GetMutex());
443544d93782SGreg Clayton                         size_t num_threads = threads.GetSize();
443644d93782SGreg Clayton                         for (size_t i = 0; i < num_threads; ++i)
443744d93782SGreg Clayton                         {
443844d93782SGreg Clayton                             ThreadSP thread_sp = threads.GetThreadAtIndex(i);
443944d93782SGreg Clayton                             char menu_char = '\0';
444044d93782SGreg Clayton                             if (i < 9)
444144d93782SGreg Clayton                                 menu_char = '1' + i;
444244d93782SGreg Clayton                             StreamString thread_menu_title;
444344d93782SGreg Clayton                             thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
444444d93782SGreg Clayton                             const char *thread_name = thread_sp->GetName();
444544d93782SGreg Clayton                             if (thread_name && thread_name[0])
444644d93782SGreg Clayton                                 thread_menu_title.Printf (" %s", thread_name);
444744d93782SGreg Clayton                             else
444844d93782SGreg Clayton                             {
444944d93782SGreg Clayton                                 const char *queue_name = thread_sp->GetQueueName();
445044d93782SGreg Clayton                                 if (queue_name && queue_name[0])
445144d93782SGreg Clayton                                     thread_menu_title.Printf (" %s", queue_name);
445244d93782SGreg Clayton                             }
4453c5dac77aSEugene Zelenko                             menu.AddSubmenu(MenuSP(new Menu(thread_menu_title.GetString().c_str(), nullptr, menu_char, thread_sp->GetID())));
445444d93782SGreg Clayton                         }
445544d93782SGreg Clayton                     }
445644d93782SGreg Clayton                     else if (submenus.size() > 7)
445744d93782SGreg Clayton                     {
445844d93782SGreg Clayton                         // Remove the separator and any other thread submenu items
445944d93782SGreg Clayton                         // that were previously added
446044d93782SGreg Clayton                         submenus.erase (submenus.begin() + 7, submenus.end());
446144d93782SGreg Clayton                     }
446244d93782SGreg Clayton                     // Since we are adding and removing items we need to recalculate the name lengths
446344d93782SGreg Clayton                     menu.RecalculateNameLengths();
446444d93782SGreg Clayton                 }
446544d93782SGreg Clayton                 return MenuActionResult::Handled;
446644d93782SGreg Clayton 
446744d93782SGreg Clayton             case eMenuID_ViewVariables:
446844d93782SGreg Clayton                 {
446944d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
447044d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
447144d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
447244d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
447344d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
447444d93782SGreg Clayton 
447544d93782SGreg Clayton                     if (variables_window_sp)
447644d93782SGreg Clayton                     {
447744d93782SGreg Clayton                         const Rect variables_bounds = variables_window_sp->GetBounds();
447844d93782SGreg Clayton 
447944d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(variables_window_sp.get());
448044d93782SGreg Clayton 
448144d93782SGreg Clayton                         if (registers_window_sp)
448244d93782SGreg Clayton                         {
448344d93782SGreg Clayton                             // We have a registers window, so give all the area back to the registers window
448444d93782SGreg Clayton                             Rect registers_bounds = variables_bounds;
448544d93782SGreg Clayton                             registers_bounds.size.width = source_bounds.size.width;
448644d93782SGreg Clayton                             registers_window_sp->SetBounds(registers_bounds);
448744d93782SGreg Clayton                         }
448844d93782SGreg Clayton                         else
448944d93782SGreg Clayton                         {
449044d93782SGreg Clayton                             // We have no registers window showing so give the bottom
449144d93782SGreg Clayton                             // area back to the source view
449244d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
449344d93782SGreg Clayton                                                       source_bounds.size.height + variables_bounds.size.height);
449444d93782SGreg Clayton                         }
449544d93782SGreg Clayton                     }
449644d93782SGreg Clayton                     else
449744d93782SGreg Clayton                     {
449844d93782SGreg Clayton                         Rect new_variables_rect;
449944d93782SGreg Clayton                         if (registers_window_sp)
450044d93782SGreg Clayton                         {
450144d93782SGreg Clayton                             // We have a registers window so split the area of the registers
450244d93782SGreg Clayton                             // window into two columns where the left hand side will be the
450344d93782SGreg Clayton                             // variables and the right hand side will be the registers
450444d93782SGreg Clayton                             const Rect variables_bounds = registers_window_sp->GetBounds();
450544d93782SGreg Clayton                             Rect new_registers_rect;
450644d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect);
450744d93782SGreg Clayton                             registers_window_sp->SetBounds (new_registers_rect);
450844d93782SGreg Clayton                         }
450944d93782SGreg Clayton                         else
451044d93782SGreg Clayton                         {
451144d93782SGreg Clayton                             // No variables window, grab the bottom part of the source window
451244d93782SGreg Clayton                             Rect new_source_rect;
451344d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect);
451444d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
451544d93782SGreg Clayton                         }
451644d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables",
451744d93782SGreg Clayton                                                                                   new_variables_rect,
451844d93782SGreg Clayton                                                                                   false);
451944d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
452044d93782SGreg Clayton                     }
452144d93782SGreg Clayton                     touchwin(stdscr);
452244d93782SGreg Clayton                 }
452344d93782SGreg Clayton                 return MenuActionResult::Handled;
452444d93782SGreg Clayton 
452544d93782SGreg Clayton             case eMenuID_ViewRegisters:
452644d93782SGreg Clayton                 {
452744d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
452844d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
452944d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
453044d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
453144d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
453244d93782SGreg Clayton 
453344d93782SGreg Clayton                     if (registers_window_sp)
453444d93782SGreg Clayton                     {
453544d93782SGreg Clayton                         if (variables_window_sp)
453644d93782SGreg Clayton                         {
453744d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
453844d93782SGreg Clayton 
453944d93782SGreg Clayton                             // We have a variables window, so give all the area back to the variables window
454044d93782SGreg Clayton                             variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(),
454144d93782SGreg Clayton                                                          variables_bounds.size.height);
454244d93782SGreg Clayton                         }
454344d93782SGreg Clayton                         else
454444d93782SGreg Clayton                         {
454544d93782SGreg Clayton                             // We have no variables window showing so give the bottom
454644d93782SGreg Clayton                             // area back to the source view
454744d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
454844d93782SGreg Clayton                                                       source_bounds.size.height + registers_window_sp->GetHeight());
454944d93782SGreg Clayton                         }
455044d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(registers_window_sp.get());
455144d93782SGreg Clayton                     }
455244d93782SGreg Clayton                     else
455344d93782SGreg Clayton                     {
455444d93782SGreg Clayton                         Rect new_regs_rect;
455544d93782SGreg Clayton                         if (variables_window_sp)
455644d93782SGreg Clayton                         {
455744d93782SGreg Clayton                             // We have a variables window, split it into two columns
455844d93782SGreg Clayton                             // where the left hand side will be the variables and the
455944d93782SGreg Clayton                             // right hand side will be the registers
456044d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
456144d93782SGreg Clayton                             Rect new_vars_rect;
456244d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect);
456344d93782SGreg Clayton                             variables_window_sp->SetBounds (new_vars_rect);
456444d93782SGreg Clayton                         }
456544d93782SGreg Clayton                         else
456644d93782SGreg Clayton                         {
456744d93782SGreg Clayton                             // No registers window, grab the bottom part of the source window
456844d93782SGreg Clayton                             Rect new_source_rect;
456944d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect);
457044d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
457144d93782SGreg Clayton                         }
457244d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers",
457344d93782SGreg Clayton                                                                                   new_regs_rect,
457444d93782SGreg Clayton                                                                                   false);
457544d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
457644d93782SGreg Clayton                     }
457744d93782SGreg Clayton                     touchwin(stdscr);
457844d93782SGreg Clayton                 }
457944d93782SGreg Clayton                 return MenuActionResult::Handled;
458044d93782SGreg Clayton 
458144d93782SGreg Clayton             case eMenuID_HelpGUIHelp:
45825fdb09bbSGreg Clayton                 m_app.GetMainWindow ()->CreateHelpSubwindow();
458344d93782SGreg Clayton                 return MenuActionResult::Handled;
458444d93782SGreg Clayton 
458544d93782SGreg Clayton             default:
458644d93782SGreg Clayton                 break;
458744d93782SGreg Clayton         }
458844d93782SGreg Clayton 
458944d93782SGreg Clayton         return MenuActionResult::NotHandled;
459044d93782SGreg Clayton     }
459144d93782SGreg Clayton protected:
459244d93782SGreg Clayton     Application &m_app;
459344d93782SGreg Clayton     Debugger &m_debugger;
459444d93782SGreg Clayton };
459544d93782SGreg Clayton 
459644d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate
459744d93782SGreg Clayton {
459844d93782SGreg Clayton public:
459944d93782SGreg Clayton     StatusBarWindowDelegate (Debugger &debugger) :
460044d93782SGreg Clayton         m_debugger (debugger)
460144d93782SGreg Clayton     {
4602554f68d3SGreg Clayton         FormatEntity::Parse("Thread: ${thread.id%tid}",
4603554f68d3SGreg Clayton                             m_format);
460444d93782SGreg Clayton     }
460544d93782SGreg Clayton 
4606315b6884SEugene Zelenko     ~StatusBarWindowDelegate() override = default;
4607bd5ae6b4SGreg Clayton 
4608bd5ae6b4SGreg Clayton     bool
4609bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
461044d93782SGreg Clayton     {
461144d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
461244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
461344d93782SGreg Clayton         Thread *thread = exe_ctx.GetThreadPtr();
461444d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
461544d93782SGreg Clayton         window.Erase();
461644d93782SGreg Clayton         window.SetBackground(2);
461744d93782SGreg Clayton         window.MoveCursor (0, 0);
461844d93782SGreg Clayton         if (process)
461944d93782SGreg Clayton         {
462044d93782SGreg Clayton             const StateType state = process->GetState();
462144d93782SGreg Clayton             window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state));
462244d93782SGreg Clayton 
462344d93782SGreg Clayton             if (StateIsStoppedState(state, true))
462444d93782SGreg Clayton             {
46255b031ebcSEd Maste                 StreamString strm;
4626c5dac77aSEugene Zelenko                 if (thread && FormatEntity::Format(m_format, strm, nullptr, &exe_ctx, nullptr, nullptr, false, false))
46275b031ebcSEd Maste                 {
462844d93782SGreg Clayton                     window.MoveCursor (40, 0);
46295b031ebcSEd Maste                     window.PutCStringTruncated(strm.GetString().c_str(), 1);
46305b031ebcSEd Maste                 }
463144d93782SGreg Clayton 
463244d93782SGreg Clayton                 window.MoveCursor (60, 0);
463344d93782SGreg Clayton                 if (frame)
463444d93782SGreg Clayton                     window.Printf ("Frame: %3u  PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr()));
463544d93782SGreg Clayton             }
463644d93782SGreg Clayton             else if (state == eStateExited)
463744d93782SGreg Clayton             {
463844d93782SGreg Clayton                 const char *exit_desc = process->GetExitDescription();
463944d93782SGreg Clayton                 const int exit_status = process->GetExitStatus();
464044d93782SGreg Clayton                 if (exit_desc && exit_desc[0])
464144d93782SGreg Clayton                     window.Printf (" with status = %i (%s)", exit_status, exit_desc);
464244d93782SGreg Clayton                 else
464344d93782SGreg Clayton                     window.Printf (" with status = %i", exit_status);
464444d93782SGreg Clayton             }
464544d93782SGreg Clayton         }
464644d93782SGreg Clayton         window.DeferredRefresh();
464744d93782SGreg Clayton         return true;
464844d93782SGreg Clayton     }
464944d93782SGreg Clayton 
465044d93782SGreg Clayton protected:
465144d93782SGreg Clayton     Debugger &m_debugger;
4652554f68d3SGreg Clayton     FormatEntity::Entry m_format;
465344d93782SGreg Clayton };
465444d93782SGreg Clayton 
465544d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate
465644d93782SGreg Clayton {
465744d93782SGreg Clayton public:
465844d93782SGreg Clayton     SourceFileWindowDelegate (Debugger &debugger) :
465944d93782SGreg Clayton         WindowDelegate (),
466044d93782SGreg Clayton         m_debugger (debugger),
466144d93782SGreg Clayton         m_sc (),
466244d93782SGreg Clayton         m_file_sp (),
4663c5dac77aSEugene Zelenko         m_disassembly_scope(nullptr),
466444d93782SGreg Clayton         m_disassembly_sp (),
466544d93782SGreg Clayton         m_disassembly_range (),
4666ec990867SGreg Clayton         m_title (),
466744d93782SGreg Clayton         m_line_width (4),
466844d93782SGreg Clayton         m_selected_line (0),
466944d93782SGreg Clayton         m_pc_line (0),
467044d93782SGreg Clayton         m_stop_id (0),
467144d93782SGreg Clayton         m_frame_idx (UINT32_MAX),
467244d93782SGreg Clayton         m_first_visible_line (0),
467344d93782SGreg Clayton         m_min_x (0),
467444d93782SGreg Clayton         m_min_y (0),
467544d93782SGreg Clayton         m_max_x (0),
467644d93782SGreg Clayton         m_max_y (0)
467744d93782SGreg Clayton     {
467844d93782SGreg Clayton     }
467944d93782SGreg Clayton 
4680315b6884SEugene Zelenko     ~SourceFileWindowDelegate() override = default;
468144d93782SGreg Clayton 
468244d93782SGreg Clayton     void
468344d93782SGreg Clayton     Update (const SymbolContext &sc)
468444d93782SGreg Clayton     {
468544d93782SGreg Clayton         m_sc = sc;
468644d93782SGreg Clayton     }
468744d93782SGreg Clayton 
468844d93782SGreg Clayton     uint32_t
468944d93782SGreg Clayton     NumVisibleLines () const
469044d93782SGreg Clayton     {
469144d93782SGreg Clayton         return m_max_y - m_min_y;
469244d93782SGreg Clayton     }
469344d93782SGreg Clayton 
4694bd5ae6b4SGreg Clayton     const char *
4695bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
469644d93782SGreg Clayton     {
469744d93782SGreg Clayton         return "Source/Disassembly window keyboard shortcuts:";
469844d93782SGreg Clayton     }
469944d93782SGreg Clayton 
4700bd5ae6b4SGreg Clayton     KeyHelp *
4701bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
470244d93782SGreg Clayton     {
470344d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
470444d93782SGreg Clayton             { KEY_RETURN, "Run to selected line with one shot breakpoint" },
470544d93782SGreg Clayton             { KEY_UP, "Select previous source line" },
470644d93782SGreg Clayton             { KEY_DOWN, "Select next source line" },
470744d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
470844d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
470944d93782SGreg Clayton             { 'b', "Set breakpoint on selected source/disassembly line" },
471044d93782SGreg Clayton             { 'c', "Continue process" },
471144d93782SGreg Clayton             { 'd', "Detach and resume process" },
471244d93782SGreg Clayton             { 'D', "Detach with process suspended" },
471344d93782SGreg Clayton             { 'h', "Show help dialog" },
471444d93782SGreg Clayton             { 'k', "Kill process" },
471544d93782SGreg Clayton             { 'n', "Step over (source line)" },
471644d93782SGreg Clayton             { 'N', "Step over (single instruction)" },
471744d93782SGreg Clayton             { 'o', "Step out" },
471844d93782SGreg Clayton             { 's', "Step in (source line)" },
471944d93782SGreg Clayton             { 'S', "Step in (single instruction)" },
472044d93782SGreg Clayton             { ',', "Page up" },
472144d93782SGreg Clayton             { '.', "Page down" },
4722c5dac77aSEugene Zelenko             { '\0', nullptr }
472344d93782SGreg Clayton         };
472444d93782SGreg Clayton         return g_source_view_key_help;
472544d93782SGreg Clayton     }
472644d93782SGreg Clayton 
4727bd5ae6b4SGreg Clayton     bool
4728bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
472944d93782SGreg Clayton     {
473044d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
473144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
4732c5dac77aSEugene Zelenko         Thread *thread = nullptr;
473344d93782SGreg Clayton 
473444d93782SGreg Clayton         bool update_location = false;
473544d93782SGreg Clayton         if (process)
473644d93782SGreg Clayton         {
473744d93782SGreg Clayton             StateType state = process->GetState();
473844d93782SGreg Clayton             if (StateIsStoppedState(state, true))
473944d93782SGreg Clayton             {
474044d93782SGreg Clayton                 // We are stopped, so it is ok to
474144d93782SGreg Clayton                 update_location = true;
474244d93782SGreg Clayton             }
474344d93782SGreg Clayton         }
474444d93782SGreg Clayton 
474544d93782SGreg Clayton         m_min_x = 1;
4746ec990867SGreg Clayton         m_min_y = 2;
474744d93782SGreg Clayton         m_max_x = window.GetMaxX()-1;
474844d93782SGreg Clayton         m_max_y = window.GetMaxY()-1;
474944d93782SGreg Clayton 
475044d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
475144d93782SGreg Clayton         StackFrameSP frame_sp;
475244d93782SGreg Clayton         bool set_selected_line_to_pc = false;
475344d93782SGreg Clayton 
475444d93782SGreg Clayton         if (update_location)
475544d93782SGreg Clayton         {
475644d93782SGreg Clayton             const bool process_alive = process ? process->IsAlive() : false;
475744d93782SGreg Clayton             bool thread_changed = false;
475844d93782SGreg Clayton             if (process_alive)
475944d93782SGreg Clayton             {
476044d93782SGreg Clayton                 thread = exe_ctx.GetThreadPtr();
476144d93782SGreg Clayton                 if (thread)
476244d93782SGreg Clayton                 {
476344d93782SGreg Clayton                     frame_sp = thread->GetSelectedFrame();
476444d93782SGreg Clayton                     auto tid = thread->GetID();
476544d93782SGreg Clayton                     thread_changed = tid != m_tid;
476644d93782SGreg Clayton                     m_tid = tid;
476744d93782SGreg Clayton                 }
476844d93782SGreg Clayton                 else
476944d93782SGreg Clayton                 {
477044d93782SGreg Clayton                     if (m_tid != LLDB_INVALID_THREAD_ID)
477144d93782SGreg Clayton                     {
477244d93782SGreg Clayton                         thread_changed = true;
477344d93782SGreg Clayton                         m_tid = LLDB_INVALID_THREAD_ID;
477444d93782SGreg Clayton                     }
477544d93782SGreg Clayton                 }
477644d93782SGreg Clayton             }
477744d93782SGreg Clayton             const uint32_t stop_id = process ? process->GetStopID() : 0;
477844d93782SGreg Clayton             const bool stop_id_changed = stop_id != m_stop_id;
477944d93782SGreg Clayton             bool frame_changed = false;
478044d93782SGreg Clayton             m_stop_id = stop_id;
4781ec990867SGreg Clayton             m_title.Clear();
478244d93782SGreg Clayton             if (frame_sp)
478344d93782SGreg Clayton             {
478444d93782SGreg Clayton                 m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
4785ec990867SGreg Clayton                 if (m_sc.module_sp)
4786ec990867SGreg Clayton                 {
4787ec990867SGreg Clayton                     m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
4788ec990867SGreg Clayton                     ConstString func_name = m_sc.GetFunctionName();
4789ec990867SGreg Clayton                     if (func_name)
4790ec990867SGreg Clayton                         m_title.Printf("`%s", func_name.GetCString());
4791ec990867SGreg Clayton                 }
479244d93782SGreg Clayton                 const uint32_t frame_idx = frame_sp->GetFrameIndex();
479344d93782SGreg Clayton                 frame_changed = frame_idx != m_frame_idx;
479444d93782SGreg Clayton                 m_frame_idx = frame_idx;
479544d93782SGreg Clayton             }
479644d93782SGreg Clayton             else
479744d93782SGreg Clayton             {
479844d93782SGreg Clayton                 m_sc.Clear(true);
479944d93782SGreg Clayton                 frame_changed = m_frame_idx != UINT32_MAX;
480044d93782SGreg Clayton                 m_frame_idx = UINT32_MAX;
480144d93782SGreg Clayton             }
480244d93782SGreg Clayton 
480344d93782SGreg Clayton             const bool context_changed = thread_changed || frame_changed || stop_id_changed;
480444d93782SGreg Clayton 
480544d93782SGreg Clayton             if (process_alive)
480644d93782SGreg Clayton             {
480744d93782SGreg Clayton                 if (m_sc.line_entry.IsValid())
480844d93782SGreg Clayton                 {
480944d93782SGreg Clayton                     m_pc_line = m_sc.line_entry.line;
481044d93782SGreg Clayton                     if (m_pc_line != UINT32_MAX)
481144d93782SGreg Clayton                         --m_pc_line; // Convert to zero based line number...
481244d93782SGreg Clayton                     // Update the selected line if the stop ID changed...
481344d93782SGreg Clayton                     if (context_changed)
481444d93782SGreg Clayton                         m_selected_line = m_pc_line;
481544d93782SGreg Clayton 
481644d93782SGreg Clayton                     if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file))
481744d93782SGreg Clayton                     {
481844d93782SGreg Clayton                         // Same file, nothing to do, we should either have the
481944d93782SGreg Clayton                         // lines or not (source file missing)
48203985c8c6SSaleem Abdulrasool                         if (m_selected_line >= static_cast<size_t>(m_first_visible_line))
482144d93782SGreg Clayton                         {
482244d93782SGreg Clayton                             if (m_selected_line >= m_first_visible_line + num_visible_lines)
482344d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
482444d93782SGreg Clayton                         }
482544d93782SGreg Clayton                         else
482644d93782SGreg Clayton                         {
482744d93782SGreg Clayton                             if (m_selected_line > 10)
482844d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
482944d93782SGreg Clayton                             else
483044d93782SGreg Clayton                                 m_first_visible_line = 0;
483144d93782SGreg Clayton                         }
483244d93782SGreg Clayton                     }
483344d93782SGreg Clayton                     else
483444d93782SGreg Clayton                     {
483544d93782SGreg Clayton                         // File changed, set selected line to the line with the PC
483644d93782SGreg Clayton                         m_selected_line = m_pc_line;
483744d93782SGreg Clayton                         m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
483844d93782SGreg Clayton                         if (m_file_sp)
483944d93782SGreg Clayton                         {
484044d93782SGreg Clayton                             const size_t num_lines = m_file_sp->GetNumLines();
484144d93782SGreg Clayton                             int m_line_width = 1;
484244d93782SGreg Clayton                             for (size_t n = num_lines; n >= 10; n = n / 10)
484344d93782SGreg Clayton                                 ++m_line_width;
484444d93782SGreg Clayton 
484544d93782SGreg Clayton                             snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width);
484644d93782SGreg Clayton                             if (num_lines < num_visible_lines || m_selected_line < num_visible_lines)
484744d93782SGreg Clayton                                 m_first_visible_line = 0;
484844d93782SGreg Clayton                             else
484944d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
485044d93782SGreg Clayton                         }
485144d93782SGreg Clayton                     }
485244d93782SGreg Clayton                 }
485344d93782SGreg Clayton                 else
485444d93782SGreg Clayton                 {
485544d93782SGreg Clayton                     m_file_sp.reset();
485644d93782SGreg Clayton                 }
485744d93782SGreg Clayton 
485844d93782SGreg Clayton                 if (!m_file_sp || m_file_sp->GetNumLines() == 0)
485944d93782SGreg Clayton                 {
486044d93782SGreg Clayton                     // Show disassembly
486144d93782SGreg Clayton                     bool prefer_file_cache = false;
486244d93782SGreg Clayton                     if (m_sc.function)
486344d93782SGreg Clayton                     {
486444d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.function)
486544d93782SGreg Clayton                         {
486644d93782SGreg Clayton                             m_disassembly_scope = m_sc.function;
4867c5dac77aSEugene Zelenko                             m_disassembly_sp = m_sc.function->GetInstructions(exe_ctx, nullptr, prefer_file_cache);
486844d93782SGreg Clayton                             if (m_disassembly_sp)
486944d93782SGreg Clayton                             {
487044d93782SGreg Clayton                                 set_selected_line_to_pc = true;
487144d93782SGreg Clayton                                 m_disassembly_range = m_sc.function->GetAddressRange();
487244d93782SGreg Clayton                             }
487344d93782SGreg Clayton                             else
487444d93782SGreg Clayton                             {
487544d93782SGreg Clayton                                 m_disassembly_range.Clear();
487644d93782SGreg Clayton                             }
487744d93782SGreg Clayton                         }
487844d93782SGreg Clayton                         else
487944d93782SGreg Clayton                         {
488044d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
488144d93782SGreg Clayton                         }
488244d93782SGreg Clayton                     }
488344d93782SGreg Clayton                     else if (m_sc.symbol)
488444d93782SGreg Clayton                     {
488544d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.symbol)
488644d93782SGreg Clayton                         {
488744d93782SGreg Clayton                             m_disassembly_scope = m_sc.symbol;
4888c5dac77aSEugene Zelenko                             m_disassembly_sp = m_sc.symbol->GetInstructions(exe_ctx, nullptr, prefer_file_cache);
488944d93782SGreg Clayton                             if (m_disassembly_sp)
489044d93782SGreg Clayton                             {
489144d93782SGreg Clayton                                 set_selected_line_to_pc = true;
489244d93782SGreg Clayton                                 m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress();
489344d93782SGreg Clayton                                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
489444d93782SGreg Clayton                             }
489544d93782SGreg Clayton                             else
489644d93782SGreg Clayton                             {
489744d93782SGreg Clayton                                 m_disassembly_range.Clear();
489844d93782SGreg Clayton                             }
489944d93782SGreg Clayton                         }
490044d93782SGreg Clayton                         else
490144d93782SGreg Clayton                         {
490244d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
490344d93782SGreg Clayton                         }
490444d93782SGreg Clayton                     }
490544d93782SGreg Clayton                 }
490644d93782SGreg Clayton             }
490744d93782SGreg Clayton             else
490844d93782SGreg Clayton             {
490944d93782SGreg Clayton                 m_pc_line = UINT32_MAX;
491044d93782SGreg Clayton             }
491144d93782SGreg Clayton         }
491244d93782SGreg Clayton 
4913ec990867SGreg Clayton         const int window_width = window.GetWidth();
491444d93782SGreg Clayton         window.Erase();
491544d93782SGreg Clayton         window.DrawTitleBox ("Sources");
4916ec990867SGreg Clayton         if (!m_title.GetString().empty())
4917ec990867SGreg Clayton         {
4918ec990867SGreg Clayton             window.AttributeOn(A_REVERSE);
4919ec990867SGreg Clayton             window.MoveCursor(1, 1);
4920ec990867SGreg Clayton             window.PutChar(' ');
4921ec990867SGreg Clayton             window.PutCStringTruncated(m_title.GetString().c_str(), 1);
4922ec990867SGreg Clayton             int x = window.GetCursorX();
4923ec990867SGreg Clayton             if (x < window_width - 1)
4924ec990867SGreg Clayton             {
4925ec990867SGreg Clayton                 window.Printf ("%*s", window_width - x - 1, "");
4926ec990867SGreg Clayton             }
4927ec990867SGreg Clayton             window.AttributeOff(A_REVERSE);
4928ec990867SGreg Clayton         }
492944d93782SGreg Clayton 
493044d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
493144d93782SGreg Clayton         const size_t num_source_lines = GetNumSourceLines();
493244d93782SGreg Clayton         if (num_source_lines > 0)
493344d93782SGreg Clayton         {
493444d93782SGreg Clayton             // Display source
493544d93782SGreg Clayton             BreakpointLines bp_lines;
493644d93782SGreg Clayton             if (target)
493744d93782SGreg Clayton             {
493844d93782SGreg Clayton                 BreakpointList &bp_list = target->GetBreakpointList();
493944d93782SGreg Clayton                 const size_t num_bps = bp_list.GetSize();
494044d93782SGreg Clayton                 for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
494144d93782SGreg Clayton                 {
494244d93782SGreg Clayton                     BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
494344d93782SGreg Clayton                     const size_t num_bps_locs = bp_sp->GetNumLocations();
494444d93782SGreg Clayton                     for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
494544d93782SGreg Clayton                     {
494644d93782SGreg Clayton                         BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
494744d93782SGreg Clayton                         LineEntry bp_loc_line_entry;
494844d93782SGreg Clayton                         if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry))
494944d93782SGreg Clayton                         {
495044d93782SGreg Clayton                             if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file)
495144d93782SGreg Clayton                             {
495244d93782SGreg Clayton                                 bp_lines.insert(bp_loc_line_entry.line);
495344d93782SGreg Clayton                             }
495444d93782SGreg Clayton                         }
495544d93782SGreg Clayton                     }
495644d93782SGreg Clayton                 }
495744d93782SGreg Clayton             }
495844d93782SGreg Clayton 
495944d93782SGreg Clayton             const attr_t selected_highlight_attr = A_REVERSE;
496044d93782SGreg Clayton             const attr_t pc_highlight_attr = COLOR_PAIR(1);
496144d93782SGreg Clayton 
49623985c8c6SSaleem Abdulrasool             for (size_t i = 0; i < num_visible_lines; ++i)
496344d93782SGreg Clayton             {
496444d93782SGreg Clayton                 const uint32_t curr_line = m_first_visible_line + i;
496544d93782SGreg Clayton                 if (curr_line < num_source_lines)
496644d93782SGreg Clayton                 {
4967ec990867SGreg Clayton                     const int line_y = m_min_y+i;
496844d93782SGreg Clayton                     window.MoveCursor(1, line_y);
496944d93782SGreg Clayton                     const bool is_pc_line = curr_line == m_pc_line;
497044d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == curr_line;
497144d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
497244d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
497344d93782SGreg Clayton                     attr_t highlight_attr = 0;
497444d93782SGreg Clayton                     attr_t bp_attr = 0;
497544d93782SGreg Clayton                     if (is_pc_line)
497644d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
497744d93782SGreg Clayton                     else if (line_is_selected)
497844d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
497944d93782SGreg Clayton 
498044d93782SGreg Clayton                     if (bp_lines.find(curr_line+1) != bp_lines.end())
498144d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
498244d93782SGreg Clayton 
498344d93782SGreg Clayton                     if (bp_attr)
498444d93782SGreg Clayton                         window.AttributeOn(bp_attr);
498544d93782SGreg Clayton 
498644d93782SGreg Clayton                     window.Printf (m_line_format, curr_line + 1);
498744d93782SGreg Clayton 
498844d93782SGreg Clayton                     if (bp_attr)
498944d93782SGreg Clayton                         window.AttributeOff(bp_attr);
499044d93782SGreg Clayton 
499144d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
499244d93782SGreg Clayton                     // Mark the line with the PC with a diamond
499344d93782SGreg Clayton                     if (is_pc_line)
499444d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
499544d93782SGreg Clayton                     else
499644d93782SGreg Clayton                         window.PutChar(' ');
499744d93782SGreg Clayton 
499844d93782SGreg Clayton                     if (highlight_attr)
499944d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
500044d93782SGreg Clayton                     const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false);
500144d93782SGreg Clayton                     if (line_len > 0)
500244d93782SGreg Clayton                         window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
500344d93782SGreg Clayton 
500444d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
500544d93782SGreg Clayton                     {
500644d93782SGreg Clayton                         StopInfoSP stop_info_sp;
500744d93782SGreg Clayton                         if (thread)
500844d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
500944d93782SGreg Clayton                         if (stop_info_sp)
501044d93782SGreg Clayton                         {
501144d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
501244d93782SGreg Clayton                             if (stop_description && stop_description[0])
501344d93782SGreg Clayton                             {
501444d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5015ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
501644d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5017ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
501844d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
501944d93782SGreg Clayton                             }
502044d93782SGreg Clayton                         }
502144d93782SGreg Clayton                         else
502244d93782SGreg Clayton                         {
5023ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
502444d93782SGreg Clayton                         }
502544d93782SGreg Clayton                     }
502644d93782SGreg Clayton                     if (highlight_attr)
502744d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
502844d93782SGreg Clayton                 }
502944d93782SGreg Clayton                 else
503044d93782SGreg Clayton                 {
503144d93782SGreg Clayton                     break;
503244d93782SGreg Clayton                 }
503344d93782SGreg Clayton             }
503444d93782SGreg Clayton         }
503544d93782SGreg Clayton         else
503644d93782SGreg Clayton         {
503744d93782SGreg Clayton             size_t num_disassembly_lines = GetNumDisassemblyLines();
503844d93782SGreg Clayton             if (num_disassembly_lines > 0)
503944d93782SGreg Clayton             {
504044d93782SGreg Clayton                 // Display disassembly
504144d93782SGreg Clayton                 BreakpointAddrs bp_file_addrs;
504244d93782SGreg Clayton                 Target *target = exe_ctx.GetTargetPtr();
504344d93782SGreg Clayton                 if (target)
504444d93782SGreg Clayton                 {
504544d93782SGreg Clayton                     BreakpointList &bp_list = target->GetBreakpointList();
504644d93782SGreg Clayton                     const size_t num_bps = bp_list.GetSize();
504744d93782SGreg Clayton                     for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
504844d93782SGreg Clayton                     {
504944d93782SGreg Clayton                         BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
505044d93782SGreg Clayton                         const size_t num_bps_locs = bp_sp->GetNumLocations();
505144d93782SGreg Clayton                         for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
505244d93782SGreg Clayton                         {
505344d93782SGreg Clayton                             BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
505444d93782SGreg Clayton                             LineEntry bp_loc_line_entry;
505544d93782SGreg Clayton                             const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress();
505644d93782SGreg Clayton                             if (file_addr != LLDB_INVALID_ADDRESS)
505744d93782SGreg Clayton                             {
505844d93782SGreg Clayton                                 if (m_disassembly_range.ContainsFileAddress(file_addr))
505944d93782SGreg Clayton                                     bp_file_addrs.insert(file_addr);
506044d93782SGreg Clayton                             }
506144d93782SGreg Clayton                         }
506244d93782SGreg Clayton                     }
506344d93782SGreg Clayton                 }
506444d93782SGreg Clayton 
506544d93782SGreg Clayton                 const attr_t selected_highlight_attr = A_REVERSE;
506644d93782SGreg Clayton                 const attr_t pc_highlight_attr = COLOR_PAIR(1);
506744d93782SGreg Clayton 
506844d93782SGreg Clayton                 StreamString strm;
506944d93782SGreg Clayton 
507044d93782SGreg Clayton                 InstructionList &insts = m_disassembly_sp->GetInstructionList();
507144d93782SGreg Clayton                 Address pc_address;
507244d93782SGreg Clayton 
507344d93782SGreg Clayton                 if (frame_sp)
507444d93782SGreg Clayton                     pc_address = frame_sp->GetFrameCodeAddress();
507544d93782SGreg Clayton                 const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX;
507644d93782SGreg Clayton                 if (set_selected_line_to_pc)
507744d93782SGreg Clayton                 {
507844d93782SGreg Clayton                     m_selected_line = pc_idx;
507944d93782SGreg Clayton                 }
508044d93782SGreg Clayton 
508144d93782SGreg Clayton                 const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
50823985c8c6SSaleem Abdulrasool                 if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
508344d93782SGreg Clayton                     m_first_visible_line = 0;
508444d93782SGreg Clayton 
508544d93782SGreg Clayton                 if (pc_idx < num_disassembly_lines)
508644d93782SGreg Clayton                 {
50873985c8c6SSaleem Abdulrasool                     if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
508844d93782SGreg Clayton                         pc_idx >= m_first_visible_line + num_visible_lines)
508944d93782SGreg Clayton                         m_first_visible_line = pc_idx - non_visible_pc_offset;
509044d93782SGreg Clayton                 }
509144d93782SGreg Clayton 
509244d93782SGreg Clayton                 for (size_t i = 0; i < num_visible_lines; ++i)
509344d93782SGreg Clayton                 {
509444d93782SGreg Clayton                     const uint32_t inst_idx = m_first_visible_line + i;
509544d93782SGreg Clayton                     Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
509644d93782SGreg Clayton                     if (!inst)
509744d93782SGreg Clayton                         break;
509844d93782SGreg Clayton 
5099ec990867SGreg Clayton                     const int line_y = m_min_y+i;
5100ec990867SGreg Clayton                     window.MoveCursor(1, line_y);
510144d93782SGreg Clayton                     const bool is_pc_line = frame_sp && inst_idx == pc_idx;
510244d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == inst_idx;
510344d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
510444d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
510544d93782SGreg Clayton                     attr_t highlight_attr = 0;
510644d93782SGreg Clayton                     attr_t bp_attr = 0;
510744d93782SGreg Clayton                     if (is_pc_line)
510844d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
510944d93782SGreg Clayton                     else if (line_is_selected)
511044d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
511144d93782SGreg Clayton 
511244d93782SGreg Clayton                     if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end())
511344d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
511444d93782SGreg Clayton 
511544d93782SGreg Clayton                     if (bp_attr)
511644d93782SGreg Clayton                         window.AttributeOn(bp_attr);
511744d93782SGreg Clayton 
5118324a1036SSaleem Abdulrasool                     window.Printf (" 0x%16.16llx ",
5119324a1036SSaleem Abdulrasool                                    static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target)));
512044d93782SGreg Clayton 
512144d93782SGreg Clayton                     if (bp_attr)
512244d93782SGreg Clayton                         window.AttributeOff(bp_attr);
512344d93782SGreg Clayton 
512444d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
512544d93782SGreg Clayton                     // Mark the line with the PC with a diamond
512644d93782SGreg Clayton                     if (is_pc_line)
512744d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
512844d93782SGreg Clayton                     else
512944d93782SGreg Clayton                         window.PutChar(' ');
513044d93782SGreg Clayton 
513144d93782SGreg Clayton                     if (highlight_attr)
513244d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
513344d93782SGreg Clayton 
513444d93782SGreg Clayton                     const char *mnemonic = inst->GetMnemonic(&exe_ctx);
513544d93782SGreg Clayton                     const char *operands = inst->GetOperands(&exe_ctx);
513644d93782SGreg Clayton                     const char *comment = inst->GetComment(&exe_ctx);
513744d93782SGreg Clayton 
5138c5dac77aSEugene Zelenko                     if (mnemonic != nullptr && mnemonic[0] == '\0')
5139c5dac77aSEugene Zelenko                         mnemonic = nullptr;
5140c5dac77aSEugene Zelenko                     if (operands != nullptr && operands[0] == '\0')
5141c5dac77aSEugene Zelenko                         operands = nullptr;
5142c5dac77aSEugene Zelenko                     if (comment != nullptr && comment[0] == '\0')
5143c5dac77aSEugene Zelenko                         comment = nullptr;
514444d93782SGreg Clayton 
514544d93782SGreg Clayton                     strm.Clear();
514644d93782SGreg Clayton 
5147c5dac77aSEugene Zelenko                     if (mnemonic != nullptr && operands != nullptr && comment != nullptr)
514844d93782SGreg Clayton                         strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment);
5149c5dac77aSEugene Zelenko                     else if (mnemonic != nullptr && operands != nullptr)
515044d93782SGreg Clayton                         strm.Printf ("%-8s %s", mnemonic, operands);
5151c5dac77aSEugene Zelenko                     else if (mnemonic != nullptr)
515244d93782SGreg Clayton                         strm.Printf ("%s", mnemonic);
515344d93782SGreg Clayton 
515444d93782SGreg Clayton                     int right_pad = 1;
515544d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
515644d93782SGreg Clayton 
515744d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
515844d93782SGreg Clayton                     {
515944d93782SGreg Clayton                         StopInfoSP stop_info_sp;
516044d93782SGreg Clayton                         if (thread)
516144d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
516244d93782SGreg Clayton                         if (stop_info_sp)
516344d93782SGreg Clayton                         {
516444d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
516544d93782SGreg Clayton                             if (stop_description && stop_description[0])
516644d93782SGreg Clayton                             {
516744d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5168ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
516944d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5170ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
517144d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
517244d93782SGreg Clayton                             }
517344d93782SGreg Clayton                         }
517444d93782SGreg Clayton                         else
517544d93782SGreg Clayton                         {
5176ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
517744d93782SGreg Clayton                         }
517844d93782SGreg Clayton                     }
517944d93782SGreg Clayton                     if (highlight_attr)
518044d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
518144d93782SGreg Clayton                 }
518244d93782SGreg Clayton             }
518344d93782SGreg Clayton         }
518444d93782SGreg Clayton         window.DeferredRefresh();
518544d93782SGreg Clayton         return true; // Drawing handled
518644d93782SGreg Clayton     }
518744d93782SGreg Clayton 
518844d93782SGreg Clayton     size_t
518944d93782SGreg Clayton     GetNumLines ()
519044d93782SGreg Clayton     {
519144d93782SGreg Clayton         size_t num_lines = GetNumSourceLines();
519244d93782SGreg Clayton         if (num_lines == 0)
519344d93782SGreg Clayton             num_lines = GetNumDisassemblyLines();
519444d93782SGreg Clayton         return num_lines;
519544d93782SGreg Clayton     }
519644d93782SGreg Clayton 
519744d93782SGreg Clayton     size_t
519844d93782SGreg Clayton     GetNumSourceLines () const
519944d93782SGreg Clayton     {
520044d93782SGreg Clayton         if (m_file_sp)
520144d93782SGreg Clayton             return m_file_sp->GetNumLines();
520244d93782SGreg Clayton         return 0;
520344d93782SGreg Clayton     }
5204315b6884SEugene Zelenko 
520544d93782SGreg Clayton     size_t
520644d93782SGreg Clayton     GetNumDisassemblyLines () const
520744d93782SGreg Clayton     {
520844d93782SGreg Clayton         if (m_disassembly_sp)
520944d93782SGreg Clayton             return m_disassembly_sp->GetInstructionList().GetSize();
521044d93782SGreg Clayton         return 0;
521144d93782SGreg Clayton     }
521244d93782SGreg Clayton 
5213bd5ae6b4SGreg Clayton     HandleCharResult
5214bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
521544d93782SGreg Clayton     {
521644d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
521744d93782SGreg Clayton         const size_t num_lines = GetNumLines ();
521844d93782SGreg Clayton 
521944d93782SGreg Clayton         switch (c)
522044d93782SGreg Clayton         {
522144d93782SGreg Clayton             case ',':
522244d93782SGreg Clayton             case KEY_PPAGE:
522344d93782SGreg Clayton                 // Page up key
52243985c8c6SSaleem Abdulrasool                 if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
522544d93782SGreg Clayton                     m_first_visible_line -= num_visible_lines;
522644d93782SGreg Clayton                 else
522744d93782SGreg Clayton                     m_first_visible_line = 0;
522844d93782SGreg Clayton                 m_selected_line = m_first_visible_line;
522944d93782SGreg Clayton                 return eKeyHandled;
523044d93782SGreg Clayton 
523144d93782SGreg Clayton             case '.':
523244d93782SGreg Clayton             case KEY_NPAGE:
523344d93782SGreg Clayton                 // Page down key
523444d93782SGreg Clayton                 {
523544d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < num_lines)
523644d93782SGreg Clayton                         m_first_visible_line += num_visible_lines;
523744d93782SGreg Clayton                     else if (num_lines < num_visible_lines)
523844d93782SGreg Clayton                         m_first_visible_line = 0;
523944d93782SGreg Clayton                     else
524044d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
524144d93782SGreg Clayton                     m_selected_line = m_first_visible_line;
524244d93782SGreg Clayton                 }
524344d93782SGreg Clayton                 return eKeyHandled;
524444d93782SGreg Clayton 
524544d93782SGreg Clayton             case KEY_UP:
524644d93782SGreg Clayton                 if (m_selected_line > 0)
524744d93782SGreg Clayton                 {
524844d93782SGreg Clayton                     m_selected_line--;
52493985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
525044d93782SGreg Clayton                         m_first_visible_line = m_selected_line;
525144d93782SGreg Clayton                 }
525244d93782SGreg Clayton                 return eKeyHandled;
525344d93782SGreg Clayton 
525444d93782SGreg Clayton             case KEY_DOWN:
525544d93782SGreg Clayton                 if (m_selected_line + 1 < num_lines)
525644d93782SGreg Clayton                 {
525744d93782SGreg Clayton                     m_selected_line++;
525844d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < m_selected_line)
525944d93782SGreg Clayton                         m_first_visible_line++;
526044d93782SGreg Clayton                 }
526144d93782SGreg Clayton                 return eKeyHandled;
526244d93782SGreg Clayton 
526344d93782SGreg Clayton             case '\r':
526444d93782SGreg Clayton             case '\n':
526544d93782SGreg Clayton             case KEY_ENTER:
526644d93782SGreg Clayton                 // Set a breakpoint and run to the line using a one shot breakpoint
526744d93782SGreg Clayton                 if (GetNumSourceLines() > 0)
526844d93782SGreg Clayton                 {
526944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
527044d93782SGreg Clayton                     if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive())
527144d93782SGreg Clayton                     {
5272c5dac77aSEugene Zelenko                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(nullptr,                   // Don't limit the breakpoint to certain modules
527344d93782SGreg Clayton                                                                                      m_file_sp->GetFileSpec(),  // Source file
527444d93782SGreg Clayton                                                                                      m_selected_line + 1,       // Source line number (m_selected_line is zero based)
52752411167fSJim Ingham                                                                                      0,                         // No offset
527644d93782SGreg Clayton                                                                                      eLazyBoolCalculate,        // Check inlines using global setting
527744d93782SGreg Clayton                                                                                      eLazyBoolCalculate,        // Skip prologue using global setting,
527844d93782SGreg Clayton                                                                                      false,                     // internal
5279055ad9beSIlia K                                                                                      false,                     // request_hardware
5280055ad9beSIlia K                                                                                      eLazyBoolCalculate);       // move_to_nearest_code
528144d93782SGreg Clayton                         // Make breakpoint one shot
528244d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
528344d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
528444d93782SGreg Clayton                     }
528544d93782SGreg Clayton                 }
528644d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
528744d93782SGreg Clayton                 {
528844d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
528944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
529044d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
529144d93782SGreg Clayton                     {
529244d93782SGreg Clayton                         Address addr = inst->GetAddress();
529344d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
529444d93782SGreg Clayton                                                                                       false,    // internal
529544d93782SGreg Clayton                                                                                       false);   // request_hardware
529644d93782SGreg Clayton                         // Make breakpoint one shot
529744d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
529844d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
529944d93782SGreg Clayton                     }
530044d93782SGreg Clayton                 }
530144d93782SGreg Clayton                 return eKeyHandled;
530244d93782SGreg Clayton 
530344d93782SGreg Clayton             case 'b':   // 'b' == toggle breakpoint on currently selected line
530444d93782SGreg Clayton                 if (m_selected_line < GetNumSourceLines())
530544d93782SGreg Clayton                 {
530644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
530744d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
530844d93782SGreg Clayton                     {
5309c5dac77aSEugene Zelenko                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint(nullptr,                   // Don't limit the breakpoint to certain modules
531044d93782SGreg Clayton                                                                                      m_file_sp->GetFileSpec(),  // Source file
531144d93782SGreg Clayton                                                                                      m_selected_line + 1,       // Source line number (m_selected_line is zero based)
53122411167fSJim Ingham                                                                                      0,                         // No offset
531344d93782SGreg Clayton                                                                                      eLazyBoolCalculate,        // Check inlines using global setting
531444d93782SGreg Clayton                                                                                      eLazyBoolCalculate,        // Skip prologue using global setting,
531544d93782SGreg Clayton                                                                                      false,                     // internal
5316055ad9beSIlia K                                                                                      false,                     // request_hardware
5317055ad9beSIlia K                                                                                      eLazyBoolCalculate);       // move_to_nearest_code
531844d93782SGreg Clayton                     }
531944d93782SGreg Clayton                 }
532044d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
532144d93782SGreg Clayton                 {
532244d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
532344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
532444d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
532544d93782SGreg Clayton                     {
532644d93782SGreg Clayton                         Address addr = inst->GetAddress();
532744d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
532844d93782SGreg Clayton                                                                                       false,    // internal
532944d93782SGreg Clayton                                                                                       false);   // request_hardware
533044d93782SGreg Clayton                     }
533144d93782SGreg Clayton                 }
533244d93782SGreg Clayton                 return eKeyHandled;
533344d93782SGreg Clayton 
533444d93782SGreg Clayton             case 'd':   // 'd' == detach and let run
533544d93782SGreg Clayton             case 'D':   // 'D' == detach and keep stopped
533644d93782SGreg Clayton                 {
533744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
533844d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
533944d93782SGreg Clayton                         exe_ctx.GetProcessRef().Detach(c == 'D');
534044d93782SGreg Clayton                 }
534144d93782SGreg Clayton                 return eKeyHandled;
534244d93782SGreg Clayton 
534344d93782SGreg Clayton             case 'k':
534444d93782SGreg Clayton                 // 'k' == kill
534544d93782SGreg Clayton                 {
534644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
534744d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
5348ede3193bSJason Molenda                         exe_ctx.GetProcessRef().Destroy(false);
534944d93782SGreg Clayton                 }
535044d93782SGreg Clayton                 return eKeyHandled;
535144d93782SGreg Clayton 
535244d93782SGreg Clayton             case 'c':
535344d93782SGreg Clayton                 // 'c' == continue
535444d93782SGreg Clayton                 {
535544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
535644d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
535744d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
535844d93782SGreg Clayton                 }
535944d93782SGreg Clayton                 return eKeyHandled;
536044d93782SGreg Clayton 
536144d93782SGreg Clayton             case 'o':
536244d93782SGreg Clayton                 // 'o' == step out
536344d93782SGreg Clayton                 {
536444d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
536544d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
536644d93782SGreg Clayton                     {
536744d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOut();
536844d93782SGreg Clayton                     }
536944d93782SGreg Clayton                 }
537044d93782SGreg Clayton                 return eKeyHandled;
5371315b6884SEugene Zelenko 
537244d93782SGreg Clayton             case 'n':   // 'n' == step over
537344d93782SGreg Clayton             case 'N':   // 'N' == step over instruction
537444d93782SGreg Clayton                 {
537544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
537644d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
537744d93782SGreg Clayton                     {
537844d93782SGreg Clayton                         bool source_step = (c == 'n');
537944d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOver(source_step);
538044d93782SGreg Clayton                     }
538144d93782SGreg Clayton                 }
538244d93782SGreg Clayton                 return eKeyHandled;
5383315b6884SEugene Zelenko 
538444d93782SGreg Clayton             case 's':   // 's' == step into
538544d93782SGreg Clayton             case 'S':   // 'S' == step into instruction
538644d93782SGreg Clayton                 {
538744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
538844d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
538944d93782SGreg Clayton                     {
539044d93782SGreg Clayton                         bool source_step = (c == 's');
53914b4b2478SJim Ingham                         exe_ctx.GetThreadRef().StepIn(source_step);
539244d93782SGreg Clayton                     }
539344d93782SGreg Clayton                 }
539444d93782SGreg Clayton                 return eKeyHandled;
539544d93782SGreg Clayton 
539644d93782SGreg Clayton             case 'h':
539744d93782SGreg Clayton                 window.CreateHelpSubwindow ();
539844d93782SGreg Clayton                 return eKeyHandled;
539944d93782SGreg Clayton 
540044d93782SGreg Clayton             default:
540144d93782SGreg Clayton                 break;
540244d93782SGreg Clayton         }
540344d93782SGreg Clayton         return eKeyNotHandled;
540444d93782SGreg Clayton     }
540544d93782SGreg Clayton 
540644d93782SGreg Clayton protected:
540744d93782SGreg Clayton     typedef std::set<uint32_t> BreakpointLines;
540844d93782SGreg Clayton     typedef std::set<lldb::addr_t> BreakpointAddrs;
540944d93782SGreg Clayton 
541044d93782SGreg Clayton     Debugger &m_debugger;
541144d93782SGreg Clayton     SymbolContext m_sc;
541244d93782SGreg Clayton     SourceManager::FileSP m_file_sp;
541344d93782SGreg Clayton     SymbolContextScope *m_disassembly_scope;
541444d93782SGreg Clayton     lldb::DisassemblerSP m_disassembly_sp;
541544d93782SGreg Clayton     AddressRange m_disassembly_range;
5416ec990867SGreg Clayton     StreamString m_title;
541744d93782SGreg Clayton     lldb::user_id_t m_tid;
541844d93782SGreg Clayton     char m_line_format[8];
541944d93782SGreg Clayton     int m_line_width;
542044d93782SGreg Clayton     uint32_t m_selected_line;       // The selected line
542144d93782SGreg Clayton     uint32_t m_pc_line;             // The line with the PC
542244d93782SGreg Clayton     uint32_t m_stop_id;
542344d93782SGreg Clayton     uint32_t m_frame_idx;
542444d93782SGreg Clayton     int m_first_visible_line;
542544d93782SGreg Clayton     int m_min_x;
542644d93782SGreg Clayton     int m_min_y;
542744d93782SGreg Clayton     int m_max_x;
542844d93782SGreg Clayton     int m_max_y;
542944d93782SGreg Clayton };
543044d93782SGreg Clayton 
543144d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true };
543244d93782SGreg Clayton 
543344d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) :
5434e30f11d9SKate Stone     IOHandler (debugger, IOHandler::Type::Curses)
543544d93782SGreg Clayton {
543644d93782SGreg Clayton }
543744d93782SGreg Clayton 
543844d93782SGreg Clayton void
543944d93782SGreg Clayton IOHandlerCursesGUI::Activate ()
544044d93782SGreg Clayton {
544144d93782SGreg Clayton     IOHandler::Activate();
544244d93782SGreg Clayton     if (!m_app_ap)
544344d93782SGreg Clayton     {
544444d93782SGreg Clayton         m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE()));
544544d93782SGreg Clayton 
544644d93782SGreg Clayton         // This is both a window and a menu delegate
544744d93782SGreg Clayton         std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger));
544844d93782SGreg Clayton 
544944d93782SGreg Clayton         MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
545044d93782SGreg Clayton         MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
5451c5dac77aSEugene Zelenko         MenuSP exit_menuitem_sp(new Menu("Exit", nullptr, 'x', ApplicationDelegate::eMenuID_LLDBExit));
545244d93782SGreg Clayton         exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
5453c5dac77aSEugene Zelenko         lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", nullptr, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
545444d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
545544d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (exit_menuitem_sp);
545644d93782SGreg Clayton 
545744d93782SGreg Clayton         MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target));
5458c5dac77aSEugene Zelenko         target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", nullptr, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
5459c5dac77aSEugene Zelenko         target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", nullptr, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
546044d93782SGreg Clayton 
546144d93782SGreg Clayton         MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process));
5462c5dac77aSEugene Zelenko         process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach"  , nullptr, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
5463c5dac77aSEugene Zelenko         process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach"  , nullptr, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
5464c5dac77aSEugene Zelenko         process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch"  , nullptr, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
546544d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
5466c5dac77aSEugene Zelenko         process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", nullptr, 'c', ApplicationDelegate::eMenuID_ProcessContinue)));
5467c5dac77aSEugene Zelenko         process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt"    , nullptr, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
5468c5dac77aSEugene Zelenko         process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill"    , nullptr, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
546944d93782SGreg Clayton 
547044d93782SGreg Clayton         MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread));
5471c5dac77aSEugene Zelenko         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In"  , nullptr, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
5472c5dac77aSEugene Zelenko         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", nullptr, 'v', ApplicationDelegate::eMenuID_ThreadStepOver)));
5473c5dac77aSEugene Zelenko         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , nullptr, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
547444d93782SGreg Clayton 
547544d93782SGreg Clayton         MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
5476c5dac77aSEugene Zelenko         view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", nullptr, 'b', ApplicationDelegate::eMenuID_ViewBacktrace)));
5477c5dac77aSEugene Zelenko         view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", nullptr, 'r', ApplicationDelegate::eMenuID_ViewRegisters)));
5478c5dac77aSEugene Zelenko         view_menu_sp->AddSubmenu (MenuSP (new Menu("Source"   , nullptr, 's', ApplicationDelegate::eMenuID_ViewSource)));
5479c5dac77aSEugene Zelenko         view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", nullptr, 'v', ApplicationDelegate::eMenuID_ViewVariables)));
548044d93782SGreg Clayton 
548144d93782SGreg Clayton         MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
5482c5dac77aSEugene Zelenko         help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", nullptr, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
548344d93782SGreg Clayton 
548444d93782SGreg Clayton         m_app_ap->Initialize();
548544d93782SGreg Clayton         WindowSP &main_window_sp = m_app_ap->GetMainWindow();
548644d93782SGreg Clayton 
548744d93782SGreg Clayton         MenuSP menubar_sp(new Menu(Menu::Type::Bar));
548844d93782SGreg Clayton         menubar_sp->AddSubmenu (lldb_menu_sp);
548944d93782SGreg Clayton         menubar_sp->AddSubmenu (target_menu_sp);
549044d93782SGreg Clayton         menubar_sp->AddSubmenu (process_menu_sp);
549144d93782SGreg Clayton         menubar_sp->AddSubmenu (thread_menu_sp);
549244d93782SGreg Clayton         menubar_sp->AddSubmenu (view_menu_sp);
549344d93782SGreg Clayton         menubar_sp->AddSubmenu (help_menu_sp);
549444d93782SGreg Clayton         menubar_sp->SetDelegate(app_menu_delegate_sp);
549544d93782SGreg Clayton 
549644d93782SGreg Clayton         Rect content_bounds = main_window_sp->GetFrame();
549744d93782SGreg Clayton         Rect menubar_bounds = content_bounds.MakeMenuBar();
549844d93782SGreg Clayton         Rect status_bounds = content_bounds.MakeStatusBar();
549944d93782SGreg Clayton         Rect source_bounds;
550044d93782SGreg Clayton         Rect variables_bounds;
550144d93782SGreg Clayton         Rect threads_bounds;
550244d93782SGreg Clayton         Rect source_variables_bounds;
550344d93782SGreg Clayton         content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds);
550444d93782SGreg Clayton         source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds);
550544d93782SGreg Clayton 
550644d93782SGreg Clayton         WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
550744d93782SGreg Clayton         // Let the menubar get keys if the active window doesn't handle the
550844d93782SGreg Clayton         // keys that are typed so it can respond to menubar key presses.
550944d93782SGreg Clayton         menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window
551044d93782SGreg Clayton         menubar_window_sp->SetDelegate(menubar_sp);
551144d93782SGreg Clayton 
551244d93782SGreg Clayton         WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source",
551344d93782SGreg Clayton                                                                    source_bounds,
551444d93782SGreg Clayton                                                                    true));
551544d93782SGreg Clayton         WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables",
551644d93782SGreg Clayton                                                                       variables_bounds,
551744d93782SGreg Clayton                                                                       false));
551844d93782SGreg Clayton         WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads",
551944d93782SGreg Clayton                                                                       threads_bounds,
552044d93782SGreg Clayton                                                                       false));
552144d93782SGreg Clayton         WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status",
552244d93782SGreg Clayton                                                                    status_bounds,
552344d93782SGreg Clayton                                                                    false));
552444d93782SGreg Clayton         status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window
552544d93782SGreg Clayton         main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
552644d93782SGreg Clayton         source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
552744d93782SGreg Clayton         variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
5528ec990867SGreg Clayton         TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger));
552944d93782SGreg Clayton         threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
553044d93782SGreg Clayton         status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
553144d93782SGreg Clayton 
55325fdb09bbSGreg Clayton         // Show the main help window once the first time the curses GUI is launched
55335fdb09bbSGreg Clayton         static bool g_showed_help = false;
55345fdb09bbSGreg Clayton         if (!g_showed_help)
55355fdb09bbSGreg Clayton         {
55365fdb09bbSGreg Clayton             g_showed_help = true;
55375fdb09bbSGreg Clayton             main_window_sp->CreateHelpSubwindow();
55385fdb09bbSGreg Clayton         }
55395fdb09bbSGreg Clayton 
554044d93782SGreg Clayton         init_pair (1, COLOR_WHITE   , COLOR_BLUE  );
554144d93782SGreg Clayton         init_pair (2, COLOR_BLACK   , COLOR_WHITE );
554244d93782SGreg Clayton         init_pair (3, COLOR_MAGENTA , COLOR_WHITE );
554344d93782SGreg Clayton         init_pair (4, COLOR_MAGENTA , COLOR_BLACK );
554444d93782SGreg Clayton         init_pair (5, COLOR_RED     , COLOR_BLACK );
554544d93782SGreg Clayton     }
554644d93782SGreg Clayton }
554744d93782SGreg Clayton 
554844d93782SGreg Clayton void
554944d93782SGreg Clayton IOHandlerCursesGUI::Deactivate ()
555044d93782SGreg Clayton {
555144d93782SGreg Clayton     m_app_ap->Terminate();
555244d93782SGreg Clayton }
555344d93782SGreg Clayton 
555444d93782SGreg Clayton void
555544d93782SGreg Clayton IOHandlerCursesGUI::Run ()
555644d93782SGreg Clayton {
555744d93782SGreg Clayton     m_app_ap->Run(m_debugger);
555844d93782SGreg Clayton     SetIsDone(true);
555944d93782SGreg Clayton }
556044d93782SGreg Clayton 
5561315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
556244d93782SGreg Clayton 
556344d93782SGreg Clayton void
5564e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel ()
5565e68f5d6bSGreg Clayton {
5566e68f5d6bSGreg Clayton }
556744d93782SGreg Clayton 
5568f0066ad0SGreg Clayton bool
556944d93782SGreg Clayton IOHandlerCursesGUI::Interrupt ()
557044d93782SGreg Clayton {
5571f0066ad0SGreg Clayton     return false;
557244d93782SGreg Clayton }
557344d93782SGreg Clayton 
557444d93782SGreg Clayton void
557544d93782SGreg Clayton IOHandlerCursesGUI::GotEOF()
557644d93782SGreg Clayton {
557744d93782SGreg Clayton }
557844d93782SGreg Clayton 
5579315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
5580