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 
1044d93782SGreg Clayton 
1144d93782SGreg Clayton #include "lldb/lldb-python.h"
1244d93782SGreg Clayton 
1344d93782SGreg Clayton #include <string>
1444d93782SGreg Clayton 
1544d93782SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h"
1644d93782SGreg Clayton #include "lldb/Core/IOHandler.h"
1744d93782SGreg Clayton #include "lldb/Core/Debugger.h"
18ec990867SGreg Clayton #include "lldb/Core/Module.h"
1944d93782SGreg Clayton #include "lldb/Core/State.h"
2044d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
2144d93782SGreg Clayton #include "lldb/Core/ValueObjectRegister.h"
22cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
2344d93782SGreg Clayton #include "lldb/Host/Editline.h"
24cacde7dfSTodd Fiala #endif
2544d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
2644d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
2744d93782SGreg Clayton #include "lldb/Symbol/Block.h"
2844d93782SGreg Clayton #include "lldb/Symbol/Function.h"
2944d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
3044d93782SGreg Clayton #include "lldb/Target/RegisterContext.h"
3144d93782SGreg Clayton #include "lldb/Target/ThreadPlan.h"
3244d93782SGreg Clayton 
33914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
3444d93782SGreg Clayton #include <ncurses.h>
3544d93782SGreg Clayton #include <panel.h>
36914b8d98SDeepak Panickal #endif
3744d93782SGreg Clayton 
3844d93782SGreg Clayton using namespace lldb;
3944d93782SGreg Clayton using namespace lldb_private;
4044d93782SGreg Clayton 
41e30f11d9SKate Stone IOHandler::IOHandler (Debugger &debugger, IOHandler::Type type) :
4244d93782SGreg Clayton     IOHandler (debugger,
43e30f11d9SKate Stone                type,
4444d93782SGreg Clayton                StreamFileSP(),  // Adopt STDIN from top input reader
4544d93782SGreg Clayton                StreamFileSP(),  // Adopt STDOUT from top input reader
46340b0309SGreg Clayton                StreamFileSP(),  // Adopt STDERR from top input reader
47340b0309SGreg Clayton                0)               // Flags
4844d93782SGreg Clayton {
4944d93782SGreg Clayton }
5044d93782SGreg Clayton 
5144d93782SGreg Clayton 
5244d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger,
53e30f11d9SKate Stone                       IOHandler::Type type,
5444d93782SGreg Clayton                       const lldb::StreamFileSP &input_sp,
5544d93782SGreg Clayton                       const lldb::StreamFileSP &output_sp,
56340b0309SGreg Clayton                       const lldb::StreamFileSP &error_sp,
57340b0309SGreg Clayton                       uint32_t flags) :
5844d93782SGreg Clayton     m_debugger (debugger),
5944d93782SGreg Clayton     m_input_sp (input_sp),
6044d93782SGreg Clayton     m_output_sp (output_sp),
6144d93782SGreg Clayton     m_error_sp (error_sp),
62e30f11d9SKate Stone     m_popped (false),
63340b0309SGreg Clayton     m_flags (flags),
64e30f11d9SKate Stone     m_type (type),
6544d93782SGreg Clayton     m_user_data (NULL),
6644d93782SGreg Clayton     m_done (false),
6744d93782SGreg Clayton     m_active (false)
6844d93782SGreg Clayton {
6944d93782SGreg Clayton     // If any files are not specified, then adopt them from the top input reader.
7044d93782SGreg Clayton     if (!m_input_sp || !m_output_sp || !m_error_sp)
7144d93782SGreg Clayton         debugger.AdoptTopIOHandlerFilesIfInvalid (m_input_sp,
7244d93782SGreg Clayton                                                   m_output_sp,
7344d93782SGreg Clayton                                                   m_error_sp);
7444d93782SGreg Clayton }
7544d93782SGreg Clayton 
7644d93782SGreg Clayton IOHandler::~IOHandler()
7744d93782SGreg Clayton {
7844d93782SGreg Clayton }
7944d93782SGreg Clayton 
8044d93782SGreg Clayton 
8144d93782SGreg Clayton int
8244d93782SGreg Clayton IOHandler::GetInputFD()
8344d93782SGreg Clayton {
8444d93782SGreg Clayton     if (m_input_sp)
8544d93782SGreg Clayton         return m_input_sp->GetFile().GetDescriptor();
8644d93782SGreg Clayton     return -1;
8744d93782SGreg Clayton }
8844d93782SGreg Clayton 
8944d93782SGreg Clayton int
9044d93782SGreg Clayton IOHandler::GetOutputFD()
9144d93782SGreg Clayton {
9244d93782SGreg Clayton     if (m_output_sp)
9344d93782SGreg Clayton         return m_output_sp->GetFile().GetDescriptor();
9444d93782SGreg Clayton     return -1;
9544d93782SGreg Clayton }
9644d93782SGreg Clayton 
9744d93782SGreg Clayton int
9844d93782SGreg Clayton IOHandler::GetErrorFD()
9944d93782SGreg Clayton {
10044d93782SGreg Clayton     if (m_error_sp)
10144d93782SGreg Clayton         return m_error_sp->GetFile().GetDescriptor();
10244d93782SGreg Clayton     return -1;
10344d93782SGreg Clayton }
10444d93782SGreg Clayton 
10544d93782SGreg Clayton FILE *
10644d93782SGreg Clayton IOHandler::GetInputFILE()
10744d93782SGreg Clayton {
10844d93782SGreg Clayton     if (m_input_sp)
10944d93782SGreg Clayton         return m_input_sp->GetFile().GetStream();
11044d93782SGreg Clayton     return NULL;
11144d93782SGreg Clayton }
11244d93782SGreg Clayton 
11344d93782SGreg Clayton FILE *
11444d93782SGreg Clayton IOHandler::GetOutputFILE()
11544d93782SGreg Clayton {
11644d93782SGreg Clayton     if (m_output_sp)
11744d93782SGreg Clayton         return m_output_sp->GetFile().GetStream();
11844d93782SGreg Clayton     return NULL;
11944d93782SGreg Clayton }
12044d93782SGreg Clayton 
12144d93782SGreg Clayton FILE *
12244d93782SGreg Clayton IOHandler::GetErrorFILE()
12344d93782SGreg Clayton {
12444d93782SGreg Clayton     if (m_error_sp)
12544d93782SGreg Clayton         return m_error_sp->GetFile().GetStream();
12644d93782SGreg Clayton     return NULL;
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 
14244d93782SGreg Clayton StreamFileSP &
14344d93782SGreg Clayton IOHandler::GetErrorStreamFile()
14444d93782SGreg Clayton {
14544d93782SGreg Clayton     return m_error_sp;
14644d93782SGreg Clayton }
14744d93782SGreg Clayton 
148340b0309SGreg Clayton bool
149340b0309SGreg Clayton IOHandler::GetIsInteractive ()
150340b0309SGreg Clayton {
151340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsInteractive ();
152340b0309SGreg Clayton }
153340b0309SGreg Clayton 
154340b0309SGreg Clayton bool
155340b0309SGreg Clayton IOHandler::GetIsRealTerminal ()
156340b0309SGreg Clayton {
157340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsRealTerminal();
158340b0309SGreg Clayton }
15944d93782SGreg Clayton 
160e30f11d9SKate Stone void
161e30f11d9SKate Stone IOHandler::SetPopped (bool b)
162e30f11d9SKate Stone {
163e30f11d9SKate Stone     m_popped.SetValue(b, eBroadcastOnChange);
164e30f11d9SKate Stone }
165e30f11d9SKate Stone 
166e30f11d9SKate Stone void
167e30f11d9SKate Stone IOHandler::WaitForPop ()
168e30f11d9SKate Stone {
169e30f11d9SKate Stone     m_popped.WaitForValueEqualTo(true);
170e30f11d9SKate Stone }
171e30f11d9SKate Stone 
17244d93782SGreg Clayton IOHandlerConfirm::IOHandlerConfirm (Debugger &debugger,
17344d93782SGreg Clayton                                     const char *prompt,
17444d93782SGreg Clayton                                     bool default_response) :
17544d93782SGreg Clayton     IOHandlerEditline(debugger,
176e30f11d9SKate Stone                       IOHandler::Type::Confirm,
17744d93782SGreg Clayton                       NULL,     // NULL editline_name means no history loaded/saved
178e30f11d9SKate Stone                       NULL,     // No prompt
179e30f11d9SKate Stone                       NULL,     // No continuation prompt
18044d93782SGreg Clayton                       false,    // Multi-line
181e30f11d9SKate Stone                       false,    // Don't colorize the prompt (i.e. the confirm message.)
182f6913cd7SGreg Clayton                       0,
18344d93782SGreg Clayton                       *this),
18444d93782SGreg Clayton     m_default_response (default_response),
18544d93782SGreg Clayton     m_user_response (default_response)
18644d93782SGreg Clayton {
18744d93782SGreg Clayton     StreamString prompt_stream;
18844d93782SGreg Clayton     prompt_stream.PutCString(prompt);
18944d93782SGreg Clayton     if (m_default_response)
19044d93782SGreg Clayton         prompt_stream.Printf(": [Y/n] ");
19144d93782SGreg Clayton     else
19244d93782SGreg Clayton         prompt_stream.Printf(": [y/N] ");
19344d93782SGreg Clayton 
19444d93782SGreg Clayton     SetPrompt (prompt_stream.GetString().c_str());
19544d93782SGreg Clayton 
19644d93782SGreg Clayton }
19744d93782SGreg Clayton 
19844d93782SGreg Clayton 
19944d93782SGreg Clayton IOHandlerConfirm::~IOHandlerConfirm ()
20044d93782SGreg Clayton {
20144d93782SGreg Clayton }
20244d93782SGreg Clayton 
20344d93782SGreg Clayton int
20444d93782SGreg Clayton IOHandlerConfirm::IOHandlerComplete (IOHandler &io_handler,
20544d93782SGreg Clayton                                      const char *current_line,
20644d93782SGreg Clayton                                      const char *cursor,
20744d93782SGreg Clayton                                      const char *last_char,
20844d93782SGreg Clayton                                      int skip_first_n_matches,
20944d93782SGreg Clayton                                      int max_matches,
21044d93782SGreg Clayton                                      StringList &matches)
21144d93782SGreg Clayton {
21244d93782SGreg Clayton     if (current_line == cursor)
21344d93782SGreg Clayton     {
21444d93782SGreg Clayton         if (m_default_response)
21544d93782SGreg Clayton         {
21644d93782SGreg Clayton             matches.AppendString("y");
21744d93782SGreg Clayton         }
21844d93782SGreg Clayton         else
21944d93782SGreg Clayton         {
22044d93782SGreg Clayton             matches.AppendString("n");
22144d93782SGreg Clayton         }
22244d93782SGreg Clayton     }
22344d93782SGreg Clayton     return matches.GetSize();
22444d93782SGreg Clayton }
22544d93782SGreg Clayton 
22644d93782SGreg Clayton void
22744d93782SGreg Clayton IOHandlerConfirm::IOHandlerInputComplete (IOHandler &io_handler, std::string &line)
22844d93782SGreg Clayton {
22944d93782SGreg Clayton     if (line.empty())
23044d93782SGreg Clayton     {
23144d93782SGreg Clayton         // User just hit enter, set the response to the default
23244d93782SGreg Clayton         m_user_response = m_default_response;
23344d93782SGreg Clayton         io_handler.SetIsDone(true);
23444d93782SGreg Clayton         return;
23544d93782SGreg Clayton     }
23644d93782SGreg Clayton 
23744d93782SGreg Clayton     if (line.size() == 1)
23844d93782SGreg Clayton     {
23944d93782SGreg Clayton         switch (line[0])
24044d93782SGreg Clayton         {
24144d93782SGreg Clayton             case 'y':
24244d93782SGreg Clayton             case 'Y':
24344d93782SGreg Clayton                 m_user_response = true;
24444d93782SGreg Clayton                 io_handler.SetIsDone(true);
24544d93782SGreg Clayton                 return;
24644d93782SGreg Clayton             case 'n':
24744d93782SGreg Clayton             case 'N':
24844d93782SGreg Clayton                 m_user_response = false;
24944d93782SGreg Clayton                 io_handler.SetIsDone(true);
25044d93782SGreg Clayton                 return;
25144d93782SGreg Clayton             default:
25244d93782SGreg Clayton                 break;
25344d93782SGreg Clayton         }
25444d93782SGreg Clayton     }
25544d93782SGreg Clayton 
25644d93782SGreg Clayton     if (line == "yes" || line == "YES" || line == "Yes")
25744d93782SGreg Clayton     {
25844d93782SGreg Clayton         m_user_response = true;
25944d93782SGreg Clayton         io_handler.SetIsDone(true);
26044d93782SGreg Clayton     }
26144d93782SGreg Clayton     else if (line == "no" || line == "NO" || line == "No")
26244d93782SGreg Clayton     {
26344d93782SGreg Clayton         m_user_response = false;
26444d93782SGreg Clayton         io_handler.SetIsDone(true);
26544d93782SGreg Clayton     }
26644d93782SGreg Clayton }
26744d93782SGreg Clayton 
26844d93782SGreg Clayton int
26944d93782SGreg Clayton IOHandlerDelegate::IOHandlerComplete (IOHandler &io_handler,
27044d93782SGreg Clayton                                       const char *current_line,
27144d93782SGreg Clayton                                       const char *cursor,
27244d93782SGreg Clayton                                       const char *last_char,
27344d93782SGreg Clayton                                       int skip_first_n_matches,
27444d93782SGreg Clayton                                       int max_matches,
27544d93782SGreg Clayton                                       StringList &matches)
27644d93782SGreg Clayton {
27744d93782SGreg Clayton     switch (m_completion)
27844d93782SGreg Clayton     {
27944d93782SGreg Clayton     case Completion::None:
28044d93782SGreg Clayton         break;
28144d93782SGreg Clayton 
28244d93782SGreg Clayton     case Completion::LLDBCommand:
28344d93782SGreg Clayton         return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion (current_line,
28444d93782SGreg Clayton                                                                                   cursor,
28544d93782SGreg Clayton                                                                                   last_char,
28644d93782SGreg Clayton                                                                                   skip_first_n_matches,
28744d93782SGreg Clayton                                                                                   max_matches,
28844d93782SGreg Clayton                                                                                   matches);
28944d93782SGreg Clayton 
29044d93782SGreg Clayton     case Completion::Expression:
29144d93782SGreg Clayton         {
29244d93782SGreg Clayton             bool word_complete = false;
29344d93782SGreg Clayton             const char *word_start = cursor;
29444d93782SGreg Clayton             if (cursor > current_line)
29544d93782SGreg Clayton                 --word_start;
29644d93782SGreg Clayton             while (word_start > current_line && !isspace(*word_start))
29744d93782SGreg Clayton                 --word_start;
29844d93782SGreg Clayton             CommandCompletions::InvokeCommonCompletionCallbacks (io_handler.GetDebugger().GetCommandInterpreter(),
29944d93782SGreg Clayton                                                                  CommandCompletions::eVariablePathCompletion,
30044d93782SGreg Clayton                                                                  word_start,
30144d93782SGreg Clayton                                                                  skip_first_n_matches,
30244d93782SGreg Clayton                                                                  max_matches,
30344d93782SGreg Clayton                                                                  NULL,
30444d93782SGreg Clayton                                                                  word_complete,
30544d93782SGreg Clayton                                                                  matches);
30644d93782SGreg Clayton 
30744d93782SGreg Clayton             size_t num_matches = matches.GetSize();
30844d93782SGreg Clayton             if (num_matches > 0)
30944d93782SGreg Clayton             {
31044d93782SGreg Clayton                 std::string common_prefix;
31144d93782SGreg Clayton                 matches.LongestCommonPrefix (common_prefix);
31244d93782SGreg Clayton                 const size_t partial_name_len = strlen(word_start);
31344d93782SGreg Clayton 
31444d93782SGreg Clayton                 // If we matched a unique single command, add a space...
31544d93782SGreg Clayton                 // Only do this if the completer told us this was a complete word, however...
31644d93782SGreg Clayton                 if (num_matches == 1 && word_complete)
31744d93782SGreg Clayton                 {
31844d93782SGreg Clayton                     common_prefix.push_back(' ');
31944d93782SGreg Clayton                 }
32044d93782SGreg Clayton                 common_prefix.erase (0, partial_name_len);
32144d93782SGreg Clayton                 matches.InsertStringAtIndex(0, std::move(common_prefix));
32244d93782SGreg Clayton             }
32344d93782SGreg Clayton             return num_matches;
32444d93782SGreg Clayton         }
32544d93782SGreg Clayton         break;
32644d93782SGreg Clayton     }
32744d93782SGreg Clayton 
32844d93782SGreg Clayton 
32944d93782SGreg Clayton     return 0;
33044d93782SGreg Clayton }
33144d93782SGreg Clayton 
33244d93782SGreg Clayton 
33344d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
334e30f11d9SKate Stone                                       IOHandler::Type type,
33544d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
33644d93782SGreg Clayton                                       const char *prompt,
337e30f11d9SKate Stone                                       const char *continuation_prompt,
33844d93782SGreg Clayton                                       bool multi_line,
339e30f11d9SKate Stone                                       bool color_prompts,
340f6913cd7SGreg Clayton                                       uint32_t line_number_start,
34144d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
34244d93782SGreg Clayton     IOHandlerEditline(debugger,
343e30f11d9SKate Stone                       type,
34444d93782SGreg Clayton                       StreamFileSP(), // Inherit input from top input reader
34544d93782SGreg Clayton                       StreamFileSP(), // Inherit output from top input reader
34644d93782SGreg Clayton                       StreamFileSP(), // Inherit error from top input reader
347340b0309SGreg Clayton                       0,              // Flags
34844d93782SGreg Clayton                       editline_name,  // Used for saving history files
34944d93782SGreg Clayton                       prompt,
350e30f11d9SKate Stone                       continuation_prompt,
35144d93782SGreg Clayton                       multi_line,
352e30f11d9SKate Stone                       color_prompts,
353f6913cd7SGreg Clayton                       line_number_start,
35444d93782SGreg Clayton                       delegate)
35544d93782SGreg Clayton {
35644d93782SGreg Clayton }
35744d93782SGreg Clayton 
35844d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
359e30f11d9SKate Stone                                       IOHandler::Type type,
36044d93782SGreg Clayton                                       const lldb::StreamFileSP &input_sp,
36144d93782SGreg Clayton                                       const lldb::StreamFileSP &output_sp,
36244d93782SGreg Clayton                                       const lldb::StreamFileSP &error_sp,
363340b0309SGreg Clayton                                       uint32_t flags,
36444d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
36544d93782SGreg Clayton                                       const char *prompt,
366e30f11d9SKate Stone                                       const char *continuation_prompt,
36744d93782SGreg Clayton                                       bool multi_line,
368e30f11d9SKate Stone                                       bool color_prompts,
369f6913cd7SGreg Clayton                                       uint32_t line_number_start,
37044d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
371e30f11d9SKate Stone     IOHandler (debugger, type, input_sp, output_sp, error_sp, flags),
372cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
37344d93782SGreg Clayton     m_editline_ap (),
374cacde7dfSTodd Fiala #endif
37544d93782SGreg Clayton     m_delegate (delegate),
37644d93782SGreg Clayton     m_prompt (),
377e30f11d9SKate Stone     m_continuation_prompt(),
378e30f11d9SKate Stone     m_current_lines_ptr (NULL),
379f6913cd7SGreg Clayton     m_base_line_number (line_number_start),
380e30f11d9SKate Stone     m_curr_line_idx (UINT32_MAX),
381e30f11d9SKate Stone     m_multi_line (multi_line),
382e30f11d9SKate Stone     m_color_prompts (color_prompts),
383e30f11d9SKate Stone     m_interrupt_exits (true)
38444d93782SGreg Clayton {
38544d93782SGreg Clayton     SetPrompt(prompt);
38644d93782SGreg Clayton 
387cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
388914b8d98SDeepak Panickal     bool use_editline = false;
389340b0309SGreg Clayton 
390340b0309SGreg Clayton     use_editline = m_input_sp->GetFile().GetIsRealTerminal();
39144d93782SGreg Clayton 
39244d93782SGreg Clayton     if (use_editline)
39344d93782SGreg Clayton     {
39444d93782SGreg Clayton         m_editline_ap.reset(new Editline (editline_name,
39544d93782SGreg Clayton                                           GetInputFILE (),
39644d93782SGreg Clayton                                           GetOutputFILE (),
397e30f11d9SKate Stone                                           GetErrorFILE (),
398e30f11d9SKate Stone                                           m_color_prompts));
399e30f11d9SKate Stone         m_editline_ap->SetIsInputCompleteCallback (IsInputCompleteCallback, this);
40044d93782SGreg Clayton         m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this);
401e30f11d9SKate Stone         // See if the delegate supports fixing indentation
402e30f11d9SKate Stone         const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
403e30f11d9SKate Stone         if (indent_chars)
404e30f11d9SKate Stone         {
405e30f11d9SKate Stone             // The delegate does support indentation, hook it up so when any indentation
406e30f11d9SKate Stone             // character is typed, the delegate gets a chance to fix it
407e30f11d9SKate Stone             m_editline_ap->SetFixIndentationCallback (FixIndentationCallback, this, indent_chars);
408e30f11d9SKate Stone         }
40944d93782SGreg Clayton     }
410cacde7dfSTodd Fiala #endif
411e30f11d9SKate Stone     SetBaseLineNumber (m_base_line_number);
412e30f11d9SKate Stone     SetPrompt(prompt ? prompt : "");
413e30f11d9SKate Stone     SetContinuationPrompt(continuation_prompt);
41444d93782SGreg Clayton }
41544d93782SGreg Clayton 
41644d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline ()
41744d93782SGreg Clayton {
418cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
41944d93782SGreg Clayton     m_editline_ap.reset();
420cacde7dfSTodd Fiala #endif
42144d93782SGreg Clayton }
42244d93782SGreg Clayton 
423e30f11d9SKate Stone void
424e30f11d9SKate Stone IOHandlerEditline::Activate ()
425e30f11d9SKate Stone {
426e30f11d9SKate Stone     IOHandler::Activate();
427e30f11d9SKate Stone     m_delegate.IOHandlerActivated(*this);
428e30f11d9SKate Stone }
429e30f11d9SKate Stone 
430e30f11d9SKate Stone void
431e30f11d9SKate Stone IOHandlerEditline::Deactivate ()
432e30f11d9SKate Stone {
433e30f11d9SKate Stone     IOHandler::Deactivate();
434e30f11d9SKate Stone     m_delegate.IOHandlerDeactivated(*this);
435e30f11d9SKate Stone }
436e30f11d9SKate Stone 
43744d93782SGreg Clayton 
43844d93782SGreg Clayton bool
439f0066ad0SGreg Clayton IOHandlerEditline::GetLine (std::string &line, bool &interrupted)
44044d93782SGreg Clayton {
441cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
44244d93782SGreg Clayton     if (m_editline_ap)
44344d93782SGreg Clayton     {
444e30f11d9SKate Stone         return m_editline_ap->GetLine (line, interrupted);
44544d93782SGreg Clayton     }
44644d93782SGreg Clayton     else
44744d93782SGreg Clayton     {
448cacde7dfSTodd Fiala #endif
44944d93782SGreg Clayton         line.clear();
45044d93782SGreg Clayton 
45144d93782SGreg Clayton         FILE *in = GetInputFILE();
45244d93782SGreg Clayton         if (in)
45344d93782SGreg Clayton         {
454340b0309SGreg Clayton             if (GetIsInteractive())
45544d93782SGreg Clayton             {
456e30f11d9SKate Stone                 const char *prompt = NULL;
457e30f11d9SKate Stone 
458e30f11d9SKate Stone                 if (m_multi_line && m_curr_line_idx > 0)
459e30f11d9SKate Stone                     prompt = GetContinuationPrompt();
460e30f11d9SKate Stone 
461e30f11d9SKate Stone                 if (prompt == NULL)
462e30f11d9SKate Stone                     prompt = GetPrompt();
463e30f11d9SKate Stone 
46444d93782SGreg Clayton                 if (prompt && prompt[0])
46544d93782SGreg Clayton                 {
46644d93782SGreg Clayton                     FILE *out = GetOutputFILE();
46744d93782SGreg Clayton                     if (out)
46844d93782SGreg Clayton                     {
46944d93782SGreg Clayton                         ::fprintf(out, "%s", prompt);
47044d93782SGreg Clayton                         ::fflush(out);
47144d93782SGreg Clayton                     }
47244d93782SGreg Clayton                 }
47344d93782SGreg Clayton             }
47444d93782SGreg Clayton             char buffer[256];
47544d93782SGreg Clayton             bool done = false;
4760f86e6e7SGreg Clayton             bool got_line = false;
47744d93782SGreg Clayton             while (!done)
47844d93782SGreg Clayton             {
47944d93782SGreg Clayton                 if (fgets(buffer, sizeof(buffer), in) == NULL)
480c9cf5798SGreg Clayton                 {
481c7797accSGreg Clayton                     const int saved_errno = errno;
482c9cf5798SGreg Clayton                     if (feof(in))
48344d93782SGreg Clayton                         done = true;
484c7797accSGreg Clayton                     else if (ferror(in))
485c7797accSGreg Clayton                     {
486c7797accSGreg Clayton                         if (saved_errno != EINTR)
487c7797accSGreg Clayton                             done = true;
488c7797accSGreg Clayton                     }
489c9cf5798SGreg Clayton                 }
49044d93782SGreg Clayton                 else
49144d93782SGreg Clayton                 {
4920f86e6e7SGreg Clayton                     got_line = true;
49344d93782SGreg Clayton                     size_t buffer_len = strlen(buffer);
49444d93782SGreg Clayton                     assert (buffer[buffer_len] == '\0');
49544d93782SGreg Clayton                     char last_char = buffer[buffer_len-1];
49644d93782SGreg Clayton                     if (last_char == '\r' || last_char == '\n')
49744d93782SGreg Clayton                     {
49844d93782SGreg Clayton                         done = true;
49944d93782SGreg Clayton                         // Strip trailing newlines
50044d93782SGreg Clayton                         while (last_char == '\r' || last_char == '\n')
50144d93782SGreg Clayton                         {
50244d93782SGreg Clayton                             --buffer_len;
50344d93782SGreg Clayton                             if (buffer_len == 0)
50444d93782SGreg Clayton                                 break;
50544d93782SGreg Clayton                             last_char = buffer[buffer_len-1];
50644d93782SGreg Clayton                         }
50744d93782SGreg Clayton                     }
50844d93782SGreg Clayton                     line.append(buffer, buffer_len);
50944d93782SGreg Clayton                 }
51044d93782SGreg Clayton             }
5110f86e6e7SGreg Clayton             // We might have gotten a newline on a line by itself
5120f86e6e7SGreg Clayton             // make sure to return true in this case.
5130f86e6e7SGreg Clayton             return got_line;
51444d93782SGreg Clayton         }
51544d93782SGreg Clayton         else
51644d93782SGreg Clayton         {
51744d93782SGreg Clayton             // No more input file, we are done...
51844d93782SGreg Clayton             SetIsDone(true);
51944d93782SGreg Clayton         }
520340b0309SGreg Clayton         return false;
521cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
52244d93782SGreg Clayton     }
523cacde7dfSTodd Fiala #endif
52444d93782SGreg Clayton }
52544d93782SGreg Clayton 
52644d93782SGreg Clayton 
527cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
528e30f11d9SKate Stone bool
529e30f11d9SKate Stone IOHandlerEditline::IsInputCompleteCallback (Editline *editline,
53044d93782SGreg Clayton                                           StringList &lines,
53144d93782SGreg Clayton                                           void *baton)
53244d93782SGreg Clayton {
53344d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
534e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader, lines);
535e30f11d9SKate Stone }
536e30f11d9SKate Stone 
537e30f11d9SKate Stone int
538e30f11d9SKate Stone IOHandlerEditline::FixIndentationCallback (Editline *editline,
539e30f11d9SKate Stone                                            const StringList &lines,
540e30f11d9SKate Stone                                            int cursor_position,
541e30f11d9SKate Stone                                            void *baton)
542e30f11d9SKate Stone {
543e30f11d9SKate Stone     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
544e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerFixIndentation(*editline_reader, lines, cursor_position);
54544d93782SGreg Clayton }
54644d93782SGreg Clayton 
54744d93782SGreg Clayton int
54844d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line,
54944d93782SGreg Clayton                                          const char *cursor,
55044d93782SGreg Clayton                                          const char *last_char,
55144d93782SGreg Clayton                                          int skip_first_n_matches,
55244d93782SGreg Clayton                                          int max_matches,
55344d93782SGreg Clayton                                          StringList &matches,
55444d93782SGreg Clayton                                          void *baton)
55544d93782SGreg Clayton {
55644d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
55744d93782SGreg Clayton     if (editline_reader)
55844d93782SGreg Clayton         return editline_reader->m_delegate.IOHandlerComplete (*editline_reader,
55944d93782SGreg Clayton                                                               current_line,
56044d93782SGreg Clayton                                                               cursor,
56144d93782SGreg Clayton                                                               last_char,
56244d93782SGreg Clayton                                                               skip_first_n_matches,
56344d93782SGreg Clayton                                                               max_matches,
56444d93782SGreg Clayton                                                               matches);
56544d93782SGreg Clayton     return 0;
56644d93782SGreg Clayton }
567cacde7dfSTodd Fiala #endif
56844d93782SGreg Clayton 
56944d93782SGreg Clayton const char *
57044d93782SGreg Clayton IOHandlerEditline::GetPrompt ()
57144d93782SGreg Clayton {
572cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
57344d93782SGreg Clayton     if (m_editline_ap)
574cacde7dfSTodd Fiala     {
57544d93782SGreg Clayton         return m_editline_ap->GetPrompt ();
576cacde7dfSTodd Fiala     }
577cacde7dfSTodd Fiala     else
578cacde7dfSTodd Fiala     {
579cacde7dfSTodd Fiala #endif
580cacde7dfSTodd Fiala         if (m_prompt.empty())
58144d93782SGreg Clayton             return NULL;
582cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
583cacde7dfSTodd Fiala     }
584cacde7dfSTodd Fiala #endif
58544d93782SGreg Clayton     return m_prompt.c_str();
58644d93782SGreg Clayton }
58744d93782SGreg Clayton 
58844d93782SGreg Clayton bool
58944d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p)
59044d93782SGreg Clayton {
59144d93782SGreg Clayton     if (p && p[0])
59244d93782SGreg Clayton         m_prompt = p;
59344d93782SGreg Clayton     else
59444d93782SGreg Clayton         m_prompt.clear();
595cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
59644d93782SGreg Clayton     if (m_editline_ap)
59744d93782SGreg Clayton         m_editline_ap->SetPrompt (m_prompt.empty() ? NULL : m_prompt.c_str());
598cacde7dfSTodd Fiala #endif
59944d93782SGreg Clayton     return true;
60044d93782SGreg Clayton }
60144d93782SGreg Clayton 
602e30f11d9SKate Stone const char *
603e30f11d9SKate Stone IOHandlerEditline::GetContinuationPrompt ()
604e30f11d9SKate Stone {
605e30f11d9SKate Stone     if (m_continuation_prompt.empty())
606e30f11d9SKate Stone         return NULL;
607e30f11d9SKate Stone     return m_continuation_prompt.c_str();
608e30f11d9SKate Stone }
609e30f11d9SKate Stone 
610e30f11d9SKate Stone 
611e30f11d9SKate Stone void
612e30f11d9SKate Stone IOHandlerEditline::SetContinuationPrompt (const char *p)
613e30f11d9SKate Stone {
614e30f11d9SKate Stone     if (p && p[0])
615e30f11d9SKate Stone         m_continuation_prompt = p;
616e30f11d9SKate Stone     else
617e30f11d9SKate Stone         m_continuation_prompt.clear();
618e30f11d9SKate Stone 
619d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
620e30f11d9SKate Stone     if (m_editline_ap)
621e30f11d9SKate Stone         m_editline_ap->SetContinuationPrompt (m_continuation_prompt.empty() ? NULL : m_continuation_prompt.c_str());
622d553d00cSZachary Turner #endif
623e30f11d9SKate Stone }
624e30f11d9SKate Stone 
625e30f11d9SKate Stone 
626f6913cd7SGreg Clayton void
627f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line)
628f6913cd7SGreg Clayton {
629f6913cd7SGreg Clayton     m_base_line_number = line;
630f6913cd7SGreg Clayton }
631e30f11d9SKate Stone 
632e30f11d9SKate Stone uint32_t
633e30f11d9SKate Stone IOHandlerEditline::GetCurrentLineIndex () const
634e30f11d9SKate Stone {
635d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
636e30f11d9SKate Stone     if (m_editline_ap)
637e30f11d9SKate Stone         return m_editline_ap->GetCurrentLine();
638e30f11d9SKate Stone #endif
639e30f11d9SKate Stone     return m_curr_line_idx;
640e30f11d9SKate Stone }
641e30f11d9SKate Stone 
64244d93782SGreg Clayton bool
643f0066ad0SGreg Clayton IOHandlerEditline::GetLines (StringList &lines, bool &interrupted)
64444d93782SGreg Clayton {
645e30f11d9SKate Stone     m_current_lines_ptr = &lines;
646e30f11d9SKate Stone 
64744d93782SGreg Clayton     bool success = false;
648cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
64944d93782SGreg Clayton     if (m_editline_ap)
65044d93782SGreg Clayton     {
651e30f11d9SKate Stone         return m_editline_ap->GetLines (m_base_line_number, lines, interrupted);
65244d93782SGreg Clayton     }
65344d93782SGreg Clayton     else
65444d93782SGreg Clayton     {
655cacde7dfSTodd Fiala #endif
656e30f11d9SKate Stone         bool done = false;
657c3d874a5SGreg Clayton         Error error;
65844d93782SGreg Clayton 
659e30f11d9SKate Stone         while (!done)
66044d93782SGreg Clayton         {
661f6913cd7SGreg Clayton             // Show line numbers if we are asked to
66244d93782SGreg Clayton             std::string line;
663f6913cd7SGreg Clayton             if (m_base_line_number > 0 && GetIsInteractive())
664f6913cd7SGreg Clayton             {
665f6913cd7SGreg Clayton                 FILE *out = GetOutputFILE();
666f6913cd7SGreg Clayton                 if (out)
667bc88d938SGreg Clayton                     ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(), GetPrompt() == NULL ? " " : "");
668f6913cd7SGreg Clayton             }
669f6913cd7SGreg Clayton 
670e30f11d9SKate Stone             m_curr_line_idx = lines.GetSize();
671e30f11d9SKate Stone 
672f0066ad0SGreg Clayton             bool interrupted = false;
673e30f11d9SKate Stone             if (GetLine(line, interrupted) && !interrupted)
67444d93782SGreg Clayton             {
67544d93782SGreg Clayton                 lines.AppendString(line);
676e30f11d9SKate Stone                 done = m_delegate.IOHandlerIsInputComplete(*this, lines);
677f0066ad0SGreg Clayton             }
67844d93782SGreg Clayton             else
67944d93782SGreg Clayton             {
680e30f11d9SKate Stone                 done = true;
68144d93782SGreg Clayton             }
68244d93782SGreg Clayton         }
68344d93782SGreg Clayton         success = lines.GetSize() > 0;
684cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
68544d93782SGreg Clayton     }
686cacde7dfSTodd Fiala #endif
68744d93782SGreg Clayton     return success;
68844d93782SGreg Clayton }
68944d93782SGreg Clayton 
69044d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data
69144d93782SGreg Clayton // from the "in" and place output into "out" and "err and return
69244d93782SGreg Clayton // when done.
69344d93782SGreg Clayton void
69444d93782SGreg Clayton IOHandlerEditline::Run ()
69544d93782SGreg Clayton {
69644d93782SGreg Clayton     std::string line;
69744d93782SGreg Clayton     while (IsActive())
69844d93782SGreg Clayton     {
699f0066ad0SGreg Clayton         bool interrupted = false;
70044d93782SGreg Clayton         if (m_multi_line)
70144d93782SGreg Clayton         {
70244d93782SGreg Clayton             StringList lines;
703f0066ad0SGreg Clayton             if (GetLines (lines, interrupted))
704f0066ad0SGreg Clayton             {
705f0066ad0SGreg Clayton                 if (interrupted)
706f0066ad0SGreg Clayton                 {
707e30f11d9SKate Stone                     m_done = m_interrupt_exits;
708e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
709e30f11d9SKate Stone 
710f0066ad0SGreg Clayton                 }
711f0066ad0SGreg Clayton                 else
71244d93782SGreg Clayton                 {
71344d93782SGreg Clayton                     line = lines.CopyList();
71444d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
71544d93782SGreg Clayton                 }
716f0066ad0SGreg Clayton             }
71744d93782SGreg Clayton             else
71844d93782SGreg Clayton             {
71944d93782SGreg Clayton                 m_done = true;
72044d93782SGreg Clayton             }
72144d93782SGreg Clayton         }
72244d93782SGreg Clayton         else
72344d93782SGreg Clayton         {
724f0066ad0SGreg Clayton             if (GetLine(line, interrupted))
72544d93782SGreg Clayton             {
726e30f11d9SKate Stone                 if (interrupted)
727e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
728e30f11d9SKate Stone                 else
72944d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
73044d93782SGreg Clayton             }
73144d93782SGreg Clayton             else
73244d93782SGreg Clayton             {
73344d93782SGreg Clayton                 m_done = true;
73444d93782SGreg Clayton             }
73544d93782SGreg Clayton         }
73644d93782SGreg Clayton     }
73744d93782SGreg Clayton }
73844d93782SGreg Clayton 
73944d93782SGreg Clayton void
74044d93782SGreg Clayton IOHandlerEditline::Hide ()
74144d93782SGreg Clayton {
742cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
743b89b7496SGreg Clayton     if (m_editline_ap)
74444d93782SGreg Clayton         m_editline_ap->Hide();
745cacde7dfSTodd Fiala #endif
74644d93782SGreg Clayton }
74744d93782SGreg Clayton 
74844d93782SGreg Clayton 
74944d93782SGreg Clayton void
75044d93782SGreg Clayton IOHandlerEditline::Refresh ()
75144d93782SGreg Clayton {
752cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
753b89b7496SGreg Clayton     if (m_editline_ap)
754b89b7496SGreg Clayton     {
75544d93782SGreg Clayton         m_editline_ap->Refresh();
756b89b7496SGreg Clayton     }
75744d93782SGreg Clayton     else
75844d93782SGreg Clayton     {
759cacde7dfSTodd Fiala #endif
76044d93782SGreg Clayton         const char *prompt = GetPrompt();
76144d93782SGreg Clayton         if (prompt && prompt[0])
76244d93782SGreg Clayton         {
76344d93782SGreg Clayton             FILE *out = GetOutputFILE();
76444d93782SGreg Clayton             if (out)
76544d93782SGreg Clayton             {
76644d93782SGreg Clayton                 ::fprintf(out, "%s", prompt);
76744d93782SGreg Clayton                 ::fflush(out);
76844d93782SGreg Clayton             }
76944d93782SGreg Clayton         }
770cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
77144d93782SGreg Clayton     }
772cacde7dfSTodd Fiala #endif
77344d93782SGreg Clayton }
77444d93782SGreg Clayton 
77544d93782SGreg Clayton void
776e68f5d6bSGreg Clayton IOHandlerEditline::Cancel ()
777e68f5d6bSGreg Clayton {
778cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
779e68f5d6bSGreg Clayton     if (m_editline_ap)
780e68f5d6bSGreg Clayton         m_editline_ap->Interrupt ();
781cacde7dfSTodd Fiala #endif
782e68f5d6bSGreg Clayton }
783e68f5d6bSGreg Clayton 
784f0066ad0SGreg Clayton bool
78544d93782SGreg Clayton IOHandlerEditline::Interrupt ()
78644d93782SGreg Clayton {
787f0066ad0SGreg Clayton     // Let the delgate handle it first
788f0066ad0SGreg Clayton     if (m_delegate.IOHandlerInterrupt(*this))
789f0066ad0SGreg Clayton         return true;
790f0066ad0SGreg Clayton 
791cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
79244d93782SGreg Clayton     if (m_editline_ap)
793f0066ad0SGreg Clayton         return m_editline_ap->Interrupt();
794cacde7dfSTodd Fiala #endif
795f0066ad0SGreg Clayton     return false;
79644d93782SGreg Clayton }
79744d93782SGreg Clayton 
79844d93782SGreg Clayton void
79944d93782SGreg Clayton IOHandlerEditline::GotEOF()
80044d93782SGreg Clayton {
801cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
80244d93782SGreg Clayton     if (m_editline_ap)
80344d93782SGreg Clayton         m_editline_ap->Interrupt();
804cacde7dfSTodd Fiala #endif
80544d93782SGreg Clayton }
80644d93782SGreg Clayton 
807914b8d98SDeepak Panickal // we may want curses to be disabled for some builds
808914b8d98SDeepak Panickal // for instance, windows
809914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
810914b8d98SDeepak Panickal 
81144d93782SGreg Clayton #include "lldb/Core/ValueObject.h"
81244d93782SGreg Clayton #include "lldb/Symbol/VariableList.h"
81344d93782SGreg Clayton #include "lldb/Target/Target.h"
81444d93782SGreg Clayton #include "lldb/Target/Process.h"
81544d93782SGreg Clayton #include "lldb/Target/Thread.h"
81644d93782SGreg Clayton #include "lldb/Target/StackFrame.h"
81744d93782SGreg Clayton 
81844d93782SGreg Clayton #define KEY_RETURN   10
81944d93782SGreg Clayton #define KEY_ESCAPE  27
82044d93782SGreg Clayton 
82144d93782SGreg Clayton namespace curses
82244d93782SGreg Clayton {
82344d93782SGreg Clayton     class Menu;
82444d93782SGreg Clayton     class MenuDelegate;
82544d93782SGreg Clayton     class Window;
82644d93782SGreg Clayton     class WindowDelegate;
82744d93782SGreg Clayton     typedef std::shared_ptr<Menu> MenuSP;
82844d93782SGreg Clayton     typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
82944d93782SGreg Clayton     typedef std::shared_ptr<Window> WindowSP;
83044d93782SGreg Clayton     typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
83144d93782SGreg Clayton     typedef std::vector<MenuSP> Menus;
83244d93782SGreg Clayton     typedef std::vector<WindowSP> Windows;
83344d93782SGreg Clayton     typedef std::vector<WindowDelegateSP> WindowDelegates;
83444d93782SGreg Clayton 
83544d93782SGreg Clayton #if 0
83644d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
83744d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
83844d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
83944d93782SGreg Clayton #endif
84044d93782SGreg Clayton     struct Point
84144d93782SGreg Clayton     {
84244d93782SGreg Clayton         int x;
84344d93782SGreg Clayton         int y;
84444d93782SGreg Clayton 
84544d93782SGreg Clayton         Point (int _x = 0, int _y = 0) :
84644d93782SGreg Clayton             x(_x),
84744d93782SGreg Clayton             y(_y)
84844d93782SGreg Clayton         {
84944d93782SGreg Clayton         }
85044d93782SGreg Clayton 
85144d93782SGreg Clayton         void
85244d93782SGreg Clayton         Clear ()
85344d93782SGreg Clayton         {
85444d93782SGreg Clayton             x = 0;
85544d93782SGreg Clayton             y = 0;
85644d93782SGreg Clayton         }
85744d93782SGreg Clayton 
85844d93782SGreg Clayton         Point &
85944d93782SGreg Clayton         operator += (const Point &rhs)
86044d93782SGreg Clayton         {
86144d93782SGreg Clayton             x += rhs.x;
86244d93782SGreg Clayton             y += rhs.y;
86344d93782SGreg Clayton             return *this;
86444d93782SGreg Clayton         }
86544d93782SGreg Clayton 
86644d93782SGreg Clayton         void
86744d93782SGreg Clayton         Dump ()
86844d93782SGreg Clayton         {
86944d93782SGreg Clayton             printf ("(x=%i, y=%i)\n", x, y);
87044d93782SGreg Clayton         }
87144d93782SGreg Clayton 
87244d93782SGreg Clayton     };
87344d93782SGreg Clayton 
87444d93782SGreg Clayton     bool operator == (const Point &lhs, const Point &rhs)
87544d93782SGreg Clayton     {
87644d93782SGreg Clayton         return lhs.x == rhs.x && lhs.y == rhs.y;
87744d93782SGreg Clayton     }
87844d93782SGreg Clayton     bool operator != (const Point &lhs, const Point &rhs)
87944d93782SGreg Clayton     {
88044d93782SGreg Clayton         return lhs.x != rhs.x || lhs.y != rhs.y;
88144d93782SGreg Clayton     }
88244d93782SGreg Clayton 
88344d93782SGreg Clayton     struct Size
88444d93782SGreg Clayton     {
88544d93782SGreg Clayton         int width;
88644d93782SGreg Clayton         int height;
88744d93782SGreg Clayton         Size (int w = 0, int h = 0) :
88844d93782SGreg Clayton             width (w),
88944d93782SGreg Clayton             height (h)
89044d93782SGreg Clayton         {
89144d93782SGreg Clayton         }
89244d93782SGreg Clayton 
89344d93782SGreg Clayton         void
89444d93782SGreg Clayton         Clear ()
89544d93782SGreg Clayton         {
89644d93782SGreg Clayton             width = 0;
89744d93782SGreg Clayton             height = 0;
89844d93782SGreg Clayton         }
89944d93782SGreg Clayton 
90044d93782SGreg Clayton         void
90144d93782SGreg Clayton         Dump ()
90244d93782SGreg Clayton         {
90344d93782SGreg Clayton             printf ("(w=%i, h=%i)\n", width, height);
90444d93782SGreg Clayton         }
90544d93782SGreg Clayton 
90644d93782SGreg Clayton     };
90744d93782SGreg Clayton 
90844d93782SGreg Clayton     bool operator == (const Size &lhs, const Size &rhs)
90944d93782SGreg Clayton     {
91044d93782SGreg Clayton         return lhs.width == rhs.width && lhs.height == rhs.height;
91144d93782SGreg Clayton     }
91244d93782SGreg Clayton     bool operator != (const Size &lhs, const Size &rhs)
91344d93782SGreg Clayton     {
91444d93782SGreg Clayton         return lhs.width != rhs.width || lhs.height != rhs.height;
91544d93782SGreg Clayton     }
91644d93782SGreg Clayton 
91744d93782SGreg Clayton     struct Rect
91844d93782SGreg Clayton     {
91944d93782SGreg Clayton         Point origin;
92044d93782SGreg Clayton         Size size;
92144d93782SGreg Clayton 
92244d93782SGreg Clayton         Rect () :
92344d93782SGreg Clayton             origin(),
92444d93782SGreg Clayton             size()
92544d93782SGreg Clayton         {
92644d93782SGreg Clayton         }
92744d93782SGreg Clayton 
92844d93782SGreg Clayton         Rect (const Point &p, const Size &s) :
92944d93782SGreg Clayton             origin (p),
93044d93782SGreg Clayton             size (s)
93144d93782SGreg Clayton         {
93244d93782SGreg Clayton         }
93344d93782SGreg Clayton 
93444d93782SGreg Clayton         void
93544d93782SGreg Clayton         Clear ()
93644d93782SGreg Clayton         {
93744d93782SGreg Clayton             origin.Clear();
93844d93782SGreg Clayton             size.Clear();
93944d93782SGreg Clayton         }
94044d93782SGreg Clayton 
94144d93782SGreg Clayton         void
94244d93782SGreg Clayton         Dump ()
94344d93782SGreg Clayton         {
94444d93782SGreg Clayton             printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height);
94544d93782SGreg Clayton         }
94644d93782SGreg Clayton 
94744d93782SGreg Clayton         void
94844d93782SGreg Clayton         Inset (int w, int h)
94944d93782SGreg Clayton         {
95044d93782SGreg Clayton             if (size.width > w*2)
95144d93782SGreg Clayton                 size.width -= w*2;
95244d93782SGreg Clayton             origin.x += w;
95344d93782SGreg Clayton 
95444d93782SGreg Clayton             if (size.height > h*2)
95544d93782SGreg Clayton                 size.height -= h*2;
95644d93782SGreg Clayton             origin.y += h;
95744d93782SGreg Clayton         }
95844d93782SGreg Clayton         // Return a status bar rectangle which is the last line of
95944d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
96044d93782SGreg Clayton         // include the status bar area.
96144d93782SGreg Clayton         Rect
96244d93782SGreg Clayton         MakeStatusBar ()
96344d93782SGreg Clayton         {
96444d93782SGreg Clayton             Rect status_bar;
96544d93782SGreg Clayton             if (size.height > 1)
96644d93782SGreg Clayton             {
96744d93782SGreg Clayton                 status_bar.origin.x = origin.x;
96844d93782SGreg Clayton                 status_bar.origin.y = size.height;
96944d93782SGreg Clayton                 status_bar.size.width = size.width;
97044d93782SGreg Clayton                 status_bar.size.height = 1;
97144d93782SGreg Clayton                 --size.height;
97244d93782SGreg Clayton             }
97344d93782SGreg Clayton             return status_bar;
97444d93782SGreg Clayton         }
97544d93782SGreg Clayton 
97644d93782SGreg Clayton         // Return a menubar rectangle which is the first line of
97744d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
97844d93782SGreg Clayton         // include the menubar area.
97944d93782SGreg Clayton         Rect
98044d93782SGreg Clayton         MakeMenuBar ()
98144d93782SGreg Clayton         {
98244d93782SGreg Clayton             Rect menubar;
98344d93782SGreg Clayton             if (size.height > 1)
98444d93782SGreg Clayton             {
98544d93782SGreg Clayton                 menubar.origin.x = origin.x;
98644d93782SGreg Clayton                 menubar.origin.y = origin.y;
98744d93782SGreg Clayton                 menubar.size.width = size.width;
98844d93782SGreg Clayton                 menubar.size.height = 1;
98944d93782SGreg Clayton                 ++origin.y;
99044d93782SGreg Clayton                 --size.height;
99144d93782SGreg Clayton             }
99244d93782SGreg Clayton             return menubar;
99344d93782SGreg Clayton         }
99444d93782SGreg Clayton 
99544d93782SGreg Clayton         void
99644d93782SGreg Clayton         HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const
99744d93782SGreg Clayton         {
99844d93782SGreg Clayton             float top_height = top_percentage * size.height;
99944d93782SGreg Clayton             HorizontalSplit (top_height, top, bottom);
100044d93782SGreg Clayton         }
100144d93782SGreg Clayton 
100244d93782SGreg Clayton         void
100344d93782SGreg Clayton         HorizontalSplit (int top_height, Rect &top, Rect &bottom) const
100444d93782SGreg Clayton         {
100544d93782SGreg Clayton             top = *this;
100644d93782SGreg Clayton             if (top_height < size.height)
100744d93782SGreg Clayton             {
100844d93782SGreg Clayton                 top.size.height = top_height;
100944d93782SGreg Clayton                 bottom.origin.x = origin.x;
101044d93782SGreg Clayton                 bottom.origin.y = origin.y + top.size.height;
101144d93782SGreg Clayton                 bottom.size.width = size.width;
101244d93782SGreg Clayton                 bottom.size.height = size.height - top.size.height;
101344d93782SGreg Clayton             }
101444d93782SGreg Clayton             else
101544d93782SGreg Clayton             {
101644d93782SGreg Clayton                 bottom.Clear();
101744d93782SGreg Clayton             }
101844d93782SGreg Clayton         }
101944d93782SGreg Clayton 
102044d93782SGreg Clayton         void
102144d93782SGreg Clayton         VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const
102244d93782SGreg Clayton         {
102344d93782SGreg Clayton             float left_width = left_percentage * size.width;
102444d93782SGreg Clayton             VerticalSplit (left_width, left, right);
102544d93782SGreg Clayton         }
102644d93782SGreg Clayton 
102744d93782SGreg Clayton 
102844d93782SGreg Clayton         void
102944d93782SGreg Clayton         VerticalSplit (int left_width, Rect &left, Rect &right) const
103044d93782SGreg Clayton         {
103144d93782SGreg Clayton             left = *this;
103244d93782SGreg Clayton             if (left_width < size.width)
103344d93782SGreg Clayton             {
103444d93782SGreg Clayton                 left.size.width = left_width;
103544d93782SGreg Clayton                 right.origin.x = origin.x + left.size.width;
103644d93782SGreg Clayton                 right.origin.y = origin.y;
103744d93782SGreg Clayton                 right.size.width = size.width - left.size.width;
103844d93782SGreg Clayton                 right.size.height = size.height;
103944d93782SGreg Clayton             }
104044d93782SGreg Clayton             else
104144d93782SGreg Clayton             {
104244d93782SGreg Clayton                 right.Clear();
104344d93782SGreg Clayton             }
104444d93782SGreg Clayton         }
104544d93782SGreg Clayton     };
104644d93782SGreg Clayton 
104744d93782SGreg Clayton     bool operator == (const Rect &lhs, const Rect &rhs)
104844d93782SGreg Clayton     {
104944d93782SGreg Clayton         return lhs.origin == rhs.origin && lhs.size == rhs.size;
105044d93782SGreg Clayton     }
105144d93782SGreg Clayton     bool operator != (const Rect &lhs, const Rect &rhs)
105244d93782SGreg Clayton     {
105344d93782SGreg Clayton         return lhs.origin != rhs.origin || lhs.size != rhs.size;
105444d93782SGreg Clayton     }
105544d93782SGreg Clayton 
105644d93782SGreg Clayton     enum HandleCharResult
105744d93782SGreg Clayton     {
105844d93782SGreg Clayton         eKeyNotHandled      = 0,
105944d93782SGreg Clayton         eKeyHandled         = 1,
106044d93782SGreg Clayton         eQuitApplication    = 2
106144d93782SGreg Clayton     };
106244d93782SGreg Clayton 
106344d93782SGreg Clayton     enum class MenuActionResult
106444d93782SGreg Clayton     {
106544d93782SGreg Clayton         Handled,
106644d93782SGreg Clayton         NotHandled,
106744d93782SGreg Clayton         Quit    // Exit all menus and quit
106844d93782SGreg Clayton     };
106944d93782SGreg Clayton 
107044d93782SGreg Clayton     struct KeyHelp
107144d93782SGreg Clayton     {
107244d93782SGreg Clayton         int ch;
107344d93782SGreg Clayton         const char *description;
107444d93782SGreg Clayton     };
107544d93782SGreg Clayton 
107644d93782SGreg Clayton     class WindowDelegate
107744d93782SGreg Clayton     {
107844d93782SGreg Clayton     public:
107944d93782SGreg Clayton         virtual
108044d93782SGreg Clayton         ~WindowDelegate()
108144d93782SGreg Clayton         {
108244d93782SGreg Clayton         }
108344d93782SGreg Clayton 
108444d93782SGreg Clayton         virtual bool
1085bd5ae6b4SGreg Clayton         WindowDelegateUpdateProcess (Window &window)
1086bd5ae6b4SGreg Clayton         {
1087bd5ae6b4SGreg Clayton         }
1088bd5ae6b4SGreg Clayton 
1089bd5ae6b4SGreg Clayton         virtual bool
1090bd5ae6b4SGreg Clayton         WindowDelegateUpdateThread (Window &window)
1091bd5ae6b4SGreg Clayton         {
1092bd5ae6b4SGreg Clayton         }
1093bd5ae6b4SGreg Clayton 
1094bd5ae6b4SGreg Clayton         virtual bool
1095bd5ae6b4SGreg Clayton         WindowDelegateUpdateStackFrame (Window &window)
1096bd5ae6b4SGreg Clayton         {
1097bd5ae6b4SGreg Clayton         }
1098bd5ae6b4SGreg Clayton 
1099bd5ae6b4SGreg Clayton         virtual bool
110044d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force)
110144d93782SGreg Clayton         {
110244d93782SGreg Clayton             return false; // Drawing not handled
110344d93782SGreg Clayton         }
110444d93782SGreg Clayton 
110544d93782SGreg Clayton         virtual HandleCharResult
110644d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key)
110744d93782SGreg Clayton         {
110844d93782SGreg Clayton             return eKeyNotHandled;
110944d93782SGreg Clayton         }
111044d93782SGreg Clayton 
111144d93782SGreg Clayton         virtual const char *
111244d93782SGreg Clayton         WindowDelegateGetHelpText ()
111344d93782SGreg Clayton         {
111444d93782SGreg Clayton             return NULL;
111544d93782SGreg Clayton         }
111644d93782SGreg Clayton 
111744d93782SGreg Clayton         virtual KeyHelp *
111844d93782SGreg Clayton         WindowDelegateGetKeyHelp ()
111944d93782SGreg Clayton         {
112044d93782SGreg Clayton             return NULL;
112144d93782SGreg Clayton         }
112244d93782SGreg Clayton     };
112344d93782SGreg Clayton 
112444d93782SGreg Clayton     class HelpDialogDelegate :
112544d93782SGreg Clayton         public WindowDelegate
112644d93782SGreg Clayton     {
112744d93782SGreg Clayton     public:
112844d93782SGreg Clayton         HelpDialogDelegate (const char *text, KeyHelp *key_help_array);
112944d93782SGreg Clayton 
1130bd5ae6b4SGreg Clayton         ~HelpDialogDelegate() override;
113144d93782SGreg Clayton 
1132bd5ae6b4SGreg Clayton         bool
1133bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
113444d93782SGreg Clayton 
1135bd5ae6b4SGreg Clayton         HandleCharResult
1136bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
113744d93782SGreg Clayton 
113844d93782SGreg Clayton         size_t
113944d93782SGreg Clayton         GetNumLines() const
114044d93782SGreg Clayton         {
114144d93782SGreg Clayton             return m_text.GetSize();
114244d93782SGreg Clayton         }
114344d93782SGreg Clayton 
114444d93782SGreg Clayton         size_t
114544d93782SGreg Clayton         GetMaxLineLength () const
114644d93782SGreg Clayton         {
114744d93782SGreg Clayton             return m_text.GetMaxStringLength();
114844d93782SGreg Clayton         }
114944d93782SGreg Clayton 
115044d93782SGreg Clayton     protected:
115144d93782SGreg Clayton         StringList m_text;
115244d93782SGreg Clayton         int m_first_visible_line;
115344d93782SGreg Clayton     };
115444d93782SGreg Clayton 
115544d93782SGreg Clayton 
115644d93782SGreg Clayton     class Window
115744d93782SGreg Clayton     {
115844d93782SGreg Clayton     public:
115944d93782SGreg Clayton 
116044d93782SGreg Clayton         Window (const char *name) :
116144d93782SGreg Clayton             m_name (name),
116244d93782SGreg Clayton             m_window (NULL),
116344d93782SGreg Clayton             m_panel (NULL),
116444d93782SGreg Clayton             m_parent (NULL),
116544d93782SGreg Clayton             m_subwindows (),
116644d93782SGreg Clayton             m_delegate_sp (),
116744d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
116844d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
116944d93782SGreg Clayton             m_delete (false),
117044d93782SGreg Clayton             m_needs_update (true),
117144d93782SGreg Clayton             m_can_activate (true),
117244d93782SGreg Clayton             m_is_subwin (false)
117344d93782SGreg Clayton         {
117444d93782SGreg Clayton         }
117544d93782SGreg Clayton 
117644d93782SGreg Clayton         Window (const char *name, WINDOW *w, bool del = true) :
117744d93782SGreg Clayton             m_name (name),
117844d93782SGreg Clayton             m_window (NULL),
117944d93782SGreg Clayton             m_panel (NULL),
118044d93782SGreg Clayton             m_parent (NULL),
118144d93782SGreg Clayton             m_subwindows (),
118244d93782SGreg Clayton             m_delegate_sp (),
118344d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
118444d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
118544d93782SGreg Clayton             m_delete (del),
118644d93782SGreg Clayton             m_needs_update (true),
118744d93782SGreg Clayton             m_can_activate (true),
118844d93782SGreg Clayton             m_is_subwin (false)
118944d93782SGreg Clayton         {
119044d93782SGreg Clayton             if (w)
119144d93782SGreg Clayton                 Reset(w);
119244d93782SGreg Clayton         }
119344d93782SGreg Clayton 
119444d93782SGreg Clayton         Window (const char *name, const Rect &bounds) :
119544d93782SGreg Clayton             m_name (name),
119644d93782SGreg Clayton             m_window (NULL),
119744d93782SGreg Clayton             m_parent (NULL),
119844d93782SGreg Clayton             m_subwindows (),
119944d93782SGreg Clayton             m_delegate_sp (),
120044d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
120144d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
120244d93782SGreg Clayton             m_delete (true),
120344d93782SGreg Clayton             m_needs_update (true),
120444d93782SGreg Clayton             m_can_activate (true),
120544d93782SGreg Clayton             m_is_subwin (false)
120644d93782SGreg Clayton         {
120744d93782SGreg Clayton             Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y));
120844d93782SGreg Clayton         }
120944d93782SGreg Clayton 
121044d93782SGreg Clayton         virtual
121144d93782SGreg Clayton         ~Window ()
121244d93782SGreg Clayton         {
121344d93782SGreg Clayton             RemoveSubWindows ();
121444d93782SGreg Clayton             Reset ();
121544d93782SGreg Clayton         }
121644d93782SGreg Clayton 
121744d93782SGreg Clayton         void
121844d93782SGreg Clayton         Reset (WINDOW *w = NULL, bool del = true)
121944d93782SGreg Clayton         {
122044d93782SGreg Clayton             if (m_window == w)
122144d93782SGreg Clayton                 return;
122244d93782SGreg Clayton 
122344d93782SGreg Clayton             if (m_panel)
122444d93782SGreg Clayton             {
122544d93782SGreg Clayton                 ::del_panel (m_panel);
122644d93782SGreg Clayton                 m_panel = NULL;
122744d93782SGreg Clayton             }
122844d93782SGreg Clayton             if (m_window && m_delete)
122944d93782SGreg Clayton             {
123044d93782SGreg Clayton                 ::delwin (m_window);
123144d93782SGreg Clayton                 m_window = NULL;
123244d93782SGreg Clayton                 m_delete = false;
123344d93782SGreg Clayton             }
123444d93782SGreg Clayton             if (w)
123544d93782SGreg Clayton             {
123644d93782SGreg Clayton                 m_window = w;
123744d93782SGreg Clayton                 m_panel = ::new_panel (m_window);
123844d93782SGreg Clayton                 m_delete = del;
123944d93782SGreg Clayton             }
124044d93782SGreg Clayton         }
124144d93782SGreg Clayton 
124244d93782SGreg Clayton         void    AttributeOn (attr_t attr)   { ::wattron (m_window, attr); }
124344d93782SGreg Clayton         void    AttributeOff (attr_t attr)  { ::wattroff (m_window, attr); }
124444d93782SGreg Clayton         void    Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); }
124544d93782SGreg Clayton         void    Clear ()    { ::wclear (m_window); }
124644d93782SGreg Clayton         void    Erase ()    { ::werase (m_window); }
124744d93782SGreg Clayton         Rect    GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window
124844d93782SGreg Clayton         int     GetChar ()  { return ::wgetch (m_window); }
124944d93782SGreg Clayton         int     GetCursorX ()     { return getcurx (m_window); }
125044d93782SGreg Clayton         int     GetCursorY ()     { return getcury (m_window); }
125144d93782SGreg Clayton         Rect    GetFrame ()    { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system
125244d93782SGreg Clayton         Point   GetParentOrigin() { return Point (GetParentX(), GetParentY()); }
125344d93782SGreg Clayton         Size    GetSize()         { return Size (GetWidth(), GetHeight()); }
125444d93782SGreg Clayton         int     GetParentX ()     { return getparx (m_window); }
125544d93782SGreg Clayton         int     GetParentY ()     { return getpary (m_window); }
125644d93782SGreg Clayton         int     GetMaxX()   { return getmaxx (m_window); }
125744d93782SGreg Clayton         int     GetMaxY()   { return getmaxy (m_window); }
125844d93782SGreg Clayton         int     GetWidth()  { return GetMaxX(); }
125944d93782SGreg Clayton         int     GetHeight() { return GetMaxY(); }
126044d93782SGreg Clayton         void    MoveCursor (int x, int y) {  ::wmove (m_window, y, x); }
126144d93782SGreg Clayton         void    MoveWindow (int x, int y) {  MoveWindow(Point(x,y)); }
126244d93782SGreg Clayton         void    Resize (int w, int h) { ::wresize(m_window, h, w); }
126344d93782SGreg Clayton         void    Resize (const Size &size) { ::wresize(m_window, size.height, size.width); }
126444d93782SGreg Clayton         void    PutChar (int ch)    { ::waddch (m_window, ch); }
126544d93782SGreg Clayton         void    PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); }
126644d93782SGreg Clayton         void    Refresh ()  { ::wrefresh (m_window); }
126744d93782SGreg Clayton         void    DeferredRefresh ()
126844d93782SGreg Clayton         {
126944d93782SGreg Clayton             // We are using panels, so we don't need to call this...
127044d93782SGreg Clayton             //::wnoutrefresh(m_window);
127144d93782SGreg Clayton         }
127244d93782SGreg Clayton         void    SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); }
127344d93782SGreg Clayton         void    UnderlineOn ()  { AttributeOn(A_UNDERLINE); }
127444d93782SGreg Clayton         void    UnderlineOff () { AttributeOff(A_UNDERLINE); }
127544d93782SGreg Clayton 
127644d93782SGreg Clayton         void    PutCStringTruncated (const char *s, int right_pad)
127744d93782SGreg Clayton         {
127844d93782SGreg Clayton             int bytes_left = GetWidth() - GetCursorX();
127944d93782SGreg Clayton             if (bytes_left > right_pad)
128044d93782SGreg Clayton             {
128144d93782SGreg Clayton                 bytes_left -= right_pad;
128244d93782SGreg Clayton                 ::waddnstr (m_window, s, bytes_left);
128344d93782SGreg Clayton             }
128444d93782SGreg Clayton         }
128544d93782SGreg Clayton 
128644d93782SGreg Clayton         void
128744d93782SGreg Clayton         MoveWindow (const Point &origin)
128844d93782SGreg Clayton         {
128944d93782SGreg Clayton             const bool moving_window = origin != GetParentOrigin();
129044d93782SGreg Clayton             if (m_is_subwin && moving_window)
129144d93782SGreg Clayton             {
129244d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
129344d93782SGreg Clayton                 Size size = GetSize();
129444d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
129544d93782SGreg Clayton                                  size.height,
129644d93782SGreg Clayton                                  size.width,
129744d93782SGreg Clayton                                  origin.y,
129844d93782SGreg Clayton                                  origin.x), true);
129944d93782SGreg Clayton             }
130044d93782SGreg Clayton             else
130144d93782SGreg Clayton             {
130244d93782SGreg Clayton                 ::mvwin (m_window, origin.y, origin.x);
130344d93782SGreg Clayton             }
130444d93782SGreg Clayton         }
130544d93782SGreg Clayton 
130644d93782SGreg Clayton         void
130744d93782SGreg Clayton         SetBounds (const Rect &bounds)
130844d93782SGreg Clayton         {
130944d93782SGreg Clayton             const bool moving_window = bounds.origin != GetParentOrigin();
131044d93782SGreg Clayton             if (m_is_subwin && moving_window)
131144d93782SGreg Clayton             {
131244d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
131344d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
131444d93782SGreg Clayton                                  bounds.size.height,
131544d93782SGreg Clayton                                  bounds.size.width,
131644d93782SGreg Clayton                                  bounds.origin.y,
131744d93782SGreg Clayton                                  bounds.origin.x), true);
131844d93782SGreg Clayton             }
131944d93782SGreg Clayton             else
132044d93782SGreg Clayton             {
132144d93782SGreg Clayton                 if (moving_window)
132244d93782SGreg Clayton                     MoveWindow(bounds.origin);
132344d93782SGreg Clayton                 Resize (bounds.size);
132444d93782SGreg Clayton             }
132544d93782SGreg Clayton         }
132644d93782SGreg Clayton 
132744d93782SGreg Clayton         void
132844d93782SGreg Clayton         Printf (const char *format, ...)  __attribute__ ((format (printf, 2, 3)))
132944d93782SGreg Clayton         {
133044d93782SGreg Clayton             va_list args;
133144d93782SGreg Clayton             va_start (args, format);
133244d93782SGreg Clayton             vwprintw(m_window, format, args);
133344d93782SGreg Clayton             va_end (args);
133444d93782SGreg Clayton         }
133544d93782SGreg Clayton 
133644d93782SGreg Clayton         void
133744d93782SGreg Clayton         Touch ()
133844d93782SGreg Clayton         {
133944d93782SGreg Clayton             ::touchwin (m_window);
134044d93782SGreg Clayton             if (m_parent)
134144d93782SGreg Clayton                 m_parent->Touch();
134244d93782SGreg Clayton         }
134344d93782SGreg Clayton 
134444d93782SGreg Clayton         WindowSP
134544d93782SGreg Clayton         CreateSubWindow (const char *name, const Rect &bounds, bool make_active)
134644d93782SGreg Clayton         {
134744d93782SGreg Clayton             WindowSP subwindow_sp;
134844d93782SGreg Clayton             if (m_window)
134944d93782SGreg Clayton             {
135044d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::subwin (m_window,
135144d93782SGreg Clayton                                                               bounds.size.height,
135244d93782SGreg Clayton                                                               bounds.size.width,
135344d93782SGreg Clayton                                                               bounds.origin.y,
135444d93782SGreg Clayton                                                               bounds.origin.x), true));
135544d93782SGreg Clayton                 subwindow_sp->m_is_subwin = true;
135644d93782SGreg Clayton             }
135744d93782SGreg Clayton             else
135844d93782SGreg Clayton             {
135944d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height,
136044d93782SGreg Clayton                                                               bounds.size.width,
136144d93782SGreg Clayton                                                               bounds.origin.y,
136244d93782SGreg Clayton                                                               bounds.origin.x), true));
136344d93782SGreg Clayton                 subwindow_sp->m_is_subwin = false;
136444d93782SGreg Clayton             }
136544d93782SGreg Clayton             subwindow_sp->m_parent = this;
136644d93782SGreg Clayton             if (make_active)
136744d93782SGreg Clayton             {
136844d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
136944d93782SGreg Clayton                 m_curr_active_window_idx = m_subwindows.size();
137044d93782SGreg Clayton             }
137144d93782SGreg Clayton             m_subwindows.push_back(subwindow_sp);
137244d93782SGreg Clayton             ::top_panel (subwindow_sp->m_panel);
137344d93782SGreg Clayton             m_needs_update = true;
137444d93782SGreg Clayton             return subwindow_sp;
137544d93782SGreg Clayton         }
137644d93782SGreg Clayton 
137744d93782SGreg Clayton         bool
137844d93782SGreg Clayton         RemoveSubWindow (Window *window)
137944d93782SGreg Clayton         {
138044d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
138144d93782SGreg Clayton             size_t i = 0;
138244d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
138344d93782SGreg Clayton             {
138444d93782SGreg Clayton                 if ((*pos).get() == window)
138544d93782SGreg Clayton                 {
138644d93782SGreg Clayton                     if (m_prev_active_window_idx == i)
138744d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
138844d93782SGreg Clayton                     else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i)
138944d93782SGreg Clayton                         --m_prev_active_window_idx;
139044d93782SGreg Clayton 
139144d93782SGreg Clayton                     if (m_curr_active_window_idx == i)
139244d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
139344d93782SGreg Clayton                     else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i)
139444d93782SGreg Clayton                         --m_curr_active_window_idx;
139544d93782SGreg Clayton                     window->Erase();
139644d93782SGreg Clayton                     m_subwindows.erase(pos);
139744d93782SGreg Clayton                     m_needs_update = true;
139844d93782SGreg Clayton                     if (m_parent)
139944d93782SGreg Clayton                         m_parent->Touch();
140044d93782SGreg Clayton                     else
140144d93782SGreg Clayton                         ::touchwin (stdscr);
140244d93782SGreg Clayton                     return true;
140344d93782SGreg Clayton                 }
140444d93782SGreg Clayton             }
140544d93782SGreg Clayton             return false;
140644d93782SGreg Clayton         }
140744d93782SGreg Clayton 
140844d93782SGreg Clayton         WindowSP
140944d93782SGreg Clayton         FindSubWindow (const char *name)
141044d93782SGreg Clayton         {
141144d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
141244d93782SGreg Clayton             size_t i = 0;
141344d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
141444d93782SGreg Clayton             {
141544d93782SGreg Clayton                 if ((*pos)->m_name.compare(name) == 0)
141644d93782SGreg Clayton                     return *pos;
141744d93782SGreg Clayton             }
141844d93782SGreg Clayton             return WindowSP();
141944d93782SGreg Clayton         }
142044d93782SGreg Clayton 
142144d93782SGreg Clayton         void
142244d93782SGreg Clayton         RemoveSubWindows ()
142344d93782SGreg Clayton         {
142444d93782SGreg Clayton             m_curr_active_window_idx = UINT32_MAX;
142544d93782SGreg Clayton             m_prev_active_window_idx = UINT32_MAX;
142644d93782SGreg Clayton             for (Windows::iterator pos = m_subwindows.begin();
142744d93782SGreg Clayton                  pos != m_subwindows.end();
142844d93782SGreg Clayton                  pos = m_subwindows.erase(pos))
142944d93782SGreg Clayton             {
143044d93782SGreg Clayton                 (*pos)->Erase();
143144d93782SGreg Clayton             }
143244d93782SGreg Clayton             if (m_parent)
143344d93782SGreg Clayton                 m_parent->Touch();
143444d93782SGreg Clayton             else
143544d93782SGreg Clayton                 ::touchwin (stdscr);
143644d93782SGreg Clayton         }
143744d93782SGreg Clayton 
143844d93782SGreg Clayton         WINDOW *
143944d93782SGreg Clayton         get()
144044d93782SGreg Clayton         {
144144d93782SGreg Clayton             return m_window;
144244d93782SGreg Clayton         }
144344d93782SGreg Clayton 
144444d93782SGreg Clayton         operator WINDOW *()
144544d93782SGreg Clayton         {
144644d93782SGreg Clayton             return m_window;
144744d93782SGreg Clayton         }
144844d93782SGreg Clayton 
144944d93782SGreg Clayton         //----------------------------------------------------------------------
145044d93782SGreg Clayton         // Window drawing utilities
145144d93782SGreg Clayton         //----------------------------------------------------------------------
145244d93782SGreg Clayton         void
145344d93782SGreg Clayton         DrawTitleBox (const char *title, const char *bottom_message = NULL)
145444d93782SGreg Clayton         {
145544d93782SGreg Clayton             attr_t attr = 0;
145644d93782SGreg Clayton             if (IsActive())
145744d93782SGreg Clayton                 attr = A_BOLD | COLOR_PAIR(2);
145844d93782SGreg Clayton             else
145944d93782SGreg Clayton                 attr = 0;
146044d93782SGreg Clayton             if (attr)
146144d93782SGreg Clayton                 AttributeOn(attr);
146244d93782SGreg Clayton 
146344d93782SGreg Clayton             Box();
146444d93782SGreg Clayton             MoveCursor(3, 0);
146544d93782SGreg Clayton 
146644d93782SGreg Clayton             if (title && title[0])
146744d93782SGreg Clayton             {
146844d93782SGreg Clayton                 PutChar ('<');
146944d93782SGreg Clayton                 PutCString (title);
147044d93782SGreg Clayton                 PutChar ('>');
147144d93782SGreg Clayton             }
147244d93782SGreg Clayton 
147344d93782SGreg Clayton             if (bottom_message && bottom_message[0])
147444d93782SGreg Clayton             {
147544d93782SGreg Clayton                 int bottom_message_length = strlen(bottom_message);
147644d93782SGreg Clayton                 int x = GetWidth() - 3 - (bottom_message_length + 2);
147744d93782SGreg Clayton 
147844d93782SGreg Clayton                 if (x > 0)
147944d93782SGreg Clayton                 {
148044d93782SGreg Clayton                     MoveCursor (x, GetHeight() - 1);
148144d93782SGreg Clayton                     PutChar ('[');
148244d93782SGreg Clayton                     PutCString(bottom_message);
148344d93782SGreg Clayton                     PutChar (']');
148444d93782SGreg Clayton                 }
148544d93782SGreg Clayton                 else
148644d93782SGreg Clayton                 {
148744d93782SGreg Clayton                     MoveCursor (1, GetHeight() - 1);
148844d93782SGreg Clayton                     PutChar ('[');
148944d93782SGreg Clayton                     PutCStringTruncated (bottom_message, 1);
149044d93782SGreg Clayton                 }
149144d93782SGreg Clayton             }
149244d93782SGreg Clayton             if (attr)
149344d93782SGreg Clayton                 AttributeOff(attr);
149444d93782SGreg Clayton 
149544d93782SGreg Clayton         }
149644d93782SGreg Clayton 
149744d93782SGreg Clayton         virtual void
149844d93782SGreg Clayton         Draw (bool force)
149944d93782SGreg Clayton         {
150044d93782SGreg Clayton             if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force))
150144d93782SGreg Clayton                 return;
150244d93782SGreg Clayton 
150344d93782SGreg Clayton             for (auto &subwindow_sp : m_subwindows)
150444d93782SGreg Clayton                 subwindow_sp->Draw(force);
150544d93782SGreg Clayton         }
150644d93782SGreg Clayton 
150744d93782SGreg Clayton         bool
150844d93782SGreg Clayton         CreateHelpSubwindow ()
150944d93782SGreg Clayton         {
151044d93782SGreg Clayton             if (m_delegate_sp)
151144d93782SGreg Clayton             {
151244d93782SGreg Clayton                 const char *text = m_delegate_sp->WindowDelegateGetHelpText ();
151344d93782SGreg Clayton                 KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp ();
151444d93782SGreg Clayton                 if ((text && text[0]) || key_help)
151544d93782SGreg Clayton                 {
151644d93782SGreg Clayton                     std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help));
151744d93782SGreg Clayton                     const size_t num_lines = help_delegate_ap->GetNumLines();
151844d93782SGreg Clayton                     const size_t max_length = help_delegate_ap->GetMaxLineLength();
151944d93782SGreg Clayton                     Rect bounds = GetBounds();
152044d93782SGreg Clayton                     bounds.Inset(1, 1);
15213985c8c6SSaleem Abdulrasool                     if (max_length + 4 < static_cast<size_t>(bounds.size.width))
152244d93782SGreg Clayton                     {
152344d93782SGreg Clayton                         bounds.origin.x += (bounds.size.width - max_length + 4)/2;
152444d93782SGreg Clayton                         bounds.size.width = max_length + 4;
152544d93782SGreg Clayton                     }
152644d93782SGreg Clayton                     else
152744d93782SGreg Clayton                     {
152844d93782SGreg Clayton                         if (bounds.size.width > 100)
152944d93782SGreg Clayton                         {
153044d93782SGreg Clayton                             const int inset_w = bounds.size.width / 4;
153144d93782SGreg Clayton                             bounds.origin.x += inset_w;
153244d93782SGreg Clayton                             bounds.size.width -= 2*inset_w;
153344d93782SGreg Clayton                         }
153444d93782SGreg Clayton                     }
153544d93782SGreg Clayton 
15363985c8c6SSaleem Abdulrasool                     if (num_lines + 2 < static_cast<size_t>(bounds.size.height))
153744d93782SGreg Clayton                     {
153844d93782SGreg Clayton                         bounds.origin.y += (bounds.size.height - num_lines + 2)/2;
153944d93782SGreg Clayton                         bounds.size.height = num_lines + 2;
154044d93782SGreg Clayton                     }
154144d93782SGreg Clayton                     else
154244d93782SGreg Clayton                     {
154344d93782SGreg Clayton                         if (bounds.size.height > 100)
154444d93782SGreg Clayton                         {
154544d93782SGreg Clayton                             const int inset_h = bounds.size.height / 4;
154644d93782SGreg Clayton                             bounds.origin.y += inset_h;
154744d93782SGreg Clayton                             bounds.size.height -= 2*inset_h;
154844d93782SGreg Clayton                         }
154944d93782SGreg Clayton                     }
15505fdb09bbSGreg Clayton                     WindowSP help_window_sp;
15515fdb09bbSGreg Clayton                     Window *parent_window = GetParent();
15525fdb09bbSGreg Clayton                     if (parent_window)
15535fdb09bbSGreg Clayton                         help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
15545fdb09bbSGreg Clayton                     else
15555fdb09bbSGreg Clayton                         help_window_sp = CreateSubWindow("Help", bounds, true);
155644d93782SGreg Clayton                     help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release()));
155744d93782SGreg Clayton                     return true;
155844d93782SGreg Clayton                 }
155944d93782SGreg Clayton             }
156044d93782SGreg Clayton             return false;
156144d93782SGreg Clayton         }
156244d93782SGreg Clayton 
156344d93782SGreg Clayton         virtual HandleCharResult
156444d93782SGreg Clayton         HandleChar (int key)
156544d93782SGreg Clayton         {
156644d93782SGreg Clayton             // Always check the active window first
156744d93782SGreg Clayton             HandleCharResult result = eKeyNotHandled;
156844d93782SGreg Clayton             WindowSP active_window_sp = GetActiveWindow ();
156944d93782SGreg Clayton             if (active_window_sp)
157044d93782SGreg Clayton             {
157144d93782SGreg Clayton                 result = active_window_sp->HandleChar (key);
157244d93782SGreg Clayton                 if (result != eKeyNotHandled)
157344d93782SGreg Clayton                     return result;
157444d93782SGreg Clayton             }
157544d93782SGreg Clayton 
157644d93782SGreg Clayton             if (m_delegate_sp)
157744d93782SGreg Clayton             {
157844d93782SGreg Clayton                 result = m_delegate_sp->WindowDelegateHandleChar (*this, key);
157944d93782SGreg Clayton                 if (result != eKeyNotHandled)
158044d93782SGreg Clayton                     return result;
158144d93782SGreg Clayton             }
158244d93782SGreg Clayton 
158344d93782SGreg Clayton             // Then check for any windows that want any keys
158444d93782SGreg Clayton             // that weren't handled. This is typically only
158544d93782SGreg Clayton             // for a menubar.
158644d93782SGreg Clayton             // Make a copy of the subwindows in case any HandleChar()
158744d93782SGreg Clayton             // functions muck with the subwindows. If we don't do this,
158844d93782SGreg Clayton             // we can crash when iterating over the subwindows.
158944d93782SGreg Clayton             Windows subwindows (m_subwindows);
159044d93782SGreg Clayton             for (auto subwindow_sp : subwindows)
159144d93782SGreg Clayton             {
159244d93782SGreg Clayton                 if (subwindow_sp->m_can_activate == false)
159344d93782SGreg Clayton                 {
159444d93782SGreg Clayton                     HandleCharResult result = subwindow_sp->HandleChar(key);
159544d93782SGreg Clayton                     if (result != eKeyNotHandled)
159644d93782SGreg Clayton                         return result;
159744d93782SGreg Clayton                 }
159844d93782SGreg Clayton             }
159944d93782SGreg Clayton 
160044d93782SGreg Clayton             return eKeyNotHandled;
160144d93782SGreg Clayton         }
160244d93782SGreg Clayton 
160344d93782SGreg Clayton         bool
160444d93782SGreg Clayton         SetActiveWindow (Window *window)
160544d93782SGreg Clayton         {
160644d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
160744d93782SGreg Clayton             for (size_t i=0; i<num_subwindows; ++i)
160844d93782SGreg Clayton             {
160944d93782SGreg Clayton                 if (m_subwindows[i].get() == window)
161044d93782SGreg Clayton                 {
161144d93782SGreg Clayton                     m_prev_active_window_idx = m_curr_active_window_idx;
161244d93782SGreg Clayton                     ::top_panel (window->m_panel);
161344d93782SGreg Clayton                     m_curr_active_window_idx = i;
161444d93782SGreg Clayton                     return true;
161544d93782SGreg Clayton                 }
161644d93782SGreg Clayton             }
161744d93782SGreg Clayton             return false;
161844d93782SGreg Clayton         }
161944d93782SGreg Clayton 
162044d93782SGreg Clayton         WindowSP
162144d93782SGreg Clayton         GetActiveWindow ()
162244d93782SGreg Clayton         {
162344d93782SGreg Clayton             if (!m_subwindows.empty())
162444d93782SGreg Clayton             {
162544d93782SGreg Clayton                 if (m_curr_active_window_idx >= m_subwindows.size())
162644d93782SGreg Clayton                 {
162744d93782SGreg Clayton                     if (m_prev_active_window_idx < m_subwindows.size())
162844d93782SGreg Clayton                     {
162944d93782SGreg Clayton                         m_curr_active_window_idx = m_prev_active_window_idx;
163044d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
163144d93782SGreg Clayton                     }
163244d93782SGreg Clayton                     else if (IsActive())
163344d93782SGreg Clayton                     {
163444d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
163544d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
163644d93782SGreg Clayton 
163744d93782SGreg Clayton                         // Find first window that wants to be active if this window is active
163844d93782SGreg Clayton                         const size_t num_subwindows = m_subwindows.size();
163944d93782SGreg Clayton                         for (size_t i=0; i<num_subwindows; ++i)
164044d93782SGreg Clayton                         {
164144d93782SGreg Clayton                             if (m_subwindows[i]->GetCanBeActive())
164244d93782SGreg Clayton                             {
164344d93782SGreg Clayton                                 m_curr_active_window_idx = i;
164444d93782SGreg Clayton                                 break;
164544d93782SGreg Clayton                             }
164644d93782SGreg Clayton                         }
164744d93782SGreg Clayton                     }
164844d93782SGreg Clayton                 }
164944d93782SGreg Clayton 
165044d93782SGreg Clayton                 if (m_curr_active_window_idx < m_subwindows.size())
165144d93782SGreg Clayton                     return m_subwindows[m_curr_active_window_idx];
165244d93782SGreg Clayton             }
165344d93782SGreg Clayton             return WindowSP();
165444d93782SGreg Clayton         }
165544d93782SGreg Clayton 
165644d93782SGreg Clayton         bool
165744d93782SGreg Clayton         GetCanBeActive () const
165844d93782SGreg Clayton         {
165944d93782SGreg Clayton             return m_can_activate;
166044d93782SGreg Clayton         }
166144d93782SGreg Clayton 
166244d93782SGreg Clayton         void
166344d93782SGreg Clayton         SetCanBeActive (bool b)
166444d93782SGreg Clayton         {
166544d93782SGreg Clayton             m_can_activate = b;
166644d93782SGreg Clayton         }
166744d93782SGreg Clayton 
166844d93782SGreg Clayton         const WindowDelegateSP &
166944d93782SGreg Clayton         GetDelegate () const
167044d93782SGreg Clayton         {
167144d93782SGreg Clayton             return m_delegate_sp;
167244d93782SGreg Clayton         }
167344d93782SGreg Clayton 
167444d93782SGreg Clayton         void
167544d93782SGreg Clayton         SetDelegate (const WindowDelegateSP &delegate_sp)
167644d93782SGreg Clayton         {
167744d93782SGreg Clayton             m_delegate_sp = delegate_sp;
167844d93782SGreg Clayton         }
167944d93782SGreg Clayton 
168044d93782SGreg Clayton         Window *
168144d93782SGreg Clayton         GetParent () const
168244d93782SGreg Clayton         {
168344d93782SGreg Clayton             return m_parent;
168444d93782SGreg Clayton         }
168544d93782SGreg Clayton 
168644d93782SGreg Clayton         bool
168744d93782SGreg Clayton         IsActive () const
168844d93782SGreg Clayton         {
168944d93782SGreg Clayton             if (m_parent)
169044d93782SGreg Clayton                 return m_parent->GetActiveWindow().get() == this;
169144d93782SGreg Clayton             else
169244d93782SGreg Clayton                 return true; // Top level window is always active
169344d93782SGreg Clayton         }
169444d93782SGreg Clayton 
169544d93782SGreg Clayton         void
169644d93782SGreg Clayton         SelectNextWindowAsActive ()
169744d93782SGreg Clayton         {
169844d93782SGreg Clayton             // Move active focus to next window
169944d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
170044d93782SGreg Clayton             if (m_curr_active_window_idx == UINT32_MAX)
170144d93782SGreg Clayton             {
170244d93782SGreg Clayton                 uint32_t idx = 0;
170344d93782SGreg Clayton                 for (auto subwindow_sp : m_subwindows)
170444d93782SGreg Clayton                 {
170544d93782SGreg Clayton                     if (subwindow_sp->GetCanBeActive())
170644d93782SGreg Clayton                     {
170744d93782SGreg Clayton                         m_curr_active_window_idx = idx;
170844d93782SGreg Clayton                         break;
170944d93782SGreg Clayton                     }
171044d93782SGreg Clayton                     ++idx;
171144d93782SGreg Clayton                 }
171244d93782SGreg Clayton             }
171344d93782SGreg Clayton             else if (m_curr_active_window_idx + 1 < num_subwindows)
171444d93782SGreg Clayton             {
171544d93782SGreg Clayton                 bool handled = false;
171644d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
171744d93782SGreg Clayton                 for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx)
171844d93782SGreg Clayton                 {
171944d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
172044d93782SGreg Clayton                     {
172144d93782SGreg Clayton                         m_curr_active_window_idx = idx;
172244d93782SGreg Clayton                         handled = true;
172344d93782SGreg Clayton                         break;
172444d93782SGreg Clayton                     }
172544d93782SGreg Clayton                 }
172644d93782SGreg Clayton                 if (!handled)
172744d93782SGreg Clayton                 {
172844d93782SGreg Clayton                     for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx)
172944d93782SGreg Clayton                     {
173044d93782SGreg Clayton                         if (m_subwindows[idx]->GetCanBeActive())
173144d93782SGreg Clayton                         {
173244d93782SGreg Clayton                             m_curr_active_window_idx = idx;
173344d93782SGreg Clayton                             break;
173444d93782SGreg Clayton                         }
173544d93782SGreg Clayton                     }
173644d93782SGreg Clayton                 }
173744d93782SGreg Clayton             }
173844d93782SGreg Clayton             else
173944d93782SGreg Clayton             {
174044d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
174144d93782SGreg Clayton                 for (size_t idx=0; idx<num_subwindows; ++idx)
174244d93782SGreg Clayton                 {
174344d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
174444d93782SGreg Clayton                     {
174544d93782SGreg Clayton                         m_curr_active_window_idx = idx;
174644d93782SGreg Clayton                         break;
174744d93782SGreg Clayton                     }
174844d93782SGreg Clayton                 }
174944d93782SGreg Clayton             }
175044d93782SGreg Clayton         }
175144d93782SGreg Clayton 
175244d93782SGreg Clayton         const char *
175344d93782SGreg Clayton         GetName () const
175444d93782SGreg Clayton         {
175544d93782SGreg Clayton             return m_name.c_str();
175644d93782SGreg Clayton         }
175744d93782SGreg Clayton     protected:
175844d93782SGreg Clayton         std::string m_name;
175944d93782SGreg Clayton         WINDOW *m_window;
176044d93782SGreg Clayton         PANEL *m_panel;
176144d93782SGreg Clayton         Window *m_parent;
176244d93782SGreg Clayton         Windows m_subwindows;
176344d93782SGreg Clayton         WindowDelegateSP m_delegate_sp;
176444d93782SGreg Clayton         uint32_t m_curr_active_window_idx;
176544d93782SGreg Clayton         uint32_t m_prev_active_window_idx;
176644d93782SGreg Clayton         bool m_delete;
176744d93782SGreg Clayton         bool m_needs_update;
176844d93782SGreg Clayton         bool m_can_activate;
176944d93782SGreg Clayton         bool m_is_subwin;
177044d93782SGreg Clayton 
177144d93782SGreg Clayton     private:
177244d93782SGreg Clayton         DISALLOW_COPY_AND_ASSIGN(Window);
177344d93782SGreg Clayton     };
177444d93782SGreg Clayton 
177544d93782SGreg Clayton     class MenuDelegate
177644d93782SGreg Clayton     {
177744d93782SGreg Clayton     public:
177844d93782SGreg Clayton         virtual ~MenuDelegate() {}
177944d93782SGreg Clayton 
178044d93782SGreg Clayton         virtual MenuActionResult
178144d93782SGreg Clayton         MenuDelegateAction (Menu &menu) = 0;
178244d93782SGreg Clayton     };
178344d93782SGreg Clayton 
178444d93782SGreg Clayton     class Menu : public WindowDelegate
178544d93782SGreg Clayton     {
178644d93782SGreg Clayton     public:
178744d93782SGreg Clayton         enum class Type
178844d93782SGreg Clayton         {
178944d93782SGreg Clayton             Invalid,
179044d93782SGreg Clayton             Bar,
179144d93782SGreg Clayton             Item,
179244d93782SGreg Clayton             Separator
179344d93782SGreg Clayton         };
179444d93782SGreg Clayton 
179544d93782SGreg Clayton         // Menubar or separator constructor
179644d93782SGreg Clayton         Menu (Type type);
179744d93782SGreg Clayton 
179844d93782SGreg Clayton         // Menuitem constructor
179944d93782SGreg Clayton         Menu (const char *name,
180044d93782SGreg Clayton               const char *key_name,
180144d93782SGreg Clayton               int key_value,
180244d93782SGreg Clayton               uint64_t identifier);
180344d93782SGreg Clayton 
1804bd5ae6b4SGreg Clayton         ~Menu () override
180544d93782SGreg Clayton         {
180644d93782SGreg Clayton         }
180744d93782SGreg Clayton 
180844d93782SGreg Clayton         const MenuDelegateSP &
180944d93782SGreg Clayton         GetDelegate () const
181044d93782SGreg Clayton         {
181144d93782SGreg Clayton             return m_delegate_sp;
181244d93782SGreg Clayton         }
181344d93782SGreg Clayton 
181444d93782SGreg Clayton         void
181544d93782SGreg Clayton         SetDelegate (const MenuDelegateSP &delegate_sp)
181644d93782SGreg Clayton         {
181744d93782SGreg Clayton             m_delegate_sp = delegate_sp;
181844d93782SGreg Clayton         }
181944d93782SGreg Clayton 
182044d93782SGreg Clayton         void
182144d93782SGreg Clayton         RecalculateNameLengths();
182244d93782SGreg Clayton 
182344d93782SGreg Clayton         void
182444d93782SGreg Clayton         AddSubmenu (const MenuSP &menu_sp);
182544d93782SGreg Clayton 
182644d93782SGreg Clayton         int
182744d93782SGreg Clayton         DrawAndRunMenu (Window &window);
182844d93782SGreg Clayton 
182944d93782SGreg Clayton         void
183044d93782SGreg Clayton         DrawMenuTitle (Window &window, bool highlight);
183144d93782SGreg Clayton 
1832bd5ae6b4SGreg Clayton         bool
1833bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
183444d93782SGreg Clayton 
1835bd5ae6b4SGreg Clayton         HandleCharResult
1836bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
183744d93782SGreg Clayton 
183844d93782SGreg Clayton         MenuActionResult
183944d93782SGreg Clayton         ActionPrivate (Menu &menu)
184044d93782SGreg Clayton         {
184144d93782SGreg Clayton             MenuActionResult result = MenuActionResult::NotHandled;
184244d93782SGreg Clayton             if (m_delegate_sp)
184344d93782SGreg Clayton             {
184444d93782SGreg Clayton                 result = m_delegate_sp->MenuDelegateAction (menu);
184544d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
184644d93782SGreg Clayton                     return result;
184744d93782SGreg Clayton             }
184844d93782SGreg Clayton             else if (m_parent)
184944d93782SGreg Clayton             {
185044d93782SGreg Clayton                 result = m_parent->ActionPrivate(menu);
185144d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
185244d93782SGreg Clayton                     return result;
185344d93782SGreg Clayton             }
185444d93782SGreg Clayton             return m_canned_result;
185544d93782SGreg Clayton         }
185644d93782SGreg Clayton 
185744d93782SGreg Clayton         MenuActionResult
185844d93782SGreg Clayton         Action ()
185944d93782SGreg Clayton         {
186044d93782SGreg Clayton             // Call the recursive action so it can try to handle it
186144d93782SGreg Clayton             // with the menu delegate, and if not, try our parent menu
186244d93782SGreg Clayton             return ActionPrivate (*this);
186344d93782SGreg Clayton         }
186444d93782SGreg Clayton 
186544d93782SGreg Clayton         void
186644d93782SGreg Clayton         SetCannedResult (MenuActionResult result)
186744d93782SGreg Clayton         {
186844d93782SGreg Clayton             m_canned_result = result;
186944d93782SGreg Clayton         }
187044d93782SGreg Clayton 
187144d93782SGreg Clayton         Menus &
187244d93782SGreg Clayton         GetSubmenus()
187344d93782SGreg Clayton         {
187444d93782SGreg Clayton             return m_submenus;
187544d93782SGreg Clayton         }
187644d93782SGreg Clayton 
187744d93782SGreg Clayton         const Menus &
187844d93782SGreg Clayton         GetSubmenus() const
187944d93782SGreg Clayton         {
188044d93782SGreg Clayton             return m_submenus;
188144d93782SGreg Clayton         }
188244d93782SGreg Clayton 
188344d93782SGreg Clayton         int
188444d93782SGreg Clayton         GetSelectedSubmenuIndex () const
188544d93782SGreg Clayton         {
188644d93782SGreg Clayton             return m_selected;
188744d93782SGreg Clayton         }
188844d93782SGreg Clayton 
188944d93782SGreg Clayton         void
189044d93782SGreg Clayton         SetSelectedSubmenuIndex (int idx)
189144d93782SGreg Clayton         {
189244d93782SGreg Clayton             m_selected = idx;
189344d93782SGreg Clayton         }
189444d93782SGreg Clayton 
189544d93782SGreg Clayton         Type
189644d93782SGreg Clayton         GetType () const
189744d93782SGreg Clayton         {
189844d93782SGreg Clayton             return m_type;
189944d93782SGreg Clayton         }
190044d93782SGreg Clayton 
190144d93782SGreg Clayton         int
190244d93782SGreg Clayton         GetStartingColumn() const
190344d93782SGreg Clayton         {
190444d93782SGreg Clayton             return m_start_col;
190544d93782SGreg Clayton         }
190644d93782SGreg Clayton 
190744d93782SGreg Clayton         void
190844d93782SGreg Clayton         SetStartingColumn(int col)
190944d93782SGreg Clayton         {
191044d93782SGreg Clayton             m_start_col = col;
191144d93782SGreg Clayton         }
191244d93782SGreg Clayton 
191344d93782SGreg Clayton         int
191444d93782SGreg Clayton         GetKeyValue() const
191544d93782SGreg Clayton         {
191644d93782SGreg Clayton             return m_key_value;
191744d93782SGreg Clayton         }
191844d93782SGreg Clayton 
191944d93782SGreg Clayton         void
192044d93782SGreg Clayton         SetKeyValue(int key_value)
192144d93782SGreg Clayton         {
192244d93782SGreg Clayton             m_key_value = key_value;
192344d93782SGreg Clayton         }
192444d93782SGreg Clayton 
192544d93782SGreg Clayton         std::string &
192644d93782SGreg Clayton         GetName()
192744d93782SGreg Clayton         {
192844d93782SGreg Clayton             return m_name;
192944d93782SGreg Clayton         }
193044d93782SGreg Clayton 
193144d93782SGreg Clayton         std::string &
193244d93782SGreg Clayton         GetKeyName()
193344d93782SGreg Clayton         {
193444d93782SGreg Clayton             return m_key_name;
193544d93782SGreg Clayton         }
193644d93782SGreg Clayton 
193744d93782SGreg Clayton         int
193844d93782SGreg Clayton         GetDrawWidth () const
193944d93782SGreg Clayton         {
194044d93782SGreg Clayton             return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
194144d93782SGreg Clayton         }
194244d93782SGreg Clayton 
194344d93782SGreg Clayton 
194444d93782SGreg Clayton         uint64_t
194544d93782SGreg Clayton         GetIdentifier() const
194644d93782SGreg Clayton         {
194744d93782SGreg Clayton             return m_identifier;
194844d93782SGreg Clayton         }
194944d93782SGreg Clayton 
195044d93782SGreg Clayton         void
195144d93782SGreg Clayton         SetIdentifier (uint64_t identifier)
195244d93782SGreg Clayton         {
195344d93782SGreg Clayton             m_identifier = identifier;
195444d93782SGreg Clayton         }
195544d93782SGreg Clayton 
195644d93782SGreg Clayton     protected:
195744d93782SGreg Clayton         std::string m_name;
195844d93782SGreg Clayton         std::string m_key_name;
195944d93782SGreg Clayton         uint64_t m_identifier;
196044d93782SGreg Clayton         Type m_type;
196144d93782SGreg Clayton         int m_key_value;
196244d93782SGreg Clayton         int m_start_col;
196344d93782SGreg Clayton         int m_max_submenu_name_length;
196444d93782SGreg Clayton         int m_max_submenu_key_name_length;
196544d93782SGreg Clayton         int m_selected;
196644d93782SGreg Clayton         Menu *m_parent;
196744d93782SGreg Clayton         Menus m_submenus;
196844d93782SGreg Clayton         WindowSP m_menu_window_sp;
196944d93782SGreg Clayton         MenuActionResult m_canned_result;
197044d93782SGreg Clayton         MenuDelegateSP m_delegate_sp;
197144d93782SGreg Clayton     };
197244d93782SGreg Clayton 
197344d93782SGreg Clayton     // Menubar or separator constructor
197444d93782SGreg Clayton     Menu::Menu (Type type) :
197544d93782SGreg Clayton         m_name (),
197644d93782SGreg Clayton         m_key_name (),
197744d93782SGreg Clayton         m_identifier (0),
197844d93782SGreg Clayton         m_type (type),
197944d93782SGreg Clayton         m_key_value (0),
198044d93782SGreg Clayton         m_start_col (0),
198144d93782SGreg Clayton         m_max_submenu_name_length (0),
198244d93782SGreg Clayton         m_max_submenu_key_name_length (0),
198344d93782SGreg Clayton         m_selected (0),
198444d93782SGreg Clayton         m_parent (NULL),
198544d93782SGreg Clayton         m_submenus (),
198644d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
198744d93782SGreg Clayton         m_delegate_sp()
198844d93782SGreg Clayton     {
198944d93782SGreg Clayton     }
199044d93782SGreg Clayton 
199144d93782SGreg Clayton     // Menuitem constructor
199244d93782SGreg Clayton     Menu::Menu (const char *name,
199344d93782SGreg Clayton                 const char *key_name,
199444d93782SGreg Clayton                 int key_value,
199544d93782SGreg Clayton                 uint64_t identifier) :
199644d93782SGreg Clayton         m_name (),
199744d93782SGreg Clayton         m_key_name (),
199844d93782SGreg Clayton         m_identifier (identifier),
199944d93782SGreg Clayton         m_type (Type::Invalid),
200044d93782SGreg Clayton         m_key_value (key_value),
200144d93782SGreg Clayton         m_start_col (0),
200244d93782SGreg Clayton         m_max_submenu_name_length (0),
200344d93782SGreg Clayton         m_max_submenu_key_name_length (0),
200444d93782SGreg Clayton         m_selected (0),
200544d93782SGreg Clayton         m_parent (NULL),
200644d93782SGreg Clayton         m_submenus (),
200744d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
200844d93782SGreg Clayton         m_delegate_sp()
200944d93782SGreg Clayton     {
201044d93782SGreg Clayton         if (name && name[0])
201144d93782SGreg Clayton         {
201244d93782SGreg Clayton             m_name = name;
201344d93782SGreg Clayton             m_type = Type::Item;
201444d93782SGreg Clayton             if (key_name && key_name[0])
201544d93782SGreg Clayton                 m_key_name = key_name;
201644d93782SGreg Clayton         }
201744d93782SGreg Clayton         else
201844d93782SGreg Clayton         {
201944d93782SGreg Clayton             m_type = Type::Separator;
202044d93782SGreg Clayton         }
202144d93782SGreg Clayton     }
202244d93782SGreg Clayton 
202344d93782SGreg Clayton     void
202444d93782SGreg Clayton     Menu::RecalculateNameLengths()
202544d93782SGreg Clayton     {
202644d93782SGreg Clayton         m_max_submenu_name_length = 0;
202744d93782SGreg Clayton         m_max_submenu_key_name_length = 0;
202844d93782SGreg Clayton         Menus &submenus = GetSubmenus();
202944d93782SGreg Clayton         const size_t num_submenus = submenus.size();
203044d93782SGreg Clayton         for (size_t i=0; i<num_submenus; ++i)
203144d93782SGreg Clayton         {
203244d93782SGreg Clayton             Menu *submenu = submenus[i].get();
20333985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
203444d93782SGreg Clayton                 m_max_submenu_name_length = submenu->m_name.size();
20353985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size())
203644d93782SGreg Clayton                 m_max_submenu_key_name_length = submenu->m_key_name.size();
203744d93782SGreg Clayton         }
203844d93782SGreg Clayton     }
203944d93782SGreg Clayton 
204044d93782SGreg Clayton     void
204144d93782SGreg Clayton     Menu::AddSubmenu (const MenuSP &menu_sp)
204244d93782SGreg Clayton     {
204344d93782SGreg Clayton         menu_sp->m_parent = this;
20443985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
204544d93782SGreg Clayton             m_max_submenu_name_length = menu_sp->m_name.size();
20463985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size())
204744d93782SGreg Clayton             m_max_submenu_key_name_length = menu_sp->m_key_name.size();
204844d93782SGreg Clayton         m_submenus.push_back(menu_sp);
204944d93782SGreg Clayton     }
205044d93782SGreg Clayton 
205144d93782SGreg Clayton     void
205244d93782SGreg Clayton     Menu::DrawMenuTitle (Window &window, bool highlight)
205344d93782SGreg Clayton     {
205444d93782SGreg Clayton         if (m_type == Type::Separator)
205544d93782SGreg Clayton         {
205644d93782SGreg Clayton             window.MoveCursor(0, window.GetCursorY());
205744d93782SGreg Clayton             window.PutChar(ACS_LTEE);
205844d93782SGreg Clayton             int width = window.GetWidth();
205944d93782SGreg Clayton             if (width > 2)
206044d93782SGreg Clayton             {
206144d93782SGreg Clayton                 width -= 2;
20623985c8c6SSaleem Abdulrasool                 for (int i=0; i< width; ++i)
206344d93782SGreg Clayton                     window.PutChar(ACS_HLINE);
206444d93782SGreg Clayton             }
206544d93782SGreg Clayton             window.PutChar(ACS_RTEE);
206644d93782SGreg Clayton         }
206744d93782SGreg Clayton         else
206844d93782SGreg Clayton         {
206944d93782SGreg Clayton             const int shortcut_key = m_key_value;
207044d93782SGreg Clayton             bool underlined_shortcut = false;
207144d93782SGreg Clayton             const attr_t hilgight_attr = A_REVERSE;
207244d93782SGreg Clayton             if (highlight)
207344d93782SGreg Clayton                 window.AttributeOn(hilgight_attr);
207444d93782SGreg Clayton             if (isprint(shortcut_key))
207544d93782SGreg Clayton             {
207644d93782SGreg Clayton                 size_t lower_pos = m_name.find(tolower(shortcut_key));
207744d93782SGreg Clayton                 size_t upper_pos = m_name.find(toupper(shortcut_key));
207844d93782SGreg Clayton                 const char *name = m_name.c_str();
207944d93782SGreg Clayton                 size_t pos = std::min<size_t>(lower_pos, upper_pos);
208044d93782SGreg Clayton                 if (pos != std::string::npos)
208144d93782SGreg Clayton                 {
208244d93782SGreg Clayton                     underlined_shortcut = true;
208344d93782SGreg Clayton                     if (pos > 0)
208444d93782SGreg Clayton                     {
208544d93782SGreg Clayton                         window.PutCString(name, pos);
208644d93782SGreg Clayton                         name += pos;
208744d93782SGreg Clayton                     }
208844d93782SGreg Clayton                     const attr_t shortcut_attr = A_UNDERLINE|A_BOLD;
208944d93782SGreg Clayton                     window.AttributeOn (shortcut_attr);
209044d93782SGreg Clayton                     window.PutChar(name[0]);
209144d93782SGreg Clayton                     window.AttributeOff(shortcut_attr);
209244d93782SGreg Clayton                     name++;
209344d93782SGreg Clayton                     if (name[0])
209444d93782SGreg Clayton                         window.PutCString(name);
209544d93782SGreg Clayton                 }
209644d93782SGreg Clayton             }
209744d93782SGreg Clayton 
209844d93782SGreg Clayton             if (!underlined_shortcut)
209944d93782SGreg Clayton             {
210044d93782SGreg Clayton                 window.PutCString(m_name.c_str());
210144d93782SGreg Clayton             }
210244d93782SGreg Clayton 
210344d93782SGreg Clayton             if (highlight)
210444d93782SGreg Clayton                 window.AttributeOff(hilgight_attr);
210544d93782SGreg Clayton 
210644d93782SGreg Clayton             if (m_key_name.empty())
210744d93782SGreg Clayton             {
210844d93782SGreg Clayton                 if (!underlined_shortcut && isprint(m_key_value))
210944d93782SGreg Clayton                 {
211044d93782SGreg Clayton                     window.AttributeOn (COLOR_PAIR(3));
211144d93782SGreg Clayton                     window.Printf (" (%c)", m_key_value);
211244d93782SGreg Clayton                     window.AttributeOff (COLOR_PAIR(3));
211344d93782SGreg Clayton                 }
211444d93782SGreg Clayton             }
211544d93782SGreg Clayton             else
211644d93782SGreg Clayton             {
211744d93782SGreg Clayton                 window.AttributeOn (COLOR_PAIR(3));
211844d93782SGreg Clayton                 window.Printf (" (%s)", m_key_name.c_str());
211944d93782SGreg Clayton                 window.AttributeOff (COLOR_PAIR(3));
212044d93782SGreg Clayton             }
212144d93782SGreg Clayton         }
212244d93782SGreg Clayton     }
212344d93782SGreg Clayton 
212444d93782SGreg Clayton     bool
212544d93782SGreg Clayton     Menu::WindowDelegateDraw (Window &window, bool force)
212644d93782SGreg Clayton     {
212744d93782SGreg Clayton         Menus &submenus = GetSubmenus();
212844d93782SGreg Clayton         const size_t num_submenus = submenus.size();
212944d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
213044d93782SGreg Clayton         Menu::Type menu_type = GetType ();
213144d93782SGreg Clayton         switch (menu_type)
213244d93782SGreg Clayton         {
213344d93782SGreg Clayton         case  Menu::Type::Bar:
213444d93782SGreg Clayton             {
213544d93782SGreg Clayton                 window.SetBackground(2);
213644d93782SGreg Clayton                 window.MoveCursor(0, 0);
213744d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
213844d93782SGreg Clayton                 {
213944d93782SGreg Clayton                     Menu *menu = submenus[i].get();
214044d93782SGreg Clayton                     if (i > 0)
214144d93782SGreg Clayton                         window.PutChar(' ');
214244d93782SGreg Clayton                     menu->SetStartingColumn (window.GetCursorX());
214344d93782SGreg Clayton                     window.PutCString("| ");
214444d93782SGreg Clayton                     menu->DrawMenuTitle (window, false);
214544d93782SGreg Clayton                 }
214644d93782SGreg Clayton                 window.PutCString(" |");
214744d93782SGreg Clayton                 window.DeferredRefresh();
214844d93782SGreg Clayton             }
214944d93782SGreg Clayton             break;
215044d93782SGreg Clayton 
215144d93782SGreg Clayton         case Menu::Type::Item:
215244d93782SGreg Clayton             {
215344d93782SGreg Clayton                 int y = 1;
215444d93782SGreg Clayton                 int x = 3;
215544d93782SGreg Clayton                 // Draw the menu
215644d93782SGreg Clayton                 int cursor_x = 0;
215744d93782SGreg Clayton                 int cursor_y = 0;
215844d93782SGreg Clayton                 window.Erase();
215944d93782SGreg Clayton                 window.SetBackground(2);
216044d93782SGreg Clayton                 window.Box();
216144d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
216244d93782SGreg Clayton                 {
21633985c8c6SSaleem Abdulrasool                     const bool is_selected =
21643985c8c6SSaleem Abdulrasool                       (i == static_cast<size_t>(selected_idx));
216544d93782SGreg Clayton                     window.MoveCursor(x, y + i);
216644d93782SGreg Clayton                     if (is_selected)
216744d93782SGreg Clayton                     {
216844d93782SGreg Clayton                         // Remember where we want the cursor to be
216944d93782SGreg Clayton                         cursor_x = x-1;
217044d93782SGreg Clayton                         cursor_y = y+i;
217144d93782SGreg Clayton                     }
217244d93782SGreg Clayton                     submenus[i]->DrawMenuTitle (window, is_selected);
217344d93782SGreg Clayton                 }
217444d93782SGreg Clayton                 window.MoveCursor(cursor_x, cursor_y);
217544d93782SGreg Clayton                 window.DeferredRefresh();
217644d93782SGreg Clayton             }
217744d93782SGreg Clayton             break;
217844d93782SGreg Clayton 
217944d93782SGreg Clayton         default:
218044d93782SGreg Clayton         case Menu::Type::Separator:
218144d93782SGreg Clayton             break;
218244d93782SGreg Clayton         }
218344d93782SGreg Clayton         return true; // Drawing handled...
218444d93782SGreg Clayton     }
218544d93782SGreg Clayton 
218644d93782SGreg Clayton     HandleCharResult
218744d93782SGreg Clayton     Menu::WindowDelegateHandleChar (Window &window, int key)
218844d93782SGreg Clayton     {
218944d93782SGreg Clayton         HandleCharResult result = eKeyNotHandled;
219044d93782SGreg Clayton 
219144d93782SGreg Clayton         Menus &submenus = GetSubmenus();
219244d93782SGreg Clayton         const size_t num_submenus = submenus.size();
219344d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
219444d93782SGreg Clayton         Menu::Type menu_type = GetType ();
219544d93782SGreg Clayton         if (menu_type == Menu::Type::Bar)
219644d93782SGreg Clayton         {
219744d93782SGreg Clayton             MenuSP run_menu_sp;
219844d93782SGreg Clayton             switch (key)
219944d93782SGreg Clayton             {
220044d93782SGreg Clayton                 case KEY_DOWN:
220144d93782SGreg Clayton                 case KEY_UP:
220244d93782SGreg Clayton                     // Show last menu or first menu
22033985c8c6SSaleem Abdulrasool                     if (selected_idx < static_cast<int>(num_submenus))
220444d93782SGreg Clayton                         run_menu_sp = submenus[selected_idx];
220544d93782SGreg Clayton                     else if (!submenus.empty())
220644d93782SGreg Clayton                         run_menu_sp = submenus.front();
220744d93782SGreg Clayton                     result = eKeyHandled;
220844d93782SGreg Clayton                     break;
220944d93782SGreg Clayton 
221044d93782SGreg Clayton                 case KEY_RIGHT:
221144d93782SGreg Clayton                 {
221244d93782SGreg Clayton                     ++m_selected;
22133985c8c6SSaleem Abdulrasool                     if (m_selected >= static_cast<int>(num_submenus))
221444d93782SGreg Clayton                         m_selected = 0;
22153985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
221644d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
221744d93782SGreg Clayton                     else if (!submenus.empty())
221844d93782SGreg Clayton                         run_menu_sp = submenus.front();
221944d93782SGreg Clayton                     result = eKeyHandled;
222044d93782SGreg Clayton                 }
222144d93782SGreg Clayton                     break;
222244d93782SGreg Clayton 
222344d93782SGreg Clayton                 case KEY_LEFT:
222444d93782SGreg Clayton                 {
222544d93782SGreg Clayton                     --m_selected;
222644d93782SGreg Clayton                     if (m_selected < 0)
222744d93782SGreg Clayton                         m_selected = num_submenus - 1;
22283985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
222944d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
223044d93782SGreg Clayton                     else if (!submenus.empty())
223144d93782SGreg Clayton                         run_menu_sp = submenus.front();
223244d93782SGreg Clayton                     result = eKeyHandled;
223344d93782SGreg Clayton                 }
223444d93782SGreg Clayton                     break;
223544d93782SGreg Clayton 
223644d93782SGreg Clayton                 default:
223744d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
223844d93782SGreg Clayton                     {
223944d93782SGreg Clayton                         if (submenus[i]->GetKeyValue() == key)
224044d93782SGreg Clayton                         {
224144d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
224244d93782SGreg Clayton                             run_menu_sp = submenus[i];
224344d93782SGreg Clayton                             result = eKeyHandled;
224444d93782SGreg Clayton                             break;
224544d93782SGreg Clayton                         }
224644d93782SGreg Clayton                     }
224744d93782SGreg Clayton                     break;
224844d93782SGreg Clayton             }
224944d93782SGreg Clayton 
225044d93782SGreg Clayton             if (run_menu_sp)
225144d93782SGreg Clayton             {
225244d93782SGreg Clayton                 // Run the action on this menu in case we need to populate the
225344d93782SGreg Clayton                 // menu with dynamic content and also in case check marks, and
225444d93782SGreg Clayton                 // any other menu decorations need to be caclulated
225544d93782SGreg Clayton                 if (run_menu_sp->Action() == MenuActionResult::Quit)
225644d93782SGreg Clayton                     return eQuitApplication;
225744d93782SGreg Clayton 
225844d93782SGreg Clayton                 Rect menu_bounds;
225944d93782SGreg Clayton                 menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
226044d93782SGreg Clayton                 menu_bounds.origin.y = 1;
226144d93782SGreg Clayton                 menu_bounds.size.width = run_menu_sp->GetDrawWidth();
226244d93782SGreg Clayton                 menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
226344d93782SGreg Clayton                 if (m_menu_window_sp)
226444d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
226544d93782SGreg Clayton 
226644d93782SGreg Clayton                 m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(),
226744d93782SGreg Clayton                                                                         menu_bounds,
226844d93782SGreg Clayton                                                                         true);
226944d93782SGreg Clayton                 m_menu_window_sp->SetDelegate (run_menu_sp);
227044d93782SGreg Clayton             }
227144d93782SGreg Clayton         }
227244d93782SGreg Clayton         else if (menu_type == Menu::Type::Item)
227344d93782SGreg Clayton         {
227444d93782SGreg Clayton             switch (key)
227544d93782SGreg Clayton             {
227644d93782SGreg Clayton                 case KEY_DOWN:
227744d93782SGreg Clayton                     if (m_submenus.size() > 1)
227844d93782SGreg Clayton                     {
227944d93782SGreg Clayton                         const int start_select = m_selected;
228044d93782SGreg Clayton                         while (++m_selected != start_select)
228144d93782SGreg Clayton                         {
22823985c8c6SSaleem Abdulrasool                             if (static_cast<size_t>(m_selected) >= num_submenus)
228344d93782SGreg Clayton                                 m_selected = 0;
228444d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
228544d93782SGreg Clayton                                 continue;
228644d93782SGreg Clayton                             else
228744d93782SGreg Clayton                                 break;
228844d93782SGreg Clayton                         }
228944d93782SGreg Clayton                         return eKeyHandled;
229044d93782SGreg Clayton                     }
229144d93782SGreg Clayton                     break;
229244d93782SGreg Clayton 
229344d93782SGreg Clayton                 case KEY_UP:
229444d93782SGreg Clayton                     if (m_submenus.size() > 1)
229544d93782SGreg Clayton                     {
229644d93782SGreg Clayton                         const int start_select = m_selected;
229744d93782SGreg Clayton                         while (--m_selected != start_select)
229844d93782SGreg Clayton                         {
22993985c8c6SSaleem Abdulrasool                             if (m_selected < static_cast<int>(0))
230044d93782SGreg Clayton                                 m_selected = num_submenus - 1;
230144d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
230244d93782SGreg Clayton                                 continue;
230344d93782SGreg Clayton                             else
230444d93782SGreg Clayton                                 break;
230544d93782SGreg Clayton                         }
230644d93782SGreg Clayton                         return eKeyHandled;
230744d93782SGreg Clayton                     }
230844d93782SGreg Clayton                     break;
230944d93782SGreg Clayton 
231044d93782SGreg Clayton                 case KEY_RETURN:
23113985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(selected_idx) < num_submenus)
231244d93782SGreg Clayton                     {
231344d93782SGreg Clayton                         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
231444d93782SGreg Clayton                             return eQuitApplication;
231544d93782SGreg Clayton                         window.GetParent()->RemoveSubWindow(&window);
231644d93782SGreg Clayton                         return eKeyHandled;
231744d93782SGreg Clayton                     }
231844d93782SGreg Clayton                     break;
231944d93782SGreg Clayton 
232044d93782SGreg Clayton                 case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences
232144d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(&window);
232244d93782SGreg Clayton                     return eKeyHandled;
232344d93782SGreg Clayton 
232444d93782SGreg Clayton                 default:
232544d93782SGreg Clayton                 {
232644d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
232744d93782SGreg Clayton                     {
232844d93782SGreg Clayton                         Menu *menu = submenus[i].get();
232944d93782SGreg Clayton                         if (menu->GetKeyValue() == key)
233044d93782SGreg Clayton                         {
233144d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
233244d93782SGreg Clayton                             window.GetParent()->RemoveSubWindow(&window);
233344d93782SGreg Clayton                             if (menu->Action() == MenuActionResult::Quit)
233444d93782SGreg Clayton                                 return eQuitApplication;
233544d93782SGreg Clayton                             return eKeyHandled;
233644d93782SGreg Clayton                         }
233744d93782SGreg Clayton                     }
233844d93782SGreg Clayton                 }
233944d93782SGreg Clayton                     break;
234044d93782SGreg Clayton 
234144d93782SGreg Clayton             }
234244d93782SGreg Clayton         }
234344d93782SGreg Clayton         else if (menu_type == Menu::Type::Separator)
234444d93782SGreg Clayton         {
234544d93782SGreg Clayton 
234644d93782SGreg Clayton         }
234744d93782SGreg Clayton         return result;
234844d93782SGreg Clayton     }
234944d93782SGreg Clayton 
235044d93782SGreg Clayton 
235144d93782SGreg Clayton     class Application
235244d93782SGreg Clayton     {
235344d93782SGreg Clayton     public:
235444d93782SGreg Clayton         Application (FILE *in, FILE *out) :
235544d93782SGreg Clayton             m_window_sp(),
235644d93782SGreg Clayton             m_screen (NULL),
235744d93782SGreg Clayton             m_in (in),
235844d93782SGreg Clayton             m_out (out)
235944d93782SGreg Clayton         {
236044d93782SGreg Clayton 
236144d93782SGreg Clayton         }
236244d93782SGreg Clayton 
236344d93782SGreg Clayton         ~Application ()
236444d93782SGreg Clayton         {
236544d93782SGreg Clayton             m_window_delegates.clear();
236644d93782SGreg Clayton             m_window_sp.reset();
236744d93782SGreg Clayton             if (m_screen)
236844d93782SGreg Clayton             {
236944d93782SGreg Clayton                 ::delscreen(m_screen);
237044d93782SGreg Clayton                 m_screen = NULL;
237144d93782SGreg Clayton             }
237244d93782SGreg Clayton         }
237344d93782SGreg Clayton 
237444d93782SGreg Clayton         void
237544d93782SGreg Clayton         Initialize ()
237644d93782SGreg Clayton         {
237744d93782SGreg Clayton             ::setlocale(LC_ALL, "");
237844d93782SGreg Clayton             ::setlocale(LC_CTYPE, "");
237944d93782SGreg Clayton #if 0
238044d93782SGreg Clayton             ::initscr();
238144d93782SGreg Clayton #else
238244d93782SGreg Clayton             m_screen = ::newterm(NULL, m_out, m_in);
238344d93782SGreg Clayton #endif
238444d93782SGreg Clayton             ::start_color();
238544d93782SGreg Clayton             ::curs_set(0);
238644d93782SGreg Clayton             ::noecho();
238744d93782SGreg Clayton             ::keypad(stdscr,TRUE);
238844d93782SGreg Clayton         }
238944d93782SGreg Clayton 
239044d93782SGreg Clayton         void
239144d93782SGreg Clayton         Terminate ()
239244d93782SGreg Clayton         {
239344d93782SGreg Clayton             ::endwin();
239444d93782SGreg Clayton         }
239544d93782SGreg Clayton 
239644d93782SGreg Clayton         void
239744d93782SGreg Clayton         Run (Debugger &debugger)
239844d93782SGreg Clayton         {
239944d93782SGreg Clayton             bool done = false;
240044d93782SGreg Clayton             int delay_in_tenths_of_a_second = 1;
240144d93782SGreg Clayton 
240244d93782SGreg Clayton             // Alas the threading model in curses is a bit lame so we need to
240344d93782SGreg Clayton             // resort to polling every 0.5 seconds. We could poll for stdin
240444d93782SGreg Clayton             // ourselves and then pass the keys down but then we need to
240544d93782SGreg Clayton             // translate all of the escape sequences ourselves. So we resort to
240644d93782SGreg Clayton             // polling for input because we need to receive async process events
240744d93782SGreg Clayton             // while in this loop.
240844d93782SGreg Clayton 
240944d93782SGreg Clayton             halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar()
241044d93782SGreg Clayton 
241144d93782SGreg Clayton             ListenerSP listener_sp (new Listener ("lldb.IOHandler.curses.Application"));
241244d93782SGreg Clayton             ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
241344d93782SGreg Clayton             ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
241444d93782SGreg Clayton             ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
241544d93782SGreg Clayton             debugger.EnableForwardEvents (listener_sp);
241644d93782SGreg Clayton 
241744d93782SGreg Clayton             bool update = true;
241844d93782SGreg Clayton #if defined(__APPLE__)
241944d93782SGreg Clayton             std::deque<int> escape_chars;
242044d93782SGreg Clayton #endif
242144d93782SGreg Clayton 
242244d93782SGreg Clayton             while (!done)
242344d93782SGreg Clayton             {
242444d93782SGreg Clayton                 if (update)
242544d93782SGreg Clayton                 {
242644d93782SGreg Clayton                     m_window_sp->Draw(false);
242744d93782SGreg Clayton                     // All windows should be calling Window::DeferredRefresh() instead
242844d93782SGreg Clayton                     // of Window::Refresh() so we can do a single update and avoid
242944d93782SGreg Clayton                     // any screen blinking
243044d93782SGreg Clayton                     update_panels();
243144d93782SGreg Clayton 
243244d93782SGreg Clayton                     // Cursor hiding isn't working on MacOSX, so hide it in the top left corner
243344d93782SGreg Clayton                     m_window_sp->MoveCursor(0, 0);
243444d93782SGreg Clayton 
243544d93782SGreg Clayton                     doupdate();
243644d93782SGreg Clayton                     update = false;
243744d93782SGreg Clayton                 }
243844d93782SGreg Clayton 
243944d93782SGreg Clayton #if defined(__APPLE__)
244044d93782SGreg Clayton                 // Terminal.app doesn't map its function keys correctly, F1-F4 default to:
244144d93782SGreg Clayton                 // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible
244244d93782SGreg Clayton                 int ch;
244344d93782SGreg Clayton                 if (escape_chars.empty())
244444d93782SGreg Clayton                     ch = m_window_sp->GetChar();
244544d93782SGreg Clayton                 else
244644d93782SGreg Clayton                 {
244744d93782SGreg Clayton                     ch = escape_chars.front();
244844d93782SGreg Clayton                     escape_chars.pop_front();
244944d93782SGreg Clayton                 }
245044d93782SGreg Clayton                 if (ch == KEY_ESCAPE)
245144d93782SGreg Clayton                 {
245244d93782SGreg Clayton                     int ch2 = m_window_sp->GetChar();
245344d93782SGreg Clayton                     if (ch2 == 'O')
245444d93782SGreg Clayton                     {
245544d93782SGreg Clayton                         int ch3 = m_window_sp->GetChar();
245644d93782SGreg Clayton                         switch (ch3)
245744d93782SGreg Clayton                         {
245844d93782SGreg Clayton                             case 'P': ch = KEY_F(1); break;
245944d93782SGreg Clayton                             case 'Q': ch = KEY_F(2); break;
246044d93782SGreg Clayton                             case 'R': ch = KEY_F(3); break;
246144d93782SGreg Clayton                             case 'S': ch = KEY_F(4); break;
246244d93782SGreg Clayton                             default:
246344d93782SGreg Clayton                                 escape_chars.push_back(ch2);
246444d93782SGreg Clayton                                 if (ch3 != -1)
246544d93782SGreg Clayton                                     escape_chars.push_back(ch3);
246644d93782SGreg Clayton                                 break;
246744d93782SGreg Clayton                         }
246844d93782SGreg Clayton                     }
246944d93782SGreg Clayton                     else if (ch2 != -1)
247044d93782SGreg Clayton                         escape_chars.push_back(ch2);
247144d93782SGreg Clayton                 }
247244d93782SGreg Clayton #else
247344d93782SGreg Clayton                 int ch = m_window_sp->GetChar();
247444d93782SGreg Clayton 
247544d93782SGreg Clayton #endif
247644d93782SGreg Clayton                 if (ch == -1)
247744d93782SGreg Clayton                 {
247844d93782SGreg Clayton                     if (feof(m_in) || ferror(m_in))
247944d93782SGreg Clayton                     {
248044d93782SGreg Clayton                         done = true;
248144d93782SGreg Clayton                     }
248244d93782SGreg Clayton                     else
248344d93782SGreg Clayton                     {
248444d93782SGreg Clayton                         // Just a timeout from using halfdelay(), check for events
248544d93782SGreg Clayton                         EventSP event_sp;
248644d93782SGreg Clayton                         while (listener_sp->PeekAtNextEvent())
248744d93782SGreg Clayton                         {
248844d93782SGreg Clayton                             listener_sp->GetNextEvent(event_sp);
248944d93782SGreg Clayton 
249044d93782SGreg Clayton                             if (event_sp)
249144d93782SGreg Clayton                             {
249244d93782SGreg Clayton                                 Broadcaster *broadcaster = event_sp->GetBroadcaster();
249344d93782SGreg Clayton                                 if (broadcaster)
249444d93782SGreg Clayton                                 {
249544d93782SGreg Clayton                                     //uint32_t event_type = event_sp->GetType();
249644d93782SGreg Clayton                                     ConstString broadcaster_class (broadcaster->GetBroadcasterClass());
249744d93782SGreg Clayton                                     if (broadcaster_class == broadcaster_class_process)
249844d93782SGreg Clayton                                     {
2499ec990867SGreg Clayton                                         debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
250044d93782SGreg Clayton                                         update = true;
250144d93782SGreg Clayton                                         continue; // Don't get any key, just update our view
250244d93782SGreg Clayton                                     }
250344d93782SGreg Clayton                                 }
250444d93782SGreg Clayton                             }
250544d93782SGreg Clayton                         }
250644d93782SGreg Clayton                     }
250744d93782SGreg Clayton                 }
250844d93782SGreg Clayton                 else
250944d93782SGreg Clayton                 {
251044d93782SGreg Clayton                     HandleCharResult key_result = m_window_sp->HandleChar(ch);
251144d93782SGreg Clayton                     switch (key_result)
251244d93782SGreg Clayton                     {
251344d93782SGreg Clayton                         case eKeyHandled:
2514ec990867SGreg Clayton                             debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
251544d93782SGreg Clayton                             update = true;
251644d93782SGreg Clayton                             break;
251744d93782SGreg Clayton                         case eKeyNotHandled:
251844d93782SGreg Clayton                             break;
251944d93782SGreg Clayton                         case eQuitApplication:
252044d93782SGreg Clayton                             done = true;
252144d93782SGreg Clayton                             break;
252244d93782SGreg Clayton                     }
252344d93782SGreg Clayton                 }
252444d93782SGreg Clayton             }
252544d93782SGreg Clayton 
252644d93782SGreg Clayton             debugger.CancelForwardEvents (listener_sp);
252744d93782SGreg Clayton 
252844d93782SGreg Clayton         }
252944d93782SGreg Clayton 
253044d93782SGreg Clayton         WindowSP &
253144d93782SGreg Clayton         GetMainWindow ()
253244d93782SGreg Clayton         {
253344d93782SGreg Clayton             if (!m_window_sp)
253444d93782SGreg Clayton                 m_window_sp.reset (new Window ("main", stdscr, false));
253544d93782SGreg Clayton             return m_window_sp;
253644d93782SGreg Clayton         }
253744d93782SGreg Clayton 
253844d93782SGreg Clayton         WindowDelegates &
253944d93782SGreg Clayton         GetWindowDelegates ()
254044d93782SGreg Clayton         {
254144d93782SGreg Clayton             return m_window_delegates;
254244d93782SGreg Clayton         }
254344d93782SGreg Clayton 
254444d93782SGreg Clayton     protected:
254544d93782SGreg Clayton         WindowSP m_window_sp;
254644d93782SGreg Clayton         WindowDelegates m_window_delegates;
254744d93782SGreg Clayton         SCREEN *m_screen;
254844d93782SGreg Clayton         FILE *m_in;
254944d93782SGreg Clayton         FILE *m_out;
255044d93782SGreg Clayton     };
255144d93782SGreg Clayton 
255244d93782SGreg Clayton 
255344d93782SGreg Clayton } // namespace curses
255444d93782SGreg Clayton 
255544d93782SGreg Clayton 
255644d93782SGreg Clayton using namespace curses;
255744d93782SGreg Clayton 
255844d93782SGreg Clayton struct Row
255944d93782SGreg Clayton {
256044d93782SGreg Clayton     ValueObjectSP valobj;
256144d93782SGreg Clayton     Row *parent;
256244d93782SGreg Clayton     int row_idx;
256344d93782SGreg Clayton     int x;
256444d93782SGreg Clayton     int y;
256544d93782SGreg Clayton     bool might_have_children;
256644d93782SGreg Clayton     bool expanded;
256744d93782SGreg Clayton     bool calculated_children;
256844d93782SGreg Clayton     std::vector<Row> children;
256944d93782SGreg Clayton 
257044d93782SGreg Clayton     Row (const ValueObjectSP &v, Row *p) :
257144d93782SGreg Clayton     valobj (v),
257244d93782SGreg Clayton     parent (p),
257344d93782SGreg Clayton     row_idx(0),
257444d93782SGreg Clayton     x(1),
257544d93782SGreg Clayton     y(1),
257644d93782SGreg Clayton     might_have_children (v ? v->MightHaveChildren() : false),
257744d93782SGreg Clayton     expanded (false),
257844d93782SGreg Clayton     calculated_children (false),
257944d93782SGreg Clayton     children()
258044d93782SGreg Clayton     {
258144d93782SGreg Clayton     }
258244d93782SGreg Clayton 
258344d93782SGreg Clayton     size_t
258444d93782SGreg Clayton     GetDepth () const
258544d93782SGreg Clayton     {
258644d93782SGreg Clayton         if (parent)
258744d93782SGreg Clayton             return 1 + parent->GetDepth();
258844d93782SGreg Clayton         return 0;
258944d93782SGreg Clayton     }
259044d93782SGreg Clayton 
259144d93782SGreg Clayton     void
259244d93782SGreg Clayton     Expand()
259344d93782SGreg Clayton     {
259444d93782SGreg Clayton         expanded = true;
259544d93782SGreg Clayton         if (!calculated_children)
259644d93782SGreg Clayton         {
259744d93782SGreg Clayton             calculated_children = true;
259844d93782SGreg Clayton             if (valobj)
259944d93782SGreg Clayton             {
260044d93782SGreg Clayton                 const size_t num_children = valobj->GetNumChildren();
260144d93782SGreg Clayton                 for (size_t i=0; i<num_children; ++i)
260244d93782SGreg Clayton                 {
260344d93782SGreg Clayton                     children.push_back(Row (valobj->GetChildAtIndex(i, true), this));
260444d93782SGreg Clayton                 }
260544d93782SGreg Clayton             }
260644d93782SGreg Clayton         }
260744d93782SGreg Clayton     }
260844d93782SGreg Clayton 
260944d93782SGreg Clayton     void
261044d93782SGreg Clayton     Unexpand ()
261144d93782SGreg Clayton     {
261244d93782SGreg Clayton         expanded = false;
261344d93782SGreg Clayton     }
261444d93782SGreg Clayton 
261544d93782SGreg Clayton     void
261644d93782SGreg Clayton     DrawTree (Window &window)
261744d93782SGreg Clayton     {
261844d93782SGreg Clayton         if (parent)
261944d93782SGreg Clayton             parent->DrawTreeForChild (window, this, 0);
262044d93782SGreg Clayton 
262144d93782SGreg Clayton         if (might_have_children)
262244d93782SGreg Clayton         {
262344d93782SGreg Clayton             // It we can get UTF8 characters to work we should try to use the "symbol"
262444d93782SGreg Clayton             // UTF8 string below
262544d93782SGreg Clayton //            const char *symbol = "";
262644d93782SGreg Clayton //            if (row.expanded)
262744d93782SGreg Clayton //                symbol = "\xe2\x96\xbd ";
262844d93782SGreg Clayton //            else
262944d93782SGreg Clayton //                symbol = "\xe2\x96\xb7 ";
263044d93782SGreg Clayton //            window.PutCString (symbol);
263144d93782SGreg Clayton 
263244d93782SGreg Clayton             // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
263344d93782SGreg Clayton             // 'v' or '>' character...
263444d93782SGreg Clayton //            if (expanded)
263544d93782SGreg Clayton //                window.PutChar (ACS_DARROW);
263644d93782SGreg Clayton //            else
263744d93782SGreg Clayton //                window.PutChar (ACS_RARROW);
263844d93782SGreg Clayton             // Since we can't find any good looking right arrow/down arrow
263944d93782SGreg Clayton             // symbols, just use a diamond...
264044d93782SGreg Clayton             window.PutChar (ACS_DIAMOND);
264144d93782SGreg Clayton             window.PutChar (ACS_HLINE);
264244d93782SGreg Clayton         }
264344d93782SGreg Clayton     }
264444d93782SGreg Clayton 
264544d93782SGreg Clayton     void
264644d93782SGreg Clayton     DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth)
264744d93782SGreg Clayton     {
264844d93782SGreg Clayton         if (parent)
264944d93782SGreg Clayton             parent->DrawTreeForChild (window, this, reverse_depth + 1);
265044d93782SGreg Clayton 
265144d93782SGreg Clayton         if (&children.back() == child)
265244d93782SGreg Clayton         {
265344d93782SGreg Clayton             // Last child
265444d93782SGreg Clayton             if (reverse_depth == 0)
265544d93782SGreg Clayton             {
265644d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
265744d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
265844d93782SGreg Clayton             }
265944d93782SGreg Clayton             else
266044d93782SGreg Clayton             {
266144d93782SGreg Clayton                 window.PutChar (' ');
266244d93782SGreg Clayton                 window.PutChar (' ');
266344d93782SGreg Clayton             }
266444d93782SGreg Clayton         }
266544d93782SGreg Clayton         else
266644d93782SGreg Clayton         {
266744d93782SGreg Clayton             if (reverse_depth == 0)
266844d93782SGreg Clayton             {
266944d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
267044d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
267144d93782SGreg Clayton             }
267244d93782SGreg Clayton             else
267344d93782SGreg Clayton             {
267444d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
267544d93782SGreg Clayton                 window.PutChar (' ');
267644d93782SGreg Clayton             }
267744d93782SGreg Clayton         }
267844d93782SGreg Clayton     }
267944d93782SGreg Clayton };
268044d93782SGreg Clayton 
268144d93782SGreg Clayton struct DisplayOptions
268244d93782SGreg Clayton {
268344d93782SGreg Clayton     bool show_types;
268444d93782SGreg Clayton };
268544d93782SGreg Clayton 
268644d93782SGreg Clayton class TreeItem;
268744d93782SGreg Clayton 
268844d93782SGreg Clayton class TreeDelegate
268944d93782SGreg Clayton {
269044d93782SGreg Clayton public:
269144d93782SGreg Clayton     TreeDelegate() {}
269244d93782SGreg Clayton     virtual ~TreeDelegate() {}
269344d93782SGreg Clayton     virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0;
269444d93782SGreg Clayton     virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0;
269544d93782SGreg Clayton     virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views
269644d93782SGreg Clayton };
269744d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
269844d93782SGreg Clayton 
269944d93782SGreg Clayton class TreeItem
270044d93782SGreg Clayton {
270144d93782SGreg Clayton public:
270244d93782SGreg Clayton 
270344d93782SGreg Clayton     TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) :
270444d93782SGreg Clayton         m_parent (parent),
270544d93782SGreg Clayton         m_delegate (delegate),
2706ec990867SGreg Clayton         m_user_data (NULL),
270744d93782SGreg Clayton         m_identifier (0),
270844d93782SGreg Clayton         m_row_idx (-1),
270944d93782SGreg Clayton         m_children (),
271044d93782SGreg Clayton         m_might_have_children (might_have_children),
271144d93782SGreg Clayton         m_is_expanded (false)
271244d93782SGreg Clayton     {
271344d93782SGreg Clayton     }
271444d93782SGreg Clayton 
271544d93782SGreg Clayton     TreeItem &
271644d93782SGreg Clayton     operator=(const TreeItem &rhs)
271744d93782SGreg Clayton     {
271844d93782SGreg Clayton         if (this != &rhs)
271944d93782SGreg Clayton         {
272044d93782SGreg Clayton             m_parent = rhs.m_parent;
272144d93782SGreg Clayton             m_delegate = rhs.m_delegate;
2722ec990867SGreg Clayton             m_user_data = rhs.m_user_data;
272344d93782SGreg Clayton             m_identifier = rhs.m_identifier;
272444d93782SGreg Clayton             m_row_idx = rhs.m_row_idx;
272544d93782SGreg Clayton             m_children = rhs.m_children;
272644d93782SGreg Clayton             m_might_have_children = rhs.m_might_have_children;
272744d93782SGreg Clayton             m_is_expanded = rhs.m_is_expanded;
272844d93782SGreg Clayton         }
272944d93782SGreg Clayton         return *this;
273044d93782SGreg Clayton     }
273144d93782SGreg Clayton 
273244d93782SGreg Clayton     size_t
273344d93782SGreg Clayton     GetDepth () const
273444d93782SGreg Clayton     {
273544d93782SGreg Clayton         if (m_parent)
273644d93782SGreg Clayton             return 1 + m_parent->GetDepth();
273744d93782SGreg Clayton         return 0;
273844d93782SGreg Clayton     }
273944d93782SGreg Clayton 
274044d93782SGreg Clayton     int
274144d93782SGreg Clayton     GetRowIndex () const
274244d93782SGreg Clayton     {
274344d93782SGreg Clayton         return m_row_idx;
274444d93782SGreg Clayton     }
274544d93782SGreg Clayton 
274644d93782SGreg Clayton     void
274744d93782SGreg Clayton     ClearChildren ()
274844d93782SGreg Clayton     {
274944d93782SGreg Clayton         m_children.clear();
275044d93782SGreg Clayton     }
275144d93782SGreg Clayton 
275244d93782SGreg Clayton     void
275344d93782SGreg Clayton     Resize (size_t n, const TreeItem &t)
275444d93782SGreg Clayton     {
275544d93782SGreg Clayton         m_children.resize(n, t);
275644d93782SGreg Clayton     }
275744d93782SGreg Clayton 
275844d93782SGreg Clayton     TreeItem &
275944d93782SGreg Clayton     operator [](size_t i)
276044d93782SGreg Clayton     {
276144d93782SGreg Clayton         return m_children[i];
276244d93782SGreg Clayton     }
276344d93782SGreg Clayton 
276444d93782SGreg Clayton     void
276544d93782SGreg Clayton     SetRowIndex (int row_idx)
276644d93782SGreg Clayton     {
276744d93782SGreg Clayton         m_row_idx = row_idx;
276844d93782SGreg Clayton     }
276944d93782SGreg Clayton 
277044d93782SGreg Clayton     size_t
277144d93782SGreg Clayton     GetNumChildren ()
277244d93782SGreg Clayton     {
277344d93782SGreg Clayton         m_delegate.TreeDelegateGenerateChildren (*this);
277444d93782SGreg Clayton         return m_children.size();
277544d93782SGreg Clayton     }
277644d93782SGreg Clayton 
277744d93782SGreg Clayton     void
277844d93782SGreg Clayton     ItemWasSelected ()
277944d93782SGreg Clayton     {
278044d93782SGreg Clayton         m_delegate.TreeDelegateItemSelected(*this);
278144d93782SGreg Clayton     }
278244d93782SGreg Clayton     void
278344d93782SGreg Clayton     CalculateRowIndexes (int &row_idx)
278444d93782SGreg Clayton     {
278544d93782SGreg Clayton         SetRowIndex(row_idx);
278644d93782SGreg Clayton         ++row_idx;
278744d93782SGreg Clayton 
2788ec990867SGreg Clayton         const bool expanded = IsExpanded();
2789ec990867SGreg Clayton 
2790ec990867SGreg Clayton         // The root item must calculate its children,
2791ec990867SGreg Clayton         // or we must calculate the number of children
2792ec990867SGreg Clayton         // if the item is expanded
2793ec990867SGreg Clayton         if (m_parent == NULL || expanded)
279444d93782SGreg Clayton             GetNumChildren();
279544d93782SGreg Clayton 
279644d93782SGreg Clayton         for (auto &item : m_children)
279744d93782SGreg Clayton         {
279844d93782SGreg Clayton             if (expanded)
279944d93782SGreg Clayton                 item.CalculateRowIndexes(row_idx);
280044d93782SGreg Clayton             else
280144d93782SGreg Clayton                 item.SetRowIndex(-1);
280244d93782SGreg Clayton         }
280344d93782SGreg Clayton     }
280444d93782SGreg Clayton 
280544d93782SGreg Clayton     TreeItem *
280644d93782SGreg Clayton     GetParent ()
280744d93782SGreg Clayton     {
280844d93782SGreg Clayton         return m_parent;
280944d93782SGreg Clayton     }
281044d93782SGreg Clayton 
281144d93782SGreg Clayton     bool
281244d93782SGreg Clayton     IsExpanded () const
281344d93782SGreg Clayton     {
281444d93782SGreg Clayton         return m_is_expanded;
281544d93782SGreg Clayton     }
281644d93782SGreg Clayton 
281744d93782SGreg Clayton     void
281844d93782SGreg Clayton     Expand()
281944d93782SGreg Clayton     {
282044d93782SGreg Clayton         m_is_expanded = true;
282144d93782SGreg Clayton     }
282244d93782SGreg Clayton 
282344d93782SGreg Clayton     void
282444d93782SGreg Clayton     Unexpand ()
282544d93782SGreg Clayton     {
282644d93782SGreg Clayton         m_is_expanded = false;
282744d93782SGreg Clayton     }
282844d93782SGreg Clayton 
282944d93782SGreg Clayton     bool
283044d93782SGreg Clayton     Draw (Window &window,
283144d93782SGreg Clayton           const int first_visible_row,
283244d93782SGreg Clayton           const uint32_t selected_row_idx,
283344d93782SGreg Clayton           int &row_idx,
283444d93782SGreg Clayton           int &num_rows_left)
283544d93782SGreg Clayton     {
283644d93782SGreg Clayton         if (num_rows_left <= 0)
283744d93782SGreg Clayton             return false;
283844d93782SGreg Clayton 
283944d93782SGreg Clayton         if (m_row_idx >= first_visible_row)
284044d93782SGreg Clayton         {
284144d93782SGreg Clayton             window.MoveCursor(2, row_idx + 1);
284244d93782SGreg Clayton 
284344d93782SGreg Clayton             if (m_parent)
284444d93782SGreg Clayton                 m_parent->DrawTreeForChild (window, this, 0);
284544d93782SGreg Clayton 
284644d93782SGreg Clayton             if (m_might_have_children)
284744d93782SGreg Clayton             {
284844d93782SGreg Clayton                 // It we can get UTF8 characters to work we should try to use the "symbol"
284944d93782SGreg Clayton                 // UTF8 string below
285044d93782SGreg Clayton                 //            const char *symbol = "";
285144d93782SGreg Clayton                 //            if (row.expanded)
285244d93782SGreg Clayton                 //                symbol = "\xe2\x96\xbd ";
285344d93782SGreg Clayton                 //            else
285444d93782SGreg Clayton                 //                symbol = "\xe2\x96\xb7 ";
285544d93782SGreg Clayton                 //            window.PutCString (symbol);
285644d93782SGreg Clayton 
285744d93782SGreg Clayton                 // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
285844d93782SGreg Clayton                 // 'v' or '>' character...
285944d93782SGreg Clayton                 //            if (expanded)
286044d93782SGreg Clayton                 //                window.PutChar (ACS_DARROW);
286144d93782SGreg Clayton                 //            else
286244d93782SGreg Clayton                 //                window.PutChar (ACS_RARROW);
286344d93782SGreg Clayton                 // Since we can't find any good looking right arrow/down arrow
286444d93782SGreg Clayton                 // symbols, just use a diamond...
286544d93782SGreg Clayton                 window.PutChar (ACS_DIAMOND);
286644d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
286744d93782SGreg Clayton             }
28683985c8c6SSaleem Abdulrasool             bool highlight =
28693985c8c6SSaleem Abdulrasool               (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive();
287044d93782SGreg Clayton 
287144d93782SGreg Clayton             if (highlight)
287244d93782SGreg Clayton                 window.AttributeOn(A_REVERSE);
287344d93782SGreg Clayton 
287444d93782SGreg Clayton             m_delegate.TreeDelegateDrawTreeItem(*this, window);
287544d93782SGreg Clayton 
287644d93782SGreg Clayton             if (highlight)
287744d93782SGreg Clayton                 window.AttributeOff(A_REVERSE);
287844d93782SGreg Clayton             ++row_idx;
287944d93782SGreg Clayton             --num_rows_left;
288044d93782SGreg Clayton         }
288144d93782SGreg Clayton 
288244d93782SGreg Clayton         if (num_rows_left <= 0)
288344d93782SGreg Clayton             return false; // We are done drawing...
288444d93782SGreg Clayton 
288544d93782SGreg Clayton         if (IsExpanded())
288644d93782SGreg Clayton         {
288744d93782SGreg Clayton             for (auto &item : m_children)
288844d93782SGreg Clayton             {
288944d93782SGreg Clayton                 // If we displayed all the rows and item.Draw() returns
289044d93782SGreg Clayton                 // false we are done drawing and can exit this for loop
289144d93782SGreg Clayton                 if (item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left) == false)
289244d93782SGreg Clayton                     break;
289344d93782SGreg Clayton             }
289444d93782SGreg Clayton         }
289544d93782SGreg Clayton         return num_rows_left >= 0; // Return true if not done drawing yet
289644d93782SGreg Clayton     }
289744d93782SGreg Clayton 
289844d93782SGreg Clayton     void
289944d93782SGreg Clayton     DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth)
290044d93782SGreg Clayton     {
290144d93782SGreg Clayton         if (m_parent)
290244d93782SGreg Clayton             m_parent->DrawTreeForChild (window, this, reverse_depth + 1);
290344d93782SGreg Clayton 
290444d93782SGreg Clayton         if (&m_children.back() == child)
290544d93782SGreg Clayton         {
290644d93782SGreg Clayton             // Last child
290744d93782SGreg Clayton             if (reverse_depth == 0)
290844d93782SGreg Clayton             {
290944d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
291044d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
291144d93782SGreg Clayton             }
291244d93782SGreg Clayton             else
291344d93782SGreg Clayton             {
291444d93782SGreg Clayton                 window.PutChar (' ');
291544d93782SGreg Clayton                 window.PutChar (' ');
291644d93782SGreg Clayton             }
291744d93782SGreg Clayton         }
291844d93782SGreg Clayton         else
291944d93782SGreg Clayton         {
292044d93782SGreg Clayton             if (reverse_depth == 0)
292144d93782SGreg Clayton             {
292244d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
292344d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
292444d93782SGreg Clayton             }
292544d93782SGreg Clayton             else
292644d93782SGreg Clayton             {
292744d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
292844d93782SGreg Clayton                 window.PutChar (' ');
292944d93782SGreg Clayton             }
293044d93782SGreg Clayton         }
293144d93782SGreg Clayton     }
293244d93782SGreg Clayton 
293344d93782SGreg Clayton     TreeItem *
293444d93782SGreg Clayton     GetItemForRowIndex (uint32_t row_idx)
293544d93782SGreg Clayton     {
29363985c8c6SSaleem Abdulrasool         if (static_cast<uint32_t>(m_row_idx) == row_idx)
293744d93782SGreg Clayton             return this;
293844d93782SGreg Clayton         if (m_children.empty())
293944d93782SGreg Clayton             return NULL;
294044d93782SGreg Clayton         if (IsExpanded())
294144d93782SGreg Clayton         {
294244d93782SGreg Clayton             for (auto &item : m_children)
294344d93782SGreg Clayton             {
294444d93782SGreg Clayton                 TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
294544d93782SGreg Clayton                 if (selected_item_ptr)
294644d93782SGreg Clayton                     return selected_item_ptr;
294744d93782SGreg Clayton             }
294844d93782SGreg Clayton         }
294944d93782SGreg Clayton         return NULL;
295044d93782SGreg Clayton     }
295144d93782SGreg Clayton 
2952ec990867SGreg Clayton     void *
2953ec990867SGreg Clayton     GetUserData() const
2954ec990867SGreg Clayton     {
2955ec990867SGreg Clayton         return m_user_data;
2956ec990867SGreg Clayton     }
2957ec990867SGreg Clayton 
2958ec990867SGreg Clayton     void
2959ec990867SGreg Clayton     SetUserData (void *user_data)
2960ec990867SGreg Clayton     {
2961ec990867SGreg Clayton         m_user_data = user_data;
2962ec990867SGreg Clayton     }
2963ec990867SGreg Clayton 
296444d93782SGreg Clayton     uint64_t
296544d93782SGreg Clayton     GetIdentifier() const
296644d93782SGreg Clayton     {
296744d93782SGreg Clayton         return m_identifier;
296844d93782SGreg Clayton     }
296944d93782SGreg Clayton 
297044d93782SGreg Clayton     void
297144d93782SGreg Clayton     SetIdentifier (uint64_t identifier)
297244d93782SGreg Clayton     {
297344d93782SGreg Clayton         m_identifier = identifier;
297444d93782SGreg Clayton     }
297544d93782SGreg Clayton 
297644d93782SGreg Clayton 
2977ec990867SGreg Clayton     void
2978ec990867SGreg Clayton     SetMightHaveChildren (bool b)
2979ec990867SGreg Clayton     {
2980ec990867SGreg Clayton         m_might_have_children = b;
2981ec990867SGreg Clayton     }
2982ec990867SGreg Clayton 
298344d93782SGreg Clayton protected:
298444d93782SGreg Clayton     TreeItem *m_parent;
298544d93782SGreg Clayton     TreeDelegate &m_delegate;
2986ec990867SGreg Clayton     void *m_user_data;
298744d93782SGreg Clayton     uint64_t m_identifier;
298844d93782SGreg Clayton     int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item
298944d93782SGreg Clayton     std::vector<TreeItem> m_children;
299044d93782SGreg Clayton     bool m_might_have_children;
299144d93782SGreg Clayton     bool m_is_expanded;
299244d93782SGreg Clayton 
299344d93782SGreg Clayton };
299444d93782SGreg Clayton 
299544d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate
299644d93782SGreg Clayton {
299744d93782SGreg Clayton public:
299844d93782SGreg Clayton     TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) :
299944d93782SGreg Clayton         m_debugger (debugger),
300044d93782SGreg Clayton         m_delegate_sp (delegate_sp),
300144d93782SGreg Clayton         m_root (NULL, *delegate_sp, true),
300244d93782SGreg Clayton         m_selected_item (NULL),
300344d93782SGreg Clayton         m_num_rows (0),
300444d93782SGreg Clayton         m_selected_row_idx (0),
300544d93782SGreg Clayton         m_first_visible_row (0),
300644d93782SGreg Clayton         m_min_x (0),
300744d93782SGreg Clayton         m_min_y (0),
300844d93782SGreg Clayton         m_max_x (0),
300944d93782SGreg Clayton         m_max_y (0)
301044d93782SGreg Clayton     {
301144d93782SGreg Clayton     }
301244d93782SGreg Clayton 
301344d93782SGreg Clayton     int
301444d93782SGreg Clayton     NumVisibleRows () const
301544d93782SGreg Clayton     {
301644d93782SGreg Clayton         return m_max_y - m_min_y;
301744d93782SGreg Clayton     }
301844d93782SGreg Clayton 
3019bd5ae6b4SGreg Clayton     bool
3020bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
302144d93782SGreg Clayton     {
302244d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
302344d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
302444d93782SGreg Clayton 
302544d93782SGreg Clayton         bool display_content = false;
302644d93782SGreg Clayton         if (process)
302744d93782SGreg Clayton         {
302844d93782SGreg Clayton             StateType state = process->GetState();
302944d93782SGreg Clayton             if (StateIsStoppedState(state, true))
303044d93782SGreg Clayton             {
303144d93782SGreg Clayton                 // We are stopped, so it is ok to
303244d93782SGreg Clayton                 display_content = true;
303344d93782SGreg Clayton             }
303444d93782SGreg Clayton             else if (StateIsRunningState(state))
303544d93782SGreg Clayton             {
303644d93782SGreg Clayton                 return true; // Don't do any updating when we are running
303744d93782SGreg Clayton             }
303844d93782SGreg Clayton         }
303944d93782SGreg Clayton 
304044d93782SGreg Clayton         m_min_x = 2;
304144d93782SGreg Clayton         m_min_y = 1;
304244d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
304344d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
304444d93782SGreg Clayton 
304544d93782SGreg Clayton         window.Erase();
304644d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
304744d93782SGreg Clayton 
304844d93782SGreg Clayton         if (display_content)
304944d93782SGreg Clayton         {
305044d93782SGreg Clayton             const int num_visible_rows = NumVisibleRows();
305144d93782SGreg Clayton             m_num_rows = 0;
305244d93782SGreg Clayton             m_root.CalculateRowIndexes(m_num_rows);
305344d93782SGreg Clayton 
305444d93782SGreg Clayton             // If we unexpanded while having something selected our
305544d93782SGreg Clayton             // total number of rows is less than the num visible rows,
305644d93782SGreg Clayton             // then make sure we show all the rows by setting the first
305744d93782SGreg Clayton             // visible row accordingly.
305844d93782SGreg Clayton             if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
305944d93782SGreg Clayton                 m_first_visible_row = 0;
306044d93782SGreg Clayton 
306144d93782SGreg Clayton             // Make sure the selected row is always visible
306244d93782SGreg Clayton             if (m_selected_row_idx < m_first_visible_row)
306344d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx;
306444d93782SGreg Clayton             else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
306544d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
306644d93782SGreg Clayton 
306744d93782SGreg Clayton             int row_idx = 0;
306844d93782SGreg Clayton             int num_rows_left = num_visible_rows;
306944d93782SGreg Clayton             m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left);
307044d93782SGreg Clayton             // Get the selected row
307144d93782SGreg Clayton             m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx);
307244d93782SGreg Clayton         }
307344d93782SGreg Clayton         else
307444d93782SGreg Clayton         {
307544d93782SGreg Clayton             m_selected_item = NULL;
307644d93782SGreg Clayton         }
307744d93782SGreg Clayton 
307844d93782SGreg Clayton         window.DeferredRefresh();
307944d93782SGreg Clayton 
308044d93782SGreg Clayton 
308144d93782SGreg Clayton         return true; // Drawing handled
308244d93782SGreg Clayton     }
308344d93782SGreg Clayton 
308444d93782SGreg Clayton 
3085bd5ae6b4SGreg Clayton     const char *
3086bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
308744d93782SGreg Clayton     {
308844d93782SGreg Clayton         return "Thread window keyboard shortcuts:";
308944d93782SGreg Clayton     }
309044d93782SGreg Clayton 
3091bd5ae6b4SGreg Clayton     KeyHelp *
3092bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
309344d93782SGreg Clayton     {
309444d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
309544d93782SGreg Clayton             { KEY_UP, "Select previous item" },
309644d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
309744d93782SGreg Clayton             { KEY_RIGHT, "Expand the selected item" },
309844d93782SGreg Clayton             { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" },
309944d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
310044d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
310144d93782SGreg Clayton             { 'h', "Show help dialog" },
310244d93782SGreg Clayton             { ' ', "Toggle item expansion" },
310344d93782SGreg Clayton             { ',', "Page up" },
310444d93782SGreg Clayton             { '.', "Page down" },
310544d93782SGreg Clayton             { '\0', NULL }
310644d93782SGreg Clayton         };
310744d93782SGreg Clayton         return g_source_view_key_help;
310844d93782SGreg Clayton     }
310944d93782SGreg Clayton 
3110bd5ae6b4SGreg Clayton     HandleCharResult
3111bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
311244d93782SGreg Clayton     {
311344d93782SGreg Clayton         switch(c)
311444d93782SGreg Clayton         {
311544d93782SGreg Clayton             case ',':
311644d93782SGreg Clayton             case KEY_PPAGE:
311744d93782SGreg Clayton                 // Page up key
311844d93782SGreg Clayton                 if (m_first_visible_row > 0)
311944d93782SGreg Clayton                 {
312044d93782SGreg Clayton                     if (m_first_visible_row > m_max_y)
312144d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
312244d93782SGreg Clayton                     else
312344d93782SGreg Clayton                         m_first_visible_row = 0;
312444d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
312544d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
312644d93782SGreg Clayton                     if (m_selected_item)
312744d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
312844d93782SGreg Clayton                 }
312944d93782SGreg Clayton                 return eKeyHandled;
313044d93782SGreg Clayton 
313144d93782SGreg Clayton             case '.':
313244d93782SGreg Clayton             case KEY_NPAGE:
313344d93782SGreg Clayton                 // Page down key
313444d93782SGreg Clayton                 if (m_num_rows > m_max_y)
313544d93782SGreg Clayton                 {
313644d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
313744d93782SGreg Clayton                     {
313844d93782SGreg Clayton                         m_first_visible_row += m_max_y;
313944d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
314044d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
314144d93782SGreg Clayton                         if (m_selected_item)
314244d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
314344d93782SGreg Clayton                     }
314444d93782SGreg Clayton                 }
314544d93782SGreg Clayton                 return eKeyHandled;
314644d93782SGreg Clayton 
314744d93782SGreg Clayton             case KEY_UP:
314844d93782SGreg Clayton                 if (m_selected_row_idx > 0)
314944d93782SGreg Clayton                 {
315044d93782SGreg Clayton                     --m_selected_row_idx;
315144d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
315244d93782SGreg Clayton                     if (m_selected_item)
315344d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
315444d93782SGreg Clayton                 }
315544d93782SGreg Clayton                 return eKeyHandled;
315644d93782SGreg Clayton             case KEY_DOWN:
315744d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
315844d93782SGreg Clayton                 {
315944d93782SGreg Clayton                     ++m_selected_row_idx;
316044d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
316144d93782SGreg Clayton                     if (m_selected_item)
316244d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
316344d93782SGreg Clayton                 }
316444d93782SGreg Clayton                 return eKeyHandled;
316544d93782SGreg Clayton 
316644d93782SGreg Clayton             case KEY_RIGHT:
316744d93782SGreg Clayton                 if (m_selected_item)
316844d93782SGreg Clayton                 {
316944d93782SGreg Clayton                     if (!m_selected_item->IsExpanded())
317044d93782SGreg Clayton                         m_selected_item->Expand();
317144d93782SGreg Clayton                 }
317244d93782SGreg Clayton                 return eKeyHandled;
317344d93782SGreg Clayton 
317444d93782SGreg Clayton             case KEY_LEFT:
317544d93782SGreg Clayton                 if (m_selected_item)
317644d93782SGreg Clayton                 {
317744d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
317844d93782SGreg Clayton                         m_selected_item->Unexpand();
317944d93782SGreg Clayton                     else if (m_selected_item->GetParent())
318044d93782SGreg Clayton                     {
318144d93782SGreg Clayton                         m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
318244d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
318344d93782SGreg Clayton                         if (m_selected_item)
318444d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
318544d93782SGreg Clayton                     }
318644d93782SGreg Clayton                 }
318744d93782SGreg Clayton                 return eKeyHandled;
318844d93782SGreg Clayton 
318944d93782SGreg Clayton             case ' ':
319044d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
319144d93782SGreg Clayton                 if (m_selected_item)
319244d93782SGreg Clayton                 {
319344d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
319444d93782SGreg Clayton                         m_selected_item->Unexpand();
319544d93782SGreg Clayton                     else
319644d93782SGreg Clayton                         m_selected_item->Expand();
319744d93782SGreg Clayton                 }
319844d93782SGreg Clayton                 return eKeyHandled;
319944d93782SGreg Clayton 
320044d93782SGreg Clayton             case 'h':
320144d93782SGreg Clayton                 window.CreateHelpSubwindow ();
320244d93782SGreg Clayton                 return eKeyHandled;
320344d93782SGreg Clayton 
320444d93782SGreg Clayton             default:
320544d93782SGreg Clayton                 break;
320644d93782SGreg Clayton         }
320744d93782SGreg Clayton         return eKeyNotHandled;
320844d93782SGreg Clayton     }
320944d93782SGreg Clayton 
321044d93782SGreg Clayton protected:
321144d93782SGreg Clayton     Debugger &m_debugger;
321244d93782SGreg Clayton     TreeDelegateSP m_delegate_sp;
321344d93782SGreg Clayton     TreeItem m_root;
321444d93782SGreg Clayton     TreeItem *m_selected_item;
321544d93782SGreg Clayton     int m_num_rows;
321644d93782SGreg Clayton     int m_selected_row_idx;
321744d93782SGreg Clayton     int m_first_visible_row;
321844d93782SGreg Clayton     int m_min_x;
321944d93782SGreg Clayton     int m_min_y;
322044d93782SGreg Clayton     int m_max_x;
322144d93782SGreg Clayton     int m_max_y;
322244d93782SGreg Clayton 
322344d93782SGreg Clayton };
322444d93782SGreg Clayton 
322544d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate
322644d93782SGreg Clayton {
322744d93782SGreg Clayton public:
3228ec990867SGreg Clayton     FrameTreeDelegate () :
3229ec990867SGreg Clayton         TreeDelegate()
323044d93782SGreg Clayton     {
3231554f68d3SGreg Clayton         FormatEntity::Parse ("frame #${frame.index}: {${function.name}${function.pc-offset}}}",
3232554f68d3SGreg Clayton                              m_format);
323344d93782SGreg Clayton     }
323444d93782SGreg Clayton 
3235bd5ae6b4SGreg Clayton     ~FrameTreeDelegate() override
323644d93782SGreg Clayton     {
323744d93782SGreg Clayton     }
323844d93782SGreg Clayton 
3239bd5ae6b4SGreg Clayton     void
3240bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
324144d93782SGreg Clayton     {
3242ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3243ec990867SGreg Clayton         if (thread)
324444d93782SGreg Clayton         {
324544d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3246ec990867SGreg Clayton             StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
324744d93782SGreg Clayton             if (frame_sp)
324844d93782SGreg Clayton             {
324944d93782SGreg Clayton                 StreamString strm;
325044d93782SGreg Clayton                 const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
325144d93782SGreg Clayton                 ExecutionContext exe_ctx (frame_sp);
3252554f68d3SGreg Clayton                 if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, NULL, NULL, false, false))
325344d93782SGreg Clayton                 {
325444d93782SGreg Clayton                     int right_pad = 1;
325544d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
325644d93782SGreg Clayton                 }
325744d93782SGreg Clayton             }
325844d93782SGreg Clayton         }
325944d93782SGreg Clayton     }
3260bd5ae6b4SGreg Clayton     void
3261bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)  override
326244d93782SGreg Clayton     {
326344d93782SGreg Clayton         // No children for frames yet...
326444d93782SGreg Clayton     }
326544d93782SGreg Clayton 
3266bd5ae6b4SGreg Clayton     bool
3267bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
326844d93782SGreg Clayton     {
3269ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3270ec990867SGreg Clayton         if (thread)
327144d93782SGreg Clayton         {
3272ec990867SGreg Clayton             thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID());
327344d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3274ec990867SGreg Clayton             thread->SetSelectedFrameByIndex(frame_idx);
327544d93782SGreg Clayton             return true;
327644d93782SGreg Clayton         }
327744d93782SGreg Clayton         return false;
327844d93782SGreg Clayton     }
3279554f68d3SGreg Clayton protected:
3280554f68d3SGreg Clayton     FormatEntity::Entry m_format;
328144d93782SGreg Clayton };
328244d93782SGreg Clayton 
328344d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate
328444d93782SGreg Clayton {
328544d93782SGreg Clayton public:
328644d93782SGreg Clayton     ThreadTreeDelegate (Debugger &debugger) :
328744d93782SGreg Clayton         TreeDelegate(),
328844d93782SGreg Clayton         m_debugger (debugger),
328944d93782SGreg Clayton         m_tid (LLDB_INVALID_THREAD_ID),
329044d93782SGreg Clayton         m_stop_id (UINT32_MAX)
329144d93782SGreg Clayton     {
3292554f68d3SGreg Clayton         FormatEntity::Parse ("thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}",
3293554f68d3SGreg Clayton                              m_format);
329444d93782SGreg Clayton     }
329544d93782SGreg Clayton 
3296bd5ae6b4SGreg Clayton     ~ThreadTreeDelegate()  override
329744d93782SGreg Clayton     {
329844d93782SGreg Clayton     }
329944d93782SGreg Clayton 
3300ec990867SGreg Clayton     ProcessSP
3301ec990867SGreg Clayton     GetProcess ()
3302ec990867SGreg Clayton     {
3303ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3304ec990867SGreg Clayton     }
3305ec990867SGreg Clayton 
3306ec990867SGreg Clayton     ThreadSP
3307ec990867SGreg Clayton     GetThread (const TreeItem &item)
3308ec990867SGreg Clayton     {
3309ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3310ec990867SGreg Clayton         if (process_sp)
3311ec990867SGreg Clayton             return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
3312ec990867SGreg Clayton         return ThreadSP();
3313ec990867SGreg Clayton     }
3314ec990867SGreg Clayton 
3315bd5ae6b4SGreg Clayton     void
3316bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
331744d93782SGreg Clayton     {
3318ec990867SGreg Clayton         ThreadSP thread_sp = GetThread (item);
331944d93782SGreg Clayton         if (thread_sp)
332044d93782SGreg Clayton         {
332144d93782SGreg Clayton             StreamString strm;
332244d93782SGreg Clayton             ExecutionContext exe_ctx (thread_sp);
3323554f68d3SGreg Clayton             if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
332444d93782SGreg Clayton             {
332544d93782SGreg Clayton                 int right_pad = 1;
332644d93782SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
332744d93782SGreg Clayton             }
332844d93782SGreg Clayton         }
332944d93782SGreg Clayton     }
3330bd5ae6b4SGreg Clayton     void
3331bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
333244d93782SGreg Clayton     {
3333ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
333444d93782SGreg Clayton         if (process_sp && process_sp->IsAlive())
333544d93782SGreg Clayton         {
333644d93782SGreg Clayton             StateType state = process_sp->GetState();
333744d93782SGreg Clayton             if (StateIsStoppedState(state, true))
333844d93782SGreg Clayton             {
3339ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
334044d93782SGreg Clayton                 if (thread_sp)
334144d93782SGreg Clayton                 {
334244d93782SGreg Clayton                     if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid)
334344d93782SGreg Clayton                         return; // Children are already up to date
3344ec990867SGreg Clayton                     if (!m_frame_delegate_sp)
334544d93782SGreg Clayton                     {
334644d93782SGreg Clayton                         // Always expand the thread item the first time we show it
3347ec990867SGreg Clayton                         m_frame_delegate_sp.reset (new FrameTreeDelegate());
334844d93782SGreg Clayton                     }
334944d93782SGreg Clayton 
335044d93782SGreg Clayton                     m_stop_id = process_sp->GetStopID();
335144d93782SGreg Clayton                     m_tid = thread_sp->GetID();
335244d93782SGreg Clayton 
335344d93782SGreg Clayton                     TreeItem t (&item, *m_frame_delegate_sp, false);
335444d93782SGreg Clayton                     size_t num_frames = thread_sp->GetStackFrameCount();
335544d93782SGreg Clayton                     item.Resize (num_frames, t);
335644d93782SGreg Clayton                     for (size_t i=0; i<num_frames; ++i)
335744d93782SGreg Clayton                     {
3358ec990867SGreg Clayton                         item[i].SetUserData(thread_sp.get());
335944d93782SGreg Clayton                         item[i].SetIdentifier(i);
336044d93782SGreg Clayton                     }
336144d93782SGreg Clayton                 }
336244d93782SGreg Clayton                 return;
336344d93782SGreg Clayton             }
336444d93782SGreg Clayton         }
336544d93782SGreg Clayton         item.ClearChildren();
336644d93782SGreg Clayton     }
336744d93782SGreg Clayton 
3368bd5ae6b4SGreg Clayton     bool
3369bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
337044d93782SGreg Clayton     {
3371ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3372ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3373ec990867SGreg Clayton         {
3374ec990867SGreg Clayton             StateType state = process_sp->GetState();
3375ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3376ec990867SGreg Clayton             {
3377ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
337844d93782SGreg Clayton                 if (thread_sp)
337944d93782SGreg Clayton                 {
338044d93782SGreg Clayton                     ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
338144d93782SGreg Clayton                     Mutex::Locker locker (thread_list.GetMutex());
338244d93782SGreg Clayton                     ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
338344d93782SGreg Clayton                     if (selected_thread_sp->GetID() != thread_sp->GetID())
338444d93782SGreg Clayton                     {
338544d93782SGreg Clayton                         thread_list.SetSelectedThreadByID(thread_sp->GetID());
338644d93782SGreg Clayton                         return true;
338744d93782SGreg Clayton                     }
338844d93782SGreg Clayton                 }
3389ec990867SGreg Clayton             }
3390ec990867SGreg Clayton         }
339144d93782SGreg Clayton         return false;
339244d93782SGreg Clayton     }
339344d93782SGreg Clayton 
339444d93782SGreg Clayton protected:
339544d93782SGreg Clayton     Debugger &m_debugger;
339644d93782SGreg Clayton     std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
339744d93782SGreg Clayton     lldb::user_id_t m_tid;
339844d93782SGreg Clayton     uint32_t m_stop_id;
3399554f68d3SGreg Clayton     FormatEntity::Entry m_format;
3400554f68d3SGreg Clayton 
340144d93782SGreg Clayton };
340244d93782SGreg Clayton 
3403ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate
3404ec990867SGreg Clayton {
3405ec990867SGreg Clayton public:
3406ec990867SGreg Clayton     ThreadsTreeDelegate (Debugger &debugger) :
3407ec990867SGreg Clayton         TreeDelegate(),
3408ec990867SGreg Clayton         m_thread_delegate_sp (),
3409ec990867SGreg Clayton         m_debugger (debugger),
3410ec990867SGreg Clayton         m_stop_id (UINT32_MAX)
3411ec990867SGreg Clayton     {
3412554f68d3SGreg Clayton         FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
3413554f68d3SGreg Clayton                             m_format);
3414ec990867SGreg Clayton     }
3415ec990867SGreg Clayton 
3416bd5ae6b4SGreg Clayton     ~ThreadsTreeDelegate() override
3417ec990867SGreg Clayton     {
3418ec990867SGreg Clayton     }
3419ec990867SGreg Clayton 
3420ec990867SGreg Clayton     ProcessSP
3421ec990867SGreg Clayton     GetProcess ()
3422ec990867SGreg Clayton     {
3423ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3424ec990867SGreg Clayton     }
3425ec990867SGreg Clayton 
3426bd5ae6b4SGreg Clayton     void
3427bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
3428ec990867SGreg Clayton     {
3429ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3430ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3431ec990867SGreg Clayton         {
3432ec990867SGreg Clayton             StreamString strm;
3433ec990867SGreg Clayton             ExecutionContext exe_ctx (process_sp);
3434554f68d3SGreg Clayton             if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
3435ec990867SGreg Clayton             {
3436ec990867SGreg Clayton                 int right_pad = 1;
3437ec990867SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
3438ec990867SGreg Clayton             }
3439ec990867SGreg Clayton         }
3440ec990867SGreg Clayton     }
3441ec990867SGreg Clayton 
3442bd5ae6b4SGreg Clayton     void
3443bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
3444ec990867SGreg Clayton     {
3445ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3446ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3447ec990867SGreg Clayton         {
3448ec990867SGreg Clayton             StateType state = process_sp->GetState();
3449ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3450ec990867SGreg Clayton             {
3451ec990867SGreg Clayton                 const uint32_t stop_id = process_sp->GetStopID();
3452ec990867SGreg Clayton                 if (m_stop_id == stop_id)
3453ec990867SGreg Clayton                     return; // Children are already up to date
3454ec990867SGreg Clayton 
3455ec990867SGreg Clayton                 m_stop_id = stop_id;
3456ec990867SGreg Clayton 
3457ec990867SGreg Clayton                 if (!m_thread_delegate_sp)
3458ec990867SGreg Clayton                 {
3459ec990867SGreg Clayton                     // Always expand the thread item the first time we show it
3460ec990867SGreg Clayton                     //item.Expand();
3461ec990867SGreg Clayton                     m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger));
3462ec990867SGreg Clayton                 }
3463ec990867SGreg Clayton 
3464ec990867SGreg Clayton                 TreeItem t (&item, *m_thread_delegate_sp, false);
3465ec990867SGreg Clayton                 ThreadList &threads = process_sp->GetThreadList();
3466ec990867SGreg Clayton                 Mutex::Locker locker (threads.GetMutex());
3467ec990867SGreg Clayton                 size_t num_threads = threads.GetSize();
3468ec990867SGreg Clayton                 item.Resize (num_threads, t);
3469ec990867SGreg Clayton                 for (size_t i=0; i<num_threads; ++i)
3470ec990867SGreg Clayton                 {
3471ec990867SGreg Clayton                     item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
3472ec990867SGreg Clayton                     item[i].SetMightHaveChildren(true);
3473ec990867SGreg Clayton                 }
3474ec990867SGreg Clayton                 return;
3475ec990867SGreg Clayton             }
3476ec990867SGreg Clayton         }
3477ec990867SGreg Clayton         item.ClearChildren();
3478ec990867SGreg Clayton     }
3479ec990867SGreg Clayton 
3480bd5ae6b4SGreg Clayton     bool
3481bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
3482ec990867SGreg Clayton     {
3483ec990867SGreg Clayton         return false;
3484ec990867SGreg Clayton     }
3485ec990867SGreg Clayton 
3486ec990867SGreg Clayton protected:
3487ec990867SGreg Clayton     std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
3488ec990867SGreg Clayton     Debugger &m_debugger;
3489ec990867SGreg Clayton     uint32_t m_stop_id;
3490554f68d3SGreg Clayton     FormatEntity::Entry m_format;
3491554f68d3SGreg Clayton 
3492ec990867SGreg Clayton };
3493ec990867SGreg Clayton 
349444d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate
349544d93782SGreg Clayton {
349644d93782SGreg Clayton public:
349744d93782SGreg Clayton     ValueObjectListDelegate () :
349844d93782SGreg Clayton         m_valobj_list (),
349944d93782SGreg Clayton         m_rows (),
350044d93782SGreg Clayton         m_selected_row (NULL),
350144d93782SGreg Clayton         m_selected_row_idx (0),
350244d93782SGreg Clayton         m_first_visible_row (0),
350344d93782SGreg Clayton         m_num_rows (0),
350444d93782SGreg Clayton         m_max_x (0),
350544d93782SGreg Clayton         m_max_y (0)
350644d93782SGreg Clayton     {
350744d93782SGreg Clayton     }
350844d93782SGreg Clayton 
350944d93782SGreg Clayton     ValueObjectListDelegate (ValueObjectList &valobj_list) :
351044d93782SGreg Clayton         m_valobj_list (valobj_list),
351144d93782SGreg Clayton         m_rows (),
351244d93782SGreg Clayton         m_selected_row (NULL),
351344d93782SGreg Clayton         m_selected_row_idx (0),
351444d93782SGreg Clayton         m_first_visible_row (0),
351544d93782SGreg Clayton         m_num_rows (0),
351644d93782SGreg Clayton         m_max_x (0),
351744d93782SGreg Clayton         m_max_y (0)
351844d93782SGreg Clayton     {
351944d93782SGreg Clayton         SetValues (valobj_list);
352044d93782SGreg Clayton     }
352144d93782SGreg Clayton 
3522bd5ae6b4SGreg Clayton     ~ValueObjectListDelegate() override
352344d93782SGreg Clayton     {
352444d93782SGreg Clayton     }
352544d93782SGreg Clayton 
352644d93782SGreg Clayton     void
352744d93782SGreg Clayton     SetValues (ValueObjectList &valobj_list)
352844d93782SGreg Clayton     {
352944d93782SGreg Clayton         m_selected_row = NULL;
353044d93782SGreg Clayton         m_selected_row_idx = 0;
353144d93782SGreg Clayton         m_first_visible_row = 0;
353244d93782SGreg Clayton         m_num_rows = 0;
353344d93782SGreg Clayton         m_rows.clear();
353444d93782SGreg Clayton         m_valobj_list = valobj_list;
353544d93782SGreg Clayton         const size_t num_values = m_valobj_list.GetSize();
353644d93782SGreg Clayton         for (size_t i=0; i<num_values; ++i)
353744d93782SGreg Clayton             m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), NULL));
353844d93782SGreg Clayton     }
353944d93782SGreg Clayton 
3540bd5ae6b4SGreg Clayton     bool
3541bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
354244d93782SGreg Clayton     {
354344d93782SGreg Clayton         m_num_rows = 0;
354444d93782SGreg Clayton         m_min_x = 2;
354544d93782SGreg Clayton         m_min_y = 1;
354644d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
354744d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
354844d93782SGreg Clayton 
354944d93782SGreg Clayton         window.Erase();
355044d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
355144d93782SGreg Clayton 
355244d93782SGreg Clayton         const int num_visible_rows = NumVisibleRows();
355344d93782SGreg Clayton         const int num_rows = CalculateTotalNumberRows (m_rows);
355444d93782SGreg Clayton 
355544d93782SGreg Clayton         // If we unexpanded while having something selected our
355644d93782SGreg Clayton         // total number of rows is less than the num visible rows,
355744d93782SGreg Clayton         // then make sure we show all the rows by setting the first
355844d93782SGreg Clayton         // visible row accordingly.
355944d93782SGreg Clayton         if (m_first_visible_row > 0 && num_rows < num_visible_rows)
356044d93782SGreg Clayton             m_first_visible_row = 0;
356144d93782SGreg Clayton 
356244d93782SGreg Clayton         // Make sure the selected row is always visible
356344d93782SGreg Clayton         if (m_selected_row_idx < m_first_visible_row)
356444d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx;
356544d93782SGreg Clayton         else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
356644d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
356744d93782SGreg Clayton 
356844d93782SGreg Clayton         DisplayRows (window, m_rows, g_options);
356944d93782SGreg Clayton 
357044d93782SGreg Clayton         window.DeferredRefresh();
357144d93782SGreg Clayton 
357244d93782SGreg Clayton         // Get the selected row
357344d93782SGreg Clayton         m_selected_row = GetRowForRowIndex (m_selected_row_idx);
357444d93782SGreg Clayton         // Keep the cursor on the selected row so the highlight and the cursor
357544d93782SGreg Clayton         // are always on the same line
357644d93782SGreg Clayton         if (m_selected_row)
357744d93782SGreg Clayton             window.MoveCursor (m_selected_row->x,
357844d93782SGreg Clayton                                m_selected_row->y);
357944d93782SGreg Clayton 
358044d93782SGreg Clayton         return true; // Drawing handled
358144d93782SGreg Clayton     }
358244d93782SGreg Clayton 
3583bd5ae6b4SGreg Clayton     KeyHelp *
3584bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
358544d93782SGreg Clayton     {
358644d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
358744d93782SGreg Clayton             { KEY_UP, "Select previous item" },
358844d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
358944d93782SGreg Clayton             { KEY_RIGHT, "Expand selected item" },
359044d93782SGreg Clayton             { KEY_LEFT, "Unexpand selected item or select parent if not expanded" },
359144d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
359244d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
359344d93782SGreg Clayton             { 'A', "Format as annotated address" },
359444d93782SGreg Clayton             { 'b', "Format as binary" },
359544d93782SGreg Clayton             { 'B', "Format as hex bytes with ASCII" },
359644d93782SGreg Clayton             { 'c', "Format as character" },
359744d93782SGreg Clayton             { 'd', "Format as a signed integer" },
359844d93782SGreg Clayton             { 'D', "Format selected value using the default format for the type" },
359944d93782SGreg Clayton             { 'f', "Format as float" },
360044d93782SGreg Clayton             { 'h', "Show help dialog" },
360144d93782SGreg Clayton             { 'i', "Format as instructions" },
360244d93782SGreg Clayton             { 'o', "Format as octal" },
360344d93782SGreg Clayton             { 'p', "Format as pointer" },
360444d93782SGreg Clayton             { 's', "Format as C string" },
360544d93782SGreg Clayton             { 't', "Toggle showing/hiding type names" },
360644d93782SGreg Clayton             { 'u', "Format as an unsigned integer" },
360744d93782SGreg Clayton             { 'x', "Format as hex" },
360844d93782SGreg Clayton             { 'X', "Format as uppercase hex" },
360944d93782SGreg Clayton             { ' ', "Toggle item expansion" },
361044d93782SGreg Clayton             { ',', "Page up" },
361144d93782SGreg Clayton             { '.', "Page down" },
361244d93782SGreg Clayton             { '\0', NULL }
361344d93782SGreg Clayton         };
361444d93782SGreg Clayton         return g_source_view_key_help;
361544d93782SGreg Clayton     }
361644d93782SGreg Clayton 
361744d93782SGreg Clayton 
3618bd5ae6b4SGreg Clayton     HandleCharResult
3619bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
362044d93782SGreg Clayton     {
362144d93782SGreg Clayton         switch(c)
362244d93782SGreg Clayton         {
362344d93782SGreg Clayton             case 'x':
362444d93782SGreg Clayton             case 'X':
362544d93782SGreg Clayton             case 'o':
362644d93782SGreg Clayton             case 's':
362744d93782SGreg Clayton             case 'u':
362844d93782SGreg Clayton             case 'd':
362944d93782SGreg Clayton             case 'D':
363044d93782SGreg Clayton             case 'i':
363144d93782SGreg Clayton             case 'A':
363244d93782SGreg Clayton             case 'p':
363344d93782SGreg Clayton             case 'c':
363444d93782SGreg Clayton             case 'b':
363544d93782SGreg Clayton             case 'B':
363644d93782SGreg Clayton             case 'f':
363744d93782SGreg Clayton                 // Change the format for the currently selected item
363844d93782SGreg Clayton                 if (m_selected_row)
363944d93782SGreg Clayton                     m_selected_row->valobj->SetFormat (FormatForChar (c));
364044d93782SGreg Clayton                 return eKeyHandled;
364144d93782SGreg Clayton 
364244d93782SGreg Clayton             case 't':
364344d93782SGreg Clayton                 // Toggle showing type names
364444d93782SGreg Clayton                 g_options.show_types = !g_options.show_types;
364544d93782SGreg Clayton                 return eKeyHandled;
364644d93782SGreg Clayton 
364744d93782SGreg Clayton             case ',':
364844d93782SGreg Clayton             case KEY_PPAGE:
364944d93782SGreg Clayton                 // Page up key
365044d93782SGreg Clayton                 if (m_first_visible_row > 0)
365144d93782SGreg Clayton                 {
36523985c8c6SSaleem Abdulrasool                     if (static_cast<int>(m_first_visible_row) > m_max_y)
365344d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
365444d93782SGreg Clayton                     else
365544d93782SGreg Clayton                         m_first_visible_row = 0;
365644d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
365744d93782SGreg Clayton                 }
365844d93782SGreg Clayton                 return eKeyHandled;
365944d93782SGreg Clayton 
366044d93782SGreg Clayton             case '.':
366144d93782SGreg Clayton             case KEY_NPAGE:
366244d93782SGreg Clayton                 // Page down key
36633985c8c6SSaleem Abdulrasool                 if (m_num_rows > static_cast<size_t>(m_max_y))
366444d93782SGreg Clayton                 {
366544d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
366644d93782SGreg Clayton                     {
366744d93782SGreg Clayton                         m_first_visible_row += m_max_y;
366844d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
366944d93782SGreg Clayton                     }
367044d93782SGreg Clayton                 }
367144d93782SGreg Clayton                 return eKeyHandled;
367244d93782SGreg Clayton 
367344d93782SGreg Clayton             case KEY_UP:
367444d93782SGreg Clayton                 if (m_selected_row_idx > 0)
367544d93782SGreg Clayton                     --m_selected_row_idx;
367644d93782SGreg Clayton                 return eKeyHandled;
367744d93782SGreg Clayton             case KEY_DOWN:
367844d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
367944d93782SGreg Clayton                     ++m_selected_row_idx;
368044d93782SGreg Clayton                 return eKeyHandled;
368144d93782SGreg Clayton 
368244d93782SGreg Clayton             case KEY_RIGHT:
368344d93782SGreg Clayton                 if (m_selected_row)
368444d93782SGreg Clayton                 {
368544d93782SGreg Clayton                     if (!m_selected_row->expanded)
368644d93782SGreg Clayton                         m_selected_row->Expand();
368744d93782SGreg Clayton                 }
368844d93782SGreg Clayton                 return eKeyHandled;
368944d93782SGreg Clayton 
369044d93782SGreg Clayton             case KEY_LEFT:
369144d93782SGreg Clayton                 if (m_selected_row)
369244d93782SGreg Clayton                 {
369344d93782SGreg Clayton                     if (m_selected_row->expanded)
369444d93782SGreg Clayton                         m_selected_row->Unexpand();
369544d93782SGreg Clayton                     else if (m_selected_row->parent)
369644d93782SGreg Clayton                         m_selected_row_idx = m_selected_row->parent->row_idx;
369744d93782SGreg Clayton                 }
369844d93782SGreg Clayton                 return eKeyHandled;
369944d93782SGreg Clayton 
370044d93782SGreg Clayton             case ' ':
370144d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
370244d93782SGreg Clayton                 if (m_selected_row)
370344d93782SGreg Clayton                 {
370444d93782SGreg Clayton                     if (m_selected_row->expanded)
370544d93782SGreg Clayton                         m_selected_row->Unexpand();
370644d93782SGreg Clayton                     else
370744d93782SGreg Clayton                         m_selected_row->Expand();
370844d93782SGreg Clayton                 }
370944d93782SGreg Clayton                 return eKeyHandled;
371044d93782SGreg Clayton 
371144d93782SGreg Clayton             case 'h':
371244d93782SGreg Clayton                 window.CreateHelpSubwindow ();
371344d93782SGreg Clayton                 return eKeyHandled;
371444d93782SGreg Clayton 
371544d93782SGreg Clayton             default:
371644d93782SGreg Clayton                 break;
371744d93782SGreg Clayton         }
371844d93782SGreg Clayton         return eKeyNotHandled;
371944d93782SGreg Clayton     }
372044d93782SGreg Clayton 
372144d93782SGreg Clayton protected:
372244d93782SGreg Clayton     ValueObjectList m_valobj_list;
372344d93782SGreg Clayton     std::vector<Row> m_rows;
372444d93782SGreg Clayton     Row *m_selected_row;
372544d93782SGreg Clayton     uint32_t m_selected_row_idx;
372644d93782SGreg Clayton     uint32_t m_first_visible_row;
372744d93782SGreg Clayton     uint32_t m_num_rows;
372844d93782SGreg Clayton     int m_min_x;
372944d93782SGreg Clayton     int m_min_y;
373044d93782SGreg Clayton     int m_max_x;
373144d93782SGreg Clayton     int m_max_y;
373244d93782SGreg Clayton 
373344d93782SGreg Clayton     static Format
373444d93782SGreg Clayton     FormatForChar (int c)
373544d93782SGreg Clayton     {
373644d93782SGreg Clayton         switch (c)
373744d93782SGreg Clayton         {
373844d93782SGreg Clayton             case 'x': return eFormatHex;
373944d93782SGreg Clayton             case 'X': return eFormatHexUppercase;
374044d93782SGreg Clayton             case 'o': return eFormatOctal;
374144d93782SGreg Clayton             case 's': return eFormatCString;
374244d93782SGreg Clayton             case 'u': return eFormatUnsigned;
374344d93782SGreg Clayton             case 'd': return eFormatDecimal;
374444d93782SGreg Clayton             case 'D': return eFormatDefault;
374544d93782SGreg Clayton             case 'i': return eFormatInstruction;
374644d93782SGreg Clayton             case 'A': return eFormatAddressInfo;
374744d93782SGreg Clayton             case 'p': return eFormatPointer;
374844d93782SGreg Clayton             case 'c': return eFormatChar;
374944d93782SGreg Clayton             case 'b': return eFormatBinary;
375044d93782SGreg Clayton             case 'B': return eFormatBytesWithASCII;
375144d93782SGreg Clayton             case 'f': return eFormatFloat;
375244d93782SGreg Clayton         }
375344d93782SGreg Clayton         return eFormatDefault;
375444d93782SGreg Clayton     }
375544d93782SGreg Clayton 
375644d93782SGreg Clayton     bool
375744d93782SGreg Clayton     DisplayRowObject (Window &window,
375844d93782SGreg Clayton                       Row &row,
375944d93782SGreg Clayton                       DisplayOptions &options,
376044d93782SGreg Clayton                       bool highlight,
376144d93782SGreg Clayton                       bool last_child)
376244d93782SGreg Clayton     {
376344d93782SGreg Clayton         ValueObject *valobj = row.valobj.get();
376444d93782SGreg Clayton 
376544d93782SGreg Clayton         if (valobj == NULL)
376644d93782SGreg Clayton             return false;
376744d93782SGreg Clayton 
376844d93782SGreg Clayton         const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : NULL;
376944d93782SGreg Clayton         const char *name = valobj->GetName().GetCString();
377044d93782SGreg Clayton         const char *value = valobj->GetValueAsCString ();
377144d93782SGreg Clayton         const char *summary = valobj->GetSummaryAsCString ();
377244d93782SGreg Clayton 
377344d93782SGreg Clayton         window.MoveCursor (row.x, row.y);
377444d93782SGreg Clayton 
377544d93782SGreg Clayton         row.DrawTree (window);
377644d93782SGreg Clayton 
377744d93782SGreg Clayton         if (highlight)
377844d93782SGreg Clayton             window.AttributeOn(A_REVERSE);
377944d93782SGreg Clayton 
378044d93782SGreg Clayton         if (type_name && type_name[0])
378144d93782SGreg Clayton             window.Printf ("(%s) ", type_name);
378244d93782SGreg Clayton 
378344d93782SGreg Clayton         if (name && name[0])
378444d93782SGreg Clayton             window.PutCString(name);
378544d93782SGreg Clayton 
378644d93782SGreg Clayton         attr_t changd_attr = 0;
378744d93782SGreg Clayton         if (valobj->GetValueDidChange())
378844d93782SGreg Clayton             changd_attr = COLOR_PAIR(5) | A_BOLD;
378944d93782SGreg Clayton 
379044d93782SGreg Clayton         if (value && value[0])
379144d93782SGreg Clayton         {
379244d93782SGreg Clayton             window.PutCString(" = ");
379344d93782SGreg Clayton             if (changd_attr)
379444d93782SGreg Clayton                 window.AttributeOn(changd_attr);
379544d93782SGreg Clayton             window.PutCString (value);
379644d93782SGreg Clayton             if (changd_attr)
379744d93782SGreg Clayton                 window.AttributeOff(changd_attr);
379844d93782SGreg Clayton         }
379944d93782SGreg Clayton 
380044d93782SGreg Clayton         if (summary && summary[0])
380144d93782SGreg Clayton         {
380244d93782SGreg Clayton             window.PutChar(' ');
380344d93782SGreg Clayton             if (changd_attr)
380444d93782SGreg Clayton                 window.AttributeOn(changd_attr);
380544d93782SGreg Clayton             window.PutCString(summary);
380644d93782SGreg Clayton             if (changd_attr)
380744d93782SGreg Clayton                 window.AttributeOff(changd_attr);
380844d93782SGreg Clayton         }
380944d93782SGreg Clayton 
381044d93782SGreg Clayton         if (highlight)
381144d93782SGreg Clayton             window.AttributeOff (A_REVERSE);
381244d93782SGreg Clayton 
381344d93782SGreg Clayton         return true;
381444d93782SGreg Clayton     }
381544d93782SGreg Clayton     void
381644d93782SGreg Clayton     DisplayRows (Window &window,
381744d93782SGreg Clayton                  std::vector<Row> &rows,
381844d93782SGreg Clayton                  DisplayOptions &options)
381944d93782SGreg Clayton     {
382044d93782SGreg Clayton         // >   0x25B7
382144d93782SGreg Clayton         // \/  0x25BD
382244d93782SGreg Clayton 
382344d93782SGreg Clayton         bool window_is_active = window.IsActive();
382444d93782SGreg Clayton         for (auto &row : rows)
382544d93782SGreg Clayton         {
382644d93782SGreg Clayton             const bool last_child = row.parent && &rows[rows.size()-1] == &row;
382744d93782SGreg Clayton             // Save the row index in each Row structure
382844d93782SGreg Clayton             row.row_idx = m_num_rows;
382944d93782SGreg Clayton             if ((m_num_rows >= m_first_visible_row) &&
38303985c8c6SSaleem Abdulrasool                 ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows())))
383144d93782SGreg Clayton             {
383244d93782SGreg Clayton                 row.x = m_min_x;
383344d93782SGreg Clayton                 row.y = m_num_rows - m_first_visible_row + 1;
383444d93782SGreg Clayton                 if (DisplayRowObject (window,
383544d93782SGreg Clayton                                       row,
383644d93782SGreg Clayton                                       options,
383744d93782SGreg Clayton                                       window_is_active && m_num_rows == m_selected_row_idx,
383844d93782SGreg Clayton                                       last_child))
383944d93782SGreg Clayton                 {
384044d93782SGreg Clayton                     ++m_num_rows;
384144d93782SGreg Clayton                 }
384244d93782SGreg Clayton                 else
384344d93782SGreg Clayton                 {
384444d93782SGreg Clayton                     row.x = 0;
384544d93782SGreg Clayton                     row.y = 0;
384644d93782SGreg Clayton                 }
384744d93782SGreg Clayton             }
384844d93782SGreg Clayton             else
384944d93782SGreg Clayton             {
385044d93782SGreg Clayton                 row.x = 0;
385144d93782SGreg Clayton                 row.y = 0;
385244d93782SGreg Clayton                 ++m_num_rows;
385344d93782SGreg Clayton             }
385444d93782SGreg Clayton 
385544d93782SGreg Clayton             if (row.expanded && !row.children.empty())
385644d93782SGreg Clayton             {
385744d93782SGreg Clayton                 DisplayRows (window,
385844d93782SGreg Clayton                              row.children,
385944d93782SGreg Clayton                              options);
386044d93782SGreg Clayton             }
386144d93782SGreg Clayton         }
386244d93782SGreg Clayton     }
386344d93782SGreg Clayton 
386444d93782SGreg Clayton     int
386544d93782SGreg Clayton     CalculateTotalNumberRows (const std::vector<Row> &rows)
386644d93782SGreg Clayton     {
386744d93782SGreg Clayton         int row_count = 0;
386844d93782SGreg Clayton         for (const auto &row : rows)
386944d93782SGreg Clayton         {
387044d93782SGreg Clayton             ++row_count;
387144d93782SGreg Clayton             if (row.expanded)
387244d93782SGreg Clayton                 row_count += CalculateTotalNumberRows(row.children);
387344d93782SGreg Clayton         }
387444d93782SGreg Clayton         return row_count;
387544d93782SGreg Clayton     }
387644d93782SGreg Clayton     static Row *
387744d93782SGreg Clayton     GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index)
387844d93782SGreg Clayton     {
387944d93782SGreg Clayton         for (auto &row : rows)
388044d93782SGreg Clayton         {
388144d93782SGreg Clayton             if (row_index == 0)
388244d93782SGreg Clayton                 return &row;
388344d93782SGreg Clayton             else
388444d93782SGreg Clayton             {
388544d93782SGreg Clayton                 --row_index;
388644d93782SGreg Clayton                 if (row.expanded && !row.children.empty())
388744d93782SGreg Clayton                 {
388844d93782SGreg Clayton                     Row *result = GetRowForRowIndexImpl (row.children, row_index);
388944d93782SGreg Clayton                     if (result)
389044d93782SGreg Clayton                         return result;
389144d93782SGreg Clayton                 }
389244d93782SGreg Clayton             }
389344d93782SGreg Clayton         }
389444d93782SGreg Clayton         return NULL;
389544d93782SGreg Clayton     }
389644d93782SGreg Clayton 
389744d93782SGreg Clayton     Row *
389844d93782SGreg Clayton     GetRowForRowIndex (size_t row_index)
389944d93782SGreg Clayton     {
390044d93782SGreg Clayton         return GetRowForRowIndexImpl (m_rows, row_index);
390144d93782SGreg Clayton     }
390244d93782SGreg Clayton 
390344d93782SGreg Clayton     int
390444d93782SGreg Clayton     NumVisibleRows () const
390544d93782SGreg Clayton     {
390644d93782SGreg Clayton         return m_max_y - m_min_y;
390744d93782SGreg Clayton     }
390844d93782SGreg Clayton 
390944d93782SGreg Clayton     static DisplayOptions g_options;
391044d93782SGreg Clayton };
391144d93782SGreg Clayton 
391244d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate
391344d93782SGreg Clayton {
391444d93782SGreg Clayton public:
391544d93782SGreg Clayton     FrameVariablesWindowDelegate (Debugger &debugger) :
391644d93782SGreg Clayton         ValueObjectListDelegate (),
391744d93782SGreg Clayton         m_debugger (debugger),
391844d93782SGreg Clayton         m_frame_block (NULL)
391944d93782SGreg Clayton     {
392044d93782SGreg Clayton     }
392144d93782SGreg Clayton 
3922bd5ae6b4SGreg Clayton     ~FrameVariablesWindowDelegate() override
392344d93782SGreg Clayton     {
392444d93782SGreg Clayton     }
392544d93782SGreg Clayton 
3926bd5ae6b4SGreg Clayton     const char *
3927bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
392844d93782SGreg Clayton     {
392944d93782SGreg Clayton         return "Frame variable window keyboard shortcuts:";
393044d93782SGreg Clayton     }
393144d93782SGreg Clayton 
3932bd5ae6b4SGreg Clayton     bool
3933bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
393444d93782SGreg Clayton     {
393544d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
393644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
393744d93782SGreg Clayton         Block *frame_block = NULL;
393844d93782SGreg Clayton         StackFrame *frame = NULL;
393944d93782SGreg Clayton 
394044d93782SGreg Clayton         if (process)
394144d93782SGreg Clayton         {
394244d93782SGreg Clayton             StateType state = process->GetState();
394344d93782SGreg Clayton             if (StateIsStoppedState(state, true))
394444d93782SGreg Clayton             {
394544d93782SGreg Clayton                 frame = exe_ctx.GetFramePtr();
394644d93782SGreg Clayton                 if (frame)
394744d93782SGreg Clayton                     frame_block = frame->GetFrameBlock ();
394844d93782SGreg Clayton             }
394944d93782SGreg Clayton             else if (StateIsRunningState(state))
395044d93782SGreg Clayton             {
395144d93782SGreg Clayton                 return true; // Don't do any updating when we are running
395244d93782SGreg Clayton             }
395344d93782SGreg Clayton         }
395444d93782SGreg Clayton 
3955*eb72dc7dSGreg Clayton 
395644d93782SGreg Clayton         ValueObjectList local_values;
395744d93782SGreg Clayton         if (frame_block)
395844d93782SGreg Clayton         {
395944d93782SGreg Clayton             // Only update the variables if they have changed
396044d93782SGreg Clayton             if (m_frame_block != frame_block)
396144d93782SGreg Clayton             {
396244d93782SGreg Clayton                 m_frame_block = frame_block;
396344d93782SGreg Clayton 
396444d93782SGreg Clayton                 VariableList *locals = frame->GetVariableList(true);
396544d93782SGreg Clayton                 if (locals)
396644d93782SGreg Clayton                 {
396744d93782SGreg Clayton                     const DynamicValueType use_dynamic = eDynamicDontRunTarget;
396844d93782SGreg Clayton                     const size_t num_locals = locals->GetSize();
396944d93782SGreg Clayton                     for (size_t i=0; i<num_locals; ++i)
3970*eb72dc7dSGreg Clayton                     {
3971*eb72dc7dSGreg Clayton                         ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic);
3972*eb72dc7dSGreg Clayton                         if (value_sp)
3973*eb72dc7dSGreg Clayton                         {
3974*eb72dc7dSGreg Clayton                             ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3975*eb72dc7dSGreg Clayton                             if (synthetic_value_sp)
3976*eb72dc7dSGreg Clayton                                 local_values.Append(synthetic_value_sp);
3977*eb72dc7dSGreg Clayton                             else
3978*eb72dc7dSGreg Clayton                                 local_values.Append(value_sp);
3979*eb72dc7dSGreg Clayton 
3980*eb72dc7dSGreg Clayton                         }
3981*eb72dc7dSGreg Clayton                     }
398244d93782SGreg Clayton                     // Update the values
398344d93782SGreg Clayton                     SetValues(local_values);
398444d93782SGreg Clayton                 }
398544d93782SGreg Clayton             }
398644d93782SGreg Clayton         }
398744d93782SGreg Clayton         else
398844d93782SGreg Clayton         {
398944d93782SGreg Clayton             m_frame_block = NULL;
399044d93782SGreg Clayton             // Update the values with an empty list if there is no frame
399144d93782SGreg Clayton             SetValues(local_values);
399244d93782SGreg Clayton         }
399344d93782SGreg Clayton 
399444d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
399544d93782SGreg Clayton 
399644d93782SGreg Clayton     }
399744d93782SGreg Clayton 
399844d93782SGreg Clayton protected:
399944d93782SGreg Clayton     Debugger &m_debugger;
400044d93782SGreg Clayton     Block *m_frame_block;
400144d93782SGreg Clayton };
400244d93782SGreg Clayton 
400344d93782SGreg Clayton 
400444d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate
400544d93782SGreg Clayton {
400644d93782SGreg Clayton public:
400744d93782SGreg Clayton     RegistersWindowDelegate (Debugger &debugger) :
400844d93782SGreg Clayton         ValueObjectListDelegate (),
400944d93782SGreg Clayton         m_debugger (debugger)
401044d93782SGreg Clayton     {
401144d93782SGreg Clayton     }
401244d93782SGreg Clayton 
401344d93782SGreg Clayton     ~RegistersWindowDelegate()
401444d93782SGreg Clayton     {
401544d93782SGreg Clayton     }
401644d93782SGreg Clayton 
4017bd5ae6b4SGreg Clayton     const char *
4018bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
401944d93782SGreg Clayton     {
402044d93782SGreg Clayton         return "Register window keyboard shortcuts:";
402144d93782SGreg Clayton     }
402244d93782SGreg Clayton 
4023bd5ae6b4SGreg Clayton     bool
4024bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
402544d93782SGreg Clayton     {
402644d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
402744d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
402844d93782SGreg Clayton 
402944d93782SGreg Clayton         ValueObjectList value_list;
403044d93782SGreg Clayton         if (frame)
403144d93782SGreg Clayton         {
403244d93782SGreg Clayton             if (frame->GetStackID() != m_stack_id)
403344d93782SGreg Clayton             {
403444d93782SGreg Clayton                 m_stack_id = frame->GetStackID();
403544d93782SGreg Clayton                 RegisterContextSP reg_ctx (frame->GetRegisterContext());
403644d93782SGreg Clayton                 if (reg_ctx)
403744d93782SGreg Clayton                 {
403844d93782SGreg Clayton                     const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
403944d93782SGreg Clayton                     for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx)
404044d93782SGreg Clayton                     {
404144d93782SGreg Clayton                         value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx));
404244d93782SGreg Clayton                     }
404344d93782SGreg Clayton                 }
404444d93782SGreg Clayton                 SetValues(value_list);
404544d93782SGreg Clayton             }
404644d93782SGreg Clayton         }
404744d93782SGreg Clayton         else
404844d93782SGreg Clayton         {
404944d93782SGreg Clayton             Process *process = exe_ctx.GetProcessPtr();
405044d93782SGreg Clayton             if (process && process->IsAlive())
405144d93782SGreg Clayton                 return true; // Don't do any updating if we are running
405244d93782SGreg Clayton             else
405344d93782SGreg Clayton             {
405444d93782SGreg Clayton                 // Update the values with an empty list if there
405544d93782SGreg Clayton                 // is no process or the process isn't alive anymore
405644d93782SGreg Clayton                 SetValues(value_list);
405744d93782SGreg Clayton             }
405844d93782SGreg Clayton         }
405944d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
406044d93782SGreg Clayton     }
406144d93782SGreg Clayton 
406244d93782SGreg Clayton protected:
406344d93782SGreg Clayton     Debugger &m_debugger;
406444d93782SGreg Clayton     StackID m_stack_id;
406544d93782SGreg Clayton };
406644d93782SGreg Clayton 
406744d93782SGreg Clayton static const char *
406844d93782SGreg Clayton CursesKeyToCString (int ch)
406944d93782SGreg Clayton {
407044d93782SGreg Clayton     static char g_desc[32];
407144d93782SGreg Clayton     if (ch >= KEY_F0 && ch < KEY_F0 + 64)
407244d93782SGreg Clayton     {
407344d93782SGreg Clayton         snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
407444d93782SGreg Clayton         return g_desc;
407544d93782SGreg Clayton     }
407644d93782SGreg Clayton     switch (ch)
407744d93782SGreg Clayton     {
407844d93782SGreg Clayton         case KEY_DOWN:  return "down";
407944d93782SGreg Clayton         case KEY_UP:    return "up";
408044d93782SGreg Clayton         case KEY_LEFT:  return "left";
408144d93782SGreg Clayton         case KEY_RIGHT: return "right";
408244d93782SGreg Clayton         case KEY_HOME:  return "home";
408344d93782SGreg Clayton         case KEY_BACKSPACE: return "backspace";
408444d93782SGreg Clayton         case KEY_DL:        return "delete-line";
408544d93782SGreg Clayton         case KEY_IL:        return "insert-line";
408644d93782SGreg Clayton         case KEY_DC:        return "delete-char";
408744d93782SGreg Clayton         case KEY_IC:        return "insert-char";
408844d93782SGreg Clayton         case KEY_CLEAR:     return "clear";
408944d93782SGreg Clayton         case KEY_EOS:       return "clear-to-eos";
409044d93782SGreg Clayton         case KEY_EOL:       return "clear-to-eol";
409144d93782SGreg Clayton         case KEY_SF:        return "scroll-forward";
409244d93782SGreg Clayton         case KEY_SR:        return "scroll-backward";
409344d93782SGreg Clayton         case KEY_NPAGE:     return "page-down";
409444d93782SGreg Clayton         case KEY_PPAGE:     return "page-up";
409544d93782SGreg Clayton         case KEY_STAB:      return "set-tab";
409644d93782SGreg Clayton         case KEY_CTAB:      return "clear-tab";
409744d93782SGreg Clayton         case KEY_CATAB:     return "clear-all-tabs";
409844d93782SGreg Clayton         case KEY_ENTER:     return "enter";
409944d93782SGreg Clayton         case KEY_PRINT:     return "print";
410044d93782SGreg Clayton         case KEY_LL:        return "lower-left key";
410144d93782SGreg Clayton         case KEY_A1:        return "upper left of keypad";
410244d93782SGreg Clayton         case KEY_A3:        return "upper right of keypad";
410344d93782SGreg Clayton         case KEY_B2:        return "center of keypad";
410444d93782SGreg Clayton         case KEY_C1:        return "lower left of keypad";
410544d93782SGreg Clayton         case KEY_C3:        return "lower right of keypad";
410644d93782SGreg Clayton         case KEY_BTAB:      return "back-tab key";
410744d93782SGreg Clayton         case KEY_BEG:       return "begin key";
410844d93782SGreg Clayton         case KEY_CANCEL:    return "cancel key";
410944d93782SGreg Clayton         case KEY_CLOSE:     return "close key";
411044d93782SGreg Clayton         case KEY_COMMAND:   return "command key";
411144d93782SGreg Clayton         case KEY_COPY:      return "copy key";
411244d93782SGreg Clayton         case KEY_CREATE:    return "create key";
411344d93782SGreg Clayton         case KEY_END:       return "end key";
411444d93782SGreg Clayton         case KEY_EXIT:      return "exit key";
411544d93782SGreg Clayton         case KEY_FIND:      return "find key";
411644d93782SGreg Clayton         case KEY_HELP:      return "help key";
411744d93782SGreg Clayton         case KEY_MARK:      return "mark key";
411844d93782SGreg Clayton         case KEY_MESSAGE:   return "message key";
411944d93782SGreg Clayton         case KEY_MOVE:      return "move key";
412044d93782SGreg Clayton         case KEY_NEXT:      return "next key";
412144d93782SGreg Clayton         case KEY_OPEN:      return "open key";
412244d93782SGreg Clayton         case KEY_OPTIONS:   return "options key";
412344d93782SGreg Clayton         case KEY_PREVIOUS:  return "previous key";
412444d93782SGreg Clayton         case KEY_REDO:      return "redo key";
412544d93782SGreg Clayton         case KEY_REFERENCE: return "reference key";
412644d93782SGreg Clayton         case KEY_REFRESH:   return "refresh key";
412744d93782SGreg Clayton         case KEY_REPLACE:   return "replace key";
412844d93782SGreg Clayton         case KEY_RESTART:   return "restart key";
412944d93782SGreg Clayton         case KEY_RESUME:    return "resume key";
413044d93782SGreg Clayton         case KEY_SAVE:      return "save key";
413144d93782SGreg Clayton         case KEY_SBEG:      return "shifted begin key";
413244d93782SGreg Clayton         case KEY_SCANCEL:   return "shifted cancel key";
413344d93782SGreg Clayton         case KEY_SCOMMAND:  return "shifted command key";
413444d93782SGreg Clayton         case KEY_SCOPY:     return "shifted copy key";
413544d93782SGreg Clayton         case KEY_SCREATE:   return "shifted create key";
413644d93782SGreg Clayton         case KEY_SDC:       return "shifted delete-character key";
413744d93782SGreg Clayton         case KEY_SDL:       return "shifted delete-line key";
413844d93782SGreg Clayton         case KEY_SELECT:    return "select key";
413944d93782SGreg Clayton         case KEY_SEND:      return "shifted end key";
414044d93782SGreg Clayton         case KEY_SEOL:      return "shifted clear-to-end-of-line key";
414144d93782SGreg Clayton         case KEY_SEXIT:     return "shifted exit key";
414244d93782SGreg Clayton         case KEY_SFIND:     return "shifted find key";
414344d93782SGreg Clayton         case KEY_SHELP:     return "shifted help key";
414444d93782SGreg Clayton         case KEY_SHOME:     return "shifted home key";
414544d93782SGreg Clayton         case KEY_SIC:       return "shifted insert-character key";
414644d93782SGreg Clayton         case KEY_SLEFT:     return "shifted left-arrow key";
414744d93782SGreg Clayton         case KEY_SMESSAGE:  return "shifted message key";
414844d93782SGreg Clayton         case KEY_SMOVE:     return "shifted move key";
414944d93782SGreg Clayton         case KEY_SNEXT:     return "shifted next key";
415044d93782SGreg Clayton         case KEY_SOPTIONS:  return "shifted options key";
415144d93782SGreg Clayton         case KEY_SPREVIOUS: return "shifted previous key";
415244d93782SGreg Clayton         case KEY_SPRINT:    return "shifted print key";
415344d93782SGreg Clayton         case KEY_SREDO:     return "shifted redo key";
415444d93782SGreg Clayton         case KEY_SREPLACE:  return "shifted replace key";
415544d93782SGreg Clayton         case KEY_SRIGHT:    return "shifted right-arrow key";
415644d93782SGreg Clayton         case KEY_SRSUME:    return "shifted resume key";
415744d93782SGreg Clayton         case KEY_SSAVE:     return "shifted save key";
415844d93782SGreg Clayton         case KEY_SSUSPEND:  return "shifted suspend key";
415944d93782SGreg Clayton         case KEY_SUNDO:     return "shifted undo key";
416044d93782SGreg Clayton         case KEY_SUSPEND:   return "suspend key";
416144d93782SGreg Clayton         case KEY_UNDO:      return "undo key";
416244d93782SGreg Clayton         case KEY_MOUSE:     return "Mouse event has occurred";
416344d93782SGreg Clayton         case KEY_RESIZE:    return "Terminal resize event";
416444d93782SGreg Clayton         case KEY_EVENT:     return "We were interrupted by an event";
416544d93782SGreg Clayton         case KEY_RETURN:    return "return";
416644d93782SGreg Clayton         case ' ':           return "space";
41675fdb09bbSGreg Clayton         case '\t':          return "tab";
416844d93782SGreg Clayton         case KEY_ESCAPE:    return "escape";
416944d93782SGreg Clayton         default:
417044d93782SGreg Clayton             if (isprint(ch))
417144d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "%c", ch);
417244d93782SGreg Clayton             else
417344d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
417444d93782SGreg Clayton             return g_desc;
417544d93782SGreg Clayton     }
417644d93782SGreg Clayton     return NULL;
417744d93782SGreg Clayton }
417844d93782SGreg Clayton 
417944d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) :
418044d93782SGreg Clayton     m_text (),
418144d93782SGreg Clayton     m_first_visible_line (0)
418244d93782SGreg Clayton {
418344d93782SGreg Clayton     if (text && text[0])
418444d93782SGreg Clayton     {
418544d93782SGreg Clayton         m_text.SplitIntoLines(text);
418644d93782SGreg Clayton         m_text.AppendString("");
418744d93782SGreg Clayton     }
418844d93782SGreg Clayton     if (key_help_array)
418944d93782SGreg Clayton     {
419044d93782SGreg Clayton         for (KeyHelp *key = key_help_array; key->ch; ++key)
419144d93782SGreg Clayton         {
419244d93782SGreg Clayton             StreamString key_description;
419344d93782SGreg Clayton             key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description);
419444d93782SGreg Clayton             m_text.AppendString(std::move(key_description.GetString()));
419544d93782SGreg Clayton         }
419644d93782SGreg Clayton     }
419744d93782SGreg Clayton }
419844d93782SGreg Clayton 
419944d93782SGreg Clayton HelpDialogDelegate::~HelpDialogDelegate()
420044d93782SGreg Clayton {
420144d93782SGreg Clayton }
420244d93782SGreg Clayton 
420344d93782SGreg Clayton bool
420444d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force)
420544d93782SGreg Clayton {
420644d93782SGreg Clayton     window.Erase();
420744d93782SGreg Clayton     const int window_height = window.GetHeight();
420844d93782SGreg Clayton     int x = 2;
420944d93782SGreg Clayton     int y = 1;
421044d93782SGreg Clayton     const int min_y = y;
421144d93782SGreg Clayton     const int max_y = window_height - 1 - y;
42123985c8c6SSaleem Abdulrasool     const size_t num_visible_lines = max_y - min_y + 1;
421344d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
421444d93782SGreg Clayton     const char *bottom_message;
421544d93782SGreg Clayton     if (num_lines <= num_visible_lines)
421644d93782SGreg Clayton         bottom_message = "Press any key to exit";
421744d93782SGreg Clayton     else
421844d93782SGreg Clayton         bottom_message = "Use arrows to scroll, any other key to exit";
421944d93782SGreg Clayton     window.DrawTitleBox(window.GetName(), bottom_message);
422044d93782SGreg Clayton     while (y <= max_y)
422144d93782SGreg Clayton     {
422244d93782SGreg Clayton         window.MoveCursor(x, y);
422344d93782SGreg Clayton         window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
422444d93782SGreg Clayton         ++y;
422544d93782SGreg Clayton     }
422644d93782SGreg Clayton     return true;
422744d93782SGreg Clayton }
422844d93782SGreg Clayton 
422944d93782SGreg Clayton HandleCharResult
423044d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key)
423144d93782SGreg Clayton {
423244d93782SGreg Clayton     bool done = false;
423344d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
423444d93782SGreg Clayton     const size_t num_visible_lines = window.GetHeight() - 2;
423544d93782SGreg Clayton 
423644d93782SGreg Clayton     if (num_lines <= num_visible_lines)
423744d93782SGreg Clayton     {
423844d93782SGreg Clayton         done = true;
423944d93782SGreg Clayton         // If we have all lines visible and don't need scrolling, then any
424044d93782SGreg Clayton         // key press will cause us to exit
424144d93782SGreg Clayton     }
424244d93782SGreg Clayton     else
424344d93782SGreg Clayton     {
424444d93782SGreg Clayton         switch (key)
424544d93782SGreg Clayton         {
424644d93782SGreg Clayton             case KEY_UP:
424744d93782SGreg Clayton                 if (m_first_visible_line > 0)
424844d93782SGreg Clayton                     --m_first_visible_line;
424944d93782SGreg Clayton                 break;
425044d93782SGreg Clayton 
425144d93782SGreg Clayton             case KEY_DOWN:
425244d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
425344d93782SGreg Clayton                     ++m_first_visible_line;
425444d93782SGreg Clayton                 break;
425544d93782SGreg Clayton 
425644d93782SGreg Clayton             case KEY_PPAGE:
425744d93782SGreg Clayton             case ',':
425844d93782SGreg Clayton                 if (m_first_visible_line > 0)
425944d93782SGreg Clayton                 {
42603985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
426144d93782SGreg Clayton                         m_first_visible_line -= num_visible_lines;
426244d93782SGreg Clayton                     else
426344d93782SGreg Clayton                         m_first_visible_line = 0;
426444d93782SGreg Clayton                 }
426544d93782SGreg Clayton                 break;
426644d93782SGreg Clayton             case KEY_NPAGE:
426744d93782SGreg Clayton             case '.':
426844d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
426944d93782SGreg Clayton                 {
427044d93782SGreg Clayton                     m_first_visible_line += num_visible_lines;
42713985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > num_lines)
427244d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
427344d93782SGreg Clayton                 }
427444d93782SGreg Clayton                 break;
427544d93782SGreg Clayton             default:
427644d93782SGreg Clayton                 done = true;
427744d93782SGreg Clayton                 break;
427844d93782SGreg Clayton         }
427944d93782SGreg Clayton     }
428044d93782SGreg Clayton     if (done)
428144d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
428244d93782SGreg Clayton     return eKeyHandled;
428344d93782SGreg Clayton }
428444d93782SGreg Clayton 
428544d93782SGreg Clayton class ApplicationDelegate :
428644d93782SGreg Clayton     public WindowDelegate,
428744d93782SGreg Clayton     public MenuDelegate
428844d93782SGreg Clayton {
428944d93782SGreg Clayton public:
429044d93782SGreg Clayton     enum {
429144d93782SGreg Clayton         eMenuID_LLDB = 1,
429244d93782SGreg Clayton         eMenuID_LLDBAbout,
429344d93782SGreg Clayton         eMenuID_LLDBExit,
429444d93782SGreg Clayton 
429544d93782SGreg Clayton         eMenuID_Target,
429644d93782SGreg Clayton         eMenuID_TargetCreate,
429744d93782SGreg Clayton         eMenuID_TargetDelete,
429844d93782SGreg Clayton 
429944d93782SGreg Clayton         eMenuID_Process,
430044d93782SGreg Clayton         eMenuID_ProcessAttach,
430144d93782SGreg Clayton         eMenuID_ProcessDetach,
430244d93782SGreg Clayton         eMenuID_ProcessLaunch,
430344d93782SGreg Clayton         eMenuID_ProcessContinue,
430444d93782SGreg Clayton         eMenuID_ProcessHalt,
430544d93782SGreg Clayton         eMenuID_ProcessKill,
430644d93782SGreg Clayton 
430744d93782SGreg Clayton         eMenuID_Thread,
430844d93782SGreg Clayton         eMenuID_ThreadStepIn,
430944d93782SGreg Clayton         eMenuID_ThreadStepOver,
431044d93782SGreg Clayton         eMenuID_ThreadStepOut,
431144d93782SGreg Clayton 
431244d93782SGreg Clayton         eMenuID_View,
431344d93782SGreg Clayton         eMenuID_ViewBacktrace,
431444d93782SGreg Clayton         eMenuID_ViewRegisters,
431544d93782SGreg Clayton         eMenuID_ViewSource,
431644d93782SGreg Clayton         eMenuID_ViewVariables,
431744d93782SGreg Clayton 
431844d93782SGreg Clayton         eMenuID_Help,
431944d93782SGreg Clayton         eMenuID_HelpGUIHelp
432044d93782SGreg Clayton     };
432144d93782SGreg Clayton 
432244d93782SGreg Clayton     ApplicationDelegate (Application &app, Debugger &debugger) :
432344d93782SGreg Clayton         WindowDelegate (),
432444d93782SGreg Clayton         MenuDelegate (),
432544d93782SGreg Clayton         m_app (app),
432644d93782SGreg Clayton         m_debugger (debugger)
432744d93782SGreg Clayton     {
432844d93782SGreg Clayton     }
432944d93782SGreg Clayton 
433044d93782SGreg Clayton     ~ApplicationDelegate ()
433144d93782SGreg Clayton     {
433244d93782SGreg Clayton     }
4333bd5ae6b4SGreg Clayton 
4334bd5ae6b4SGreg Clayton     bool
4335bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
433644d93782SGreg Clayton     {
433744d93782SGreg Clayton         return false; // Drawing not handled, let standard window drawing happen
433844d93782SGreg Clayton     }
433944d93782SGreg Clayton 
4340bd5ae6b4SGreg Clayton     HandleCharResult
4341bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int key) override
434244d93782SGreg Clayton     {
43435fdb09bbSGreg Clayton         switch (key)
434444d93782SGreg Clayton         {
43455fdb09bbSGreg Clayton             case '\t':
434644d93782SGreg Clayton                 window.SelectNextWindowAsActive();
434744d93782SGreg Clayton                 return eKeyHandled;
43485fdb09bbSGreg Clayton 
43495fdb09bbSGreg Clayton             case 'h':
43505fdb09bbSGreg Clayton                 window.CreateHelpSubwindow();
43515fdb09bbSGreg Clayton                 return eKeyHandled;
43525fdb09bbSGreg Clayton 
43535fdb09bbSGreg Clayton             case KEY_ESCAPE:
43545fdb09bbSGreg Clayton                 return eQuitApplication;
43555fdb09bbSGreg Clayton 
43565fdb09bbSGreg Clayton             default:
43575fdb09bbSGreg Clayton                 break;
435844d93782SGreg Clayton         }
435944d93782SGreg Clayton         return eKeyNotHandled;
436044d93782SGreg Clayton     }
436144d93782SGreg Clayton 
43625fdb09bbSGreg Clayton 
4363bd5ae6b4SGreg Clayton     const char *
4364bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
43655fdb09bbSGreg Clayton     {
43665fdb09bbSGreg Clayton         return "Welcome to the LLDB curses GUI.\n\n"
43675fdb09bbSGreg Clayton         "Press the TAB key to change the selected view.\n"
43685fdb09bbSGreg Clayton         "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n"
43695fdb09bbSGreg Clayton         "Common key bindings for all views:";
43705fdb09bbSGreg Clayton     }
43715fdb09bbSGreg Clayton 
4372bd5ae6b4SGreg Clayton     KeyHelp *
4373bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
43745fdb09bbSGreg Clayton     {
43755fdb09bbSGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
43765fdb09bbSGreg Clayton             { '\t', "Select next view" },
43775fdb09bbSGreg Clayton             { 'h', "Show help dialog with view specific key bindings" },
43785fdb09bbSGreg Clayton             { ',', "Page up" },
43795fdb09bbSGreg Clayton             { '.', "Page down" },
43805fdb09bbSGreg Clayton             { KEY_UP, "Select previous" },
43815fdb09bbSGreg Clayton             { KEY_DOWN, "Select next" },
43825fdb09bbSGreg Clayton             { KEY_LEFT, "Unexpand or select parent" },
43835fdb09bbSGreg Clayton             { KEY_RIGHT, "Expand" },
43845fdb09bbSGreg Clayton             { KEY_PPAGE, "Page up" },
43855fdb09bbSGreg Clayton             { KEY_NPAGE, "Page down" },
43865fdb09bbSGreg Clayton             { '\0', NULL }
43875fdb09bbSGreg Clayton         };
43885fdb09bbSGreg Clayton         return g_source_view_key_help;
43895fdb09bbSGreg Clayton     }
43905fdb09bbSGreg Clayton 
4391bd5ae6b4SGreg Clayton     MenuActionResult
4392bd5ae6b4SGreg Clayton     MenuDelegateAction (Menu &menu) override
439344d93782SGreg Clayton     {
439444d93782SGreg Clayton         switch (menu.GetIdentifier())
439544d93782SGreg Clayton         {
439644d93782SGreg Clayton             case eMenuID_ThreadStepIn:
439744d93782SGreg Clayton                 {
439844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
439944d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
440044d93782SGreg Clayton                     {
440144d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
440244d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
44034b4b2478SJim Ingham                             exe_ctx.GetThreadRef().StepIn(true);
440444d93782SGreg Clayton                     }
440544d93782SGreg Clayton                 }
440644d93782SGreg Clayton                 return MenuActionResult::Handled;
440744d93782SGreg Clayton 
440844d93782SGreg Clayton             case eMenuID_ThreadStepOut:
440944d93782SGreg Clayton                 {
441044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
441144d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
441244d93782SGreg Clayton                     {
441344d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
441444d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
441544d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOut();
441644d93782SGreg Clayton                     }
441744d93782SGreg Clayton                 }
441844d93782SGreg Clayton                 return MenuActionResult::Handled;
441944d93782SGreg Clayton 
442044d93782SGreg Clayton             case eMenuID_ThreadStepOver:
442144d93782SGreg Clayton                 {
442244d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
442344d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
442444d93782SGreg Clayton                     {
442544d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
442644d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
442744d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOver(true);
442844d93782SGreg Clayton                     }
442944d93782SGreg Clayton                 }
443044d93782SGreg Clayton                 return MenuActionResult::Handled;
443144d93782SGreg Clayton 
443244d93782SGreg Clayton             case eMenuID_ProcessContinue:
443344d93782SGreg Clayton                 {
443444d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
443544d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
443644d93782SGreg Clayton                     {
443744d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
443844d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
443944d93782SGreg Clayton                             process->Resume();
444044d93782SGreg Clayton                     }
444144d93782SGreg Clayton                 }
444244d93782SGreg Clayton                 return MenuActionResult::Handled;
444344d93782SGreg Clayton 
444444d93782SGreg Clayton             case eMenuID_ProcessKill:
444544d93782SGreg Clayton                 {
444644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
444744d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
444844d93782SGreg Clayton                     {
444944d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
445044d93782SGreg Clayton                         if (process && process->IsAlive())
445144d93782SGreg Clayton                             process->Destroy();
445244d93782SGreg Clayton                     }
445344d93782SGreg Clayton                 }
445444d93782SGreg Clayton                 return MenuActionResult::Handled;
445544d93782SGreg Clayton 
445644d93782SGreg Clayton             case eMenuID_ProcessHalt:
445744d93782SGreg Clayton                 {
445844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
445944d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
446044d93782SGreg Clayton                     {
446144d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
446244d93782SGreg Clayton                         if (process && process->IsAlive())
446344d93782SGreg Clayton                             process->Halt();
446444d93782SGreg Clayton                     }
446544d93782SGreg Clayton                 }
446644d93782SGreg Clayton                 return MenuActionResult::Handled;
446744d93782SGreg Clayton 
446844d93782SGreg Clayton             case eMenuID_ProcessDetach:
446944d93782SGreg Clayton                 {
447044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
447144d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
447244d93782SGreg Clayton                     {
447344d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
447444d93782SGreg Clayton                         if (process && process->IsAlive())
447544d93782SGreg Clayton                             process->Detach(false);
447644d93782SGreg Clayton                     }
447744d93782SGreg Clayton                 }
447844d93782SGreg Clayton                 return MenuActionResult::Handled;
447944d93782SGreg Clayton 
448044d93782SGreg Clayton             case eMenuID_Process:
448144d93782SGreg Clayton                 {
448244d93782SGreg Clayton                     // Populate the menu with all of the threads if the process is stopped when
448344d93782SGreg Clayton                     // the Process menu gets selected and is about to display its submenu.
448444d93782SGreg Clayton                     Menus &submenus = menu.GetSubmenus();
448544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
448644d93782SGreg Clayton                     Process *process = exe_ctx.GetProcessPtr();
448744d93782SGreg Clayton                     if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
448844d93782SGreg Clayton                     {
448944d93782SGreg Clayton                         if (submenus.size() == 7)
449044d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
449144d93782SGreg Clayton                         else if (submenus.size() > 8)
449244d93782SGreg Clayton                             submenus.erase (submenus.begin() + 8, submenus.end());
449344d93782SGreg Clayton 
449444d93782SGreg Clayton                         ThreadList &threads = process->GetThreadList();
449544d93782SGreg Clayton                         Mutex::Locker locker (threads.GetMutex());
449644d93782SGreg Clayton                         size_t num_threads = threads.GetSize();
449744d93782SGreg Clayton                         for (size_t i=0; i<num_threads; ++i)
449844d93782SGreg Clayton                         {
449944d93782SGreg Clayton                             ThreadSP thread_sp = threads.GetThreadAtIndex(i);
450044d93782SGreg Clayton                             char menu_char = '\0';
450144d93782SGreg Clayton                             if (i < 9)
450244d93782SGreg Clayton                                 menu_char = '1' + i;
450344d93782SGreg Clayton                             StreamString thread_menu_title;
450444d93782SGreg Clayton                             thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
450544d93782SGreg Clayton                             const char *thread_name = thread_sp->GetName();
450644d93782SGreg Clayton                             if (thread_name && thread_name[0])
450744d93782SGreg Clayton                                 thread_menu_title.Printf (" %s", thread_name);
450844d93782SGreg Clayton                             else
450944d93782SGreg Clayton                             {
451044d93782SGreg Clayton                                 const char *queue_name = thread_sp->GetQueueName();
451144d93782SGreg Clayton                                 if (queue_name && queue_name[0])
451244d93782SGreg Clayton                                     thread_menu_title.Printf (" %s", queue_name);
451344d93782SGreg Clayton                             }
451444d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(thread_menu_title.GetString().c_str(), NULL, menu_char, thread_sp->GetID())));
451544d93782SGreg Clayton                         }
451644d93782SGreg Clayton                     }
451744d93782SGreg Clayton                     else if (submenus.size() > 7)
451844d93782SGreg Clayton                     {
451944d93782SGreg Clayton                         // Remove the separator and any other thread submenu items
452044d93782SGreg Clayton                         // that were previously added
452144d93782SGreg Clayton                         submenus.erase (submenus.begin() + 7, submenus.end());
452244d93782SGreg Clayton                     }
452344d93782SGreg Clayton                     // Since we are adding and removing items we need to recalculate the name lengths
452444d93782SGreg Clayton                     menu.RecalculateNameLengths();
452544d93782SGreg Clayton                 }
452644d93782SGreg Clayton                 return MenuActionResult::Handled;
452744d93782SGreg Clayton 
452844d93782SGreg Clayton             case eMenuID_ViewVariables:
452944d93782SGreg Clayton                 {
453044d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
453144d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
453244d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
453344d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
453444d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
453544d93782SGreg Clayton 
453644d93782SGreg Clayton                     if (variables_window_sp)
453744d93782SGreg Clayton                     {
453844d93782SGreg Clayton                         const Rect variables_bounds = variables_window_sp->GetBounds();
453944d93782SGreg Clayton 
454044d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(variables_window_sp.get());
454144d93782SGreg Clayton 
454244d93782SGreg Clayton                         if (registers_window_sp)
454344d93782SGreg Clayton                         {
454444d93782SGreg Clayton                             // We have a registers window, so give all the area back to the registers window
454544d93782SGreg Clayton                             Rect registers_bounds = variables_bounds;
454644d93782SGreg Clayton                             registers_bounds.size.width = source_bounds.size.width;
454744d93782SGreg Clayton                             registers_window_sp->SetBounds(registers_bounds);
454844d93782SGreg Clayton                         }
454944d93782SGreg Clayton                         else
455044d93782SGreg Clayton                         {
455144d93782SGreg Clayton                             // We have no registers window showing so give the bottom
455244d93782SGreg Clayton                             // area back to the source view
455344d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
455444d93782SGreg Clayton                                                       source_bounds.size.height + variables_bounds.size.height);
455544d93782SGreg Clayton                         }
455644d93782SGreg Clayton                     }
455744d93782SGreg Clayton                     else
455844d93782SGreg Clayton                     {
455944d93782SGreg Clayton                         Rect new_variables_rect;
456044d93782SGreg Clayton                         if (registers_window_sp)
456144d93782SGreg Clayton                         {
456244d93782SGreg Clayton                             // We have a registers window so split the area of the registers
456344d93782SGreg Clayton                             // window into two columns where the left hand side will be the
456444d93782SGreg Clayton                             // variables and the right hand side will be the registers
456544d93782SGreg Clayton                             const Rect variables_bounds = registers_window_sp->GetBounds();
456644d93782SGreg Clayton                             Rect new_registers_rect;
456744d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect);
456844d93782SGreg Clayton                             registers_window_sp->SetBounds (new_registers_rect);
456944d93782SGreg Clayton                         }
457044d93782SGreg Clayton                         else
457144d93782SGreg Clayton                         {
457244d93782SGreg Clayton                             // No variables window, grab the bottom part of the source window
457344d93782SGreg Clayton                             Rect new_source_rect;
457444d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect);
457544d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
457644d93782SGreg Clayton                         }
457744d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables",
457844d93782SGreg Clayton                                                                                   new_variables_rect,
457944d93782SGreg Clayton                                                                                   false);
458044d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
458144d93782SGreg Clayton                     }
458244d93782SGreg Clayton                     touchwin(stdscr);
458344d93782SGreg Clayton                 }
458444d93782SGreg Clayton                 return MenuActionResult::Handled;
458544d93782SGreg Clayton 
458644d93782SGreg Clayton             case eMenuID_ViewRegisters:
458744d93782SGreg Clayton                 {
458844d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
458944d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
459044d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
459144d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
459244d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
459344d93782SGreg Clayton 
459444d93782SGreg Clayton                     if (registers_window_sp)
459544d93782SGreg Clayton                     {
459644d93782SGreg Clayton                         if (variables_window_sp)
459744d93782SGreg Clayton                         {
459844d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
459944d93782SGreg Clayton 
460044d93782SGreg Clayton                             // We have a variables window, so give all the area back to the variables window
460144d93782SGreg Clayton                             variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(),
460244d93782SGreg Clayton                                                          variables_bounds.size.height);
460344d93782SGreg Clayton                         }
460444d93782SGreg Clayton                         else
460544d93782SGreg Clayton                         {
460644d93782SGreg Clayton                             // We have no variables window showing so give the bottom
460744d93782SGreg Clayton                             // area back to the source view
460844d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
460944d93782SGreg Clayton                                                       source_bounds.size.height + registers_window_sp->GetHeight());
461044d93782SGreg Clayton                         }
461144d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(registers_window_sp.get());
461244d93782SGreg Clayton                     }
461344d93782SGreg Clayton                     else
461444d93782SGreg Clayton                     {
461544d93782SGreg Clayton                         Rect new_regs_rect;
461644d93782SGreg Clayton                         if (variables_window_sp)
461744d93782SGreg Clayton                         {
461844d93782SGreg Clayton                             // We have a variables window, split it into two columns
461944d93782SGreg Clayton                             // where the left hand side will be the variables and the
462044d93782SGreg Clayton                             // right hand side will be the registers
462144d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
462244d93782SGreg Clayton                             Rect new_vars_rect;
462344d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect);
462444d93782SGreg Clayton                             variables_window_sp->SetBounds (new_vars_rect);
462544d93782SGreg Clayton                         }
462644d93782SGreg Clayton                         else
462744d93782SGreg Clayton                         {
462844d93782SGreg Clayton                             // No registers window, grab the bottom part of the source window
462944d93782SGreg Clayton                             Rect new_source_rect;
463044d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect);
463144d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
463244d93782SGreg Clayton                         }
463344d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers",
463444d93782SGreg Clayton                                                                                   new_regs_rect,
463544d93782SGreg Clayton                                                                                   false);
463644d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
463744d93782SGreg Clayton                     }
463844d93782SGreg Clayton                     touchwin(stdscr);
463944d93782SGreg Clayton                 }
464044d93782SGreg Clayton                 return MenuActionResult::Handled;
464144d93782SGreg Clayton 
464244d93782SGreg Clayton             case eMenuID_HelpGUIHelp:
46435fdb09bbSGreg Clayton                 m_app.GetMainWindow ()->CreateHelpSubwindow();
464444d93782SGreg Clayton                 return MenuActionResult::Handled;
464544d93782SGreg Clayton 
464644d93782SGreg Clayton             default:
464744d93782SGreg Clayton                 break;
464844d93782SGreg Clayton         }
464944d93782SGreg Clayton 
465044d93782SGreg Clayton         return MenuActionResult::NotHandled;
465144d93782SGreg Clayton     }
465244d93782SGreg Clayton protected:
465344d93782SGreg Clayton     Application &m_app;
465444d93782SGreg Clayton     Debugger &m_debugger;
465544d93782SGreg Clayton };
465644d93782SGreg Clayton 
465744d93782SGreg Clayton 
465844d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate
465944d93782SGreg Clayton {
466044d93782SGreg Clayton public:
466144d93782SGreg Clayton     StatusBarWindowDelegate (Debugger &debugger) :
466244d93782SGreg Clayton         m_debugger (debugger)
466344d93782SGreg Clayton     {
4664554f68d3SGreg Clayton         FormatEntity::Parse("Thread: ${thread.id%tid}",
4665554f68d3SGreg Clayton                             m_format);
466644d93782SGreg Clayton     }
466744d93782SGreg Clayton 
466844d93782SGreg Clayton     ~StatusBarWindowDelegate ()
466944d93782SGreg Clayton     {
467044d93782SGreg Clayton     }
4671bd5ae6b4SGreg Clayton 
4672bd5ae6b4SGreg Clayton     bool
4673bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
467444d93782SGreg Clayton     {
467544d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
467644d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
467744d93782SGreg Clayton         Thread *thread = exe_ctx.GetThreadPtr();
467844d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
467944d93782SGreg Clayton         window.Erase();
468044d93782SGreg Clayton         window.SetBackground(2);
468144d93782SGreg Clayton         window.MoveCursor (0, 0);
468244d93782SGreg Clayton         if (process)
468344d93782SGreg Clayton         {
468444d93782SGreg Clayton             const StateType state = process->GetState();
468544d93782SGreg Clayton             window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state));
468644d93782SGreg Clayton 
468744d93782SGreg Clayton             if (StateIsStoppedState(state, true))
468844d93782SGreg Clayton             {
46895b031ebcSEd Maste                 StreamString strm;
4690554f68d3SGreg Clayton                 if (thread && FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
46915b031ebcSEd Maste                 {
469244d93782SGreg Clayton                     window.MoveCursor (40, 0);
46935b031ebcSEd Maste                     window.PutCStringTruncated(strm.GetString().c_str(), 1);
46945b031ebcSEd Maste                 }
469544d93782SGreg Clayton 
469644d93782SGreg Clayton                 window.MoveCursor (60, 0);
469744d93782SGreg Clayton                 if (frame)
469844d93782SGreg Clayton                     window.Printf ("Frame: %3u  PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr()));
469944d93782SGreg Clayton             }
470044d93782SGreg Clayton             else if (state == eStateExited)
470144d93782SGreg Clayton             {
470244d93782SGreg Clayton                 const char *exit_desc = process->GetExitDescription();
470344d93782SGreg Clayton                 const int exit_status = process->GetExitStatus();
470444d93782SGreg Clayton                 if (exit_desc && exit_desc[0])
470544d93782SGreg Clayton                     window.Printf (" with status = %i (%s)", exit_status, exit_desc);
470644d93782SGreg Clayton                 else
470744d93782SGreg Clayton                     window.Printf (" with status = %i", exit_status);
470844d93782SGreg Clayton             }
470944d93782SGreg Clayton         }
471044d93782SGreg Clayton         window.DeferredRefresh();
471144d93782SGreg Clayton         return true;
471244d93782SGreg Clayton     }
471344d93782SGreg Clayton 
471444d93782SGreg Clayton protected:
471544d93782SGreg Clayton     Debugger &m_debugger;
4716554f68d3SGreg Clayton     FormatEntity::Entry m_format;
471744d93782SGreg Clayton };
471844d93782SGreg Clayton 
471944d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate
472044d93782SGreg Clayton {
472144d93782SGreg Clayton public:
472244d93782SGreg Clayton     SourceFileWindowDelegate (Debugger &debugger) :
472344d93782SGreg Clayton         WindowDelegate (),
472444d93782SGreg Clayton         m_debugger (debugger),
472544d93782SGreg Clayton         m_sc (),
472644d93782SGreg Clayton         m_file_sp (),
472744d93782SGreg Clayton         m_disassembly_scope (NULL),
472844d93782SGreg Clayton         m_disassembly_sp (),
472944d93782SGreg Clayton         m_disassembly_range (),
4730ec990867SGreg Clayton         m_title (),
473144d93782SGreg Clayton         m_line_width (4),
473244d93782SGreg Clayton         m_selected_line (0),
473344d93782SGreg Clayton         m_pc_line (0),
473444d93782SGreg Clayton         m_stop_id (0),
473544d93782SGreg Clayton         m_frame_idx (UINT32_MAX),
473644d93782SGreg Clayton         m_first_visible_line (0),
473744d93782SGreg Clayton         m_min_x (0),
473844d93782SGreg Clayton         m_min_y (0),
473944d93782SGreg Clayton         m_max_x (0),
474044d93782SGreg Clayton         m_max_y (0)
474144d93782SGreg Clayton     {
474244d93782SGreg Clayton     }
474344d93782SGreg Clayton 
4744bd5ae6b4SGreg Clayton     ~SourceFileWindowDelegate() override
474544d93782SGreg Clayton     {
474644d93782SGreg Clayton     }
474744d93782SGreg Clayton 
474844d93782SGreg Clayton     void
474944d93782SGreg Clayton     Update (const SymbolContext &sc)
475044d93782SGreg Clayton     {
475144d93782SGreg Clayton         m_sc = sc;
475244d93782SGreg Clayton     }
475344d93782SGreg Clayton 
475444d93782SGreg Clayton     uint32_t
475544d93782SGreg Clayton     NumVisibleLines () const
475644d93782SGreg Clayton     {
475744d93782SGreg Clayton         return m_max_y - m_min_y;
475844d93782SGreg Clayton     }
475944d93782SGreg Clayton 
4760bd5ae6b4SGreg Clayton     const char *
4761bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
476244d93782SGreg Clayton     {
476344d93782SGreg Clayton         return "Source/Disassembly window keyboard shortcuts:";
476444d93782SGreg Clayton     }
476544d93782SGreg Clayton 
4766bd5ae6b4SGreg Clayton     KeyHelp *
4767bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
476844d93782SGreg Clayton     {
476944d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
477044d93782SGreg Clayton             { KEY_RETURN, "Run to selected line with one shot breakpoint" },
477144d93782SGreg Clayton             { KEY_UP, "Select previous source line" },
477244d93782SGreg Clayton             { KEY_DOWN, "Select next source line" },
477344d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
477444d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
477544d93782SGreg Clayton             { 'b', "Set breakpoint on selected source/disassembly line" },
477644d93782SGreg Clayton             { 'c', "Continue process" },
477744d93782SGreg Clayton             { 'd', "Detach and resume process" },
477844d93782SGreg Clayton             { 'D', "Detach with process suspended" },
477944d93782SGreg Clayton             { 'h', "Show help dialog" },
478044d93782SGreg Clayton             { 'k', "Kill process" },
478144d93782SGreg Clayton             { 'n', "Step over (source line)" },
478244d93782SGreg Clayton             { 'N', "Step over (single instruction)" },
478344d93782SGreg Clayton             { 'o', "Step out" },
478444d93782SGreg Clayton             { 's', "Step in (source line)" },
478544d93782SGreg Clayton             { 'S', "Step in (single instruction)" },
478644d93782SGreg Clayton             { ',', "Page up" },
478744d93782SGreg Clayton             { '.', "Page down" },
478844d93782SGreg Clayton             { '\0', NULL }
478944d93782SGreg Clayton         };
479044d93782SGreg Clayton         return g_source_view_key_help;
479144d93782SGreg Clayton     }
479244d93782SGreg Clayton 
4793bd5ae6b4SGreg Clayton     bool
4794bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
479544d93782SGreg Clayton     {
479644d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
479744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
479844d93782SGreg Clayton         Thread *thread = NULL;
479944d93782SGreg Clayton 
480044d93782SGreg Clayton         bool update_location = false;
480144d93782SGreg Clayton         if (process)
480244d93782SGreg Clayton         {
480344d93782SGreg Clayton             StateType state = process->GetState();
480444d93782SGreg Clayton             if (StateIsStoppedState(state, true))
480544d93782SGreg Clayton             {
480644d93782SGreg Clayton                 // We are stopped, so it is ok to
480744d93782SGreg Clayton                 update_location = true;
480844d93782SGreg Clayton             }
480944d93782SGreg Clayton         }
481044d93782SGreg Clayton 
481144d93782SGreg Clayton         m_min_x = 1;
4812ec990867SGreg Clayton         m_min_y = 2;
481344d93782SGreg Clayton         m_max_x = window.GetMaxX()-1;
481444d93782SGreg Clayton         m_max_y = window.GetMaxY()-1;
481544d93782SGreg Clayton 
481644d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
481744d93782SGreg Clayton         StackFrameSP frame_sp;
481844d93782SGreg Clayton         bool set_selected_line_to_pc = false;
481944d93782SGreg Clayton 
482044d93782SGreg Clayton         if (update_location)
482144d93782SGreg Clayton         {
482244d93782SGreg Clayton             const bool process_alive = process ? process->IsAlive() : false;
482344d93782SGreg Clayton             bool thread_changed = false;
482444d93782SGreg Clayton             if (process_alive)
482544d93782SGreg Clayton             {
482644d93782SGreg Clayton                 thread = exe_ctx.GetThreadPtr();
482744d93782SGreg Clayton                 if (thread)
482844d93782SGreg Clayton                 {
482944d93782SGreg Clayton                     frame_sp = thread->GetSelectedFrame();
483044d93782SGreg Clayton                     auto tid = thread->GetID();
483144d93782SGreg Clayton                     thread_changed = tid != m_tid;
483244d93782SGreg Clayton                     m_tid = tid;
483344d93782SGreg Clayton                 }
483444d93782SGreg Clayton                 else
483544d93782SGreg Clayton                 {
483644d93782SGreg Clayton                     if (m_tid != LLDB_INVALID_THREAD_ID)
483744d93782SGreg Clayton                     {
483844d93782SGreg Clayton                         thread_changed = true;
483944d93782SGreg Clayton                         m_tid = LLDB_INVALID_THREAD_ID;
484044d93782SGreg Clayton                     }
484144d93782SGreg Clayton                 }
484244d93782SGreg Clayton             }
484344d93782SGreg Clayton             const uint32_t stop_id = process ? process->GetStopID() : 0;
484444d93782SGreg Clayton             const bool stop_id_changed = stop_id != m_stop_id;
484544d93782SGreg Clayton             bool frame_changed = false;
484644d93782SGreg Clayton             m_stop_id = stop_id;
4847ec990867SGreg Clayton             m_title.Clear();
484844d93782SGreg Clayton             if (frame_sp)
484944d93782SGreg Clayton             {
485044d93782SGreg Clayton                 m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
4851ec990867SGreg Clayton                 if (m_sc.module_sp)
4852ec990867SGreg Clayton                 {
4853ec990867SGreg Clayton                     m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
4854ec990867SGreg Clayton                     ConstString func_name = m_sc.GetFunctionName();
4855ec990867SGreg Clayton                     if (func_name)
4856ec990867SGreg Clayton                         m_title.Printf("`%s", func_name.GetCString());
4857ec990867SGreg Clayton                 }
485844d93782SGreg Clayton                 const uint32_t frame_idx = frame_sp->GetFrameIndex();
485944d93782SGreg Clayton                 frame_changed = frame_idx != m_frame_idx;
486044d93782SGreg Clayton                 m_frame_idx = frame_idx;
486144d93782SGreg Clayton             }
486244d93782SGreg Clayton             else
486344d93782SGreg Clayton             {
486444d93782SGreg Clayton                 m_sc.Clear(true);
486544d93782SGreg Clayton                 frame_changed = m_frame_idx != UINT32_MAX;
486644d93782SGreg Clayton                 m_frame_idx = UINT32_MAX;
486744d93782SGreg Clayton             }
486844d93782SGreg Clayton 
486944d93782SGreg Clayton             const bool context_changed = thread_changed || frame_changed || stop_id_changed;
487044d93782SGreg Clayton 
487144d93782SGreg Clayton             if (process_alive)
487244d93782SGreg Clayton             {
487344d93782SGreg Clayton                 if (m_sc.line_entry.IsValid())
487444d93782SGreg Clayton                 {
487544d93782SGreg Clayton                     m_pc_line = m_sc.line_entry.line;
487644d93782SGreg Clayton                     if (m_pc_line != UINT32_MAX)
487744d93782SGreg Clayton                         --m_pc_line; // Convert to zero based line number...
487844d93782SGreg Clayton                     // Update the selected line if the stop ID changed...
487944d93782SGreg Clayton                     if (context_changed)
488044d93782SGreg Clayton                         m_selected_line = m_pc_line;
488144d93782SGreg Clayton 
488244d93782SGreg Clayton                     if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file))
488344d93782SGreg Clayton                     {
488444d93782SGreg Clayton                         // Same file, nothing to do, we should either have the
488544d93782SGreg Clayton                         // lines or not (source file missing)
48863985c8c6SSaleem Abdulrasool                         if (m_selected_line >= static_cast<size_t>(m_first_visible_line))
488744d93782SGreg Clayton                         {
488844d93782SGreg Clayton                             if (m_selected_line >= m_first_visible_line + num_visible_lines)
488944d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
489044d93782SGreg Clayton                         }
489144d93782SGreg Clayton                         else
489244d93782SGreg Clayton                         {
489344d93782SGreg Clayton                             if (m_selected_line > 10)
489444d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
489544d93782SGreg Clayton                             else
489644d93782SGreg Clayton                                 m_first_visible_line = 0;
489744d93782SGreg Clayton                         }
489844d93782SGreg Clayton                     }
489944d93782SGreg Clayton                     else
490044d93782SGreg Clayton                     {
490144d93782SGreg Clayton                         // File changed, set selected line to the line with the PC
490244d93782SGreg Clayton                         m_selected_line = m_pc_line;
490344d93782SGreg Clayton                         m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
490444d93782SGreg Clayton                         if (m_file_sp)
490544d93782SGreg Clayton                         {
490644d93782SGreg Clayton                             const size_t num_lines = m_file_sp->GetNumLines();
490744d93782SGreg Clayton                             int m_line_width = 1;
490844d93782SGreg Clayton                             for (size_t n = num_lines; n >= 10; n = n / 10)
490944d93782SGreg Clayton                                 ++m_line_width;
491044d93782SGreg Clayton 
491144d93782SGreg Clayton                             snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width);
491244d93782SGreg Clayton                             if (num_lines < num_visible_lines || m_selected_line < num_visible_lines)
491344d93782SGreg Clayton                                 m_first_visible_line = 0;
491444d93782SGreg Clayton                             else
491544d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
491644d93782SGreg Clayton                         }
491744d93782SGreg Clayton                     }
491844d93782SGreg Clayton                 }
491944d93782SGreg Clayton                 else
492044d93782SGreg Clayton                 {
492144d93782SGreg Clayton                     m_file_sp.reset();
492244d93782SGreg Clayton                 }
492344d93782SGreg Clayton 
492444d93782SGreg Clayton                 if (!m_file_sp || m_file_sp->GetNumLines() == 0)
492544d93782SGreg Clayton                 {
492644d93782SGreg Clayton                     // Show disassembly
492744d93782SGreg Clayton                     bool prefer_file_cache = false;
492844d93782SGreg Clayton                     if (m_sc.function)
492944d93782SGreg Clayton                     {
493044d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.function)
493144d93782SGreg Clayton                         {
493244d93782SGreg Clayton                             m_disassembly_scope = m_sc.function;
493344d93782SGreg Clayton                             m_disassembly_sp = m_sc.function->GetInstructions (exe_ctx, NULL, prefer_file_cache);
493444d93782SGreg Clayton                             if (m_disassembly_sp)
493544d93782SGreg Clayton                             {
493644d93782SGreg Clayton                                 set_selected_line_to_pc = true;
493744d93782SGreg Clayton                                 m_disassembly_range = m_sc.function->GetAddressRange();
493844d93782SGreg Clayton                             }
493944d93782SGreg Clayton                             else
494044d93782SGreg Clayton                             {
494144d93782SGreg Clayton                                 m_disassembly_range.Clear();
494244d93782SGreg Clayton                             }
494344d93782SGreg Clayton                         }
494444d93782SGreg Clayton                         else
494544d93782SGreg Clayton                         {
494644d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
494744d93782SGreg Clayton                         }
494844d93782SGreg Clayton                     }
494944d93782SGreg Clayton                     else if (m_sc.symbol)
495044d93782SGreg Clayton                     {
495144d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.symbol)
495244d93782SGreg Clayton                         {
495344d93782SGreg Clayton                             m_disassembly_scope = m_sc.symbol;
495444d93782SGreg Clayton                             m_disassembly_sp = m_sc.symbol->GetInstructions (exe_ctx, NULL, prefer_file_cache);
495544d93782SGreg Clayton                             if (m_disassembly_sp)
495644d93782SGreg Clayton                             {
495744d93782SGreg Clayton                                 set_selected_line_to_pc = true;
495844d93782SGreg Clayton                                 m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress();
495944d93782SGreg Clayton                                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
496044d93782SGreg Clayton                             }
496144d93782SGreg Clayton                             else
496244d93782SGreg Clayton                             {
496344d93782SGreg Clayton                                 m_disassembly_range.Clear();
496444d93782SGreg Clayton                             }
496544d93782SGreg Clayton                         }
496644d93782SGreg Clayton                         else
496744d93782SGreg Clayton                         {
496844d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
496944d93782SGreg Clayton                         }
497044d93782SGreg Clayton                     }
497144d93782SGreg Clayton                 }
497244d93782SGreg Clayton             }
497344d93782SGreg Clayton             else
497444d93782SGreg Clayton             {
497544d93782SGreg Clayton                 m_pc_line = UINT32_MAX;
497644d93782SGreg Clayton             }
497744d93782SGreg Clayton         }
497844d93782SGreg Clayton 
4979ec990867SGreg Clayton         const int window_width = window.GetWidth();
498044d93782SGreg Clayton         window.Erase();
498144d93782SGreg Clayton         window.DrawTitleBox ("Sources");
4982ec990867SGreg Clayton         if (!m_title.GetString().empty())
4983ec990867SGreg Clayton         {
4984ec990867SGreg Clayton             window.AttributeOn(A_REVERSE);
4985ec990867SGreg Clayton             window.MoveCursor(1, 1);
4986ec990867SGreg Clayton             window.PutChar(' ');
4987ec990867SGreg Clayton             window.PutCStringTruncated(m_title.GetString().c_str(), 1);
4988ec990867SGreg Clayton             int x = window.GetCursorX();
4989ec990867SGreg Clayton             if (x < window_width - 1)
4990ec990867SGreg Clayton             {
4991ec990867SGreg Clayton                 window.Printf ("%*s", window_width - x - 1, "");
4992ec990867SGreg Clayton             }
4993ec990867SGreg Clayton             window.AttributeOff(A_REVERSE);
4994ec990867SGreg Clayton         }
499544d93782SGreg Clayton 
499644d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
499744d93782SGreg Clayton         const size_t num_source_lines = GetNumSourceLines();
499844d93782SGreg Clayton         if (num_source_lines > 0)
499944d93782SGreg Clayton         {
500044d93782SGreg Clayton             // Display source
500144d93782SGreg Clayton             BreakpointLines bp_lines;
500244d93782SGreg Clayton             if (target)
500344d93782SGreg Clayton             {
500444d93782SGreg Clayton                 BreakpointList &bp_list = target->GetBreakpointList();
500544d93782SGreg Clayton                 const size_t num_bps = bp_list.GetSize();
500644d93782SGreg Clayton                 for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
500744d93782SGreg Clayton                 {
500844d93782SGreg Clayton                     BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
500944d93782SGreg Clayton                     const size_t num_bps_locs = bp_sp->GetNumLocations();
501044d93782SGreg Clayton                     for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
501144d93782SGreg Clayton                     {
501244d93782SGreg Clayton                         BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
501344d93782SGreg Clayton                         LineEntry bp_loc_line_entry;
501444d93782SGreg Clayton                         if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry))
501544d93782SGreg Clayton                         {
501644d93782SGreg Clayton                             if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file)
501744d93782SGreg Clayton                             {
501844d93782SGreg Clayton                                 bp_lines.insert(bp_loc_line_entry.line);
501944d93782SGreg Clayton                             }
502044d93782SGreg Clayton                         }
502144d93782SGreg Clayton                     }
502244d93782SGreg Clayton                 }
502344d93782SGreg Clayton             }
502444d93782SGreg Clayton 
502544d93782SGreg Clayton             const attr_t selected_highlight_attr = A_REVERSE;
502644d93782SGreg Clayton             const attr_t pc_highlight_attr = COLOR_PAIR(1);
502744d93782SGreg Clayton 
50283985c8c6SSaleem Abdulrasool             for (size_t i=0; i<num_visible_lines; ++i)
502944d93782SGreg Clayton             {
503044d93782SGreg Clayton                 const uint32_t curr_line = m_first_visible_line + i;
503144d93782SGreg Clayton                 if (curr_line < num_source_lines)
503244d93782SGreg Clayton                 {
5033ec990867SGreg Clayton                     const int line_y = m_min_y+i;
503444d93782SGreg Clayton                     window.MoveCursor(1, line_y);
503544d93782SGreg Clayton                     const bool is_pc_line = curr_line == m_pc_line;
503644d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == curr_line;
503744d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
503844d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
503944d93782SGreg Clayton                     attr_t highlight_attr = 0;
504044d93782SGreg Clayton                     attr_t bp_attr = 0;
504144d93782SGreg Clayton                     if (is_pc_line)
504244d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
504344d93782SGreg Clayton                     else if (line_is_selected)
504444d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
504544d93782SGreg Clayton 
504644d93782SGreg Clayton                     if (bp_lines.find(curr_line+1) != bp_lines.end())
504744d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
504844d93782SGreg Clayton 
504944d93782SGreg Clayton                     if (bp_attr)
505044d93782SGreg Clayton                         window.AttributeOn(bp_attr);
505144d93782SGreg Clayton 
505244d93782SGreg Clayton                     window.Printf (m_line_format, curr_line + 1);
505344d93782SGreg Clayton 
505444d93782SGreg Clayton                     if (bp_attr)
505544d93782SGreg Clayton                         window.AttributeOff(bp_attr);
505644d93782SGreg Clayton 
505744d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
505844d93782SGreg Clayton                     // Mark the line with the PC with a diamond
505944d93782SGreg Clayton                     if (is_pc_line)
506044d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
506144d93782SGreg Clayton                     else
506244d93782SGreg Clayton                         window.PutChar(' ');
506344d93782SGreg Clayton 
506444d93782SGreg Clayton                     if (highlight_attr)
506544d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
506644d93782SGreg Clayton                     const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false);
506744d93782SGreg Clayton                     if (line_len > 0)
506844d93782SGreg Clayton                         window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
506944d93782SGreg Clayton 
507044d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
507144d93782SGreg Clayton                     {
507244d93782SGreg Clayton                         StopInfoSP stop_info_sp;
507344d93782SGreg Clayton                         if (thread)
507444d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
507544d93782SGreg Clayton                         if (stop_info_sp)
507644d93782SGreg Clayton                         {
507744d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
507844d93782SGreg Clayton                             if (stop_description && stop_description[0])
507944d93782SGreg Clayton                             {
508044d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5081ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
508244d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5083ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
508444d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
508544d93782SGreg Clayton                             }
508644d93782SGreg Clayton                         }
508744d93782SGreg Clayton                         else
508844d93782SGreg Clayton                         {
5089ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
509044d93782SGreg Clayton                         }
509144d93782SGreg Clayton                     }
509244d93782SGreg Clayton                     if (highlight_attr)
509344d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
509444d93782SGreg Clayton 
509544d93782SGreg Clayton                 }
509644d93782SGreg Clayton                 else
509744d93782SGreg Clayton                 {
509844d93782SGreg Clayton                     break;
509944d93782SGreg Clayton                 }
510044d93782SGreg Clayton             }
510144d93782SGreg Clayton         }
510244d93782SGreg Clayton         else
510344d93782SGreg Clayton         {
510444d93782SGreg Clayton             size_t num_disassembly_lines = GetNumDisassemblyLines();
510544d93782SGreg Clayton             if (num_disassembly_lines > 0)
510644d93782SGreg Clayton             {
510744d93782SGreg Clayton                 // Display disassembly
510844d93782SGreg Clayton                 BreakpointAddrs bp_file_addrs;
510944d93782SGreg Clayton                 Target *target = exe_ctx.GetTargetPtr();
511044d93782SGreg Clayton                 if (target)
511144d93782SGreg Clayton                 {
511244d93782SGreg Clayton                     BreakpointList &bp_list = target->GetBreakpointList();
511344d93782SGreg Clayton                     const size_t num_bps = bp_list.GetSize();
511444d93782SGreg Clayton                     for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
511544d93782SGreg Clayton                     {
511644d93782SGreg Clayton                         BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
511744d93782SGreg Clayton                         const size_t num_bps_locs = bp_sp->GetNumLocations();
511844d93782SGreg Clayton                         for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
511944d93782SGreg Clayton                         {
512044d93782SGreg Clayton                             BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
512144d93782SGreg Clayton                             LineEntry bp_loc_line_entry;
512244d93782SGreg Clayton                             const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress();
512344d93782SGreg Clayton                             if (file_addr != LLDB_INVALID_ADDRESS)
512444d93782SGreg Clayton                             {
512544d93782SGreg Clayton                                 if (m_disassembly_range.ContainsFileAddress(file_addr))
512644d93782SGreg Clayton                                     bp_file_addrs.insert(file_addr);
512744d93782SGreg Clayton                             }
512844d93782SGreg Clayton                         }
512944d93782SGreg Clayton                     }
513044d93782SGreg Clayton                 }
513144d93782SGreg Clayton 
513244d93782SGreg Clayton                 const attr_t selected_highlight_attr = A_REVERSE;
513344d93782SGreg Clayton                 const attr_t pc_highlight_attr = COLOR_PAIR(1);
513444d93782SGreg Clayton 
513544d93782SGreg Clayton                 StreamString strm;
513644d93782SGreg Clayton 
513744d93782SGreg Clayton                 InstructionList &insts = m_disassembly_sp->GetInstructionList();
513844d93782SGreg Clayton                 Address pc_address;
513944d93782SGreg Clayton 
514044d93782SGreg Clayton                 if (frame_sp)
514144d93782SGreg Clayton                     pc_address = frame_sp->GetFrameCodeAddress();
514244d93782SGreg Clayton                 const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX;
514344d93782SGreg Clayton                 if (set_selected_line_to_pc)
514444d93782SGreg Clayton                 {
514544d93782SGreg Clayton                     m_selected_line = pc_idx;
514644d93782SGreg Clayton                 }
514744d93782SGreg Clayton 
514844d93782SGreg Clayton                 const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
51493985c8c6SSaleem Abdulrasool                 if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
515044d93782SGreg Clayton                     m_first_visible_line = 0;
515144d93782SGreg Clayton 
515244d93782SGreg Clayton                 if (pc_idx < num_disassembly_lines)
515344d93782SGreg Clayton                 {
51543985c8c6SSaleem Abdulrasool                     if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
515544d93782SGreg Clayton                         pc_idx >= m_first_visible_line + num_visible_lines)
515644d93782SGreg Clayton                         m_first_visible_line = pc_idx - non_visible_pc_offset;
515744d93782SGreg Clayton                 }
515844d93782SGreg Clayton 
515944d93782SGreg Clayton                 for (size_t i=0; i<num_visible_lines; ++i)
516044d93782SGreg Clayton                 {
516144d93782SGreg Clayton                     const uint32_t inst_idx = m_first_visible_line + i;
516244d93782SGreg Clayton                     Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
516344d93782SGreg Clayton                     if (!inst)
516444d93782SGreg Clayton                         break;
516544d93782SGreg Clayton 
5166ec990867SGreg Clayton                     const int line_y = m_min_y+i;
5167ec990867SGreg Clayton                     window.MoveCursor(1, line_y);
516844d93782SGreg Clayton                     const bool is_pc_line = frame_sp && inst_idx == pc_idx;
516944d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == inst_idx;
517044d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
517144d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
517244d93782SGreg Clayton                     attr_t highlight_attr = 0;
517344d93782SGreg Clayton                     attr_t bp_attr = 0;
517444d93782SGreg Clayton                     if (is_pc_line)
517544d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
517644d93782SGreg Clayton                     else if (line_is_selected)
517744d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
517844d93782SGreg Clayton 
517944d93782SGreg Clayton                     if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end())
518044d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
518144d93782SGreg Clayton 
518244d93782SGreg Clayton                     if (bp_attr)
518344d93782SGreg Clayton                         window.AttributeOn(bp_attr);
518444d93782SGreg Clayton 
5185324a1036SSaleem Abdulrasool                     window.Printf (" 0x%16.16llx ",
5186324a1036SSaleem Abdulrasool                                    static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target)));
518744d93782SGreg Clayton 
518844d93782SGreg Clayton                     if (bp_attr)
518944d93782SGreg Clayton                         window.AttributeOff(bp_attr);
519044d93782SGreg Clayton 
519144d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
519244d93782SGreg Clayton                     // Mark the line with the PC with a diamond
519344d93782SGreg Clayton                     if (is_pc_line)
519444d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
519544d93782SGreg Clayton                     else
519644d93782SGreg Clayton                         window.PutChar(' ');
519744d93782SGreg Clayton 
519844d93782SGreg Clayton                     if (highlight_attr)
519944d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
520044d93782SGreg Clayton 
520144d93782SGreg Clayton                     const char *mnemonic = inst->GetMnemonic(&exe_ctx);
520244d93782SGreg Clayton                     const char *operands = inst->GetOperands(&exe_ctx);
520344d93782SGreg Clayton                     const char *comment = inst->GetComment(&exe_ctx);
520444d93782SGreg Clayton 
520544d93782SGreg Clayton                     if (mnemonic && mnemonic[0] == '\0')
520644d93782SGreg Clayton                         mnemonic = NULL;
520744d93782SGreg Clayton                     if (operands && operands[0] == '\0')
520844d93782SGreg Clayton                         operands = NULL;
520944d93782SGreg Clayton                     if (comment && comment[0] == '\0')
521044d93782SGreg Clayton                         comment = NULL;
521144d93782SGreg Clayton 
521244d93782SGreg Clayton                     strm.Clear();
521344d93782SGreg Clayton 
521444d93782SGreg Clayton                     if (mnemonic && operands && comment)
521544d93782SGreg Clayton                         strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment);
521644d93782SGreg Clayton                     else if (mnemonic && operands)
521744d93782SGreg Clayton                         strm.Printf ("%-8s %s", mnemonic, operands);
521844d93782SGreg Clayton                     else if (mnemonic)
521944d93782SGreg Clayton                         strm.Printf ("%s", mnemonic);
522044d93782SGreg Clayton 
522144d93782SGreg Clayton                     int right_pad = 1;
522244d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
522344d93782SGreg Clayton 
522444d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
522544d93782SGreg Clayton                     {
522644d93782SGreg Clayton                         StopInfoSP stop_info_sp;
522744d93782SGreg Clayton                         if (thread)
522844d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
522944d93782SGreg Clayton                         if (stop_info_sp)
523044d93782SGreg Clayton                         {
523144d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
523244d93782SGreg Clayton                             if (stop_description && stop_description[0])
523344d93782SGreg Clayton                             {
523444d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5235ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
523644d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5237ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
523844d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
523944d93782SGreg Clayton                             }
524044d93782SGreg Clayton                         }
524144d93782SGreg Clayton                         else
524244d93782SGreg Clayton                         {
5243ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
524444d93782SGreg Clayton                         }
524544d93782SGreg Clayton                     }
524644d93782SGreg Clayton                     if (highlight_attr)
524744d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
524844d93782SGreg Clayton                 }
524944d93782SGreg Clayton             }
525044d93782SGreg Clayton         }
525144d93782SGreg Clayton         window.DeferredRefresh();
525244d93782SGreg Clayton         return true; // Drawing handled
525344d93782SGreg Clayton     }
525444d93782SGreg Clayton 
525544d93782SGreg Clayton     size_t
525644d93782SGreg Clayton     GetNumLines ()
525744d93782SGreg Clayton     {
525844d93782SGreg Clayton         size_t num_lines = GetNumSourceLines();
525944d93782SGreg Clayton         if (num_lines == 0)
526044d93782SGreg Clayton             num_lines = GetNumDisassemblyLines();
526144d93782SGreg Clayton         return num_lines;
526244d93782SGreg Clayton     }
526344d93782SGreg Clayton 
526444d93782SGreg Clayton     size_t
526544d93782SGreg Clayton     GetNumSourceLines () const
526644d93782SGreg Clayton     {
526744d93782SGreg Clayton         if (m_file_sp)
526844d93782SGreg Clayton             return m_file_sp->GetNumLines();
526944d93782SGreg Clayton         return 0;
527044d93782SGreg Clayton     }
527144d93782SGreg Clayton     size_t
527244d93782SGreg Clayton     GetNumDisassemblyLines () const
527344d93782SGreg Clayton     {
527444d93782SGreg Clayton         if (m_disassembly_sp)
527544d93782SGreg Clayton             return m_disassembly_sp->GetInstructionList().GetSize();
527644d93782SGreg Clayton         return 0;
527744d93782SGreg Clayton     }
527844d93782SGreg Clayton 
5279bd5ae6b4SGreg Clayton     HandleCharResult
5280bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
528144d93782SGreg Clayton     {
528244d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
528344d93782SGreg Clayton         const size_t num_lines = GetNumLines ();
528444d93782SGreg Clayton 
528544d93782SGreg Clayton         switch (c)
528644d93782SGreg Clayton         {
528744d93782SGreg Clayton             case ',':
528844d93782SGreg Clayton             case KEY_PPAGE:
528944d93782SGreg Clayton                 // Page up key
52903985c8c6SSaleem Abdulrasool                 if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
529144d93782SGreg Clayton                     m_first_visible_line -= num_visible_lines;
529244d93782SGreg Clayton                 else
529344d93782SGreg Clayton                     m_first_visible_line = 0;
529444d93782SGreg Clayton                 m_selected_line = m_first_visible_line;
529544d93782SGreg Clayton                 return eKeyHandled;
529644d93782SGreg Clayton 
529744d93782SGreg Clayton             case '.':
529844d93782SGreg Clayton             case KEY_NPAGE:
529944d93782SGreg Clayton                 // Page down key
530044d93782SGreg Clayton                 {
530144d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < num_lines)
530244d93782SGreg Clayton                         m_first_visible_line += num_visible_lines;
530344d93782SGreg Clayton                     else if (num_lines < num_visible_lines)
530444d93782SGreg Clayton                         m_first_visible_line = 0;
530544d93782SGreg Clayton                     else
530644d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
530744d93782SGreg Clayton                     m_selected_line = m_first_visible_line;
530844d93782SGreg Clayton                 }
530944d93782SGreg Clayton                 return eKeyHandled;
531044d93782SGreg Clayton 
531144d93782SGreg Clayton             case KEY_UP:
531244d93782SGreg Clayton                 if (m_selected_line > 0)
531344d93782SGreg Clayton                 {
531444d93782SGreg Clayton                     m_selected_line--;
53153985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
531644d93782SGreg Clayton                         m_first_visible_line = m_selected_line;
531744d93782SGreg Clayton                 }
531844d93782SGreg Clayton                 return eKeyHandled;
531944d93782SGreg Clayton 
532044d93782SGreg Clayton             case KEY_DOWN:
532144d93782SGreg Clayton                 if (m_selected_line + 1 < num_lines)
532244d93782SGreg Clayton                 {
532344d93782SGreg Clayton                     m_selected_line++;
532444d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < m_selected_line)
532544d93782SGreg Clayton                         m_first_visible_line++;
532644d93782SGreg Clayton                 }
532744d93782SGreg Clayton                 return eKeyHandled;
532844d93782SGreg Clayton 
532944d93782SGreg Clayton             case '\r':
533044d93782SGreg Clayton             case '\n':
533144d93782SGreg Clayton             case KEY_ENTER:
533244d93782SGreg Clayton                 // Set a breakpoint and run to the line using a one shot breakpoint
533344d93782SGreg Clayton                 if (GetNumSourceLines() > 0)
533444d93782SGreg Clayton                 {
533544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
533644d93782SGreg Clayton                     if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive())
533744d93782SGreg Clayton                     {
533844d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
533944d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
534044d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
534144d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
534244d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
534344d93782SGreg Clayton                                                                                       false,                     // internal
534444d93782SGreg Clayton                                                                                       false);                    // request_hardware
534544d93782SGreg Clayton                         // Make breakpoint one shot
534644d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
534744d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
534844d93782SGreg Clayton                     }
534944d93782SGreg Clayton                 }
535044d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
535144d93782SGreg Clayton                 {
535244d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
535344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
535444d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
535544d93782SGreg Clayton                     {
535644d93782SGreg Clayton                         Address addr = inst->GetAddress();
535744d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
535844d93782SGreg Clayton                                                                                       false,    // internal
535944d93782SGreg Clayton                                                                                       false);   // request_hardware
536044d93782SGreg Clayton                         // Make breakpoint one shot
536144d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
536244d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
536344d93782SGreg Clayton                     }
536444d93782SGreg Clayton                 }
536544d93782SGreg Clayton                 return eKeyHandled;
536644d93782SGreg Clayton 
536744d93782SGreg Clayton             case 'b':   // 'b' == toggle breakpoint on currently selected line
536844d93782SGreg Clayton                 if (m_selected_line < GetNumSourceLines())
536944d93782SGreg Clayton                 {
537044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
537144d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
537244d93782SGreg Clayton                     {
537344d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
537444d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
537544d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
537644d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
537744d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
537844d93782SGreg Clayton                                                                                       false,                     // internal
537944d93782SGreg Clayton                                                                                       false);                    // request_hardware
538044d93782SGreg Clayton                     }
538144d93782SGreg Clayton                 }
538244d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
538344d93782SGreg Clayton                 {
538444d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
538544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
538644d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
538744d93782SGreg Clayton                     {
538844d93782SGreg Clayton                         Address addr = inst->GetAddress();
538944d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
539044d93782SGreg Clayton                                                                                       false,    // internal
539144d93782SGreg Clayton                                                                                       false);   // request_hardware
539244d93782SGreg Clayton                     }
539344d93782SGreg Clayton                 }
539444d93782SGreg Clayton                 return eKeyHandled;
539544d93782SGreg Clayton 
539644d93782SGreg Clayton             case 'd':   // 'd' == detach and let run
539744d93782SGreg Clayton             case 'D':   // 'D' == detach and keep stopped
539844d93782SGreg Clayton                 {
539944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
540044d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
540144d93782SGreg Clayton                         exe_ctx.GetProcessRef().Detach(c == 'D');
540244d93782SGreg Clayton                 }
540344d93782SGreg Clayton                 return eKeyHandled;
540444d93782SGreg Clayton 
540544d93782SGreg Clayton             case 'k':
540644d93782SGreg Clayton                 // 'k' == kill
540744d93782SGreg Clayton                 {
540844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
540944d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
541044d93782SGreg Clayton                         exe_ctx.GetProcessRef().Destroy();
541144d93782SGreg Clayton                 }
541244d93782SGreg Clayton                 return eKeyHandled;
541344d93782SGreg Clayton 
541444d93782SGreg Clayton             case 'c':
541544d93782SGreg Clayton                 // 'c' == continue
541644d93782SGreg Clayton                 {
541744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
541844d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
541944d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
542044d93782SGreg Clayton                 }
542144d93782SGreg Clayton                 return eKeyHandled;
542244d93782SGreg Clayton 
542344d93782SGreg Clayton             case 'o':
542444d93782SGreg Clayton                 // 'o' == step out
542544d93782SGreg Clayton                 {
542644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
542744d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
542844d93782SGreg Clayton                     {
542944d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOut();
543044d93782SGreg Clayton                     }
543144d93782SGreg Clayton                 }
543244d93782SGreg Clayton                 return eKeyHandled;
543344d93782SGreg Clayton             case 'n':   // 'n' == step over
543444d93782SGreg Clayton             case 'N':   // 'N' == step over instruction
543544d93782SGreg Clayton                 {
543644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
543744d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
543844d93782SGreg Clayton                     {
543944d93782SGreg Clayton                         bool source_step = (c == 'n');
544044d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOver(source_step);
544144d93782SGreg Clayton                     }
544244d93782SGreg Clayton                 }
544344d93782SGreg Clayton                 return eKeyHandled;
544444d93782SGreg Clayton             case 's':   // 's' == step into
544544d93782SGreg Clayton             case 'S':   // 'S' == step into instruction
544644d93782SGreg Clayton                 {
544744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
544844d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
544944d93782SGreg Clayton                     {
545044d93782SGreg Clayton                         bool source_step = (c == 's');
54514b4b2478SJim Ingham                         exe_ctx.GetThreadRef().StepIn(source_step);
545244d93782SGreg Clayton                     }
545344d93782SGreg Clayton                 }
545444d93782SGreg Clayton                 return eKeyHandled;
545544d93782SGreg Clayton 
545644d93782SGreg Clayton             case 'h':
545744d93782SGreg Clayton                 window.CreateHelpSubwindow ();
545844d93782SGreg Clayton                 return eKeyHandled;
545944d93782SGreg Clayton 
546044d93782SGreg Clayton             default:
546144d93782SGreg Clayton                 break;
546244d93782SGreg Clayton         }
546344d93782SGreg Clayton         return eKeyNotHandled;
546444d93782SGreg Clayton     }
546544d93782SGreg Clayton 
546644d93782SGreg Clayton protected:
546744d93782SGreg Clayton     typedef std::set<uint32_t> BreakpointLines;
546844d93782SGreg Clayton     typedef std::set<lldb::addr_t> BreakpointAddrs;
546944d93782SGreg Clayton 
547044d93782SGreg Clayton     Debugger &m_debugger;
547144d93782SGreg Clayton     SymbolContext m_sc;
547244d93782SGreg Clayton     SourceManager::FileSP m_file_sp;
547344d93782SGreg Clayton     SymbolContextScope *m_disassembly_scope;
547444d93782SGreg Clayton     lldb::DisassemblerSP m_disassembly_sp;
547544d93782SGreg Clayton     AddressRange m_disassembly_range;
5476ec990867SGreg Clayton     StreamString m_title;
547744d93782SGreg Clayton     lldb::user_id_t m_tid;
547844d93782SGreg Clayton     char m_line_format[8];
547944d93782SGreg Clayton     int m_line_width;
548044d93782SGreg Clayton     uint32_t m_selected_line;       // The selected line
548144d93782SGreg Clayton     uint32_t m_pc_line;             // The line with the PC
548244d93782SGreg Clayton     uint32_t m_stop_id;
548344d93782SGreg Clayton     uint32_t m_frame_idx;
548444d93782SGreg Clayton     int m_first_visible_line;
548544d93782SGreg Clayton     int m_min_x;
548644d93782SGreg Clayton     int m_min_y;
548744d93782SGreg Clayton     int m_max_x;
548844d93782SGreg Clayton     int m_max_y;
548944d93782SGreg Clayton 
549044d93782SGreg Clayton };
549144d93782SGreg Clayton 
549244d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true };
549344d93782SGreg Clayton 
549444d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) :
5495e30f11d9SKate Stone     IOHandler (debugger, IOHandler::Type::Curses)
549644d93782SGreg Clayton {
549744d93782SGreg Clayton }
549844d93782SGreg Clayton 
549944d93782SGreg Clayton void
550044d93782SGreg Clayton IOHandlerCursesGUI::Activate ()
550144d93782SGreg Clayton {
550244d93782SGreg Clayton     IOHandler::Activate();
550344d93782SGreg Clayton     if (!m_app_ap)
550444d93782SGreg Clayton     {
550544d93782SGreg Clayton         m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE()));
550644d93782SGreg Clayton 
550744d93782SGreg Clayton 
550844d93782SGreg Clayton         // This is both a window and a menu delegate
550944d93782SGreg Clayton         std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger));
551044d93782SGreg Clayton 
551144d93782SGreg Clayton         MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
551244d93782SGreg Clayton         MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
551344d93782SGreg Clayton         MenuSP exit_menuitem_sp(new Menu("Exit", NULL, 'x', ApplicationDelegate::eMenuID_LLDBExit));
551444d93782SGreg Clayton         exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
551544d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", NULL, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
551644d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
551744d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (exit_menuitem_sp);
551844d93782SGreg Clayton 
551944d93782SGreg Clayton         MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target));
552044d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", NULL, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
552144d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", NULL, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
552244d93782SGreg Clayton 
552344d93782SGreg Clayton         MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process));
552444d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach"  , NULL, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
552544d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach"  , NULL, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
552644d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch"  , NULL, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
552744d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
552844d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", NULL, 'c', ApplicationDelegate::eMenuID_ProcessContinue)));
552944d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt"    , NULL, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
553044d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill"    , NULL, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
553144d93782SGreg Clayton 
553244d93782SGreg Clayton         MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread));
553344d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In"  , NULL, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
553444d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", NULL, 'v', ApplicationDelegate::eMenuID_ThreadStepOver)));
553544d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , NULL, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
553644d93782SGreg Clayton 
553744d93782SGreg Clayton         MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
553844d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", NULL, 'b', ApplicationDelegate::eMenuID_ViewBacktrace)));
553944d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", NULL, 'r', ApplicationDelegate::eMenuID_ViewRegisters)));
554044d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Source"   , NULL, 's', ApplicationDelegate::eMenuID_ViewSource)));
554144d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", NULL, 'v', ApplicationDelegate::eMenuID_ViewVariables)));
554244d93782SGreg Clayton 
554344d93782SGreg Clayton         MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
554444d93782SGreg Clayton         help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", NULL, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
554544d93782SGreg Clayton 
554644d93782SGreg Clayton         m_app_ap->Initialize();
554744d93782SGreg Clayton         WindowSP &main_window_sp = m_app_ap->GetMainWindow();
554844d93782SGreg Clayton 
554944d93782SGreg Clayton         MenuSP menubar_sp(new Menu(Menu::Type::Bar));
555044d93782SGreg Clayton         menubar_sp->AddSubmenu (lldb_menu_sp);
555144d93782SGreg Clayton         menubar_sp->AddSubmenu (target_menu_sp);
555244d93782SGreg Clayton         menubar_sp->AddSubmenu (process_menu_sp);
555344d93782SGreg Clayton         menubar_sp->AddSubmenu (thread_menu_sp);
555444d93782SGreg Clayton         menubar_sp->AddSubmenu (view_menu_sp);
555544d93782SGreg Clayton         menubar_sp->AddSubmenu (help_menu_sp);
555644d93782SGreg Clayton         menubar_sp->SetDelegate(app_menu_delegate_sp);
555744d93782SGreg Clayton 
555844d93782SGreg Clayton         Rect content_bounds = main_window_sp->GetFrame();
555944d93782SGreg Clayton         Rect menubar_bounds = content_bounds.MakeMenuBar();
556044d93782SGreg Clayton         Rect status_bounds = content_bounds.MakeStatusBar();
556144d93782SGreg Clayton         Rect source_bounds;
556244d93782SGreg Clayton         Rect variables_bounds;
556344d93782SGreg Clayton         Rect threads_bounds;
556444d93782SGreg Clayton         Rect source_variables_bounds;
556544d93782SGreg Clayton         content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds);
556644d93782SGreg Clayton         source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds);
556744d93782SGreg Clayton 
556844d93782SGreg Clayton         WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
556944d93782SGreg Clayton         // Let the menubar get keys if the active window doesn't handle the
557044d93782SGreg Clayton         // keys that are typed so it can respond to menubar key presses.
557144d93782SGreg Clayton         menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window
557244d93782SGreg Clayton         menubar_window_sp->SetDelegate(menubar_sp);
557344d93782SGreg Clayton 
557444d93782SGreg Clayton         WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source",
557544d93782SGreg Clayton                                                                    source_bounds,
557644d93782SGreg Clayton                                                                    true));
557744d93782SGreg Clayton         WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables",
557844d93782SGreg Clayton                                                                       variables_bounds,
557944d93782SGreg Clayton                                                                       false));
558044d93782SGreg Clayton         WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads",
558144d93782SGreg Clayton                                                                       threads_bounds,
558244d93782SGreg Clayton                                                                       false));
558344d93782SGreg Clayton         WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status",
558444d93782SGreg Clayton                                                                    status_bounds,
558544d93782SGreg Clayton                                                                    false));
558644d93782SGreg Clayton         status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window
558744d93782SGreg Clayton         main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
558844d93782SGreg Clayton         source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
558944d93782SGreg Clayton         variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
5590ec990867SGreg Clayton         TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger));
559144d93782SGreg Clayton         threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
559244d93782SGreg Clayton         status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
559344d93782SGreg Clayton 
55945fdb09bbSGreg Clayton         // Show the main help window once the first time the curses GUI is launched
55955fdb09bbSGreg Clayton         static bool g_showed_help = false;
55965fdb09bbSGreg Clayton         if (!g_showed_help)
55975fdb09bbSGreg Clayton         {
55985fdb09bbSGreg Clayton             g_showed_help = true;
55995fdb09bbSGreg Clayton             main_window_sp->CreateHelpSubwindow();
56005fdb09bbSGreg Clayton         }
56015fdb09bbSGreg Clayton 
560244d93782SGreg Clayton         init_pair (1, COLOR_WHITE   , COLOR_BLUE  );
560344d93782SGreg Clayton         init_pair (2, COLOR_BLACK   , COLOR_WHITE );
560444d93782SGreg Clayton         init_pair (3, COLOR_MAGENTA , COLOR_WHITE );
560544d93782SGreg Clayton         init_pair (4, COLOR_MAGENTA , COLOR_BLACK );
560644d93782SGreg Clayton         init_pair (5, COLOR_RED     , COLOR_BLACK );
560744d93782SGreg Clayton 
560844d93782SGreg Clayton     }
560944d93782SGreg Clayton }
561044d93782SGreg Clayton 
561144d93782SGreg Clayton void
561244d93782SGreg Clayton IOHandlerCursesGUI::Deactivate ()
561344d93782SGreg Clayton {
561444d93782SGreg Clayton     m_app_ap->Terminate();
561544d93782SGreg Clayton }
561644d93782SGreg Clayton 
561744d93782SGreg Clayton void
561844d93782SGreg Clayton IOHandlerCursesGUI::Run ()
561944d93782SGreg Clayton {
562044d93782SGreg Clayton     m_app_ap->Run(m_debugger);
562144d93782SGreg Clayton     SetIsDone(true);
562244d93782SGreg Clayton }
562344d93782SGreg Clayton 
562444d93782SGreg Clayton 
562544d93782SGreg Clayton IOHandlerCursesGUI::~IOHandlerCursesGUI ()
562644d93782SGreg Clayton {
562744d93782SGreg Clayton 
562844d93782SGreg Clayton }
562944d93782SGreg Clayton 
563044d93782SGreg Clayton void
563144d93782SGreg Clayton IOHandlerCursesGUI::Hide ()
563244d93782SGreg Clayton {
563344d93782SGreg Clayton }
563444d93782SGreg Clayton 
563544d93782SGreg Clayton 
563644d93782SGreg Clayton void
563744d93782SGreg Clayton IOHandlerCursesGUI::Refresh ()
563844d93782SGreg Clayton {
563944d93782SGreg Clayton }
564044d93782SGreg Clayton 
5641e68f5d6bSGreg Clayton void
5642e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel ()
5643e68f5d6bSGreg Clayton {
5644e68f5d6bSGreg Clayton }
564544d93782SGreg Clayton 
5646f0066ad0SGreg Clayton bool
564744d93782SGreg Clayton IOHandlerCursesGUI::Interrupt ()
564844d93782SGreg Clayton {
5649f0066ad0SGreg Clayton     return false;
565044d93782SGreg Clayton }
565144d93782SGreg Clayton 
565244d93782SGreg Clayton 
565344d93782SGreg Clayton void
565444d93782SGreg Clayton IOHandlerCursesGUI::GotEOF()
565544d93782SGreg Clayton {
565644d93782SGreg Clayton }
565744d93782SGreg Clayton 
5658914b8d98SDeepak Panickal #endif // #ifndef LLDB_DISABLE_CURSES
5659