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),
383*e034a04eSGreg Clayton     m_interrupt_exits (true),
384*e034a04eSGreg Clayton     m_editing (false)
38544d93782SGreg Clayton {
38644d93782SGreg Clayton     SetPrompt(prompt);
38744d93782SGreg Clayton 
388cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
389914b8d98SDeepak Panickal     bool use_editline = false;
390340b0309SGreg Clayton 
391340b0309SGreg Clayton     use_editline = m_input_sp->GetFile().GetIsRealTerminal();
39244d93782SGreg Clayton 
39344d93782SGreg Clayton     if (use_editline)
39444d93782SGreg Clayton     {
39544d93782SGreg Clayton         m_editline_ap.reset(new Editline (editline_name,
39644d93782SGreg Clayton                                           GetInputFILE (),
39744d93782SGreg Clayton                                           GetOutputFILE (),
398e30f11d9SKate Stone                                           GetErrorFILE (),
399e30f11d9SKate Stone                                           m_color_prompts));
400e30f11d9SKate Stone         m_editline_ap->SetIsInputCompleteCallback (IsInputCompleteCallback, this);
40144d93782SGreg Clayton         m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this);
402e30f11d9SKate Stone         // See if the delegate supports fixing indentation
403e30f11d9SKate Stone         const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
404e30f11d9SKate Stone         if (indent_chars)
405e30f11d9SKate Stone         {
406e30f11d9SKate Stone             // The delegate does support indentation, hook it up so when any indentation
407e30f11d9SKate Stone             // character is typed, the delegate gets a chance to fix it
408e30f11d9SKate Stone             m_editline_ap->SetFixIndentationCallback (FixIndentationCallback, this, indent_chars);
409e30f11d9SKate Stone         }
41044d93782SGreg Clayton     }
411cacde7dfSTodd Fiala #endif
412e30f11d9SKate Stone     SetBaseLineNumber (m_base_line_number);
413e30f11d9SKate Stone     SetPrompt(prompt ? prompt : "");
414e30f11d9SKate Stone     SetContinuationPrompt(continuation_prompt);
41544d93782SGreg Clayton }
41644d93782SGreg Clayton 
41744d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline ()
41844d93782SGreg Clayton {
419cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
42044d93782SGreg Clayton     m_editline_ap.reset();
421cacde7dfSTodd Fiala #endif
42244d93782SGreg Clayton }
42344d93782SGreg Clayton 
424e30f11d9SKate Stone void
425e30f11d9SKate Stone IOHandlerEditline::Activate ()
426e30f11d9SKate Stone {
427e30f11d9SKate Stone     IOHandler::Activate();
428e30f11d9SKate Stone     m_delegate.IOHandlerActivated(*this);
429e30f11d9SKate Stone }
430e30f11d9SKate Stone 
431e30f11d9SKate Stone void
432e30f11d9SKate Stone IOHandlerEditline::Deactivate ()
433e30f11d9SKate Stone {
434e30f11d9SKate Stone     IOHandler::Deactivate();
435e30f11d9SKate Stone     m_delegate.IOHandlerDeactivated(*this);
436e30f11d9SKate Stone }
437e30f11d9SKate Stone 
43844d93782SGreg Clayton 
43944d93782SGreg Clayton bool
440f0066ad0SGreg Clayton IOHandlerEditline::GetLine (std::string &line, bool &interrupted)
44144d93782SGreg Clayton {
442cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
44344d93782SGreg Clayton     if (m_editline_ap)
44444d93782SGreg Clayton     {
445e30f11d9SKate Stone         return m_editline_ap->GetLine (line, interrupted);
44644d93782SGreg Clayton     }
44744d93782SGreg Clayton     else
44844d93782SGreg Clayton     {
449cacde7dfSTodd Fiala #endif
45044d93782SGreg Clayton         line.clear();
45144d93782SGreg Clayton 
45244d93782SGreg Clayton         FILE *in = GetInputFILE();
45344d93782SGreg Clayton         if (in)
45444d93782SGreg Clayton         {
455340b0309SGreg Clayton             if (GetIsInteractive())
45644d93782SGreg Clayton             {
457e30f11d9SKate Stone                 const char *prompt = NULL;
458e30f11d9SKate Stone 
459e30f11d9SKate Stone                 if (m_multi_line && m_curr_line_idx > 0)
460e30f11d9SKate Stone                     prompt = GetContinuationPrompt();
461e30f11d9SKate Stone 
462e30f11d9SKate Stone                 if (prompt == NULL)
463e30f11d9SKate Stone                     prompt = GetPrompt();
464e30f11d9SKate Stone 
46544d93782SGreg Clayton                 if (prompt && prompt[0])
46644d93782SGreg Clayton                 {
46744d93782SGreg Clayton                     FILE *out = GetOutputFILE();
46844d93782SGreg Clayton                     if (out)
46944d93782SGreg Clayton                     {
47044d93782SGreg Clayton                         ::fprintf(out, "%s", prompt);
47144d93782SGreg Clayton                         ::fflush(out);
47244d93782SGreg Clayton                     }
47344d93782SGreg Clayton                 }
47444d93782SGreg Clayton             }
47544d93782SGreg Clayton             char buffer[256];
47644d93782SGreg Clayton             bool done = false;
4770f86e6e7SGreg Clayton             bool got_line = false;
478*e034a04eSGreg Clayton             m_editing = true;
47944d93782SGreg Clayton             while (!done)
48044d93782SGreg Clayton             {
48144d93782SGreg Clayton                 if (fgets(buffer, sizeof(buffer), in) == NULL)
482c9cf5798SGreg Clayton                 {
483c7797accSGreg Clayton                     const int saved_errno = errno;
484c9cf5798SGreg Clayton                     if (feof(in))
48544d93782SGreg Clayton                         done = true;
486c7797accSGreg Clayton                     else if (ferror(in))
487c7797accSGreg Clayton                     {
488c7797accSGreg Clayton                         if (saved_errno != EINTR)
489c7797accSGreg Clayton                             done = true;
490c7797accSGreg Clayton                     }
491c9cf5798SGreg Clayton                 }
49244d93782SGreg Clayton                 else
49344d93782SGreg Clayton                 {
4940f86e6e7SGreg Clayton                     got_line = true;
49544d93782SGreg Clayton                     size_t buffer_len = strlen(buffer);
49644d93782SGreg Clayton                     assert (buffer[buffer_len] == '\0');
49744d93782SGreg Clayton                     char last_char = buffer[buffer_len-1];
49844d93782SGreg Clayton                     if (last_char == '\r' || last_char == '\n')
49944d93782SGreg Clayton                     {
50044d93782SGreg Clayton                         done = true;
50144d93782SGreg Clayton                         // Strip trailing newlines
50244d93782SGreg Clayton                         while (last_char == '\r' || last_char == '\n')
50344d93782SGreg Clayton                         {
50444d93782SGreg Clayton                             --buffer_len;
50544d93782SGreg Clayton                             if (buffer_len == 0)
50644d93782SGreg Clayton                                 break;
50744d93782SGreg Clayton                             last_char = buffer[buffer_len-1];
50844d93782SGreg Clayton                         }
50944d93782SGreg Clayton                     }
51044d93782SGreg Clayton                     line.append(buffer, buffer_len);
51144d93782SGreg Clayton                 }
51244d93782SGreg Clayton             }
513*e034a04eSGreg Clayton             m_editing = false;
5140f86e6e7SGreg Clayton             // We might have gotten a newline on a line by itself
5150f86e6e7SGreg Clayton             // make sure to return true in this case.
5160f86e6e7SGreg Clayton             return got_line;
51744d93782SGreg Clayton         }
51844d93782SGreg Clayton         else
51944d93782SGreg Clayton         {
52044d93782SGreg Clayton             // No more input file, we are done...
52144d93782SGreg Clayton             SetIsDone(true);
52244d93782SGreg Clayton         }
523340b0309SGreg Clayton         return false;
524cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
52544d93782SGreg Clayton     }
526cacde7dfSTodd Fiala #endif
52744d93782SGreg Clayton }
52844d93782SGreg Clayton 
52944d93782SGreg Clayton 
530cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
531e30f11d9SKate Stone bool
532e30f11d9SKate Stone IOHandlerEditline::IsInputCompleteCallback (Editline *editline,
53344d93782SGreg Clayton                                           StringList &lines,
53444d93782SGreg Clayton                                           void *baton)
53544d93782SGreg Clayton {
53644d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
537e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader, lines);
538e30f11d9SKate Stone }
539e30f11d9SKate Stone 
540e30f11d9SKate Stone int
541e30f11d9SKate Stone IOHandlerEditline::FixIndentationCallback (Editline *editline,
542e30f11d9SKate Stone                                            const StringList &lines,
543e30f11d9SKate Stone                                            int cursor_position,
544e30f11d9SKate Stone                                            void *baton)
545e30f11d9SKate Stone {
546e30f11d9SKate Stone     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
547e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerFixIndentation(*editline_reader, lines, cursor_position);
54844d93782SGreg Clayton }
54944d93782SGreg Clayton 
55044d93782SGreg Clayton int
55144d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line,
55244d93782SGreg Clayton                                          const char *cursor,
55344d93782SGreg Clayton                                          const char *last_char,
55444d93782SGreg Clayton                                          int skip_first_n_matches,
55544d93782SGreg Clayton                                          int max_matches,
55644d93782SGreg Clayton                                          StringList &matches,
55744d93782SGreg Clayton                                          void *baton)
55844d93782SGreg Clayton {
55944d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
56044d93782SGreg Clayton     if (editline_reader)
56144d93782SGreg Clayton         return editline_reader->m_delegate.IOHandlerComplete (*editline_reader,
56244d93782SGreg Clayton                                                               current_line,
56344d93782SGreg Clayton                                                               cursor,
56444d93782SGreg Clayton                                                               last_char,
56544d93782SGreg Clayton                                                               skip_first_n_matches,
56644d93782SGreg Clayton                                                               max_matches,
56744d93782SGreg Clayton                                                               matches);
56844d93782SGreg Clayton     return 0;
56944d93782SGreg Clayton }
570cacde7dfSTodd Fiala #endif
57144d93782SGreg Clayton 
57244d93782SGreg Clayton const char *
57344d93782SGreg Clayton IOHandlerEditline::GetPrompt ()
57444d93782SGreg Clayton {
575cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
57644d93782SGreg Clayton     if (m_editline_ap)
577cacde7dfSTodd Fiala     {
57844d93782SGreg Clayton         return m_editline_ap->GetPrompt ();
579cacde7dfSTodd Fiala     }
580cacde7dfSTodd Fiala     else
581cacde7dfSTodd Fiala     {
582cacde7dfSTodd Fiala #endif
583cacde7dfSTodd Fiala         if (m_prompt.empty())
58444d93782SGreg Clayton             return NULL;
585cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
586cacde7dfSTodd Fiala     }
587cacde7dfSTodd Fiala #endif
58844d93782SGreg Clayton     return m_prompt.c_str();
58944d93782SGreg Clayton }
59044d93782SGreg Clayton 
59144d93782SGreg Clayton bool
59244d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p)
59344d93782SGreg Clayton {
59444d93782SGreg Clayton     if (p && p[0])
59544d93782SGreg Clayton         m_prompt = p;
59644d93782SGreg Clayton     else
59744d93782SGreg Clayton         m_prompt.clear();
598cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
59944d93782SGreg Clayton     if (m_editline_ap)
60044d93782SGreg Clayton         m_editline_ap->SetPrompt (m_prompt.empty() ? NULL : m_prompt.c_str());
601cacde7dfSTodd Fiala #endif
60244d93782SGreg Clayton     return true;
60344d93782SGreg Clayton }
60444d93782SGreg Clayton 
605e30f11d9SKate Stone const char *
606e30f11d9SKate Stone IOHandlerEditline::GetContinuationPrompt ()
607e30f11d9SKate Stone {
608e30f11d9SKate Stone     if (m_continuation_prompt.empty())
609e30f11d9SKate Stone         return NULL;
610e30f11d9SKate Stone     return m_continuation_prompt.c_str();
611e30f11d9SKate Stone }
612e30f11d9SKate Stone 
613e30f11d9SKate Stone 
614e30f11d9SKate Stone void
615e30f11d9SKate Stone IOHandlerEditline::SetContinuationPrompt (const char *p)
616e30f11d9SKate Stone {
617e30f11d9SKate Stone     if (p && p[0])
618e30f11d9SKate Stone         m_continuation_prompt = p;
619e30f11d9SKate Stone     else
620e30f11d9SKate Stone         m_continuation_prompt.clear();
621e30f11d9SKate Stone 
622d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
623e30f11d9SKate Stone     if (m_editline_ap)
624e30f11d9SKate Stone         m_editline_ap->SetContinuationPrompt (m_continuation_prompt.empty() ? NULL : m_continuation_prompt.c_str());
625d553d00cSZachary Turner #endif
626e30f11d9SKate Stone }
627e30f11d9SKate Stone 
628e30f11d9SKate Stone 
629f6913cd7SGreg Clayton void
630f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line)
631f6913cd7SGreg Clayton {
632f6913cd7SGreg Clayton     m_base_line_number = line;
633f6913cd7SGreg Clayton }
634e30f11d9SKate Stone 
635e30f11d9SKate Stone uint32_t
636e30f11d9SKate Stone IOHandlerEditline::GetCurrentLineIndex () const
637e30f11d9SKate Stone {
638d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
639e30f11d9SKate Stone     if (m_editline_ap)
640e30f11d9SKate Stone         return m_editline_ap->GetCurrentLine();
641e30f11d9SKate Stone #endif
642e30f11d9SKate Stone     return m_curr_line_idx;
643e30f11d9SKate Stone }
644e30f11d9SKate Stone 
64544d93782SGreg Clayton bool
646f0066ad0SGreg Clayton IOHandlerEditline::GetLines (StringList &lines, bool &interrupted)
64744d93782SGreg Clayton {
648e30f11d9SKate Stone     m_current_lines_ptr = &lines;
649e30f11d9SKate Stone 
65044d93782SGreg Clayton     bool success = false;
651cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
65244d93782SGreg Clayton     if (m_editline_ap)
65344d93782SGreg Clayton     {
654e30f11d9SKate Stone         return m_editline_ap->GetLines (m_base_line_number, lines, interrupted);
65544d93782SGreg Clayton     }
65644d93782SGreg Clayton     else
65744d93782SGreg Clayton     {
658cacde7dfSTodd Fiala #endif
659e30f11d9SKate Stone         bool done = false;
660c3d874a5SGreg Clayton         Error error;
66144d93782SGreg Clayton 
662e30f11d9SKate Stone         while (!done)
66344d93782SGreg Clayton         {
664f6913cd7SGreg Clayton             // Show line numbers if we are asked to
66544d93782SGreg Clayton             std::string line;
666f6913cd7SGreg Clayton             if (m_base_line_number > 0 && GetIsInteractive())
667f6913cd7SGreg Clayton             {
668f6913cd7SGreg Clayton                 FILE *out = GetOutputFILE();
669f6913cd7SGreg Clayton                 if (out)
670bc88d938SGreg Clayton                     ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(), GetPrompt() == NULL ? " " : "");
671f6913cd7SGreg Clayton             }
672f6913cd7SGreg Clayton 
673e30f11d9SKate Stone             m_curr_line_idx = lines.GetSize();
674e30f11d9SKate Stone 
675f0066ad0SGreg Clayton             bool interrupted = false;
676e30f11d9SKate Stone             if (GetLine(line, interrupted) && !interrupted)
67744d93782SGreg Clayton             {
67844d93782SGreg Clayton                 lines.AppendString(line);
679e30f11d9SKate Stone                 done = m_delegate.IOHandlerIsInputComplete(*this, lines);
680f0066ad0SGreg Clayton             }
68144d93782SGreg Clayton             else
68244d93782SGreg Clayton             {
683e30f11d9SKate Stone                 done = true;
68444d93782SGreg Clayton             }
68544d93782SGreg Clayton         }
68644d93782SGreg Clayton         success = lines.GetSize() > 0;
687cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
68844d93782SGreg Clayton     }
689cacde7dfSTodd Fiala #endif
69044d93782SGreg Clayton     return success;
69144d93782SGreg Clayton }
69244d93782SGreg Clayton 
69344d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data
69444d93782SGreg Clayton // from the "in" and place output into "out" and "err and return
69544d93782SGreg Clayton // when done.
69644d93782SGreg Clayton void
69744d93782SGreg Clayton IOHandlerEditline::Run ()
69844d93782SGreg Clayton {
69944d93782SGreg Clayton     std::string line;
70044d93782SGreg Clayton     while (IsActive())
70144d93782SGreg Clayton     {
702f0066ad0SGreg Clayton         bool interrupted = false;
70344d93782SGreg Clayton         if (m_multi_line)
70444d93782SGreg Clayton         {
70544d93782SGreg Clayton             StringList lines;
706f0066ad0SGreg Clayton             if (GetLines (lines, interrupted))
707f0066ad0SGreg Clayton             {
708f0066ad0SGreg Clayton                 if (interrupted)
709f0066ad0SGreg Clayton                 {
710e30f11d9SKate Stone                     m_done = m_interrupt_exits;
711e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
712e30f11d9SKate Stone 
713f0066ad0SGreg Clayton                 }
714f0066ad0SGreg Clayton                 else
71544d93782SGreg Clayton                 {
71644d93782SGreg Clayton                     line = lines.CopyList();
71744d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
71844d93782SGreg Clayton                 }
719f0066ad0SGreg Clayton             }
72044d93782SGreg Clayton             else
72144d93782SGreg Clayton             {
72244d93782SGreg Clayton                 m_done = true;
72344d93782SGreg Clayton             }
72444d93782SGreg Clayton         }
72544d93782SGreg Clayton         else
72644d93782SGreg Clayton         {
727f0066ad0SGreg Clayton             if (GetLine(line, interrupted))
72844d93782SGreg Clayton             {
729e30f11d9SKate Stone                 if (interrupted)
730e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
731e30f11d9SKate Stone                 else
73244d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
73344d93782SGreg Clayton             }
73444d93782SGreg Clayton             else
73544d93782SGreg Clayton             {
73644d93782SGreg Clayton                 m_done = true;
73744d93782SGreg Clayton             }
73844d93782SGreg Clayton         }
73944d93782SGreg Clayton     }
74044d93782SGreg Clayton }
74144d93782SGreg Clayton 
74244d93782SGreg Clayton void
74344d93782SGreg Clayton IOHandlerEditline::Hide ()
74444d93782SGreg Clayton {
745cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
746b89b7496SGreg Clayton     if (m_editline_ap)
74744d93782SGreg Clayton         m_editline_ap->Hide();
748cacde7dfSTodd Fiala #endif
74944d93782SGreg Clayton }
75044d93782SGreg Clayton 
75144d93782SGreg Clayton 
75244d93782SGreg Clayton void
75344d93782SGreg Clayton IOHandlerEditline::Refresh ()
75444d93782SGreg Clayton {
755cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
756b89b7496SGreg Clayton     if (m_editline_ap)
757b89b7496SGreg Clayton     {
75844d93782SGreg Clayton         m_editline_ap->Refresh();
759b89b7496SGreg Clayton     }
760*e034a04eSGreg Clayton     else if (m_editing)
76144d93782SGreg Clayton     {
762cacde7dfSTodd Fiala #endif
76344d93782SGreg Clayton         const char *prompt = GetPrompt();
76444d93782SGreg Clayton         if (prompt && prompt[0])
76544d93782SGreg Clayton         {
76644d93782SGreg Clayton             FILE *out = GetOutputFILE();
76744d93782SGreg Clayton             if (out)
76844d93782SGreg Clayton             {
76944d93782SGreg Clayton                 ::fprintf(out, "%s", prompt);
77044d93782SGreg Clayton                 ::fflush(out);
77144d93782SGreg Clayton             }
77244d93782SGreg Clayton         }
773cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
77444d93782SGreg Clayton     }
775cacde7dfSTodd Fiala #endif
77644d93782SGreg Clayton }
77744d93782SGreg Clayton 
77844d93782SGreg Clayton void
779e68f5d6bSGreg Clayton IOHandlerEditline::Cancel ()
780e68f5d6bSGreg Clayton {
781cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
782e68f5d6bSGreg Clayton     if (m_editline_ap)
783e68f5d6bSGreg Clayton         m_editline_ap->Interrupt ();
784cacde7dfSTodd Fiala #endif
785e68f5d6bSGreg Clayton }
786e68f5d6bSGreg Clayton 
787f0066ad0SGreg Clayton bool
78844d93782SGreg Clayton IOHandlerEditline::Interrupt ()
78944d93782SGreg Clayton {
790f0066ad0SGreg Clayton     // Let the delgate handle it first
791f0066ad0SGreg Clayton     if (m_delegate.IOHandlerInterrupt(*this))
792f0066ad0SGreg Clayton         return true;
793f0066ad0SGreg Clayton 
794cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
79544d93782SGreg Clayton     if (m_editline_ap)
796f0066ad0SGreg Clayton         return m_editline_ap->Interrupt();
797cacde7dfSTodd Fiala #endif
798f0066ad0SGreg Clayton     return false;
79944d93782SGreg Clayton }
80044d93782SGreg Clayton 
80144d93782SGreg Clayton void
80244d93782SGreg Clayton IOHandlerEditline::GotEOF()
80344d93782SGreg Clayton {
804cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
80544d93782SGreg Clayton     if (m_editline_ap)
80644d93782SGreg Clayton         m_editline_ap->Interrupt();
807cacde7dfSTodd Fiala #endif
80844d93782SGreg Clayton }
80944d93782SGreg Clayton 
810914b8d98SDeepak Panickal // we may want curses to be disabled for some builds
811914b8d98SDeepak Panickal // for instance, windows
812914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
813914b8d98SDeepak Panickal 
81444d93782SGreg Clayton #include "lldb/Core/ValueObject.h"
81544d93782SGreg Clayton #include "lldb/Symbol/VariableList.h"
81644d93782SGreg Clayton #include "lldb/Target/Target.h"
81744d93782SGreg Clayton #include "lldb/Target/Process.h"
81844d93782SGreg Clayton #include "lldb/Target/Thread.h"
81944d93782SGreg Clayton #include "lldb/Target/StackFrame.h"
82044d93782SGreg Clayton 
82144d93782SGreg Clayton #define KEY_RETURN   10
82244d93782SGreg Clayton #define KEY_ESCAPE  27
82344d93782SGreg Clayton 
82444d93782SGreg Clayton namespace curses
82544d93782SGreg Clayton {
82644d93782SGreg Clayton     class Menu;
82744d93782SGreg Clayton     class MenuDelegate;
82844d93782SGreg Clayton     class Window;
82944d93782SGreg Clayton     class WindowDelegate;
83044d93782SGreg Clayton     typedef std::shared_ptr<Menu> MenuSP;
83144d93782SGreg Clayton     typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
83244d93782SGreg Clayton     typedef std::shared_ptr<Window> WindowSP;
83344d93782SGreg Clayton     typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
83444d93782SGreg Clayton     typedef std::vector<MenuSP> Menus;
83544d93782SGreg Clayton     typedef std::vector<WindowSP> Windows;
83644d93782SGreg Clayton     typedef std::vector<WindowDelegateSP> WindowDelegates;
83744d93782SGreg Clayton 
83844d93782SGreg Clayton #if 0
83944d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
84044d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
84144d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
84244d93782SGreg Clayton #endif
84344d93782SGreg Clayton     struct Point
84444d93782SGreg Clayton     {
84544d93782SGreg Clayton         int x;
84644d93782SGreg Clayton         int y;
84744d93782SGreg Clayton 
84844d93782SGreg Clayton         Point (int _x = 0, int _y = 0) :
84944d93782SGreg Clayton             x(_x),
85044d93782SGreg Clayton             y(_y)
85144d93782SGreg Clayton         {
85244d93782SGreg Clayton         }
85344d93782SGreg Clayton 
85444d93782SGreg Clayton         void
85544d93782SGreg Clayton         Clear ()
85644d93782SGreg Clayton         {
85744d93782SGreg Clayton             x = 0;
85844d93782SGreg Clayton             y = 0;
85944d93782SGreg Clayton         }
86044d93782SGreg Clayton 
86144d93782SGreg Clayton         Point &
86244d93782SGreg Clayton         operator += (const Point &rhs)
86344d93782SGreg Clayton         {
86444d93782SGreg Clayton             x += rhs.x;
86544d93782SGreg Clayton             y += rhs.y;
86644d93782SGreg Clayton             return *this;
86744d93782SGreg Clayton         }
86844d93782SGreg Clayton 
86944d93782SGreg Clayton         void
87044d93782SGreg Clayton         Dump ()
87144d93782SGreg Clayton         {
87244d93782SGreg Clayton             printf ("(x=%i, y=%i)\n", x, y);
87344d93782SGreg Clayton         }
87444d93782SGreg Clayton 
87544d93782SGreg Clayton     };
87644d93782SGreg Clayton 
87744d93782SGreg Clayton     bool operator == (const Point &lhs, const Point &rhs)
87844d93782SGreg Clayton     {
87944d93782SGreg Clayton         return lhs.x == rhs.x && lhs.y == rhs.y;
88044d93782SGreg Clayton     }
88144d93782SGreg Clayton     bool operator != (const Point &lhs, const Point &rhs)
88244d93782SGreg Clayton     {
88344d93782SGreg Clayton         return lhs.x != rhs.x || lhs.y != rhs.y;
88444d93782SGreg Clayton     }
88544d93782SGreg Clayton 
88644d93782SGreg Clayton     struct Size
88744d93782SGreg Clayton     {
88844d93782SGreg Clayton         int width;
88944d93782SGreg Clayton         int height;
89044d93782SGreg Clayton         Size (int w = 0, int h = 0) :
89144d93782SGreg Clayton             width (w),
89244d93782SGreg Clayton             height (h)
89344d93782SGreg Clayton         {
89444d93782SGreg Clayton         }
89544d93782SGreg Clayton 
89644d93782SGreg Clayton         void
89744d93782SGreg Clayton         Clear ()
89844d93782SGreg Clayton         {
89944d93782SGreg Clayton             width = 0;
90044d93782SGreg Clayton             height = 0;
90144d93782SGreg Clayton         }
90244d93782SGreg Clayton 
90344d93782SGreg Clayton         void
90444d93782SGreg Clayton         Dump ()
90544d93782SGreg Clayton         {
90644d93782SGreg Clayton             printf ("(w=%i, h=%i)\n", width, height);
90744d93782SGreg Clayton         }
90844d93782SGreg Clayton 
90944d93782SGreg Clayton     };
91044d93782SGreg Clayton 
91144d93782SGreg Clayton     bool operator == (const Size &lhs, const Size &rhs)
91244d93782SGreg Clayton     {
91344d93782SGreg Clayton         return lhs.width == rhs.width && lhs.height == rhs.height;
91444d93782SGreg Clayton     }
91544d93782SGreg Clayton     bool operator != (const Size &lhs, const Size &rhs)
91644d93782SGreg Clayton     {
91744d93782SGreg Clayton         return lhs.width != rhs.width || lhs.height != rhs.height;
91844d93782SGreg Clayton     }
91944d93782SGreg Clayton 
92044d93782SGreg Clayton     struct Rect
92144d93782SGreg Clayton     {
92244d93782SGreg Clayton         Point origin;
92344d93782SGreg Clayton         Size size;
92444d93782SGreg Clayton 
92544d93782SGreg Clayton         Rect () :
92644d93782SGreg Clayton             origin(),
92744d93782SGreg Clayton             size()
92844d93782SGreg Clayton         {
92944d93782SGreg Clayton         }
93044d93782SGreg Clayton 
93144d93782SGreg Clayton         Rect (const Point &p, const Size &s) :
93244d93782SGreg Clayton             origin (p),
93344d93782SGreg Clayton             size (s)
93444d93782SGreg Clayton         {
93544d93782SGreg Clayton         }
93644d93782SGreg Clayton 
93744d93782SGreg Clayton         void
93844d93782SGreg Clayton         Clear ()
93944d93782SGreg Clayton         {
94044d93782SGreg Clayton             origin.Clear();
94144d93782SGreg Clayton             size.Clear();
94244d93782SGreg Clayton         }
94344d93782SGreg Clayton 
94444d93782SGreg Clayton         void
94544d93782SGreg Clayton         Dump ()
94644d93782SGreg Clayton         {
94744d93782SGreg Clayton             printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height);
94844d93782SGreg Clayton         }
94944d93782SGreg Clayton 
95044d93782SGreg Clayton         void
95144d93782SGreg Clayton         Inset (int w, int h)
95244d93782SGreg Clayton         {
95344d93782SGreg Clayton             if (size.width > w*2)
95444d93782SGreg Clayton                 size.width -= w*2;
95544d93782SGreg Clayton             origin.x += w;
95644d93782SGreg Clayton 
95744d93782SGreg Clayton             if (size.height > h*2)
95844d93782SGreg Clayton                 size.height -= h*2;
95944d93782SGreg Clayton             origin.y += h;
96044d93782SGreg Clayton         }
96144d93782SGreg Clayton         // Return a status bar rectangle which is the last line of
96244d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
96344d93782SGreg Clayton         // include the status bar area.
96444d93782SGreg Clayton         Rect
96544d93782SGreg Clayton         MakeStatusBar ()
96644d93782SGreg Clayton         {
96744d93782SGreg Clayton             Rect status_bar;
96844d93782SGreg Clayton             if (size.height > 1)
96944d93782SGreg Clayton             {
97044d93782SGreg Clayton                 status_bar.origin.x = origin.x;
97144d93782SGreg Clayton                 status_bar.origin.y = size.height;
97244d93782SGreg Clayton                 status_bar.size.width = size.width;
97344d93782SGreg Clayton                 status_bar.size.height = 1;
97444d93782SGreg Clayton                 --size.height;
97544d93782SGreg Clayton             }
97644d93782SGreg Clayton             return status_bar;
97744d93782SGreg Clayton         }
97844d93782SGreg Clayton 
97944d93782SGreg Clayton         // Return a menubar rectangle which is the first line of
98044d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
98144d93782SGreg Clayton         // include the menubar area.
98244d93782SGreg Clayton         Rect
98344d93782SGreg Clayton         MakeMenuBar ()
98444d93782SGreg Clayton         {
98544d93782SGreg Clayton             Rect menubar;
98644d93782SGreg Clayton             if (size.height > 1)
98744d93782SGreg Clayton             {
98844d93782SGreg Clayton                 menubar.origin.x = origin.x;
98944d93782SGreg Clayton                 menubar.origin.y = origin.y;
99044d93782SGreg Clayton                 menubar.size.width = size.width;
99144d93782SGreg Clayton                 menubar.size.height = 1;
99244d93782SGreg Clayton                 ++origin.y;
99344d93782SGreg Clayton                 --size.height;
99444d93782SGreg Clayton             }
99544d93782SGreg Clayton             return menubar;
99644d93782SGreg Clayton         }
99744d93782SGreg Clayton 
99844d93782SGreg Clayton         void
99944d93782SGreg Clayton         HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const
100044d93782SGreg Clayton         {
100144d93782SGreg Clayton             float top_height = top_percentage * size.height;
100244d93782SGreg Clayton             HorizontalSplit (top_height, top, bottom);
100344d93782SGreg Clayton         }
100444d93782SGreg Clayton 
100544d93782SGreg Clayton         void
100644d93782SGreg Clayton         HorizontalSplit (int top_height, Rect &top, Rect &bottom) const
100744d93782SGreg Clayton         {
100844d93782SGreg Clayton             top = *this;
100944d93782SGreg Clayton             if (top_height < size.height)
101044d93782SGreg Clayton             {
101144d93782SGreg Clayton                 top.size.height = top_height;
101244d93782SGreg Clayton                 bottom.origin.x = origin.x;
101344d93782SGreg Clayton                 bottom.origin.y = origin.y + top.size.height;
101444d93782SGreg Clayton                 bottom.size.width = size.width;
101544d93782SGreg Clayton                 bottom.size.height = size.height - top.size.height;
101644d93782SGreg Clayton             }
101744d93782SGreg Clayton             else
101844d93782SGreg Clayton             {
101944d93782SGreg Clayton                 bottom.Clear();
102044d93782SGreg Clayton             }
102144d93782SGreg Clayton         }
102244d93782SGreg Clayton 
102344d93782SGreg Clayton         void
102444d93782SGreg Clayton         VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const
102544d93782SGreg Clayton         {
102644d93782SGreg Clayton             float left_width = left_percentage * size.width;
102744d93782SGreg Clayton             VerticalSplit (left_width, left, right);
102844d93782SGreg Clayton         }
102944d93782SGreg Clayton 
103044d93782SGreg Clayton 
103144d93782SGreg Clayton         void
103244d93782SGreg Clayton         VerticalSplit (int left_width, Rect &left, Rect &right) const
103344d93782SGreg Clayton         {
103444d93782SGreg Clayton             left = *this;
103544d93782SGreg Clayton             if (left_width < size.width)
103644d93782SGreg Clayton             {
103744d93782SGreg Clayton                 left.size.width = left_width;
103844d93782SGreg Clayton                 right.origin.x = origin.x + left.size.width;
103944d93782SGreg Clayton                 right.origin.y = origin.y;
104044d93782SGreg Clayton                 right.size.width = size.width - left.size.width;
104144d93782SGreg Clayton                 right.size.height = size.height;
104244d93782SGreg Clayton             }
104344d93782SGreg Clayton             else
104444d93782SGreg Clayton             {
104544d93782SGreg Clayton                 right.Clear();
104644d93782SGreg Clayton             }
104744d93782SGreg Clayton         }
104844d93782SGreg Clayton     };
104944d93782SGreg Clayton 
105044d93782SGreg Clayton     bool operator == (const Rect &lhs, const Rect &rhs)
105144d93782SGreg Clayton     {
105244d93782SGreg Clayton         return lhs.origin == rhs.origin && lhs.size == rhs.size;
105344d93782SGreg Clayton     }
105444d93782SGreg Clayton     bool operator != (const Rect &lhs, const Rect &rhs)
105544d93782SGreg Clayton     {
105644d93782SGreg Clayton         return lhs.origin != rhs.origin || lhs.size != rhs.size;
105744d93782SGreg Clayton     }
105844d93782SGreg Clayton 
105944d93782SGreg Clayton     enum HandleCharResult
106044d93782SGreg Clayton     {
106144d93782SGreg Clayton         eKeyNotHandled      = 0,
106244d93782SGreg Clayton         eKeyHandled         = 1,
106344d93782SGreg Clayton         eQuitApplication    = 2
106444d93782SGreg Clayton     };
106544d93782SGreg Clayton 
106644d93782SGreg Clayton     enum class MenuActionResult
106744d93782SGreg Clayton     {
106844d93782SGreg Clayton         Handled,
106944d93782SGreg Clayton         NotHandled,
107044d93782SGreg Clayton         Quit    // Exit all menus and quit
107144d93782SGreg Clayton     };
107244d93782SGreg Clayton 
107344d93782SGreg Clayton     struct KeyHelp
107444d93782SGreg Clayton     {
107544d93782SGreg Clayton         int ch;
107644d93782SGreg Clayton         const char *description;
107744d93782SGreg Clayton     };
107844d93782SGreg Clayton 
107944d93782SGreg Clayton     class WindowDelegate
108044d93782SGreg Clayton     {
108144d93782SGreg Clayton     public:
108244d93782SGreg Clayton         virtual
108344d93782SGreg Clayton         ~WindowDelegate()
108444d93782SGreg Clayton         {
108544d93782SGreg Clayton         }
108644d93782SGreg Clayton 
108744d93782SGreg Clayton         virtual bool
108844d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force)
108944d93782SGreg Clayton         {
109044d93782SGreg Clayton             return false; // Drawing not handled
109144d93782SGreg Clayton         }
109244d93782SGreg Clayton 
109344d93782SGreg Clayton         virtual HandleCharResult
109444d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key)
109544d93782SGreg Clayton         {
109644d93782SGreg Clayton             return eKeyNotHandled;
109744d93782SGreg Clayton         }
109844d93782SGreg Clayton 
109944d93782SGreg Clayton         virtual const char *
110044d93782SGreg Clayton         WindowDelegateGetHelpText ()
110144d93782SGreg Clayton         {
110244d93782SGreg Clayton             return NULL;
110344d93782SGreg Clayton         }
110444d93782SGreg Clayton 
110544d93782SGreg Clayton         virtual KeyHelp *
110644d93782SGreg Clayton         WindowDelegateGetKeyHelp ()
110744d93782SGreg Clayton         {
110844d93782SGreg Clayton             return NULL;
110944d93782SGreg Clayton         }
111044d93782SGreg Clayton     };
111144d93782SGreg Clayton 
111244d93782SGreg Clayton     class HelpDialogDelegate :
111344d93782SGreg Clayton         public WindowDelegate
111444d93782SGreg Clayton     {
111544d93782SGreg Clayton     public:
111644d93782SGreg Clayton         HelpDialogDelegate (const char *text, KeyHelp *key_help_array);
111744d93782SGreg Clayton 
1118bd5ae6b4SGreg Clayton         ~HelpDialogDelegate() override;
111944d93782SGreg Clayton 
1120bd5ae6b4SGreg Clayton         bool
1121bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
112244d93782SGreg Clayton 
1123bd5ae6b4SGreg Clayton         HandleCharResult
1124bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
112544d93782SGreg Clayton 
112644d93782SGreg Clayton         size_t
112744d93782SGreg Clayton         GetNumLines() const
112844d93782SGreg Clayton         {
112944d93782SGreg Clayton             return m_text.GetSize();
113044d93782SGreg Clayton         }
113144d93782SGreg Clayton 
113244d93782SGreg Clayton         size_t
113344d93782SGreg Clayton         GetMaxLineLength () const
113444d93782SGreg Clayton         {
113544d93782SGreg Clayton             return m_text.GetMaxStringLength();
113644d93782SGreg Clayton         }
113744d93782SGreg Clayton 
113844d93782SGreg Clayton     protected:
113944d93782SGreg Clayton         StringList m_text;
114044d93782SGreg Clayton         int m_first_visible_line;
114144d93782SGreg Clayton     };
114244d93782SGreg Clayton 
114344d93782SGreg Clayton 
114444d93782SGreg Clayton     class Window
114544d93782SGreg Clayton     {
114644d93782SGreg Clayton     public:
114744d93782SGreg Clayton 
114844d93782SGreg Clayton         Window (const char *name) :
114944d93782SGreg Clayton             m_name (name),
115044d93782SGreg Clayton             m_window (NULL),
115144d93782SGreg Clayton             m_panel (NULL),
115244d93782SGreg Clayton             m_parent (NULL),
115344d93782SGreg Clayton             m_subwindows (),
115444d93782SGreg Clayton             m_delegate_sp (),
115544d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
115644d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
115744d93782SGreg Clayton             m_delete (false),
115844d93782SGreg Clayton             m_needs_update (true),
115944d93782SGreg Clayton             m_can_activate (true),
116044d93782SGreg Clayton             m_is_subwin (false)
116144d93782SGreg Clayton         {
116244d93782SGreg Clayton         }
116344d93782SGreg Clayton 
116444d93782SGreg Clayton         Window (const char *name, WINDOW *w, bool del = true) :
116544d93782SGreg Clayton             m_name (name),
116644d93782SGreg Clayton             m_window (NULL),
116744d93782SGreg Clayton             m_panel (NULL),
116844d93782SGreg Clayton             m_parent (NULL),
116944d93782SGreg Clayton             m_subwindows (),
117044d93782SGreg Clayton             m_delegate_sp (),
117144d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
117244d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
117344d93782SGreg Clayton             m_delete (del),
117444d93782SGreg Clayton             m_needs_update (true),
117544d93782SGreg Clayton             m_can_activate (true),
117644d93782SGreg Clayton             m_is_subwin (false)
117744d93782SGreg Clayton         {
117844d93782SGreg Clayton             if (w)
117944d93782SGreg Clayton                 Reset(w);
118044d93782SGreg Clayton         }
118144d93782SGreg Clayton 
118244d93782SGreg Clayton         Window (const char *name, const Rect &bounds) :
118344d93782SGreg Clayton             m_name (name),
118444d93782SGreg Clayton             m_window (NULL),
118544d93782SGreg Clayton             m_parent (NULL),
118644d93782SGreg Clayton             m_subwindows (),
118744d93782SGreg Clayton             m_delegate_sp (),
118844d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
118944d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
119044d93782SGreg Clayton             m_delete (true),
119144d93782SGreg Clayton             m_needs_update (true),
119244d93782SGreg Clayton             m_can_activate (true),
119344d93782SGreg Clayton             m_is_subwin (false)
119444d93782SGreg Clayton         {
119544d93782SGreg Clayton             Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y));
119644d93782SGreg Clayton         }
119744d93782SGreg Clayton 
119844d93782SGreg Clayton         virtual
119944d93782SGreg Clayton         ~Window ()
120044d93782SGreg Clayton         {
120144d93782SGreg Clayton             RemoveSubWindows ();
120244d93782SGreg Clayton             Reset ();
120344d93782SGreg Clayton         }
120444d93782SGreg Clayton 
120544d93782SGreg Clayton         void
120644d93782SGreg Clayton         Reset (WINDOW *w = NULL, bool del = true)
120744d93782SGreg Clayton         {
120844d93782SGreg Clayton             if (m_window == w)
120944d93782SGreg Clayton                 return;
121044d93782SGreg Clayton 
121144d93782SGreg Clayton             if (m_panel)
121244d93782SGreg Clayton             {
121344d93782SGreg Clayton                 ::del_panel (m_panel);
121444d93782SGreg Clayton                 m_panel = NULL;
121544d93782SGreg Clayton             }
121644d93782SGreg Clayton             if (m_window && m_delete)
121744d93782SGreg Clayton             {
121844d93782SGreg Clayton                 ::delwin (m_window);
121944d93782SGreg Clayton                 m_window = NULL;
122044d93782SGreg Clayton                 m_delete = false;
122144d93782SGreg Clayton             }
122244d93782SGreg Clayton             if (w)
122344d93782SGreg Clayton             {
122444d93782SGreg Clayton                 m_window = w;
122544d93782SGreg Clayton                 m_panel = ::new_panel (m_window);
122644d93782SGreg Clayton                 m_delete = del;
122744d93782SGreg Clayton             }
122844d93782SGreg Clayton         }
122944d93782SGreg Clayton 
123044d93782SGreg Clayton         void    AttributeOn (attr_t attr)   { ::wattron (m_window, attr); }
123144d93782SGreg Clayton         void    AttributeOff (attr_t attr)  { ::wattroff (m_window, attr); }
123244d93782SGreg Clayton         void    Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); }
123344d93782SGreg Clayton         void    Clear ()    { ::wclear (m_window); }
123444d93782SGreg Clayton         void    Erase ()    { ::werase (m_window); }
123544d93782SGreg Clayton         Rect    GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window
123644d93782SGreg Clayton         int     GetChar ()  { return ::wgetch (m_window); }
123744d93782SGreg Clayton         int     GetCursorX ()     { return getcurx (m_window); }
123844d93782SGreg Clayton         int     GetCursorY ()     { return getcury (m_window); }
123944d93782SGreg Clayton         Rect    GetFrame ()    { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system
124044d93782SGreg Clayton         Point   GetParentOrigin() { return Point (GetParentX(), GetParentY()); }
124144d93782SGreg Clayton         Size    GetSize()         { return Size (GetWidth(), GetHeight()); }
124244d93782SGreg Clayton         int     GetParentX ()     { return getparx (m_window); }
124344d93782SGreg Clayton         int     GetParentY ()     { return getpary (m_window); }
124444d93782SGreg Clayton         int     GetMaxX()   { return getmaxx (m_window); }
124544d93782SGreg Clayton         int     GetMaxY()   { return getmaxy (m_window); }
124644d93782SGreg Clayton         int     GetWidth()  { return GetMaxX(); }
124744d93782SGreg Clayton         int     GetHeight() { return GetMaxY(); }
124844d93782SGreg Clayton         void    MoveCursor (int x, int y) {  ::wmove (m_window, y, x); }
124944d93782SGreg Clayton         void    MoveWindow (int x, int y) {  MoveWindow(Point(x,y)); }
125044d93782SGreg Clayton         void    Resize (int w, int h) { ::wresize(m_window, h, w); }
125144d93782SGreg Clayton         void    Resize (const Size &size) { ::wresize(m_window, size.height, size.width); }
125244d93782SGreg Clayton         void    PutChar (int ch)    { ::waddch (m_window, ch); }
125344d93782SGreg Clayton         void    PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); }
125444d93782SGreg Clayton         void    Refresh ()  { ::wrefresh (m_window); }
125544d93782SGreg Clayton         void    DeferredRefresh ()
125644d93782SGreg Clayton         {
125744d93782SGreg Clayton             // We are using panels, so we don't need to call this...
125844d93782SGreg Clayton             //::wnoutrefresh(m_window);
125944d93782SGreg Clayton         }
126044d93782SGreg Clayton         void    SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); }
126144d93782SGreg Clayton         void    UnderlineOn ()  { AttributeOn(A_UNDERLINE); }
126244d93782SGreg Clayton         void    UnderlineOff () { AttributeOff(A_UNDERLINE); }
126344d93782SGreg Clayton 
126444d93782SGreg Clayton         void    PutCStringTruncated (const char *s, int right_pad)
126544d93782SGreg Clayton         {
126644d93782SGreg Clayton             int bytes_left = GetWidth() - GetCursorX();
126744d93782SGreg Clayton             if (bytes_left > right_pad)
126844d93782SGreg Clayton             {
126944d93782SGreg Clayton                 bytes_left -= right_pad;
127044d93782SGreg Clayton                 ::waddnstr (m_window, s, bytes_left);
127144d93782SGreg Clayton             }
127244d93782SGreg Clayton         }
127344d93782SGreg Clayton 
127444d93782SGreg Clayton         void
127544d93782SGreg Clayton         MoveWindow (const Point &origin)
127644d93782SGreg Clayton         {
127744d93782SGreg Clayton             const bool moving_window = origin != GetParentOrigin();
127844d93782SGreg Clayton             if (m_is_subwin && moving_window)
127944d93782SGreg Clayton             {
128044d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
128144d93782SGreg Clayton                 Size size = GetSize();
128244d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
128344d93782SGreg Clayton                                  size.height,
128444d93782SGreg Clayton                                  size.width,
128544d93782SGreg Clayton                                  origin.y,
128644d93782SGreg Clayton                                  origin.x), true);
128744d93782SGreg Clayton             }
128844d93782SGreg Clayton             else
128944d93782SGreg Clayton             {
129044d93782SGreg Clayton                 ::mvwin (m_window, origin.y, origin.x);
129144d93782SGreg Clayton             }
129244d93782SGreg Clayton         }
129344d93782SGreg Clayton 
129444d93782SGreg Clayton         void
129544d93782SGreg Clayton         SetBounds (const Rect &bounds)
129644d93782SGreg Clayton         {
129744d93782SGreg Clayton             const bool moving_window = bounds.origin != GetParentOrigin();
129844d93782SGreg Clayton             if (m_is_subwin && moving_window)
129944d93782SGreg Clayton             {
130044d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
130144d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
130244d93782SGreg Clayton                                  bounds.size.height,
130344d93782SGreg Clayton                                  bounds.size.width,
130444d93782SGreg Clayton                                  bounds.origin.y,
130544d93782SGreg Clayton                                  bounds.origin.x), true);
130644d93782SGreg Clayton             }
130744d93782SGreg Clayton             else
130844d93782SGreg Clayton             {
130944d93782SGreg Clayton                 if (moving_window)
131044d93782SGreg Clayton                     MoveWindow(bounds.origin);
131144d93782SGreg Clayton                 Resize (bounds.size);
131244d93782SGreg Clayton             }
131344d93782SGreg Clayton         }
131444d93782SGreg Clayton 
131544d93782SGreg Clayton         void
131644d93782SGreg Clayton         Printf (const char *format, ...)  __attribute__ ((format (printf, 2, 3)))
131744d93782SGreg Clayton         {
131844d93782SGreg Clayton             va_list args;
131944d93782SGreg Clayton             va_start (args, format);
132044d93782SGreg Clayton             vwprintw(m_window, format, args);
132144d93782SGreg Clayton             va_end (args);
132244d93782SGreg Clayton         }
132344d93782SGreg Clayton 
132444d93782SGreg Clayton         void
132544d93782SGreg Clayton         Touch ()
132644d93782SGreg Clayton         {
132744d93782SGreg Clayton             ::touchwin (m_window);
132844d93782SGreg Clayton             if (m_parent)
132944d93782SGreg Clayton                 m_parent->Touch();
133044d93782SGreg Clayton         }
133144d93782SGreg Clayton 
133244d93782SGreg Clayton         WindowSP
133344d93782SGreg Clayton         CreateSubWindow (const char *name, const Rect &bounds, bool make_active)
133444d93782SGreg Clayton         {
133544d93782SGreg Clayton             WindowSP subwindow_sp;
133644d93782SGreg Clayton             if (m_window)
133744d93782SGreg Clayton             {
133844d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::subwin (m_window,
133944d93782SGreg Clayton                                                               bounds.size.height,
134044d93782SGreg Clayton                                                               bounds.size.width,
134144d93782SGreg Clayton                                                               bounds.origin.y,
134244d93782SGreg Clayton                                                               bounds.origin.x), true));
134344d93782SGreg Clayton                 subwindow_sp->m_is_subwin = true;
134444d93782SGreg Clayton             }
134544d93782SGreg Clayton             else
134644d93782SGreg Clayton             {
134744d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height,
134844d93782SGreg Clayton                                                               bounds.size.width,
134944d93782SGreg Clayton                                                               bounds.origin.y,
135044d93782SGreg Clayton                                                               bounds.origin.x), true));
135144d93782SGreg Clayton                 subwindow_sp->m_is_subwin = false;
135244d93782SGreg Clayton             }
135344d93782SGreg Clayton             subwindow_sp->m_parent = this;
135444d93782SGreg Clayton             if (make_active)
135544d93782SGreg Clayton             {
135644d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
135744d93782SGreg Clayton                 m_curr_active_window_idx = m_subwindows.size();
135844d93782SGreg Clayton             }
135944d93782SGreg Clayton             m_subwindows.push_back(subwindow_sp);
136044d93782SGreg Clayton             ::top_panel (subwindow_sp->m_panel);
136144d93782SGreg Clayton             m_needs_update = true;
136244d93782SGreg Clayton             return subwindow_sp;
136344d93782SGreg Clayton         }
136444d93782SGreg Clayton 
136544d93782SGreg Clayton         bool
136644d93782SGreg Clayton         RemoveSubWindow (Window *window)
136744d93782SGreg Clayton         {
136844d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
136944d93782SGreg Clayton             size_t i = 0;
137044d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
137144d93782SGreg Clayton             {
137244d93782SGreg Clayton                 if ((*pos).get() == window)
137344d93782SGreg Clayton                 {
137444d93782SGreg Clayton                     if (m_prev_active_window_idx == i)
137544d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
137644d93782SGreg Clayton                     else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i)
137744d93782SGreg Clayton                         --m_prev_active_window_idx;
137844d93782SGreg Clayton 
137944d93782SGreg Clayton                     if (m_curr_active_window_idx == i)
138044d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
138144d93782SGreg Clayton                     else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i)
138244d93782SGreg Clayton                         --m_curr_active_window_idx;
138344d93782SGreg Clayton                     window->Erase();
138444d93782SGreg Clayton                     m_subwindows.erase(pos);
138544d93782SGreg Clayton                     m_needs_update = true;
138644d93782SGreg Clayton                     if (m_parent)
138744d93782SGreg Clayton                         m_parent->Touch();
138844d93782SGreg Clayton                     else
138944d93782SGreg Clayton                         ::touchwin (stdscr);
139044d93782SGreg Clayton                     return true;
139144d93782SGreg Clayton                 }
139244d93782SGreg Clayton             }
139344d93782SGreg Clayton             return false;
139444d93782SGreg Clayton         }
139544d93782SGreg Clayton 
139644d93782SGreg Clayton         WindowSP
139744d93782SGreg Clayton         FindSubWindow (const char *name)
139844d93782SGreg Clayton         {
139944d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
140044d93782SGreg Clayton             size_t i = 0;
140144d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
140244d93782SGreg Clayton             {
140344d93782SGreg Clayton                 if ((*pos)->m_name.compare(name) == 0)
140444d93782SGreg Clayton                     return *pos;
140544d93782SGreg Clayton             }
140644d93782SGreg Clayton             return WindowSP();
140744d93782SGreg Clayton         }
140844d93782SGreg Clayton 
140944d93782SGreg Clayton         void
141044d93782SGreg Clayton         RemoveSubWindows ()
141144d93782SGreg Clayton         {
141244d93782SGreg Clayton             m_curr_active_window_idx = UINT32_MAX;
141344d93782SGreg Clayton             m_prev_active_window_idx = UINT32_MAX;
141444d93782SGreg Clayton             for (Windows::iterator pos = m_subwindows.begin();
141544d93782SGreg Clayton                  pos != m_subwindows.end();
141644d93782SGreg Clayton                  pos = m_subwindows.erase(pos))
141744d93782SGreg Clayton             {
141844d93782SGreg Clayton                 (*pos)->Erase();
141944d93782SGreg Clayton             }
142044d93782SGreg Clayton             if (m_parent)
142144d93782SGreg Clayton                 m_parent->Touch();
142244d93782SGreg Clayton             else
142344d93782SGreg Clayton                 ::touchwin (stdscr);
142444d93782SGreg Clayton         }
142544d93782SGreg Clayton 
142644d93782SGreg Clayton         WINDOW *
142744d93782SGreg Clayton         get()
142844d93782SGreg Clayton         {
142944d93782SGreg Clayton             return m_window;
143044d93782SGreg Clayton         }
143144d93782SGreg Clayton 
143244d93782SGreg Clayton         operator WINDOW *()
143344d93782SGreg Clayton         {
143444d93782SGreg Clayton             return m_window;
143544d93782SGreg Clayton         }
143644d93782SGreg Clayton 
143744d93782SGreg Clayton         //----------------------------------------------------------------------
143844d93782SGreg Clayton         // Window drawing utilities
143944d93782SGreg Clayton         //----------------------------------------------------------------------
144044d93782SGreg Clayton         void
144144d93782SGreg Clayton         DrawTitleBox (const char *title, const char *bottom_message = NULL)
144244d93782SGreg Clayton         {
144344d93782SGreg Clayton             attr_t attr = 0;
144444d93782SGreg Clayton             if (IsActive())
144544d93782SGreg Clayton                 attr = A_BOLD | COLOR_PAIR(2);
144644d93782SGreg Clayton             else
144744d93782SGreg Clayton                 attr = 0;
144844d93782SGreg Clayton             if (attr)
144944d93782SGreg Clayton                 AttributeOn(attr);
145044d93782SGreg Clayton 
145144d93782SGreg Clayton             Box();
145244d93782SGreg Clayton             MoveCursor(3, 0);
145344d93782SGreg Clayton 
145444d93782SGreg Clayton             if (title && title[0])
145544d93782SGreg Clayton             {
145644d93782SGreg Clayton                 PutChar ('<');
145744d93782SGreg Clayton                 PutCString (title);
145844d93782SGreg Clayton                 PutChar ('>');
145944d93782SGreg Clayton             }
146044d93782SGreg Clayton 
146144d93782SGreg Clayton             if (bottom_message && bottom_message[0])
146244d93782SGreg Clayton             {
146344d93782SGreg Clayton                 int bottom_message_length = strlen(bottom_message);
146444d93782SGreg Clayton                 int x = GetWidth() - 3 - (bottom_message_length + 2);
146544d93782SGreg Clayton 
146644d93782SGreg Clayton                 if (x > 0)
146744d93782SGreg Clayton                 {
146844d93782SGreg Clayton                     MoveCursor (x, GetHeight() - 1);
146944d93782SGreg Clayton                     PutChar ('[');
147044d93782SGreg Clayton                     PutCString(bottom_message);
147144d93782SGreg Clayton                     PutChar (']');
147244d93782SGreg Clayton                 }
147344d93782SGreg Clayton                 else
147444d93782SGreg Clayton                 {
147544d93782SGreg Clayton                     MoveCursor (1, GetHeight() - 1);
147644d93782SGreg Clayton                     PutChar ('[');
147744d93782SGreg Clayton                     PutCStringTruncated (bottom_message, 1);
147844d93782SGreg Clayton                 }
147944d93782SGreg Clayton             }
148044d93782SGreg Clayton             if (attr)
148144d93782SGreg Clayton                 AttributeOff(attr);
148244d93782SGreg Clayton 
148344d93782SGreg Clayton         }
148444d93782SGreg Clayton 
148544d93782SGreg Clayton         virtual void
148644d93782SGreg Clayton         Draw (bool force)
148744d93782SGreg Clayton         {
148844d93782SGreg Clayton             if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force))
148944d93782SGreg Clayton                 return;
149044d93782SGreg Clayton 
149144d93782SGreg Clayton             for (auto &subwindow_sp : m_subwindows)
149244d93782SGreg Clayton                 subwindow_sp->Draw(force);
149344d93782SGreg Clayton         }
149444d93782SGreg Clayton 
149544d93782SGreg Clayton         bool
149644d93782SGreg Clayton         CreateHelpSubwindow ()
149744d93782SGreg Clayton         {
149844d93782SGreg Clayton             if (m_delegate_sp)
149944d93782SGreg Clayton             {
150044d93782SGreg Clayton                 const char *text = m_delegate_sp->WindowDelegateGetHelpText ();
150144d93782SGreg Clayton                 KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp ();
150244d93782SGreg Clayton                 if ((text && text[0]) || key_help)
150344d93782SGreg Clayton                 {
150444d93782SGreg Clayton                     std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help));
150544d93782SGreg Clayton                     const size_t num_lines = help_delegate_ap->GetNumLines();
150644d93782SGreg Clayton                     const size_t max_length = help_delegate_ap->GetMaxLineLength();
150744d93782SGreg Clayton                     Rect bounds = GetBounds();
150844d93782SGreg Clayton                     bounds.Inset(1, 1);
15093985c8c6SSaleem Abdulrasool                     if (max_length + 4 < static_cast<size_t>(bounds.size.width))
151044d93782SGreg Clayton                     {
151144d93782SGreg Clayton                         bounds.origin.x += (bounds.size.width - max_length + 4)/2;
151244d93782SGreg Clayton                         bounds.size.width = max_length + 4;
151344d93782SGreg Clayton                     }
151444d93782SGreg Clayton                     else
151544d93782SGreg Clayton                     {
151644d93782SGreg Clayton                         if (bounds.size.width > 100)
151744d93782SGreg Clayton                         {
151844d93782SGreg Clayton                             const int inset_w = bounds.size.width / 4;
151944d93782SGreg Clayton                             bounds.origin.x += inset_w;
152044d93782SGreg Clayton                             bounds.size.width -= 2*inset_w;
152144d93782SGreg Clayton                         }
152244d93782SGreg Clayton                     }
152344d93782SGreg Clayton 
15243985c8c6SSaleem Abdulrasool                     if (num_lines + 2 < static_cast<size_t>(bounds.size.height))
152544d93782SGreg Clayton                     {
152644d93782SGreg Clayton                         bounds.origin.y += (bounds.size.height - num_lines + 2)/2;
152744d93782SGreg Clayton                         bounds.size.height = num_lines + 2;
152844d93782SGreg Clayton                     }
152944d93782SGreg Clayton                     else
153044d93782SGreg Clayton                     {
153144d93782SGreg Clayton                         if (bounds.size.height > 100)
153244d93782SGreg Clayton                         {
153344d93782SGreg Clayton                             const int inset_h = bounds.size.height / 4;
153444d93782SGreg Clayton                             bounds.origin.y += inset_h;
153544d93782SGreg Clayton                             bounds.size.height -= 2*inset_h;
153644d93782SGreg Clayton                         }
153744d93782SGreg Clayton                     }
15385fdb09bbSGreg Clayton                     WindowSP help_window_sp;
15395fdb09bbSGreg Clayton                     Window *parent_window = GetParent();
15405fdb09bbSGreg Clayton                     if (parent_window)
15415fdb09bbSGreg Clayton                         help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
15425fdb09bbSGreg Clayton                     else
15435fdb09bbSGreg Clayton                         help_window_sp = CreateSubWindow("Help", bounds, true);
154444d93782SGreg Clayton                     help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release()));
154544d93782SGreg Clayton                     return true;
154644d93782SGreg Clayton                 }
154744d93782SGreg Clayton             }
154844d93782SGreg Clayton             return false;
154944d93782SGreg Clayton         }
155044d93782SGreg Clayton 
155144d93782SGreg Clayton         virtual HandleCharResult
155244d93782SGreg Clayton         HandleChar (int key)
155344d93782SGreg Clayton         {
155444d93782SGreg Clayton             // Always check the active window first
155544d93782SGreg Clayton             HandleCharResult result = eKeyNotHandled;
155644d93782SGreg Clayton             WindowSP active_window_sp = GetActiveWindow ();
155744d93782SGreg Clayton             if (active_window_sp)
155844d93782SGreg Clayton             {
155944d93782SGreg Clayton                 result = active_window_sp->HandleChar (key);
156044d93782SGreg Clayton                 if (result != eKeyNotHandled)
156144d93782SGreg Clayton                     return result;
156244d93782SGreg Clayton             }
156344d93782SGreg Clayton 
156444d93782SGreg Clayton             if (m_delegate_sp)
156544d93782SGreg Clayton             {
156644d93782SGreg Clayton                 result = m_delegate_sp->WindowDelegateHandleChar (*this, key);
156744d93782SGreg Clayton                 if (result != eKeyNotHandled)
156844d93782SGreg Clayton                     return result;
156944d93782SGreg Clayton             }
157044d93782SGreg Clayton 
157144d93782SGreg Clayton             // Then check for any windows that want any keys
157244d93782SGreg Clayton             // that weren't handled. This is typically only
157344d93782SGreg Clayton             // for a menubar.
157444d93782SGreg Clayton             // Make a copy of the subwindows in case any HandleChar()
157544d93782SGreg Clayton             // functions muck with the subwindows. If we don't do this,
157644d93782SGreg Clayton             // we can crash when iterating over the subwindows.
157744d93782SGreg Clayton             Windows subwindows (m_subwindows);
157844d93782SGreg Clayton             for (auto subwindow_sp : subwindows)
157944d93782SGreg Clayton             {
158044d93782SGreg Clayton                 if (subwindow_sp->m_can_activate == false)
158144d93782SGreg Clayton                 {
158244d93782SGreg Clayton                     HandleCharResult result = subwindow_sp->HandleChar(key);
158344d93782SGreg Clayton                     if (result != eKeyNotHandled)
158444d93782SGreg Clayton                         return result;
158544d93782SGreg Clayton                 }
158644d93782SGreg Clayton             }
158744d93782SGreg Clayton 
158844d93782SGreg Clayton             return eKeyNotHandled;
158944d93782SGreg Clayton         }
159044d93782SGreg Clayton 
159144d93782SGreg Clayton         bool
159244d93782SGreg Clayton         SetActiveWindow (Window *window)
159344d93782SGreg Clayton         {
159444d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
159544d93782SGreg Clayton             for (size_t i=0; i<num_subwindows; ++i)
159644d93782SGreg Clayton             {
159744d93782SGreg Clayton                 if (m_subwindows[i].get() == window)
159844d93782SGreg Clayton                 {
159944d93782SGreg Clayton                     m_prev_active_window_idx = m_curr_active_window_idx;
160044d93782SGreg Clayton                     ::top_panel (window->m_panel);
160144d93782SGreg Clayton                     m_curr_active_window_idx = i;
160244d93782SGreg Clayton                     return true;
160344d93782SGreg Clayton                 }
160444d93782SGreg Clayton             }
160544d93782SGreg Clayton             return false;
160644d93782SGreg Clayton         }
160744d93782SGreg Clayton 
160844d93782SGreg Clayton         WindowSP
160944d93782SGreg Clayton         GetActiveWindow ()
161044d93782SGreg Clayton         {
161144d93782SGreg Clayton             if (!m_subwindows.empty())
161244d93782SGreg Clayton             {
161344d93782SGreg Clayton                 if (m_curr_active_window_idx >= m_subwindows.size())
161444d93782SGreg Clayton                 {
161544d93782SGreg Clayton                     if (m_prev_active_window_idx < m_subwindows.size())
161644d93782SGreg Clayton                     {
161744d93782SGreg Clayton                         m_curr_active_window_idx = m_prev_active_window_idx;
161844d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
161944d93782SGreg Clayton                     }
162044d93782SGreg Clayton                     else if (IsActive())
162144d93782SGreg Clayton                     {
162244d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
162344d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
162444d93782SGreg Clayton 
162544d93782SGreg Clayton                         // Find first window that wants to be active if this window is active
162644d93782SGreg Clayton                         const size_t num_subwindows = m_subwindows.size();
162744d93782SGreg Clayton                         for (size_t i=0; i<num_subwindows; ++i)
162844d93782SGreg Clayton                         {
162944d93782SGreg Clayton                             if (m_subwindows[i]->GetCanBeActive())
163044d93782SGreg Clayton                             {
163144d93782SGreg Clayton                                 m_curr_active_window_idx = i;
163244d93782SGreg Clayton                                 break;
163344d93782SGreg Clayton                             }
163444d93782SGreg Clayton                         }
163544d93782SGreg Clayton                     }
163644d93782SGreg Clayton                 }
163744d93782SGreg Clayton 
163844d93782SGreg Clayton                 if (m_curr_active_window_idx < m_subwindows.size())
163944d93782SGreg Clayton                     return m_subwindows[m_curr_active_window_idx];
164044d93782SGreg Clayton             }
164144d93782SGreg Clayton             return WindowSP();
164244d93782SGreg Clayton         }
164344d93782SGreg Clayton 
164444d93782SGreg Clayton         bool
164544d93782SGreg Clayton         GetCanBeActive () const
164644d93782SGreg Clayton         {
164744d93782SGreg Clayton             return m_can_activate;
164844d93782SGreg Clayton         }
164944d93782SGreg Clayton 
165044d93782SGreg Clayton         void
165144d93782SGreg Clayton         SetCanBeActive (bool b)
165244d93782SGreg Clayton         {
165344d93782SGreg Clayton             m_can_activate = b;
165444d93782SGreg Clayton         }
165544d93782SGreg Clayton 
165644d93782SGreg Clayton         const WindowDelegateSP &
165744d93782SGreg Clayton         GetDelegate () const
165844d93782SGreg Clayton         {
165944d93782SGreg Clayton             return m_delegate_sp;
166044d93782SGreg Clayton         }
166144d93782SGreg Clayton 
166244d93782SGreg Clayton         void
166344d93782SGreg Clayton         SetDelegate (const WindowDelegateSP &delegate_sp)
166444d93782SGreg Clayton         {
166544d93782SGreg Clayton             m_delegate_sp = delegate_sp;
166644d93782SGreg Clayton         }
166744d93782SGreg Clayton 
166844d93782SGreg Clayton         Window *
166944d93782SGreg Clayton         GetParent () const
167044d93782SGreg Clayton         {
167144d93782SGreg Clayton             return m_parent;
167244d93782SGreg Clayton         }
167344d93782SGreg Clayton 
167444d93782SGreg Clayton         bool
167544d93782SGreg Clayton         IsActive () const
167644d93782SGreg Clayton         {
167744d93782SGreg Clayton             if (m_parent)
167844d93782SGreg Clayton                 return m_parent->GetActiveWindow().get() == this;
167944d93782SGreg Clayton             else
168044d93782SGreg Clayton                 return true; // Top level window is always active
168144d93782SGreg Clayton         }
168244d93782SGreg Clayton 
168344d93782SGreg Clayton         void
168444d93782SGreg Clayton         SelectNextWindowAsActive ()
168544d93782SGreg Clayton         {
168644d93782SGreg Clayton             // Move active focus to next window
168744d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
168844d93782SGreg Clayton             if (m_curr_active_window_idx == UINT32_MAX)
168944d93782SGreg Clayton             {
169044d93782SGreg Clayton                 uint32_t idx = 0;
169144d93782SGreg Clayton                 for (auto subwindow_sp : m_subwindows)
169244d93782SGreg Clayton                 {
169344d93782SGreg Clayton                     if (subwindow_sp->GetCanBeActive())
169444d93782SGreg Clayton                     {
169544d93782SGreg Clayton                         m_curr_active_window_idx = idx;
169644d93782SGreg Clayton                         break;
169744d93782SGreg Clayton                     }
169844d93782SGreg Clayton                     ++idx;
169944d93782SGreg Clayton                 }
170044d93782SGreg Clayton             }
170144d93782SGreg Clayton             else if (m_curr_active_window_idx + 1 < num_subwindows)
170244d93782SGreg Clayton             {
170344d93782SGreg Clayton                 bool handled = false;
170444d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
170544d93782SGreg Clayton                 for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx)
170644d93782SGreg Clayton                 {
170744d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
170844d93782SGreg Clayton                     {
170944d93782SGreg Clayton                         m_curr_active_window_idx = idx;
171044d93782SGreg Clayton                         handled = true;
171144d93782SGreg Clayton                         break;
171244d93782SGreg Clayton                     }
171344d93782SGreg Clayton                 }
171444d93782SGreg Clayton                 if (!handled)
171544d93782SGreg Clayton                 {
171644d93782SGreg Clayton                     for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx)
171744d93782SGreg Clayton                     {
171844d93782SGreg Clayton                         if (m_subwindows[idx]->GetCanBeActive())
171944d93782SGreg Clayton                         {
172044d93782SGreg Clayton                             m_curr_active_window_idx = idx;
172144d93782SGreg Clayton                             break;
172244d93782SGreg Clayton                         }
172344d93782SGreg Clayton                     }
172444d93782SGreg Clayton                 }
172544d93782SGreg Clayton             }
172644d93782SGreg Clayton             else
172744d93782SGreg Clayton             {
172844d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
172944d93782SGreg Clayton                 for (size_t idx=0; idx<num_subwindows; ++idx)
173044d93782SGreg Clayton                 {
173144d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
173244d93782SGreg Clayton                     {
173344d93782SGreg Clayton                         m_curr_active_window_idx = idx;
173444d93782SGreg Clayton                         break;
173544d93782SGreg Clayton                     }
173644d93782SGreg Clayton                 }
173744d93782SGreg Clayton             }
173844d93782SGreg Clayton         }
173944d93782SGreg Clayton 
174044d93782SGreg Clayton         const char *
174144d93782SGreg Clayton         GetName () const
174244d93782SGreg Clayton         {
174344d93782SGreg Clayton             return m_name.c_str();
174444d93782SGreg Clayton         }
174544d93782SGreg Clayton     protected:
174644d93782SGreg Clayton         std::string m_name;
174744d93782SGreg Clayton         WINDOW *m_window;
174844d93782SGreg Clayton         PANEL *m_panel;
174944d93782SGreg Clayton         Window *m_parent;
175044d93782SGreg Clayton         Windows m_subwindows;
175144d93782SGreg Clayton         WindowDelegateSP m_delegate_sp;
175244d93782SGreg Clayton         uint32_t m_curr_active_window_idx;
175344d93782SGreg Clayton         uint32_t m_prev_active_window_idx;
175444d93782SGreg Clayton         bool m_delete;
175544d93782SGreg Clayton         bool m_needs_update;
175644d93782SGreg Clayton         bool m_can_activate;
175744d93782SGreg Clayton         bool m_is_subwin;
175844d93782SGreg Clayton 
175944d93782SGreg Clayton     private:
176044d93782SGreg Clayton         DISALLOW_COPY_AND_ASSIGN(Window);
176144d93782SGreg Clayton     };
176244d93782SGreg Clayton 
176344d93782SGreg Clayton     class MenuDelegate
176444d93782SGreg Clayton     {
176544d93782SGreg Clayton     public:
176644d93782SGreg Clayton         virtual ~MenuDelegate() {}
176744d93782SGreg Clayton 
176844d93782SGreg Clayton         virtual MenuActionResult
176944d93782SGreg Clayton         MenuDelegateAction (Menu &menu) = 0;
177044d93782SGreg Clayton     };
177144d93782SGreg Clayton 
177244d93782SGreg Clayton     class Menu : public WindowDelegate
177344d93782SGreg Clayton     {
177444d93782SGreg Clayton     public:
177544d93782SGreg Clayton         enum class Type
177644d93782SGreg Clayton         {
177744d93782SGreg Clayton             Invalid,
177844d93782SGreg Clayton             Bar,
177944d93782SGreg Clayton             Item,
178044d93782SGreg Clayton             Separator
178144d93782SGreg Clayton         };
178244d93782SGreg Clayton 
178344d93782SGreg Clayton         // Menubar or separator constructor
178444d93782SGreg Clayton         Menu (Type type);
178544d93782SGreg Clayton 
178644d93782SGreg Clayton         // Menuitem constructor
178744d93782SGreg Clayton         Menu (const char *name,
178844d93782SGreg Clayton               const char *key_name,
178944d93782SGreg Clayton               int key_value,
179044d93782SGreg Clayton               uint64_t identifier);
179144d93782SGreg Clayton 
1792bd5ae6b4SGreg Clayton         ~Menu () override
179344d93782SGreg Clayton         {
179444d93782SGreg Clayton         }
179544d93782SGreg Clayton 
179644d93782SGreg Clayton         const MenuDelegateSP &
179744d93782SGreg Clayton         GetDelegate () const
179844d93782SGreg Clayton         {
179944d93782SGreg Clayton             return m_delegate_sp;
180044d93782SGreg Clayton         }
180144d93782SGreg Clayton 
180244d93782SGreg Clayton         void
180344d93782SGreg Clayton         SetDelegate (const MenuDelegateSP &delegate_sp)
180444d93782SGreg Clayton         {
180544d93782SGreg Clayton             m_delegate_sp = delegate_sp;
180644d93782SGreg Clayton         }
180744d93782SGreg Clayton 
180844d93782SGreg Clayton         void
180944d93782SGreg Clayton         RecalculateNameLengths();
181044d93782SGreg Clayton 
181144d93782SGreg Clayton         void
181244d93782SGreg Clayton         AddSubmenu (const MenuSP &menu_sp);
181344d93782SGreg Clayton 
181444d93782SGreg Clayton         int
181544d93782SGreg Clayton         DrawAndRunMenu (Window &window);
181644d93782SGreg Clayton 
181744d93782SGreg Clayton         void
181844d93782SGreg Clayton         DrawMenuTitle (Window &window, bool highlight);
181944d93782SGreg Clayton 
1820bd5ae6b4SGreg Clayton         bool
1821bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
182244d93782SGreg Clayton 
1823bd5ae6b4SGreg Clayton         HandleCharResult
1824bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
182544d93782SGreg Clayton 
182644d93782SGreg Clayton         MenuActionResult
182744d93782SGreg Clayton         ActionPrivate (Menu &menu)
182844d93782SGreg Clayton         {
182944d93782SGreg Clayton             MenuActionResult result = MenuActionResult::NotHandled;
183044d93782SGreg Clayton             if (m_delegate_sp)
183144d93782SGreg Clayton             {
183244d93782SGreg Clayton                 result = m_delegate_sp->MenuDelegateAction (menu);
183344d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
183444d93782SGreg Clayton                     return result;
183544d93782SGreg Clayton             }
183644d93782SGreg Clayton             else if (m_parent)
183744d93782SGreg Clayton             {
183844d93782SGreg Clayton                 result = m_parent->ActionPrivate(menu);
183944d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
184044d93782SGreg Clayton                     return result;
184144d93782SGreg Clayton             }
184244d93782SGreg Clayton             return m_canned_result;
184344d93782SGreg Clayton         }
184444d93782SGreg Clayton 
184544d93782SGreg Clayton         MenuActionResult
184644d93782SGreg Clayton         Action ()
184744d93782SGreg Clayton         {
184844d93782SGreg Clayton             // Call the recursive action so it can try to handle it
184944d93782SGreg Clayton             // with the menu delegate, and if not, try our parent menu
185044d93782SGreg Clayton             return ActionPrivate (*this);
185144d93782SGreg Clayton         }
185244d93782SGreg Clayton 
185344d93782SGreg Clayton         void
185444d93782SGreg Clayton         SetCannedResult (MenuActionResult result)
185544d93782SGreg Clayton         {
185644d93782SGreg Clayton             m_canned_result = result;
185744d93782SGreg Clayton         }
185844d93782SGreg Clayton 
185944d93782SGreg Clayton         Menus &
186044d93782SGreg Clayton         GetSubmenus()
186144d93782SGreg Clayton         {
186244d93782SGreg Clayton             return m_submenus;
186344d93782SGreg Clayton         }
186444d93782SGreg Clayton 
186544d93782SGreg Clayton         const Menus &
186644d93782SGreg Clayton         GetSubmenus() const
186744d93782SGreg Clayton         {
186844d93782SGreg Clayton             return m_submenus;
186944d93782SGreg Clayton         }
187044d93782SGreg Clayton 
187144d93782SGreg Clayton         int
187244d93782SGreg Clayton         GetSelectedSubmenuIndex () const
187344d93782SGreg Clayton         {
187444d93782SGreg Clayton             return m_selected;
187544d93782SGreg Clayton         }
187644d93782SGreg Clayton 
187744d93782SGreg Clayton         void
187844d93782SGreg Clayton         SetSelectedSubmenuIndex (int idx)
187944d93782SGreg Clayton         {
188044d93782SGreg Clayton             m_selected = idx;
188144d93782SGreg Clayton         }
188244d93782SGreg Clayton 
188344d93782SGreg Clayton         Type
188444d93782SGreg Clayton         GetType () const
188544d93782SGreg Clayton         {
188644d93782SGreg Clayton             return m_type;
188744d93782SGreg Clayton         }
188844d93782SGreg Clayton 
188944d93782SGreg Clayton         int
189044d93782SGreg Clayton         GetStartingColumn() const
189144d93782SGreg Clayton         {
189244d93782SGreg Clayton             return m_start_col;
189344d93782SGreg Clayton         }
189444d93782SGreg Clayton 
189544d93782SGreg Clayton         void
189644d93782SGreg Clayton         SetStartingColumn(int col)
189744d93782SGreg Clayton         {
189844d93782SGreg Clayton             m_start_col = col;
189944d93782SGreg Clayton         }
190044d93782SGreg Clayton 
190144d93782SGreg Clayton         int
190244d93782SGreg Clayton         GetKeyValue() const
190344d93782SGreg Clayton         {
190444d93782SGreg Clayton             return m_key_value;
190544d93782SGreg Clayton         }
190644d93782SGreg Clayton 
190744d93782SGreg Clayton         void
190844d93782SGreg Clayton         SetKeyValue(int key_value)
190944d93782SGreg Clayton         {
191044d93782SGreg Clayton             m_key_value = key_value;
191144d93782SGreg Clayton         }
191244d93782SGreg Clayton 
191344d93782SGreg Clayton         std::string &
191444d93782SGreg Clayton         GetName()
191544d93782SGreg Clayton         {
191644d93782SGreg Clayton             return m_name;
191744d93782SGreg Clayton         }
191844d93782SGreg Clayton 
191944d93782SGreg Clayton         std::string &
192044d93782SGreg Clayton         GetKeyName()
192144d93782SGreg Clayton         {
192244d93782SGreg Clayton             return m_key_name;
192344d93782SGreg Clayton         }
192444d93782SGreg Clayton 
192544d93782SGreg Clayton         int
192644d93782SGreg Clayton         GetDrawWidth () const
192744d93782SGreg Clayton         {
192844d93782SGreg Clayton             return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
192944d93782SGreg Clayton         }
193044d93782SGreg Clayton 
193144d93782SGreg Clayton 
193244d93782SGreg Clayton         uint64_t
193344d93782SGreg Clayton         GetIdentifier() const
193444d93782SGreg Clayton         {
193544d93782SGreg Clayton             return m_identifier;
193644d93782SGreg Clayton         }
193744d93782SGreg Clayton 
193844d93782SGreg Clayton         void
193944d93782SGreg Clayton         SetIdentifier (uint64_t identifier)
194044d93782SGreg Clayton         {
194144d93782SGreg Clayton             m_identifier = identifier;
194244d93782SGreg Clayton         }
194344d93782SGreg Clayton 
194444d93782SGreg Clayton     protected:
194544d93782SGreg Clayton         std::string m_name;
194644d93782SGreg Clayton         std::string m_key_name;
194744d93782SGreg Clayton         uint64_t m_identifier;
194844d93782SGreg Clayton         Type m_type;
194944d93782SGreg Clayton         int m_key_value;
195044d93782SGreg Clayton         int m_start_col;
195144d93782SGreg Clayton         int m_max_submenu_name_length;
195244d93782SGreg Clayton         int m_max_submenu_key_name_length;
195344d93782SGreg Clayton         int m_selected;
195444d93782SGreg Clayton         Menu *m_parent;
195544d93782SGreg Clayton         Menus m_submenus;
195644d93782SGreg Clayton         WindowSP m_menu_window_sp;
195744d93782SGreg Clayton         MenuActionResult m_canned_result;
195844d93782SGreg Clayton         MenuDelegateSP m_delegate_sp;
195944d93782SGreg Clayton     };
196044d93782SGreg Clayton 
196144d93782SGreg Clayton     // Menubar or separator constructor
196244d93782SGreg Clayton     Menu::Menu (Type type) :
196344d93782SGreg Clayton         m_name (),
196444d93782SGreg Clayton         m_key_name (),
196544d93782SGreg Clayton         m_identifier (0),
196644d93782SGreg Clayton         m_type (type),
196744d93782SGreg Clayton         m_key_value (0),
196844d93782SGreg Clayton         m_start_col (0),
196944d93782SGreg Clayton         m_max_submenu_name_length (0),
197044d93782SGreg Clayton         m_max_submenu_key_name_length (0),
197144d93782SGreg Clayton         m_selected (0),
197244d93782SGreg Clayton         m_parent (NULL),
197344d93782SGreg Clayton         m_submenus (),
197444d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
197544d93782SGreg Clayton         m_delegate_sp()
197644d93782SGreg Clayton     {
197744d93782SGreg Clayton     }
197844d93782SGreg Clayton 
197944d93782SGreg Clayton     // Menuitem constructor
198044d93782SGreg Clayton     Menu::Menu (const char *name,
198144d93782SGreg Clayton                 const char *key_name,
198244d93782SGreg Clayton                 int key_value,
198344d93782SGreg Clayton                 uint64_t identifier) :
198444d93782SGreg Clayton         m_name (),
198544d93782SGreg Clayton         m_key_name (),
198644d93782SGreg Clayton         m_identifier (identifier),
198744d93782SGreg Clayton         m_type (Type::Invalid),
198844d93782SGreg Clayton         m_key_value (key_value),
198944d93782SGreg Clayton         m_start_col (0),
199044d93782SGreg Clayton         m_max_submenu_name_length (0),
199144d93782SGreg Clayton         m_max_submenu_key_name_length (0),
199244d93782SGreg Clayton         m_selected (0),
199344d93782SGreg Clayton         m_parent (NULL),
199444d93782SGreg Clayton         m_submenus (),
199544d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
199644d93782SGreg Clayton         m_delegate_sp()
199744d93782SGreg Clayton     {
199844d93782SGreg Clayton         if (name && name[0])
199944d93782SGreg Clayton         {
200044d93782SGreg Clayton             m_name = name;
200144d93782SGreg Clayton             m_type = Type::Item;
200244d93782SGreg Clayton             if (key_name && key_name[0])
200344d93782SGreg Clayton                 m_key_name = key_name;
200444d93782SGreg Clayton         }
200544d93782SGreg Clayton         else
200644d93782SGreg Clayton         {
200744d93782SGreg Clayton             m_type = Type::Separator;
200844d93782SGreg Clayton         }
200944d93782SGreg Clayton     }
201044d93782SGreg Clayton 
201144d93782SGreg Clayton     void
201244d93782SGreg Clayton     Menu::RecalculateNameLengths()
201344d93782SGreg Clayton     {
201444d93782SGreg Clayton         m_max_submenu_name_length = 0;
201544d93782SGreg Clayton         m_max_submenu_key_name_length = 0;
201644d93782SGreg Clayton         Menus &submenus = GetSubmenus();
201744d93782SGreg Clayton         const size_t num_submenus = submenus.size();
201844d93782SGreg Clayton         for (size_t i=0; i<num_submenus; ++i)
201944d93782SGreg Clayton         {
202044d93782SGreg Clayton             Menu *submenu = submenus[i].get();
20213985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
202244d93782SGreg Clayton                 m_max_submenu_name_length = submenu->m_name.size();
20233985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size())
202444d93782SGreg Clayton                 m_max_submenu_key_name_length = submenu->m_key_name.size();
202544d93782SGreg Clayton         }
202644d93782SGreg Clayton     }
202744d93782SGreg Clayton 
202844d93782SGreg Clayton     void
202944d93782SGreg Clayton     Menu::AddSubmenu (const MenuSP &menu_sp)
203044d93782SGreg Clayton     {
203144d93782SGreg Clayton         menu_sp->m_parent = this;
20323985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
203344d93782SGreg Clayton             m_max_submenu_name_length = menu_sp->m_name.size();
20343985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size())
203544d93782SGreg Clayton             m_max_submenu_key_name_length = menu_sp->m_key_name.size();
203644d93782SGreg Clayton         m_submenus.push_back(menu_sp);
203744d93782SGreg Clayton     }
203844d93782SGreg Clayton 
203944d93782SGreg Clayton     void
204044d93782SGreg Clayton     Menu::DrawMenuTitle (Window &window, bool highlight)
204144d93782SGreg Clayton     {
204244d93782SGreg Clayton         if (m_type == Type::Separator)
204344d93782SGreg Clayton         {
204444d93782SGreg Clayton             window.MoveCursor(0, window.GetCursorY());
204544d93782SGreg Clayton             window.PutChar(ACS_LTEE);
204644d93782SGreg Clayton             int width = window.GetWidth();
204744d93782SGreg Clayton             if (width > 2)
204844d93782SGreg Clayton             {
204944d93782SGreg Clayton                 width -= 2;
20503985c8c6SSaleem Abdulrasool                 for (int i=0; i< width; ++i)
205144d93782SGreg Clayton                     window.PutChar(ACS_HLINE);
205244d93782SGreg Clayton             }
205344d93782SGreg Clayton             window.PutChar(ACS_RTEE);
205444d93782SGreg Clayton         }
205544d93782SGreg Clayton         else
205644d93782SGreg Clayton         {
205744d93782SGreg Clayton             const int shortcut_key = m_key_value;
205844d93782SGreg Clayton             bool underlined_shortcut = false;
205944d93782SGreg Clayton             const attr_t hilgight_attr = A_REVERSE;
206044d93782SGreg Clayton             if (highlight)
206144d93782SGreg Clayton                 window.AttributeOn(hilgight_attr);
206244d93782SGreg Clayton             if (isprint(shortcut_key))
206344d93782SGreg Clayton             {
206444d93782SGreg Clayton                 size_t lower_pos = m_name.find(tolower(shortcut_key));
206544d93782SGreg Clayton                 size_t upper_pos = m_name.find(toupper(shortcut_key));
206644d93782SGreg Clayton                 const char *name = m_name.c_str();
206744d93782SGreg Clayton                 size_t pos = std::min<size_t>(lower_pos, upper_pos);
206844d93782SGreg Clayton                 if (pos != std::string::npos)
206944d93782SGreg Clayton                 {
207044d93782SGreg Clayton                     underlined_shortcut = true;
207144d93782SGreg Clayton                     if (pos > 0)
207244d93782SGreg Clayton                     {
207344d93782SGreg Clayton                         window.PutCString(name, pos);
207444d93782SGreg Clayton                         name += pos;
207544d93782SGreg Clayton                     }
207644d93782SGreg Clayton                     const attr_t shortcut_attr = A_UNDERLINE|A_BOLD;
207744d93782SGreg Clayton                     window.AttributeOn (shortcut_attr);
207844d93782SGreg Clayton                     window.PutChar(name[0]);
207944d93782SGreg Clayton                     window.AttributeOff(shortcut_attr);
208044d93782SGreg Clayton                     name++;
208144d93782SGreg Clayton                     if (name[0])
208244d93782SGreg Clayton                         window.PutCString(name);
208344d93782SGreg Clayton                 }
208444d93782SGreg Clayton             }
208544d93782SGreg Clayton 
208644d93782SGreg Clayton             if (!underlined_shortcut)
208744d93782SGreg Clayton             {
208844d93782SGreg Clayton                 window.PutCString(m_name.c_str());
208944d93782SGreg Clayton             }
209044d93782SGreg Clayton 
209144d93782SGreg Clayton             if (highlight)
209244d93782SGreg Clayton                 window.AttributeOff(hilgight_attr);
209344d93782SGreg Clayton 
209444d93782SGreg Clayton             if (m_key_name.empty())
209544d93782SGreg Clayton             {
209644d93782SGreg Clayton                 if (!underlined_shortcut && isprint(m_key_value))
209744d93782SGreg Clayton                 {
209844d93782SGreg Clayton                     window.AttributeOn (COLOR_PAIR(3));
209944d93782SGreg Clayton                     window.Printf (" (%c)", m_key_value);
210044d93782SGreg Clayton                     window.AttributeOff (COLOR_PAIR(3));
210144d93782SGreg Clayton                 }
210244d93782SGreg Clayton             }
210344d93782SGreg Clayton             else
210444d93782SGreg Clayton             {
210544d93782SGreg Clayton                 window.AttributeOn (COLOR_PAIR(3));
210644d93782SGreg Clayton                 window.Printf (" (%s)", m_key_name.c_str());
210744d93782SGreg Clayton                 window.AttributeOff (COLOR_PAIR(3));
210844d93782SGreg Clayton             }
210944d93782SGreg Clayton         }
211044d93782SGreg Clayton     }
211144d93782SGreg Clayton 
211244d93782SGreg Clayton     bool
211344d93782SGreg Clayton     Menu::WindowDelegateDraw (Window &window, bool force)
211444d93782SGreg Clayton     {
211544d93782SGreg Clayton         Menus &submenus = GetSubmenus();
211644d93782SGreg Clayton         const size_t num_submenus = submenus.size();
211744d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
211844d93782SGreg Clayton         Menu::Type menu_type = GetType ();
211944d93782SGreg Clayton         switch (menu_type)
212044d93782SGreg Clayton         {
212144d93782SGreg Clayton         case  Menu::Type::Bar:
212244d93782SGreg Clayton             {
212344d93782SGreg Clayton                 window.SetBackground(2);
212444d93782SGreg Clayton                 window.MoveCursor(0, 0);
212544d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
212644d93782SGreg Clayton                 {
212744d93782SGreg Clayton                     Menu *menu = submenus[i].get();
212844d93782SGreg Clayton                     if (i > 0)
212944d93782SGreg Clayton                         window.PutChar(' ');
213044d93782SGreg Clayton                     menu->SetStartingColumn (window.GetCursorX());
213144d93782SGreg Clayton                     window.PutCString("| ");
213244d93782SGreg Clayton                     menu->DrawMenuTitle (window, false);
213344d93782SGreg Clayton                 }
213444d93782SGreg Clayton                 window.PutCString(" |");
213544d93782SGreg Clayton                 window.DeferredRefresh();
213644d93782SGreg Clayton             }
213744d93782SGreg Clayton             break;
213844d93782SGreg Clayton 
213944d93782SGreg Clayton         case Menu::Type::Item:
214044d93782SGreg Clayton             {
214144d93782SGreg Clayton                 int y = 1;
214244d93782SGreg Clayton                 int x = 3;
214344d93782SGreg Clayton                 // Draw the menu
214444d93782SGreg Clayton                 int cursor_x = 0;
214544d93782SGreg Clayton                 int cursor_y = 0;
214644d93782SGreg Clayton                 window.Erase();
214744d93782SGreg Clayton                 window.SetBackground(2);
214844d93782SGreg Clayton                 window.Box();
214944d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
215044d93782SGreg Clayton                 {
21513985c8c6SSaleem Abdulrasool                     const bool is_selected =
21523985c8c6SSaleem Abdulrasool                       (i == static_cast<size_t>(selected_idx));
215344d93782SGreg Clayton                     window.MoveCursor(x, y + i);
215444d93782SGreg Clayton                     if (is_selected)
215544d93782SGreg Clayton                     {
215644d93782SGreg Clayton                         // Remember where we want the cursor to be
215744d93782SGreg Clayton                         cursor_x = x-1;
215844d93782SGreg Clayton                         cursor_y = y+i;
215944d93782SGreg Clayton                     }
216044d93782SGreg Clayton                     submenus[i]->DrawMenuTitle (window, is_selected);
216144d93782SGreg Clayton                 }
216244d93782SGreg Clayton                 window.MoveCursor(cursor_x, cursor_y);
216344d93782SGreg Clayton                 window.DeferredRefresh();
216444d93782SGreg Clayton             }
216544d93782SGreg Clayton             break;
216644d93782SGreg Clayton 
216744d93782SGreg Clayton         default:
216844d93782SGreg Clayton         case Menu::Type::Separator:
216944d93782SGreg Clayton             break;
217044d93782SGreg Clayton         }
217144d93782SGreg Clayton         return true; // Drawing handled...
217244d93782SGreg Clayton     }
217344d93782SGreg Clayton 
217444d93782SGreg Clayton     HandleCharResult
217544d93782SGreg Clayton     Menu::WindowDelegateHandleChar (Window &window, int key)
217644d93782SGreg Clayton     {
217744d93782SGreg Clayton         HandleCharResult result = eKeyNotHandled;
217844d93782SGreg Clayton 
217944d93782SGreg Clayton         Menus &submenus = GetSubmenus();
218044d93782SGreg Clayton         const size_t num_submenus = submenus.size();
218144d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
218244d93782SGreg Clayton         Menu::Type menu_type = GetType ();
218344d93782SGreg Clayton         if (menu_type == Menu::Type::Bar)
218444d93782SGreg Clayton         {
218544d93782SGreg Clayton             MenuSP run_menu_sp;
218644d93782SGreg Clayton             switch (key)
218744d93782SGreg Clayton             {
218844d93782SGreg Clayton                 case KEY_DOWN:
218944d93782SGreg Clayton                 case KEY_UP:
219044d93782SGreg Clayton                     // Show last menu or first menu
21913985c8c6SSaleem Abdulrasool                     if (selected_idx < static_cast<int>(num_submenus))
219244d93782SGreg Clayton                         run_menu_sp = submenus[selected_idx];
219344d93782SGreg Clayton                     else if (!submenus.empty())
219444d93782SGreg Clayton                         run_menu_sp = submenus.front();
219544d93782SGreg Clayton                     result = eKeyHandled;
219644d93782SGreg Clayton                     break;
219744d93782SGreg Clayton 
219844d93782SGreg Clayton                 case KEY_RIGHT:
219944d93782SGreg Clayton                 {
220044d93782SGreg Clayton                     ++m_selected;
22013985c8c6SSaleem Abdulrasool                     if (m_selected >= static_cast<int>(num_submenus))
220244d93782SGreg Clayton                         m_selected = 0;
22033985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
220444d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
220544d93782SGreg Clayton                     else if (!submenus.empty())
220644d93782SGreg Clayton                         run_menu_sp = submenus.front();
220744d93782SGreg Clayton                     result = eKeyHandled;
220844d93782SGreg Clayton                 }
220944d93782SGreg Clayton                     break;
221044d93782SGreg Clayton 
221144d93782SGreg Clayton                 case KEY_LEFT:
221244d93782SGreg Clayton                 {
221344d93782SGreg Clayton                     --m_selected;
221444d93782SGreg Clayton                     if (m_selected < 0)
221544d93782SGreg Clayton                         m_selected = num_submenus - 1;
22163985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
221744d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
221844d93782SGreg Clayton                     else if (!submenus.empty())
221944d93782SGreg Clayton                         run_menu_sp = submenus.front();
222044d93782SGreg Clayton                     result = eKeyHandled;
222144d93782SGreg Clayton                 }
222244d93782SGreg Clayton                     break;
222344d93782SGreg Clayton 
222444d93782SGreg Clayton                 default:
222544d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
222644d93782SGreg Clayton                     {
222744d93782SGreg Clayton                         if (submenus[i]->GetKeyValue() == key)
222844d93782SGreg Clayton                         {
222944d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
223044d93782SGreg Clayton                             run_menu_sp = submenus[i];
223144d93782SGreg Clayton                             result = eKeyHandled;
223244d93782SGreg Clayton                             break;
223344d93782SGreg Clayton                         }
223444d93782SGreg Clayton                     }
223544d93782SGreg Clayton                     break;
223644d93782SGreg Clayton             }
223744d93782SGreg Clayton 
223844d93782SGreg Clayton             if (run_menu_sp)
223944d93782SGreg Clayton             {
224044d93782SGreg Clayton                 // Run the action on this menu in case we need to populate the
224144d93782SGreg Clayton                 // menu with dynamic content and also in case check marks, and
224244d93782SGreg Clayton                 // any other menu decorations need to be caclulated
224344d93782SGreg Clayton                 if (run_menu_sp->Action() == MenuActionResult::Quit)
224444d93782SGreg Clayton                     return eQuitApplication;
224544d93782SGreg Clayton 
224644d93782SGreg Clayton                 Rect menu_bounds;
224744d93782SGreg Clayton                 menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
224844d93782SGreg Clayton                 menu_bounds.origin.y = 1;
224944d93782SGreg Clayton                 menu_bounds.size.width = run_menu_sp->GetDrawWidth();
225044d93782SGreg Clayton                 menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
225144d93782SGreg Clayton                 if (m_menu_window_sp)
225244d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
225344d93782SGreg Clayton 
225444d93782SGreg Clayton                 m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(),
225544d93782SGreg Clayton                                                                         menu_bounds,
225644d93782SGreg Clayton                                                                         true);
225744d93782SGreg Clayton                 m_menu_window_sp->SetDelegate (run_menu_sp);
225844d93782SGreg Clayton             }
225944d93782SGreg Clayton         }
226044d93782SGreg Clayton         else if (menu_type == Menu::Type::Item)
226144d93782SGreg Clayton         {
226244d93782SGreg Clayton             switch (key)
226344d93782SGreg Clayton             {
226444d93782SGreg Clayton                 case KEY_DOWN:
226544d93782SGreg Clayton                     if (m_submenus.size() > 1)
226644d93782SGreg Clayton                     {
226744d93782SGreg Clayton                         const int start_select = m_selected;
226844d93782SGreg Clayton                         while (++m_selected != start_select)
226944d93782SGreg Clayton                         {
22703985c8c6SSaleem Abdulrasool                             if (static_cast<size_t>(m_selected) >= num_submenus)
227144d93782SGreg Clayton                                 m_selected = 0;
227244d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
227344d93782SGreg Clayton                                 continue;
227444d93782SGreg Clayton                             else
227544d93782SGreg Clayton                                 break;
227644d93782SGreg Clayton                         }
227744d93782SGreg Clayton                         return eKeyHandled;
227844d93782SGreg Clayton                     }
227944d93782SGreg Clayton                     break;
228044d93782SGreg Clayton 
228144d93782SGreg Clayton                 case KEY_UP:
228244d93782SGreg Clayton                     if (m_submenus.size() > 1)
228344d93782SGreg Clayton                     {
228444d93782SGreg Clayton                         const int start_select = m_selected;
228544d93782SGreg Clayton                         while (--m_selected != start_select)
228644d93782SGreg Clayton                         {
22873985c8c6SSaleem Abdulrasool                             if (m_selected < static_cast<int>(0))
228844d93782SGreg Clayton                                 m_selected = num_submenus - 1;
228944d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
229044d93782SGreg Clayton                                 continue;
229144d93782SGreg Clayton                             else
229244d93782SGreg Clayton                                 break;
229344d93782SGreg Clayton                         }
229444d93782SGreg Clayton                         return eKeyHandled;
229544d93782SGreg Clayton                     }
229644d93782SGreg Clayton                     break;
229744d93782SGreg Clayton 
229844d93782SGreg Clayton                 case KEY_RETURN:
22993985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(selected_idx) < num_submenus)
230044d93782SGreg Clayton                     {
230144d93782SGreg Clayton                         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
230244d93782SGreg Clayton                             return eQuitApplication;
230344d93782SGreg Clayton                         window.GetParent()->RemoveSubWindow(&window);
230444d93782SGreg Clayton                         return eKeyHandled;
230544d93782SGreg Clayton                     }
230644d93782SGreg Clayton                     break;
230744d93782SGreg Clayton 
230844d93782SGreg Clayton                 case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences
230944d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(&window);
231044d93782SGreg Clayton                     return eKeyHandled;
231144d93782SGreg Clayton 
231244d93782SGreg Clayton                 default:
231344d93782SGreg Clayton                 {
231444d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
231544d93782SGreg Clayton                     {
231644d93782SGreg Clayton                         Menu *menu = submenus[i].get();
231744d93782SGreg Clayton                         if (menu->GetKeyValue() == key)
231844d93782SGreg Clayton                         {
231944d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
232044d93782SGreg Clayton                             window.GetParent()->RemoveSubWindow(&window);
232144d93782SGreg Clayton                             if (menu->Action() == MenuActionResult::Quit)
232244d93782SGreg Clayton                                 return eQuitApplication;
232344d93782SGreg Clayton                             return eKeyHandled;
232444d93782SGreg Clayton                         }
232544d93782SGreg Clayton                     }
232644d93782SGreg Clayton                 }
232744d93782SGreg Clayton                     break;
232844d93782SGreg Clayton 
232944d93782SGreg Clayton             }
233044d93782SGreg Clayton         }
233144d93782SGreg Clayton         else if (menu_type == Menu::Type::Separator)
233244d93782SGreg Clayton         {
233344d93782SGreg Clayton 
233444d93782SGreg Clayton         }
233544d93782SGreg Clayton         return result;
233644d93782SGreg Clayton     }
233744d93782SGreg Clayton 
233844d93782SGreg Clayton 
233944d93782SGreg Clayton     class Application
234044d93782SGreg Clayton     {
234144d93782SGreg Clayton     public:
234244d93782SGreg Clayton         Application (FILE *in, FILE *out) :
234344d93782SGreg Clayton             m_window_sp(),
234444d93782SGreg Clayton             m_screen (NULL),
234544d93782SGreg Clayton             m_in (in),
234644d93782SGreg Clayton             m_out (out)
234744d93782SGreg Clayton         {
234844d93782SGreg Clayton 
234944d93782SGreg Clayton         }
235044d93782SGreg Clayton 
235144d93782SGreg Clayton         ~Application ()
235244d93782SGreg Clayton         {
235344d93782SGreg Clayton             m_window_delegates.clear();
235444d93782SGreg Clayton             m_window_sp.reset();
235544d93782SGreg Clayton             if (m_screen)
235644d93782SGreg Clayton             {
235744d93782SGreg Clayton                 ::delscreen(m_screen);
235844d93782SGreg Clayton                 m_screen = NULL;
235944d93782SGreg Clayton             }
236044d93782SGreg Clayton         }
236144d93782SGreg Clayton 
236244d93782SGreg Clayton         void
236344d93782SGreg Clayton         Initialize ()
236444d93782SGreg Clayton         {
236544d93782SGreg Clayton             ::setlocale(LC_ALL, "");
236644d93782SGreg Clayton             ::setlocale(LC_CTYPE, "");
236744d93782SGreg Clayton #if 0
236844d93782SGreg Clayton             ::initscr();
236944d93782SGreg Clayton #else
237044d93782SGreg Clayton             m_screen = ::newterm(NULL, m_out, m_in);
237144d93782SGreg Clayton #endif
237244d93782SGreg Clayton             ::start_color();
237344d93782SGreg Clayton             ::curs_set(0);
237444d93782SGreg Clayton             ::noecho();
237544d93782SGreg Clayton             ::keypad(stdscr,TRUE);
237644d93782SGreg Clayton         }
237744d93782SGreg Clayton 
237844d93782SGreg Clayton         void
237944d93782SGreg Clayton         Terminate ()
238044d93782SGreg Clayton         {
238144d93782SGreg Clayton             ::endwin();
238244d93782SGreg Clayton         }
238344d93782SGreg Clayton 
238444d93782SGreg Clayton         void
238544d93782SGreg Clayton         Run (Debugger &debugger)
238644d93782SGreg Clayton         {
238744d93782SGreg Clayton             bool done = false;
238844d93782SGreg Clayton             int delay_in_tenths_of_a_second = 1;
238944d93782SGreg Clayton 
239044d93782SGreg Clayton             // Alas the threading model in curses is a bit lame so we need to
239144d93782SGreg Clayton             // resort to polling every 0.5 seconds. We could poll for stdin
239244d93782SGreg Clayton             // ourselves and then pass the keys down but then we need to
239344d93782SGreg Clayton             // translate all of the escape sequences ourselves. So we resort to
239444d93782SGreg Clayton             // polling for input because we need to receive async process events
239544d93782SGreg Clayton             // while in this loop.
239644d93782SGreg Clayton 
239744d93782SGreg Clayton             halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar()
239844d93782SGreg Clayton 
239944d93782SGreg Clayton             ListenerSP listener_sp (new Listener ("lldb.IOHandler.curses.Application"));
240044d93782SGreg Clayton             ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
240144d93782SGreg Clayton             ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
240244d93782SGreg Clayton             ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
240344d93782SGreg Clayton             debugger.EnableForwardEvents (listener_sp);
240444d93782SGreg Clayton 
240544d93782SGreg Clayton             bool update = true;
240644d93782SGreg Clayton #if defined(__APPLE__)
240744d93782SGreg Clayton             std::deque<int> escape_chars;
240844d93782SGreg Clayton #endif
240944d93782SGreg Clayton 
241044d93782SGreg Clayton             while (!done)
241144d93782SGreg Clayton             {
241244d93782SGreg Clayton                 if (update)
241344d93782SGreg Clayton                 {
241444d93782SGreg Clayton                     m_window_sp->Draw(false);
241544d93782SGreg Clayton                     // All windows should be calling Window::DeferredRefresh() instead
241644d93782SGreg Clayton                     // of Window::Refresh() so we can do a single update and avoid
241744d93782SGreg Clayton                     // any screen blinking
241844d93782SGreg Clayton                     update_panels();
241944d93782SGreg Clayton 
242044d93782SGreg Clayton                     // Cursor hiding isn't working on MacOSX, so hide it in the top left corner
242144d93782SGreg Clayton                     m_window_sp->MoveCursor(0, 0);
242244d93782SGreg Clayton 
242344d93782SGreg Clayton                     doupdate();
242444d93782SGreg Clayton                     update = false;
242544d93782SGreg Clayton                 }
242644d93782SGreg Clayton 
242744d93782SGreg Clayton #if defined(__APPLE__)
242844d93782SGreg Clayton                 // Terminal.app doesn't map its function keys correctly, F1-F4 default to:
242944d93782SGreg Clayton                 // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible
243044d93782SGreg Clayton                 int ch;
243144d93782SGreg Clayton                 if (escape_chars.empty())
243244d93782SGreg Clayton                     ch = m_window_sp->GetChar();
243344d93782SGreg Clayton                 else
243444d93782SGreg Clayton                 {
243544d93782SGreg Clayton                     ch = escape_chars.front();
243644d93782SGreg Clayton                     escape_chars.pop_front();
243744d93782SGreg Clayton                 }
243844d93782SGreg Clayton                 if (ch == KEY_ESCAPE)
243944d93782SGreg Clayton                 {
244044d93782SGreg Clayton                     int ch2 = m_window_sp->GetChar();
244144d93782SGreg Clayton                     if (ch2 == 'O')
244244d93782SGreg Clayton                     {
244344d93782SGreg Clayton                         int ch3 = m_window_sp->GetChar();
244444d93782SGreg Clayton                         switch (ch3)
244544d93782SGreg Clayton                         {
244644d93782SGreg Clayton                             case 'P': ch = KEY_F(1); break;
244744d93782SGreg Clayton                             case 'Q': ch = KEY_F(2); break;
244844d93782SGreg Clayton                             case 'R': ch = KEY_F(3); break;
244944d93782SGreg Clayton                             case 'S': ch = KEY_F(4); break;
245044d93782SGreg Clayton                             default:
245144d93782SGreg Clayton                                 escape_chars.push_back(ch2);
245244d93782SGreg Clayton                                 if (ch3 != -1)
245344d93782SGreg Clayton                                     escape_chars.push_back(ch3);
245444d93782SGreg Clayton                                 break;
245544d93782SGreg Clayton                         }
245644d93782SGreg Clayton                     }
245744d93782SGreg Clayton                     else if (ch2 != -1)
245844d93782SGreg Clayton                         escape_chars.push_back(ch2);
245944d93782SGreg Clayton                 }
246044d93782SGreg Clayton #else
246144d93782SGreg Clayton                 int ch = m_window_sp->GetChar();
246244d93782SGreg Clayton 
246344d93782SGreg Clayton #endif
246444d93782SGreg Clayton                 if (ch == -1)
246544d93782SGreg Clayton                 {
246644d93782SGreg Clayton                     if (feof(m_in) || ferror(m_in))
246744d93782SGreg Clayton                     {
246844d93782SGreg Clayton                         done = true;
246944d93782SGreg Clayton                     }
247044d93782SGreg Clayton                     else
247144d93782SGreg Clayton                     {
247244d93782SGreg Clayton                         // Just a timeout from using halfdelay(), check for events
247344d93782SGreg Clayton                         EventSP event_sp;
247444d93782SGreg Clayton                         while (listener_sp->PeekAtNextEvent())
247544d93782SGreg Clayton                         {
247644d93782SGreg Clayton                             listener_sp->GetNextEvent(event_sp);
247744d93782SGreg Clayton 
247844d93782SGreg Clayton                             if (event_sp)
247944d93782SGreg Clayton                             {
248044d93782SGreg Clayton                                 Broadcaster *broadcaster = event_sp->GetBroadcaster();
248144d93782SGreg Clayton                                 if (broadcaster)
248244d93782SGreg Clayton                                 {
248344d93782SGreg Clayton                                     //uint32_t event_type = event_sp->GetType();
248444d93782SGreg Clayton                                     ConstString broadcaster_class (broadcaster->GetBroadcasterClass());
248544d93782SGreg Clayton                                     if (broadcaster_class == broadcaster_class_process)
248644d93782SGreg Clayton                                     {
2487ec990867SGreg Clayton                                         debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
248844d93782SGreg Clayton                                         update = true;
248944d93782SGreg Clayton                                         continue; // Don't get any key, just update our view
249044d93782SGreg Clayton                                     }
249144d93782SGreg Clayton                                 }
249244d93782SGreg Clayton                             }
249344d93782SGreg Clayton                         }
249444d93782SGreg Clayton                     }
249544d93782SGreg Clayton                 }
249644d93782SGreg Clayton                 else
249744d93782SGreg Clayton                 {
249844d93782SGreg Clayton                     HandleCharResult key_result = m_window_sp->HandleChar(ch);
249944d93782SGreg Clayton                     switch (key_result)
250044d93782SGreg Clayton                     {
250144d93782SGreg Clayton                         case eKeyHandled:
2502ec990867SGreg Clayton                             debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
250344d93782SGreg Clayton                             update = true;
250444d93782SGreg Clayton                             break;
250544d93782SGreg Clayton                         case eKeyNotHandled:
250644d93782SGreg Clayton                             break;
250744d93782SGreg Clayton                         case eQuitApplication:
250844d93782SGreg Clayton                             done = true;
250944d93782SGreg Clayton                             break;
251044d93782SGreg Clayton                     }
251144d93782SGreg Clayton                 }
251244d93782SGreg Clayton             }
251344d93782SGreg Clayton 
251444d93782SGreg Clayton             debugger.CancelForwardEvents (listener_sp);
251544d93782SGreg Clayton 
251644d93782SGreg Clayton         }
251744d93782SGreg Clayton 
251844d93782SGreg Clayton         WindowSP &
251944d93782SGreg Clayton         GetMainWindow ()
252044d93782SGreg Clayton         {
252144d93782SGreg Clayton             if (!m_window_sp)
252244d93782SGreg Clayton                 m_window_sp.reset (new Window ("main", stdscr, false));
252344d93782SGreg Clayton             return m_window_sp;
252444d93782SGreg Clayton         }
252544d93782SGreg Clayton 
252644d93782SGreg Clayton         WindowDelegates &
252744d93782SGreg Clayton         GetWindowDelegates ()
252844d93782SGreg Clayton         {
252944d93782SGreg Clayton             return m_window_delegates;
253044d93782SGreg Clayton         }
253144d93782SGreg Clayton 
253244d93782SGreg Clayton     protected:
253344d93782SGreg Clayton         WindowSP m_window_sp;
253444d93782SGreg Clayton         WindowDelegates m_window_delegates;
253544d93782SGreg Clayton         SCREEN *m_screen;
253644d93782SGreg Clayton         FILE *m_in;
253744d93782SGreg Clayton         FILE *m_out;
253844d93782SGreg Clayton     };
253944d93782SGreg Clayton 
254044d93782SGreg Clayton 
254144d93782SGreg Clayton } // namespace curses
254244d93782SGreg Clayton 
254344d93782SGreg Clayton 
254444d93782SGreg Clayton using namespace curses;
254544d93782SGreg Clayton 
254644d93782SGreg Clayton struct Row
254744d93782SGreg Clayton {
254844d93782SGreg Clayton     ValueObjectSP valobj;
254944d93782SGreg Clayton     Row *parent;
255044d93782SGreg Clayton     int row_idx;
255144d93782SGreg Clayton     int x;
255244d93782SGreg Clayton     int y;
255344d93782SGreg Clayton     bool might_have_children;
255444d93782SGreg Clayton     bool expanded;
255544d93782SGreg Clayton     bool calculated_children;
255644d93782SGreg Clayton     std::vector<Row> children;
255744d93782SGreg Clayton 
255844d93782SGreg Clayton     Row (const ValueObjectSP &v, Row *p) :
255944d93782SGreg Clayton     valobj (v),
256044d93782SGreg Clayton     parent (p),
256144d93782SGreg Clayton     row_idx(0),
256244d93782SGreg Clayton     x(1),
256344d93782SGreg Clayton     y(1),
256444d93782SGreg Clayton     might_have_children (v ? v->MightHaveChildren() : false),
256544d93782SGreg Clayton     expanded (false),
256644d93782SGreg Clayton     calculated_children (false),
256744d93782SGreg Clayton     children()
256844d93782SGreg Clayton     {
256944d93782SGreg Clayton     }
257044d93782SGreg Clayton 
257144d93782SGreg Clayton     size_t
257244d93782SGreg Clayton     GetDepth () const
257344d93782SGreg Clayton     {
257444d93782SGreg Clayton         if (parent)
257544d93782SGreg Clayton             return 1 + parent->GetDepth();
257644d93782SGreg Clayton         return 0;
257744d93782SGreg Clayton     }
257844d93782SGreg Clayton 
257944d93782SGreg Clayton     void
258044d93782SGreg Clayton     Expand()
258144d93782SGreg Clayton     {
258244d93782SGreg Clayton         expanded = true;
258344d93782SGreg Clayton         if (!calculated_children)
258444d93782SGreg Clayton         {
258544d93782SGreg Clayton             calculated_children = true;
258644d93782SGreg Clayton             if (valobj)
258744d93782SGreg Clayton             {
258844d93782SGreg Clayton                 const size_t num_children = valobj->GetNumChildren();
258944d93782SGreg Clayton                 for (size_t i=0; i<num_children; ++i)
259044d93782SGreg Clayton                 {
259144d93782SGreg Clayton                     children.push_back(Row (valobj->GetChildAtIndex(i, true), this));
259244d93782SGreg Clayton                 }
259344d93782SGreg Clayton             }
259444d93782SGreg Clayton         }
259544d93782SGreg Clayton     }
259644d93782SGreg Clayton 
259744d93782SGreg Clayton     void
259844d93782SGreg Clayton     Unexpand ()
259944d93782SGreg Clayton     {
260044d93782SGreg Clayton         expanded = false;
260144d93782SGreg Clayton     }
260244d93782SGreg Clayton 
260344d93782SGreg Clayton     void
260444d93782SGreg Clayton     DrawTree (Window &window)
260544d93782SGreg Clayton     {
260644d93782SGreg Clayton         if (parent)
260744d93782SGreg Clayton             parent->DrawTreeForChild (window, this, 0);
260844d93782SGreg Clayton 
260944d93782SGreg Clayton         if (might_have_children)
261044d93782SGreg Clayton         {
261144d93782SGreg Clayton             // It we can get UTF8 characters to work we should try to use the "symbol"
261244d93782SGreg Clayton             // UTF8 string below
261344d93782SGreg Clayton //            const char *symbol = "";
261444d93782SGreg Clayton //            if (row.expanded)
261544d93782SGreg Clayton //                symbol = "\xe2\x96\xbd ";
261644d93782SGreg Clayton //            else
261744d93782SGreg Clayton //                symbol = "\xe2\x96\xb7 ";
261844d93782SGreg Clayton //            window.PutCString (symbol);
261944d93782SGreg Clayton 
262044d93782SGreg Clayton             // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
262144d93782SGreg Clayton             // 'v' or '>' character...
262244d93782SGreg Clayton //            if (expanded)
262344d93782SGreg Clayton //                window.PutChar (ACS_DARROW);
262444d93782SGreg Clayton //            else
262544d93782SGreg Clayton //                window.PutChar (ACS_RARROW);
262644d93782SGreg Clayton             // Since we can't find any good looking right arrow/down arrow
262744d93782SGreg Clayton             // symbols, just use a diamond...
262844d93782SGreg Clayton             window.PutChar (ACS_DIAMOND);
262944d93782SGreg Clayton             window.PutChar (ACS_HLINE);
263044d93782SGreg Clayton         }
263144d93782SGreg Clayton     }
263244d93782SGreg Clayton 
263344d93782SGreg Clayton     void
263444d93782SGreg Clayton     DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth)
263544d93782SGreg Clayton     {
263644d93782SGreg Clayton         if (parent)
263744d93782SGreg Clayton             parent->DrawTreeForChild (window, this, reverse_depth + 1);
263844d93782SGreg Clayton 
263944d93782SGreg Clayton         if (&children.back() == child)
264044d93782SGreg Clayton         {
264144d93782SGreg Clayton             // Last child
264244d93782SGreg Clayton             if (reverse_depth == 0)
264344d93782SGreg Clayton             {
264444d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
264544d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
264644d93782SGreg Clayton             }
264744d93782SGreg Clayton             else
264844d93782SGreg Clayton             {
264944d93782SGreg Clayton                 window.PutChar (' ');
265044d93782SGreg Clayton                 window.PutChar (' ');
265144d93782SGreg Clayton             }
265244d93782SGreg Clayton         }
265344d93782SGreg Clayton         else
265444d93782SGreg Clayton         {
265544d93782SGreg Clayton             if (reverse_depth == 0)
265644d93782SGreg Clayton             {
265744d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
265844d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
265944d93782SGreg Clayton             }
266044d93782SGreg Clayton             else
266144d93782SGreg Clayton             {
266244d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
266344d93782SGreg Clayton                 window.PutChar (' ');
266444d93782SGreg Clayton             }
266544d93782SGreg Clayton         }
266644d93782SGreg Clayton     }
266744d93782SGreg Clayton };
266844d93782SGreg Clayton 
266944d93782SGreg Clayton struct DisplayOptions
267044d93782SGreg Clayton {
267144d93782SGreg Clayton     bool show_types;
267244d93782SGreg Clayton };
267344d93782SGreg Clayton 
267444d93782SGreg Clayton class TreeItem;
267544d93782SGreg Clayton 
267644d93782SGreg Clayton class TreeDelegate
267744d93782SGreg Clayton {
267844d93782SGreg Clayton public:
267944d93782SGreg Clayton     TreeDelegate() {}
268044d93782SGreg Clayton     virtual ~TreeDelegate() {}
268144d93782SGreg Clayton     virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0;
268244d93782SGreg Clayton     virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0;
268344d93782SGreg Clayton     virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views
268444d93782SGreg Clayton };
268544d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
268644d93782SGreg Clayton 
268744d93782SGreg Clayton class TreeItem
268844d93782SGreg Clayton {
268944d93782SGreg Clayton public:
269044d93782SGreg Clayton 
269144d93782SGreg Clayton     TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) :
269244d93782SGreg Clayton         m_parent (parent),
269344d93782SGreg Clayton         m_delegate (delegate),
2694ec990867SGreg Clayton         m_user_data (NULL),
269544d93782SGreg Clayton         m_identifier (0),
269644d93782SGreg Clayton         m_row_idx (-1),
269744d93782SGreg Clayton         m_children (),
269844d93782SGreg Clayton         m_might_have_children (might_have_children),
269944d93782SGreg Clayton         m_is_expanded (false)
270044d93782SGreg Clayton     {
270144d93782SGreg Clayton     }
270244d93782SGreg Clayton 
270344d93782SGreg Clayton     TreeItem &
270444d93782SGreg Clayton     operator=(const TreeItem &rhs)
270544d93782SGreg Clayton     {
270644d93782SGreg Clayton         if (this != &rhs)
270744d93782SGreg Clayton         {
270844d93782SGreg Clayton             m_parent = rhs.m_parent;
270944d93782SGreg Clayton             m_delegate = rhs.m_delegate;
2710ec990867SGreg Clayton             m_user_data = rhs.m_user_data;
271144d93782SGreg Clayton             m_identifier = rhs.m_identifier;
271244d93782SGreg Clayton             m_row_idx = rhs.m_row_idx;
271344d93782SGreg Clayton             m_children = rhs.m_children;
271444d93782SGreg Clayton             m_might_have_children = rhs.m_might_have_children;
271544d93782SGreg Clayton             m_is_expanded = rhs.m_is_expanded;
271644d93782SGreg Clayton         }
271744d93782SGreg Clayton         return *this;
271844d93782SGreg Clayton     }
271944d93782SGreg Clayton 
272044d93782SGreg Clayton     size_t
272144d93782SGreg Clayton     GetDepth () const
272244d93782SGreg Clayton     {
272344d93782SGreg Clayton         if (m_parent)
272444d93782SGreg Clayton             return 1 + m_parent->GetDepth();
272544d93782SGreg Clayton         return 0;
272644d93782SGreg Clayton     }
272744d93782SGreg Clayton 
272844d93782SGreg Clayton     int
272944d93782SGreg Clayton     GetRowIndex () const
273044d93782SGreg Clayton     {
273144d93782SGreg Clayton         return m_row_idx;
273244d93782SGreg Clayton     }
273344d93782SGreg Clayton 
273444d93782SGreg Clayton     void
273544d93782SGreg Clayton     ClearChildren ()
273644d93782SGreg Clayton     {
273744d93782SGreg Clayton         m_children.clear();
273844d93782SGreg Clayton     }
273944d93782SGreg Clayton 
274044d93782SGreg Clayton     void
274144d93782SGreg Clayton     Resize (size_t n, const TreeItem &t)
274244d93782SGreg Clayton     {
274344d93782SGreg Clayton         m_children.resize(n, t);
274444d93782SGreg Clayton     }
274544d93782SGreg Clayton 
274644d93782SGreg Clayton     TreeItem &
274744d93782SGreg Clayton     operator [](size_t i)
274844d93782SGreg Clayton     {
274944d93782SGreg Clayton         return m_children[i];
275044d93782SGreg Clayton     }
275144d93782SGreg Clayton 
275244d93782SGreg Clayton     void
275344d93782SGreg Clayton     SetRowIndex (int row_idx)
275444d93782SGreg Clayton     {
275544d93782SGreg Clayton         m_row_idx = row_idx;
275644d93782SGreg Clayton     }
275744d93782SGreg Clayton 
275844d93782SGreg Clayton     size_t
275944d93782SGreg Clayton     GetNumChildren ()
276044d93782SGreg Clayton     {
276144d93782SGreg Clayton         m_delegate.TreeDelegateGenerateChildren (*this);
276244d93782SGreg Clayton         return m_children.size();
276344d93782SGreg Clayton     }
276444d93782SGreg Clayton 
276544d93782SGreg Clayton     void
276644d93782SGreg Clayton     ItemWasSelected ()
276744d93782SGreg Clayton     {
276844d93782SGreg Clayton         m_delegate.TreeDelegateItemSelected(*this);
276944d93782SGreg Clayton     }
277044d93782SGreg Clayton     void
277144d93782SGreg Clayton     CalculateRowIndexes (int &row_idx)
277244d93782SGreg Clayton     {
277344d93782SGreg Clayton         SetRowIndex(row_idx);
277444d93782SGreg Clayton         ++row_idx;
277544d93782SGreg Clayton 
2776ec990867SGreg Clayton         const bool expanded = IsExpanded();
2777ec990867SGreg Clayton 
2778ec990867SGreg Clayton         // The root item must calculate its children,
2779ec990867SGreg Clayton         // or we must calculate the number of children
2780ec990867SGreg Clayton         // if the item is expanded
2781ec990867SGreg Clayton         if (m_parent == NULL || expanded)
278244d93782SGreg Clayton             GetNumChildren();
278344d93782SGreg Clayton 
278444d93782SGreg Clayton         for (auto &item : m_children)
278544d93782SGreg Clayton         {
278644d93782SGreg Clayton             if (expanded)
278744d93782SGreg Clayton                 item.CalculateRowIndexes(row_idx);
278844d93782SGreg Clayton             else
278944d93782SGreg Clayton                 item.SetRowIndex(-1);
279044d93782SGreg Clayton         }
279144d93782SGreg Clayton     }
279244d93782SGreg Clayton 
279344d93782SGreg Clayton     TreeItem *
279444d93782SGreg Clayton     GetParent ()
279544d93782SGreg Clayton     {
279644d93782SGreg Clayton         return m_parent;
279744d93782SGreg Clayton     }
279844d93782SGreg Clayton 
279944d93782SGreg Clayton     bool
280044d93782SGreg Clayton     IsExpanded () const
280144d93782SGreg Clayton     {
280244d93782SGreg Clayton         return m_is_expanded;
280344d93782SGreg Clayton     }
280444d93782SGreg Clayton 
280544d93782SGreg Clayton     void
280644d93782SGreg Clayton     Expand()
280744d93782SGreg Clayton     {
280844d93782SGreg Clayton         m_is_expanded = true;
280944d93782SGreg Clayton     }
281044d93782SGreg Clayton 
281144d93782SGreg Clayton     void
281244d93782SGreg Clayton     Unexpand ()
281344d93782SGreg Clayton     {
281444d93782SGreg Clayton         m_is_expanded = false;
281544d93782SGreg Clayton     }
281644d93782SGreg Clayton 
281744d93782SGreg Clayton     bool
281844d93782SGreg Clayton     Draw (Window &window,
281944d93782SGreg Clayton           const int first_visible_row,
282044d93782SGreg Clayton           const uint32_t selected_row_idx,
282144d93782SGreg Clayton           int &row_idx,
282244d93782SGreg Clayton           int &num_rows_left)
282344d93782SGreg Clayton     {
282444d93782SGreg Clayton         if (num_rows_left <= 0)
282544d93782SGreg Clayton             return false;
282644d93782SGreg Clayton 
282744d93782SGreg Clayton         if (m_row_idx >= first_visible_row)
282844d93782SGreg Clayton         {
282944d93782SGreg Clayton             window.MoveCursor(2, row_idx + 1);
283044d93782SGreg Clayton 
283144d93782SGreg Clayton             if (m_parent)
283244d93782SGreg Clayton                 m_parent->DrawTreeForChild (window, this, 0);
283344d93782SGreg Clayton 
283444d93782SGreg Clayton             if (m_might_have_children)
283544d93782SGreg Clayton             {
283644d93782SGreg Clayton                 // It we can get UTF8 characters to work we should try to use the "symbol"
283744d93782SGreg Clayton                 // UTF8 string below
283844d93782SGreg Clayton                 //            const char *symbol = "";
283944d93782SGreg Clayton                 //            if (row.expanded)
284044d93782SGreg Clayton                 //                symbol = "\xe2\x96\xbd ";
284144d93782SGreg Clayton                 //            else
284244d93782SGreg Clayton                 //                symbol = "\xe2\x96\xb7 ";
284344d93782SGreg Clayton                 //            window.PutCString (symbol);
284444d93782SGreg Clayton 
284544d93782SGreg Clayton                 // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
284644d93782SGreg Clayton                 // 'v' or '>' character...
284744d93782SGreg Clayton                 //            if (expanded)
284844d93782SGreg Clayton                 //                window.PutChar (ACS_DARROW);
284944d93782SGreg Clayton                 //            else
285044d93782SGreg Clayton                 //                window.PutChar (ACS_RARROW);
285144d93782SGreg Clayton                 // Since we can't find any good looking right arrow/down arrow
285244d93782SGreg Clayton                 // symbols, just use a diamond...
285344d93782SGreg Clayton                 window.PutChar (ACS_DIAMOND);
285444d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
285544d93782SGreg Clayton             }
28563985c8c6SSaleem Abdulrasool             bool highlight =
28573985c8c6SSaleem Abdulrasool               (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive();
285844d93782SGreg Clayton 
285944d93782SGreg Clayton             if (highlight)
286044d93782SGreg Clayton                 window.AttributeOn(A_REVERSE);
286144d93782SGreg Clayton 
286244d93782SGreg Clayton             m_delegate.TreeDelegateDrawTreeItem(*this, window);
286344d93782SGreg Clayton 
286444d93782SGreg Clayton             if (highlight)
286544d93782SGreg Clayton                 window.AttributeOff(A_REVERSE);
286644d93782SGreg Clayton             ++row_idx;
286744d93782SGreg Clayton             --num_rows_left;
286844d93782SGreg Clayton         }
286944d93782SGreg Clayton 
287044d93782SGreg Clayton         if (num_rows_left <= 0)
287144d93782SGreg Clayton             return false; // We are done drawing...
287244d93782SGreg Clayton 
287344d93782SGreg Clayton         if (IsExpanded())
287444d93782SGreg Clayton         {
287544d93782SGreg Clayton             for (auto &item : m_children)
287644d93782SGreg Clayton             {
287744d93782SGreg Clayton                 // If we displayed all the rows and item.Draw() returns
287844d93782SGreg Clayton                 // false we are done drawing and can exit this for loop
287944d93782SGreg Clayton                 if (item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left) == false)
288044d93782SGreg Clayton                     break;
288144d93782SGreg Clayton             }
288244d93782SGreg Clayton         }
288344d93782SGreg Clayton         return num_rows_left >= 0; // Return true if not done drawing yet
288444d93782SGreg Clayton     }
288544d93782SGreg Clayton 
288644d93782SGreg Clayton     void
288744d93782SGreg Clayton     DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth)
288844d93782SGreg Clayton     {
288944d93782SGreg Clayton         if (m_parent)
289044d93782SGreg Clayton             m_parent->DrawTreeForChild (window, this, reverse_depth + 1);
289144d93782SGreg Clayton 
289244d93782SGreg Clayton         if (&m_children.back() == child)
289344d93782SGreg Clayton         {
289444d93782SGreg Clayton             // Last child
289544d93782SGreg Clayton             if (reverse_depth == 0)
289644d93782SGreg Clayton             {
289744d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
289844d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
289944d93782SGreg Clayton             }
290044d93782SGreg Clayton             else
290144d93782SGreg Clayton             {
290244d93782SGreg Clayton                 window.PutChar (' ');
290344d93782SGreg Clayton                 window.PutChar (' ');
290444d93782SGreg Clayton             }
290544d93782SGreg Clayton         }
290644d93782SGreg Clayton         else
290744d93782SGreg Clayton         {
290844d93782SGreg Clayton             if (reverse_depth == 0)
290944d93782SGreg Clayton             {
291044d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
291144d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
291244d93782SGreg Clayton             }
291344d93782SGreg Clayton             else
291444d93782SGreg Clayton             {
291544d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
291644d93782SGreg Clayton                 window.PutChar (' ');
291744d93782SGreg Clayton             }
291844d93782SGreg Clayton         }
291944d93782SGreg Clayton     }
292044d93782SGreg Clayton 
292144d93782SGreg Clayton     TreeItem *
292244d93782SGreg Clayton     GetItemForRowIndex (uint32_t row_idx)
292344d93782SGreg Clayton     {
29243985c8c6SSaleem Abdulrasool         if (static_cast<uint32_t>(m_row_idx) == row_idx)
292544d93782SGreg Clayton             return this;
292644d93782SGreg Clayton         if (m_children.empty())
292744d93782SGreg Clayton             return NULL;
292844d93782SGreg Clayton         if (IsExpanded())
292944d93782SGreg Clayton         {
293044d93782SGreg Clayton             for (auto &item : m_children)
293144d93782SGreg Clayton             {
293244d93782SGreg Clayton                 TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
293344d93782SGreg Clayton                 if (selected_item_ptr)
293444d93782SGreg Clayton                     return selected_item_ptr;
293544d93782SGreg Clayton             }
293644d93782SGreg Clayton         }
293744d93782SGreg Clayton         return NULL;
293844d93782SGreg Clayton     }
293944d93782SGreg Clayton 
2940ec990867SGreg Clayton     void *
2941ec990867SGreg Clayton     GetUserData() const
2942ec990867SGreg Clayton     {
2943ec990867SGreg Clayton         return m_user_data;
2944ec990867SGreg Clayton     }
2945ec990867SGreg Clayton 
2946ec990867SGreg Clayton     void
2947ec990867SGreg Clayton     SetUserData (void *user_data)
2948ec990867SGreg Clayton     {
2949ec990867SGreg Clayton         m_user_data = user_data;
2950ec990867SGreg Clayton     }
2951ec990867SGreg Clayton 
295244d93782SGreg Clayton     uint64_t
295344d93782SGreg Clayton     GetIdentifier() const
295444d93782SGreg Clayton     {
295544d93782SGreg Clayton         return m_identifier;
295644d93782SGreg Clayton     }
295744d93782SGreg Clayton 
295844d93782SGreg Clayton     void
295944d93782SGreg Clayton     SetIdentifier (uint64_t identifier)
296044d93782SGreg Clayton     {
296144d93782SGreg Clayton         m_identifier = identifier;
296244d93782SGreg Clayton     }
296344d93782SGreg Clayton 
296444d93782SGreg Clayton 
2965ec990867SGreg Clayton     void
2966ec990867SGreg Clayton     SetMightHaveChildren (bool b)
2967ec990867SGreg Clayton     {
2968ec990867SGreg Clayton         m_might_have_children = b;
2969ec990867SGreg Clayton     }
2970ec990867SGreg Clayton 
297144d93782SGreg Clayton protected:
297244d93782SGreg Clayton     TreeItem *m_parent;
297344d93782SGreg Clayton     TreeDelegate &m_delegate;
2974ec990867SGreg Clayton     void *m_user_data;
297544d93782SGreg Clayton     uint64_t m_identifier;
297644d93782SGreg Clayton     int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item
297744d93782SGreg Clayton     std::vector<TreeItem> m_children;
297844d93782SGreg Clayton     bool m_might_have_children;
297944d93782SGreg Clayton     bool m_is_expanded;
298044d93782SGreg Clayton 
298144d93782SGreg Clayton };
298244d93782SGreg Clayton 
298344d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate
298444d93782SGreg Clayton {
298544d93782SGreg Clayton public:
298644d93782SGreg Clayton     TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) :
298744d93782SGreg Clayton         m_debugger (debugger),
298844d93782SGreg Clayton         m_delegate_sp (delegate_sp),
298944d93782SGreg Clayton         m_root (NULL, *delegate_sp, true),
299044d93782SGreg Clayton         m_selected_item (NULL),
299144d93782SGreg Clayton         m_num_rows (0),
299244d93782SGreg Clayton         m_selected_row_idx (0),
299344d93782SGreg Clayton         m_first_visible_row (0),
299444d93782SGreg Clayton         m_min_x (0),
299544d93782SGreg Clayton         m_min_y (0),
299644d93782SGreg Clayton         m_max_x (0),
299744d93782SGreg Clayton         m_max_y (0)
299844d93782SGreg Clayton     {
299944d93782SGreg Clayton     }
300044d93782SGreg Clayton 
300144d93782SGreg Clayton     int
300244d93782SGreg Clayton     NumVisibleRows () const
300344d93782SGreg Clayton     {
300444d93782SGreg Clayton         return m_max_y - m_min_y;
300544d93782SGreg Clayton     }
300644d93782SGreg Clayton 
3007bd5ae6b4SGreg Clayton     bool
3008bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
300944d93782SGreg Clayton     {
301044d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
301144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
301244d93782SGreg Clayton 
301344d93782SGreg Clayton         bool display_content = false;
301444d93782SGreg Clayton         if (process)
301544d93782SGreg Clayton         {
301644d93782SGreg Clayton             StateType state = process->GetState();
301744d93782SGreg Clayton             if (StateIsStoppedState(state, true))
301844d93782SGreg Clayton             {
301944d93782SGreg Clayton                 // We are stopped, so it is ok to
302044d93782SGreg Clayton                 display_content = true;
302144d93782SGreg Clayton             }
302244d93782SGreg Clayton             else if (StateIsRunningState(state))
302344d93782SGreg Clayton             {
302444d93782SGreg Clayton                 return true; // Don't do any updating when we are running
302544d93782SGreg Clayton             }
302644d93782SGreg Clayton         }
302744d93782SGreg Clayton 
302844d93782SGreg Clayton         m_min_x = 2;
302944d93782SGreg Clayton         m_min_y = 1;
303044d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
303144d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
303244d93782SGreg Clayton 
303344d93782SGreg Clayton         window.Erase();
303444d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
303544d93782SGreg Clayton 
303644d93782SGreg Clayton         if (display_content)
303744d93782SGreg Clayton         {
303844d93782SGreg Clayton             const int num_visible_rows = NumVisibleRows();
303944d93782SGreg Clayton             m_num_rows = 0;
304044d93782SGreg Clayton             m_root.CalculateRowIndexes(m_num_rows);
304144d93782SGreg Clayton 
304244d93782SGreg Clayton             // If we unexpanded while having something selected our
304344d93782SGreg Clayton             // total number of rows is less than the num visible rows,
304444d93782SGreg Clayton             // then make sure we show all the rows by setting the first
304544d93782SGreg Clayton             // visible row accordingly.
304644d93782SGreg Clayton             if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
304744d93782SGreg Clayton                 m_first_visible_row = 0;
304844d93782SGreg Clayton 
304944d93782SGreg Clayton             // Make sure the selected row is always visible
305044d93782SGreg Clayton             if (m_selected_row_idx < m_first_visible_row)
305144d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx;
305244d93782SGreg Clayton             else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
305344d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
305444d93782SGreg Clayton 
305544d93782SGreg Clayton             int row_idx = 0;
305644d93782SGreg Clayton             int num_rows_left = num_visible_rows;
305744d93782SGreg Clayton             m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left);
305844d93782SGreg Clayton             // Get the selected row
305944d93782SGreg Clayton             m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx);
306044d93782SGreg Clayton         }
306144d93782SGreg Clayton         else
306244d93782SGreg Clayton         {
306344d93782SGreg Clayton             m_selected_item = NULL;
306444d93782SGreg Clayton         }
306544d93782SGreg Clayton 
306644d93782SGreg Clayton         window.DeferredRefresh();
306744d93782SGreg Clayton 
306844d93782SGreg Clayton 
306944d93782SGreg Clayton         return true; // Drawing handled
307044d93782SGreg Clayton     }
307144d93782SGreg Clayton 
307244d93782SGreg Clayton 
3073bd5ae6b4SGreg Clayton     const char *
3074bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
307544d93782SGreg Clayton     {
307644d93782SGreg Clayton         return "Thread window keyboard shortcuts:";
307744d93782SGreg Clayton     }
307844d93782SGreg Clayton 
3079bd5ae6b4SGreg Clayton     KeyHelp *
3080bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
308144d93782SGreg Clayton     {
308244d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
308344d93782SGreg Clayton             { KEY_UP, "Select previous item" },
308444d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
308544d93782SGreg Clayton             { KEY_RIGHT, "Expand the selected item" },
308644d93782SGreg Clayton             { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" },
308744d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
308844d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
308944d93782SGreg Clayton             { 'h', "Show help dialog" },
309044d93782SGreg Clayton             { ' ', "Toggle item expansion" },
309144d93782SGreg Clayton             { ',', "Page up" },
309244d93782SGreg Clayton             { '.', "Page down" },
309344d93782SGreg Clayton             { '\0', NULL }
309444d93782SGreg Clayton         };
309544d93782SGreg Clayton         return g_source_view_key_help;
309644d93782SGreg Clayton     }
309744d93782SGreg Clayton 
3098bd5ae6b4SGreg Clayton     HandleCharResult
3099bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
310044d93782SGreg Clayton     {
310144d93782SGreg Clayton         switch(c)
310244d93782SGreg Clayton         {
310344d93782SGreg Clayton             case ',':
310444d93782SGreg Clayton             case KEY_PPAGE:
310544d93782SGreg Clayton                 // Page up key
310644d93782SGreg Clayton                 if (m_first_visible_row > 0)
310744d93782SGreg Clayton                 {
310844d93782SGreg Clayton                     if (m_first_visible_row > m_max_y)
310944d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
311044d93782SGreg Clayton                     else
311144d93782SGreg Clayton                         m_first_visible_row = 0;
311244d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
311344d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
311444d93782SGreg Clayton                     if (m_selected_item)
311544d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
311644d93782SGreg Clayton                 }
311744d93782SGreg Clayton                 return eKeyHandled;
311844d93782SGreg Clayton 
311944d93782SGreg Clayton             case '.':
312044d93782SGreg Clayton             case KEY_NPAGE:
312144d93782SGreg Clayton                 // Page down key
312244d93782SGreg Clayton                 if (m_num_rows > m_max_y)
312344d93782SGreg Clayton                 {
312444d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
312544d93782SGreg Clayton                     {
312644d93782SGreg Clayton                         m_first_visible_row += m_max_y;
312744d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
312844d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
312944d93782SGreg Clayton                         if (m_selected_item)
313044d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
313144d93782SGreg Clayton                     }
313244d93782SGreg Clayton                 }
313344d93782SGreg Clayton                 return eKeyHandled;
313444d93782SGreg Clayton 
313544d93782SGreg Clayton             case KEY_UP:
313644d93782SGreg Clayton                 if (m_selected_row_idx > 0)
313744d93782SGreg Clayton                 {
313844d93782SGreg Clayton                     --m_selected_row_idx;
313944d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
314044d93782SGreg Clayton                     if (m_selected_item)
314144d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
314244d93782SGreg Clayton                 }
314344d93782SGreg Clayton                 return eKeyHandled;
314444d93782SGreg Clayton             case KEY_DOWN:
314544d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
314644d93782SGreg Clayton                 {
314744d93782SGreg Clayton                     ++m_selected_row_idx;
314844d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
314944d93782SGreg Clayton                     if (m_selected_item)
315044d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
315144d93782SGreg Clayton                 }
315244d93782SGreg Clayton                 return eKeyHandled;
315344d93782SGreg Clayton 
315444d93782SGreg Clayton             case KEY_RIGHT:
315544d93782SGreg Clayton                 if (m_selected_item)
315644d93782SGreg Clayton                 {
315744d93782SGreg Clayton                     if (!m_selected_item->IsExpanded())
315844d93782SGreg Clayton                         m_selected_item->Expand();
315944d93782SGreg Clayton                 }
316044d93782SGreg Clayton                 return eKeyHandled;
316144d93782SGreg Clayton 
316244d93782SGreg Clayton             case KEY_LEFT:
316344d93782SGreg Clayton                 if (m_selected_item)
316444d93782SGreg Clayton                 {
316544d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
316644d93782SGreg Clayton                         m_selected_item->Unexpand();
316744d93782SGreg Clayton                     else if (m_selected_item->GetParent())
316844d93782SGreg Clayton                     {
316944d93782SGreg Clayton                         m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
317044d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
317144d93782SGreg Clayton                         if (m_selected_item)
317244d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
317344d93782SGreg Clayton                     }
317444d93782SGreg Clayton                 }
317544d93782SGreg Clayton                 return eKeyHandled;
317644d93782SGreg Clayton 
317744d93782SGreg Clayton             case ' ':
317844d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
317944d93782SGreg Clayton                 if (m_selected_item)
318044d93782SGreg Clayton                 {
318144d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
318244d93782SGreg Clayton                         m_selected_item->Unexpand();
318344d93782SGreg Clayton                     else
318444d93782SGreg Clayton                         m_selected_item->Expand();
318544d93782SGreg Clayton                 }
318644d93782SGreg Clayton                 return eKeyHandled;
318744d93782SGreg Clayton 
318844d93782SGreg Clayton             case 'h':
318944d93782SGreg Clayton                 window.CreateHelpSubwindow ();
319044d93782SGreg Clayton                 return eKeyHandled;
319144d93782SGreg Clayton 
319244d93782SGreg Clayton             default:
319344d93782SGreg Clayton                 break;
319444d93782SGreg Clayton         }
319544d93782SGreg Clayton         return eKeyNotHandled;
319644d93782SGreg Clayton     }
319744d93782SGreg Clayton 
319844d93782SGreg Clayton protected:
319944d93782SGreg Clayton     Debugger &m_debugger;
320044d93782SGreg Clayton     TreeDelegateSP m_delegate_sp;
320144d93782SGreg Clayton     TreeItem m_root;
320244d93782SGreg Clayton     TreeItem *m_selected_item;
320344d93782SGreg Clayton     int m_num_rows;
320444d93782SGreg Clayton     int m_selected_row_idx;
320544d93782SGreg Clayton     int m_first_visible_row;
320644d93782SGreg Clayton     int m_min_x;
320744d93782SGreg Clayton     int m_min_y;
320844d93782SGreg Clayton     int m_max_x;
320944d93782SGreg Clayton     int m_max_y;
321044d93782SGreg Clayton 
321144d93782SGreg Clayton };
321244d93782SGreg Clayton 
321344d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate
321444d93782SGreg Clayton {
321544d93782SGreg Clayton public:
3216ec990867SGreg Clayton     FrameTreeDelegate () :
3217ec990867SGreg Clayton         TreeDelegate()
321844d93782SGreg Clayton     {
3219554f68d3SGreg Clayton         FormatEntity::Parse ("frame #${frame.index}: {${function.name}${function.pc-offset}}}",
3220554f68d3SGreg Clayton                              m_format);
322144d93782SGreg Clayton     }
322244d93782SGreg Clayton 
3223bd5ae6b4SGreg Clayton     ~FrameTreeDelegate() override
322444d93782SGreg Clayton     {
322544d93782SGreg Clayton     }
322644d93782SGreg Clayton 
3227bd5ae6b4SGreg Clayton     void
3228bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
322944d93782SGreg Clayton     {
3230ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3231ec990867SGreg Clayton         if (thread)
323244d93782SGreg Clayton         {
323344d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3234ec990867SGreg Clayton             StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
323544d93782SGreg Clayton             if (frame_sp)
323644d93782SGreg Clayton             {
323744d93782SGreg Clayton                 StreamString strm;
323844d93782SGreg Clayton                 const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
323944d93782SGreg Clayton                 ExecutionContext exe_ctx (frame_sp);
3240554f68d3SGreg Clayton                 if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, NULL, NULL, false, false))
324144d93782SGreg Clayton                 {
324244d93782SGreg Clayton                     int right_pad = 1;
324344d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
324444d93782SGreg Clayton                 }
324544d93782SGreg Clayton             }
324644d93782SGreg Clayton         }
324744d93782SGreg Clayton     }
3248bd5ae6b4SGreg Clayton     void
3249bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)  override
325044d93782SGreg Clayton     {
325144d93782SGreg Clayton         // No children for frames yet...
325244d93782SGreg Clayton     }
325344d93782SGreg Clayton 
3254bd5ae6b4SGreg Clayton     bool
3255bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
325644d93782SGreg Clayton     {
3257ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3258ec990867SGreg Clayton         if (thread)
325944d93782SGreg Clayton         {
3260ec990867SGreg Clayton             thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID());
326144d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3262ec990867SGreg Clayton             thread->SetSelectedFrameByIndex(frame_idx);
326344d93782SGreg Clayton             return true;
326444d93782SGreg Clayton         }
326544d93782SGreg Clayton         return false;
326644d93782SGreg Clayton     }
3267554f68d3SGreg Clayton protected:
3268554f68d3SGreg Clayton     FormatEntity::Entry m_format;
326944d93782SGreg Clayton };
327044d93782SGreg Clayton 
327144d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate
327244d93782SGreg Clayton {
327344d93782SGreg Clayton public:
327444d93782SGreg Clayton     ThreadTreeDelegate (Debugger &debugger) :
327544d93782SGreg Clayton         TreeDelegate(),
327644d93782SGreg Clayton         m_debugger (debugger),
327744d93782SGreg Clayton         m_tid (LLDB_INVALID_THREAD_ID),
327844d93782SGreg Clayton         m_stop_id (UINT32_MAX)
327944d93782SGreg Clayton     {
3280554f68d3SGreg Clayton         FormatEntity::Parse ("thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}",
3281554f68d3SGreg Clayton                              m_format);
328244d93782SGreg Clayton     }
328344d93782SGreg Clayton 
3284bd5ae6b4SGreg Clayton     ~ThreadTreeDelegate()  override
328544d93782SGreg Clayton     {
328644d93782SGreg Clayton     }
328744d93782SGreg Clayton 
3288ec990867SGreg Clayton     ProcessSP
3289ec990867SGreg Clayton     GetProcess ()
3290ec990867SGreg Clayton     {
3291ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3292ec990867SGreg Clayton     }
3293ec990867SGreg Clayton 
3294ec990867SGreg Clayton     ThreadSP
3295ec990867SGreg Clayton     GetThread (const TreeItem &item)
3296ec990867SGreg Clayton     {
3297ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3298ec990867SGreg Clayton         if (process_sp)
3299ec990867SGreg Clayton             return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
3300ec990867SGreg Clayton         return ThreadSP();
3301ec990867SGreg Clayton     }
3302ec990867SGreg Clayton 
3303bd5ae6b4SGreg Clayton     void
3304bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
330544d93782SGreg Clayton     {
3306ec990867SGreg Clayton         ThreadSP thread_sp = GetThread (item);
330744d93782SGreg Clayton         if (thread_sp)
330844d93782SGreg Clayton         {
330944d93782SGreg Clayton             StreamString strm;
331044d93782SGreg Clayton             ExecutionContext exe_ctx (thread_sp);
3311554f68d3SGreg Clayton             if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
331244d93782SGreg Clayton             {
331344d93782SGreg Clayton                 int right_pad = 1;
331444d93782SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
331544d93782SGreg Clayton             }
331644d93782SGreg Clayton         }
331744d93782SGreg Clayton     }
3318bd5ae6b4SGreg Clayton     void
3319bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
332044d93782SGreg Clayton     {
3321ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
332244d93782SGreg Clayton         if (process_sp && process_sp->IsAlive())
332344d93782SGreg Clayton         {
332444d93782SGreg Clayton             StateType state = process_sp->GetState();
332544d93782SGreg Clayton             if (StateIsStoppedState(state, true))
332644d93782SGreg Clayton             {
3327ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
332844d93782SGreg Clayton                 if (thread_sp)
332944d93782SGreg Clayton                 {
333044d93782SGreg Clayton                     if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid)
333144d93782SGreg Clayton                         return; // Children are already up to date
3332ec990867SGreg Clayton                     if (!m_frame_delegate_sp)
333344d93782SGreg Clayton                     {
333444d93782SGreg Clayton                         // Always expand the thread item the first time we show it
3335ec990867SGreg Clayton                         m_frame_delegate_sp.reset (new FrameTreeDelegate());
333644d93782SGreg Clayton                     }
333744d93782SGreg Clayton 
333844d93782SGreg Clayton                     m_stop_id = process_sp->GetStopID();
333944d93782SGreg Clayton                     m_tid = thread_sp->GetID();
334044d93782SGreg Clayton 
334144d93782SGreg Clayton                     TreeItem t (&item, *m_frame_delegate_sp, false);
334244d93782SGreg Clayton                     size_t num_frames = thread_sp->GetStackFrameCount();
334344d93782SGreg Clayton                     item.Resize (num_frames, t);
334444d93782SGreg Clayton                     for (size_t i=0; i<num_frames; ++i)
334544d93782SGreg Clayton                     {
3346ec990867SGreg Clayton                         item[i].SetUserData(thread_sp.get());
334744d93782SGreg Clayton                         item[i].SetIdentifier(i);
334844d93782SGreg Clayton                     }
334944d93782SGreg Clayton                 }
335044d93782SGreg Clayton                 return;
335144d93782SGreg Clayton             }
335244d93782SGreg Clayton         }
335344d93782SGreg Clayton         item.ClearChildren();
335444d93782SGreg Clayton     }
335544d93782SGreg Clayton 
3356bd5ae6b4SGreg Clayton     bool
3357bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
335844d93782SGreg Clayton     {
3359ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3360ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3361ec990867SGreg Clayton         {
3362ec990867SGreg Clayton             StateType state = process_sp->GetState();
3363ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3364ec990867SGreg Clayton             {
3365ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
336644d93782SGreg Clayton                 if (thread_sp)
336744d93782SGreg Clayton                 {
336844d93782SGreg Clayton                     ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
336944d93782SGreg Clayton                     Mutex::Locker locker (thread_list.GetMutex());
337044d93782SGreg Clayton                     ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
337144d93782SGreg Clayton                     if (selected_thread_sp->GetID() != thread_sp->GetID())
337244d93782SGreg Clayton                     {
337344d93782SGreg Clayton                         thread_list.SetSelectedThreadByID(thread_sp->GetID());
337444d93782SGreg Clayton                         return true;
337544d93782SGreg Clayton                     }
337644d93782SGreg Clayton                 }
3377ec990867SGreg Clayton             }
3378ec990867SGreg Clayton         }
337944d93782SGreg Clayton         return false;
338044d93782SGreg Clayton     }
338144d93782SGreg Clayton 
338244d93782SGreg Clayton protected:
338344d93782SGreg Clayton     Debugger &m_debugger;
338444d93782SGreg Clayton     std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
338544d93782SGreg Clayton     lldb::user_id_t m_tid;
338644d93782SGreg Clayton     uint32_t m_stop_id;
3387554f68d3SGreg Clayton     FormatEntity::Entry m_format;
3388554f68d3SGreg Clayton 
338944d93782SGreg Clayton };
339044d93782SGreg Clayton 
3391ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate
3392ec990867SGreg Clayton {
3393ec990867SGreg Clayton public:
3394ec990867SGreg Clayton     ThreadsTreeDelegate (Debugger &debugger) :
3395ec990867SGreg Clayton         TreeDelegate(),
3396ec990867SGreg Clayton         m_thread_delegate_sp (),
3397ec990867SGreg Clayton         m_debugger (debugger),
3398ec990867SGreg Clayton         m_stop_id (UINT32_MAX)
3399ec990867SGreg Clayton     {
3400554f68d3SGreg Clayton         FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
3401554f68d3SGreg Clayton                             m_format);
3402ec990867SGreg Clayton     }
3403ec990867SGreg Clayton 
3404bd5ae6b4SGreg Clayton     ~ThreadsTreeDelegate() override
3405ec990867SGreg Clayton     {
3406ec990867SGreg Clayton     }
3407ec990867SGreg Clayton 
3408ec990867SGreg Clayton     ProcessSP
3409ec990867SGreg Clayton     GetProcess ()
3410ec990867SGreg Clayton     {
3411ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3412ec990867SGreg Clayton     }
3413ec990867SGreg Clayton 
3414bd5ae6b4SGreg Clayton     void
3415bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
3416ec990867SGreg Clayton     {
3417ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3418ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3419ec990867SGreg Clayton         {
3420ec990867SGreg Clayton             StreamString strm;
3421ec990867SGreg Clayton             ExecutionContext exe_ctx (process_sp);
3422554f68d3SGreg Clayton             if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
3423ec990867SGreg Clayton             {
3424ec990867SGreg Clayton                 int right_pad = 1;
3425ec990867SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
3426ec990867SGreg Clayton             }
3427ec990867SGreg Clayton         }
3428ec990867SGreg Clayton     }
3429ec990867SGreg Clayton 
3430bd5ae6b4SGreg Clayton     void
3431bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
3432ec990867SGreg Clayton     {
3433ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3434ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3435ec990867SGreg Clayton         {
3436ec990867SGreg Clayton             StateType state = process_sp->GetState();
3437ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3438ec990867SGreg Clayton             {
3439ec990867SGreg Clayton                 const uint32_t stop_id = process_sp->GetStopID();
3440ec990867SGreg Clayton                 if (m_stop_id == stop_id)
3441ec990867SGreg Clayton                     return; // Children are already up to date
3442ec990867SGreg Clayton 
3443ec990867SGreg Clayton                 m_stop_id = stop_id;
3444ec990867SGreg Clayton 
3445ec990867SGreg Clayton                 if (!m_thread_delegate_sp)
3446ec990867SGreg Clayton                 {
3447ec990867SGreg Clayton                     // Always expand the thread item the first time we show it
3448ec990867SGreg Clayton                     //item.Expand();
3449ec990867SGreg Clayton                     m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger));
3450ec990867SGreg Clayton                 }
3451ec990867SGreg Clayton 
3452ec990867SGreg Clayton                 TreeItem t (&item, *m_thread_delegate_sp, false);
3453ec990867SGreg Clayton                 ThreadList &threads = process_sp->GetThreadList();
3454ec990867SGreg Clayton                 Mutex::Locker locker (threads.GetMutex());
3455ec990867SGreg Clayton                 size_t num_threads = threads.GetSize();
3456ec990867SGreg Clayton                 item.Resize (num_threads, t);
3457ec990867SGreg Clayton                 for (size_t i=0; i<num_threads; ++i)
3458ec990867SGreg Clayton                 {
3459ec990867SGreg Clayton                     item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
3460ec990867SGreg Clayton                     item[i].SetMightHaveChildren(true);
3461ec990867SGreg Clayton                 }
3462ec990867SGreg Clayton                 return;
3463ec990867SGreg Clayton             }
3464ec990867SGreg Clayton         }
3465ec990867SGreg Clayton         item.ClearChildren();
3466ec990867SGreg Clayton     }
3467ec990867SGreg Clayton 
3468bd5ae6b4SGreg Clayton     bool
3469bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
3470ec990867SGreg Clayton     {
3471ec990867SGreg Clayton         return false;
3472ec990867SGreg Clayton     }
3473ec990867SGreg Clayton 
3474ec990867SGreg Clayton protected:
3475ec990867SGreg Clayton     std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
3476ec990867SGreg Clayton     Debugger &m_debugger;
3477ec990867SGreg Clayton     uint32_t m_stop_id;
3478554f68d3SGreg Clayton     FormatEntity::Entry m_format;
3479554f68d3SGreg Clayton 
3480ec990867SGreg Clayton };
3481ec990867SGreg Clayton 
348244d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate
348344d93782SGreg Clayton {
348444d93782SGreg Clayton public:
348544d93782SGreg Clayton     ValueObjectListDelegate () :
348644d93782SGreg Clayton         m_valobj_list (),
348744d93782SGreg Clayton         m_rows (),
348844d93782SGreg Clayton         m_selected_row (NULL),
348944d93782SGreg Clayton         m_selected_row_idx (0),
349044d93782SGreg Clayton         m_first_visible_row (0),
349144d93782SGreg Clayton         m_num_rows (0),
349244d93782SGreg Clayton         m_max_x (0),
349344d93782SGreg Clayton         m_max_y (0)
349444d93782SGreg Clayton     {
349544d93782SGreg Clayton     }
349644d93782SGreg Clayton 
349744d93782SGreg Clayton     ValueObjectListDelegate (ValueObjectList &valobj_list) :
349844d93782SGreg Clayton         m_valobj_list (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         SetValues (valobj_list);
350844d93782SGreg Clayton     }
350944d93782SGreg Clayton 
3510bd5ae6b4SGreg Clayton     ~ValueObjectListDelegate() override
351144d93782SGreg Clayton     {
351244d93782SGreg Clayton     }
351344d93782SGreg Clayton 
351444d93782SGreg Clayton     void
351544d93782SGreg Clayton     SetValues (ValueObjectList &valobj_list)
351644d93782SGreg Clayton     {
351744d93782SGreg Clayton         m_selected_row = NULL;
351844d93782SGreg Clayton         m_selected_row_idx = 0;
351944d93782SGreg Clayton         m_first_visible_row = 0;
352044d93782SGreg Clayton         m_num_rows = 0;
352144d93782SGreg Clayton         m_rows.clear();
352244d93782SGreg Clayton         m_valobj_list = valobj_list;
352344d93782SGreg Clayton         const size_t num_values = m_valobj_list.GetSize();
352444d93782SGreg Clayton         for (size_t i=0; i<num_values; ++i)
352544d93782SGreg Clayton             m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), NULL));
352644d93782SGreg Clayton     }
352744d93782SGreg Clayton 
3528bd5ae6b4SGreg Clayton     bool
3529bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
353044d93782SGreg Clayton     {
353144d93782SGreg Clayton         m_num_rows = 0;
353244d93782SGreg Clayton         m_min_x = 2;
353344d93782SGreg Clayton         m_min_y = 1;
353444d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
353544d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
353644d93782SGreg Clayton 
353744d93782SGreg Clayton         window.Erase();
353844d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
353944d93782SGreg Clayton 
354044d93782SGreg Clayton         const int num_visible_rows = NumVisibleRows();
354144d93782SGreg Clayton         const int num_rows = CalculateTotalNumberRows (m_rows);
354244d93782SGreg Clayton 
354344d93782SGreg Clayton         // If we unexpanded while having something selected our
354444d93782SGreg Clayton         // total number of rows is less than the num visible rows,
354544d93782SGreg Clayton         // then make sure we show all the rows by setting the first
354644d93782SGreg Clayton         // visible row accordingly.
354744d93782SGreg Clayton         if (m_first_visible_row > 0 && num_rows < num_visible_rows)
354844d93782SGreg Clayton             m_first_visible_row = 0;
354944d93782SGreg Clayton 
355044d93782SGreg Clayton         // Make sure the selected row is always visible
355144d93782SGreg Clayton         if (m_selected_row_idx < m_first_visible_row)
355244d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx;
355344d93782SGreg Clayton         else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
355444d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
355544d93782SGreg Clayton 
355644d93782SGreg Clayton         DisplayRows (window, m_rows, g_options);
355744d93782SGreg Clayton 
355844d93782SGreg Clayton         window.DeferredRefresh();
355944d93782SGreg Clayton 
356044d93782SGreg Clayton         // Get the selected row
356144d93782SGreg Clayton         m_selected_row = GetRowForRowIndex (m_selected_row_idx);
356244d93782SGreg Clayton         // Keep the cursor on the selected row so the highlight and the cursor
356344d93782SGreg Clayton         // are always on the same line
356444d93782SGreg Clayton         if (m_selected_row)
356544d93782SGreg Clayton             window.MoveCursor (m_selected_row->x,
356644d93782SGreg Clayton                                m_selected_row->y);
356744d93782SGreg Clayton 
356844d93782SGreg Clayton         return true; // Drawing handled
356944d93782SGreg Clayton     }
357044d93782SGreg Clayton 
3571bd5ae6b4SGreg Clayton     KeyHelp *
3572bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
357344d93782SGreg Clayton     {
357444d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
357544d93782SGreg Clayton             { KEY_UP, "Select previous item" },
357644d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
357744d93782SGreg Clayton             { KEY_RIGHT, "Expand selected item" },
357844d93782SGreg Clayton             { KEY_LEFT, "Unexpand selected item or select parent if not expanded" },
357944d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
358044d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
358144d93782SGreg Clayton             { 'A', "Format as annotated address" },
358244d93782SGreg Clayton             { 'b', "Format as binary" },
358344d93782SGreg Clayton             { 'B', "Format as hex bytes with ASCII" },
358444d93782SGreg Clayton             { 'c', "Format as character" },
358544d93782SGreg Clayton             { 'd', "Format as a signed integer" },
358644d93782SGreg Clayton             { 'D', "Format selected value using the default format for the type" },
358744d93782SGreg Clayton             { 'f', "Format as float" },
358844d93782SGreg Clayton             { 'h', "Show help dialog" },
358944d93782SGreg Clayton             { 'i', "Format as instructions" },
359044d93782SGreg Clayton             { 'o', "Format as octal" },
359144d93782SGreg Clayton             { 'p', "Format as pointer" },
359244d93782SGreg Clayton             { 's', "Format as C string" },
359344d93782SGreg Clayton             { 't', "Toggle showing/hiding type names" },
359444d93782SGreg Clayton             { 'u', "Format as an unsigned integer" },
359544d93782SGreg Clayton             { 'x', "Format as hex" },
359644d93782SGreg Clayton             { 'X', "Format as uppercase hex" },
359744d93782SGreg Clayton             { ' ', "Toggle item expansion" },
359844d93782SGreg Clayton             { ',', "Page up" },
359944d93782SGreg Clayton             { '.', "Page down" },
360044d93782SGreg Clayton             { '\0', NULL }
360144d93782SGreg Clayton         };
360244d93782SGreg Clayton         return g_source_view_key_help;
360344d93782SGreg Clayton     }
360444d93782SGreg Clayton 
360544d93782SGreg Clayton 
3606bd5ae6b4SGreg Clayton     HandleCharResult
3607bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
360844d93782SGreg Clayton     {
360944d93782SGreg Clayton         switch(c)
361044d93782SGreg Clayton         {
361144d93782SGreg Clayton             case 'x':
361244d93782SGreg Clayton             case 'X':
361344d93782SGreg Clayton             case 'o':
361444d93782SGreg Clayton             case 's':
361544d93782SGreg Clayton             case 'u':
361644d93782SGreg Clayton             case 'd':
361744d93782SGreg Clayton             case 'D':
361844d93782SGreg Clayton             case 'i':
361944d93782SGreg Clayton             case 'A':
362044d93782SGreg Clayton             case 'p':
362144d93782SGreg Clayton             case 'c':
362244d93782SGreg Clayton             case 'b':
362344d93782SGreg Clayton             case 'B':
362444d93782SGreg Clayton             case 'f':
362544d93782SGreg Clayton                 // Change the format for the currently selected item
362644d93782SGreg Clayton                 if (m_selected_row)
362744d93782SGreg Clayton                     m_selected_row->valobj->SetFormat (FormatForChar (c));
362844d93782SGreg Clayton                 return eKeyHandled;
362944d93782SGreg Clayton 
363044d93782SGreg Clayton             case 't':
363144d93782SGreg Clayton                 // Toggle showing type names
363244d93782SGreg Clayton                 g_options.show_types = !g_options.show_types;
363344d93782SGreg Clayton                 return eKeyHandled;
363444d93782SGreg Clayton 
363544d93782SGreg Clayton             case ',':
363644d93782SGreg Clayton             case KEY_PPAGE:
363744d93782SGreg Clayton                 // Page up key
363844d93782SGreg Clayton                 if (m_first_visible_row > 0)
363944d93782SGreg Clayton                 {
36403985c8c6SSaleem Abdulrasool                     if (static_cast<int>(m_first_visible_row) > m_max_y)
364144d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
364244d93782SGreg Clayton                     else
364344d93782SGreg Clayton                         m_first_visible_row = 0;
364444d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
364544d93782SGreg Clayton                 }
364644d93782SGreg Clayton                 return eKeyHandled;
364744d93782SGreg Clayton 
364844d93782SGreg Clayton             case '.':
364944d93782SGreg Clayton             case KEY_NPAGE:
365044d93782SGreg Clayton                 // Page down key
36513985c8c6SSaleem Abdulrasool                 if (m_num_rows > static_cast<size_t>(m_max_y))
365244d93782SGreg Clayton                 {
365344d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
365444d93782SGreg Clayton                     {
365544d93782SGreg Clayton                         m_first_visible_row += m_max_y;
365644d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
365744d93782SGreg Clayton                     }
365844d93782SGreg Clayton                 }
365944d93782SGreg Clayton                 return eKeyHandled;
366044d93782SGreg Clayton 
366144d93782SGreg Clayton             case KEY_UP:
366244d93782SGreg Clayton                 if (m_selected_row_idx > 0)
366344d93782SGreg Clayton                     --m_selected_row_idx;
366444d93782SGreg Clayton                 return eKeyHandled;
366544d93782SGreg Clayton             case KEY_DOWN:
366644d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
366744d93782SGreg Clayton                     ++m_selected_row_idx;
366844d93782SGreg Clayton                 return eKeyHandled;
366944d93782SGreg Clayton 
367044d93782SGreg Clayton             case KEY_RIGHT:
367144d93782SGreg Clayton                 if (m_selected_row)
367244d93782SGreg Clayton                 {
367344d93782SGreg Clayton                     if (!m_selected_row->expanded)
367444d93782SGreg Clayton                         m_selected_row->Expand();
367544d93782SGreg Clayton                 }
367644d93782SGreg Clayton                 return eKeyHandled;
367744d93782SGreg Clayton 
367844d93782SGreg Clayton             case KEY_LEFT:
367944d93782SGreg Clayton                 if (m_selected_row)
368044d93782SGreg Clayton                 {
368144d93782SGreg Clayton                     if (m_selected_row->expanded)
368244d93782SGreg Clayton                         m_selected_row->Unexpand();
368344d93782SGreg Clayton                     else if (m_selected_row->parent)
368444d93782SGreg Clayton                         m_selected_row_idx = m_selected_row->parent->row_idx;
368544d93782SGreg Clayton                 }
368644d93782SGreg Clayton                 return eKeyHandled;
368744d93782SGreg Clayton 
368844d93782SGreg Clayton             case ' ':
368944d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
369044d93782SGreg Clayton                 if (m_selected_row)
369144d93782SGreg Clayton                 {
369244d93782SGreg Clayton                     if (m_selected_row->expanded)
369344d93782SGreg Clayton                         m_selected_row->Unexpand();
369444d93782SGreg Clayton                     else
369544d93782SGreg Clayton                         m_selected_row->Expand();
369644d93782SGreg Clayton                 }
369744d93782SGreg Clayton                 return eKeyHandled;
369844d93782SGreg Clayton 
369944d93782SGreg Clayton             case 'h':
370044d93782SGreg Clayton                 window.CreateHelpSubwindow ();
370144d93782SGreg Clayton                 return eKeyHandled;
370244d93782SGreg Clayton 
370344d93782SGreg Clayton             default:
370444d93782SGreg Clayton                 break;
370544d93782SGreg Clayton         }
370644d93782SGreg Clayton         return eKeyNotHandled;
370744d93782SGreg Clayton     }
370844d93782SGreg Clayton 
370944d93782SGreg Clayton protected:
371044d93782SGreg Clayton     ValueObjectList m_valobj_list;
371144d93782SGreg Clayton     std::vector<Row> m_rows;
371244d93782SGreg Clayton     Row *m_selected_row;
371344d93782SGreg Clayton     uint32_t m_selected_row_idx;
371444d93782SGreg Clayton     uint32_t m_first_visible_row;
371544d93782SGreg Clayton     uint32_t m_num_rows;
371644d93782SGreg Clayton     int m_min_x;
371744d93782SGreg Clayton     int m_min_y;
371844d93782SGreg Clayton     int m_max_x;
371944d93782SGreg Clayton     int m_max_y;
372044d93782SGreg Clayton 
372144d93782SGreg Clayton     static Format
372244d93782SGreg Clayton     FormatForChar (int c)
372344d93782SGreg Clayton     {
372444d93782SGreg Clayton         switch (c)
372544d93782SGreg Clayton         {
372644d93782SGreg Clayton             case 'x': return eFormatHex;
372744d93782SGreg Clayton             case 'X': return eFormatHexUppercase;
372844d93782SGreg Clayton             case 'o': return eFormatOctal;
372944d93782SGreg Clayton             case 's': return eFormatCString;
373044d93782SGreg Clayton             case 'u': return eFormatUnsigned;
373144d93782SGreg Clayton             case 'd': return eFormatDecimal;
373244d93782SGreg Clayton             case 'D': return eFormatDefault;
373344d93782SGreg Clayton             case 'i': return eFormatInstruction;
373444d93782SGreg Clayton             case 'A': return eFormatAddressInfo;
373544d93782SGreg Clayton             case 'p': return eFormatPointer;
373644d93782SGreg Clayton             case 'c': return eFormatChar;
373744d93782SGreg Clayton             case 'b': return eFormatBinary;
373844d93782SGreg Clayton             case 'B': return eFormatBytesWithASCII;
373944d93782SGreg Clayton             case 'f': return eFormatFloat;
374044d93782SGreg Clayton         }
374144d93782SGreg Clayton         return eFormatDefault;
374244d93782SGreg Clayton     }
374344d93782SGreg Clayton 
374444d93782SGreg Clayton     bool
374544d93782SGreg Clayton     DisplayRowObject (Window &window,
374644d93782SGreg Clayton                       Row &row,
374744d93782SGreg Clayton                       DisplayOptions &options,
374844d93782SGreg Clayton                       bool highlight,
374944d93782SGreg Clayton                       bool last_child)
375044d93782SGreg Clayton     {
375144d93782SGreg Clayton         ValueObject *valobj = row.valobj.get();
375244d93782SGreg Clayton 
375344d93782SGreg Clayton         if (valobj == NULL)
375444d93782SGreg Clayton             return false;
375544d93782SGreg Clayton 
375644d93782SGreg Clayton         const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : NULL;
375744d93782SGreg Clayton         const char *name = valobj->GetName().GetCString();
375844d93782SGreg Clayton         const char *value = valobj->GetValueAsCString ();
375944d93782SGreg Clayton         const char *summary = valobj->GetSummaryAsCString ();
376044d93782SGreg Clayton 
376144d93782SGreg Clayton         window.MoveCursor (row.x, row.y);
376244d93782SGreg Clayton 
376344d93782SGreg Clayton         row.DrawTree (window);
376444d93782SGreg Clayton 
376544d93782SGreg Clayton         if (highlight)
376644d93782SGreg Clayton             window.AttributeOn(A_REVERSE);
376744d93782SGreg Clayton 
376844d93782SGreg Clayton         if (type_name && type_name[0])
376944d93782SGreg Clayton             window.Printf ("(%s) ", type_name);
377044d93782SGreg Clayton 
377144d93782SGreg Clayton         if (name && name[0])
377244d93782SGreg Clayton             window.PutCString(name);
377344d93782SGreg Clayton 
377444d93782SGreg Clayton         attr_t changd_attr = 0;
377544d93782SGreg Clayton         if (valobj->GetValueDidChange())
377644d93782SGreg Clayton             changd_attr = COLOR_PAIR(5) | A_BOLD;
377744d93782SGreg Clayton 
377844d93782SGreg Clayton         if (value && value[0])
377944d93782SGreg Clayton         {
378044d93782SGreg Clayton             window.PutCString(" = ");
378144d93782SGreg Clayton             if (changd_attr)
378244d93782SGreg Clayton                 window.AttributeOn(changd_attr);
378344d93782SGreg Clayton             window.PutCString (value);
378444d93782SGreg Clayton             if (changd_attr)
378544d93782SGreg Clayton                 window.AttributeOff(changd_attr);
378644d93782SGreg Clayton         }
378744d93782SGreg Clayton 
378844d93782SGreg Clayton         if (summary && summary[0])
378944d93782SGreg Clayton         {
379044d93782SGreg Clayton             window.PutChar(' ');
379144d93782SGreg Clayton             if (changd_attr)
379244d93782SGreg Clayton                 window.AttributeOn(changd_attr);
379344d93782SGreg Clayton             window.PutCString(summary);
379444d93782SGreg Clayton             if (changd_attr)
379544d93782SGreg Clayton                 window.AttributeOff(changd_attr);
379644d93782SGreg Clayton         }
379744d93782SGreg Clayton 
379844d93782SGreg Clayton         if (highlight)
379944d93782SGreg Clayton             window.AttributeOff (A_REVERSE);
380044d93782SGreg Clayton 
380144d93782SGreg Clayton         return true;
380244d93782SGreg Clayton     }
380344d93782SGreg Clayton     void
380444d93782SGreg Clayton     DisplayRows (Window &window,
380544d93782SGreg Clayton                  std::vector<Row> &rows,
380644d93782SGreg Clayton                  DisplayOptions &options)
380744d93782SGreg Clayton     {
380844d93782SGreg Clayton         // >   0x25B7
380944d93782SGreg Clayton         // \/  0x25BD
381044d93782SGreg Clayton 
381144d93782SGreg Clayton         bool window_is_active = window.IsActive();
381244d93782SGreg Clayton         for (auto &row : rows)
381344d93782SGreg Clayton         {
381444d93782SGreg Clayton             const bool last_child = row.parent && &rows[rows.size()-1] == &row;
381544d93782SGreg Clayton             // Save the row index in each Row structure
381644d93782SGreg Clayton             row.row_idx = m_num_rows;
381744d93782SGreg Clayton             if ((m_num_rows >= m_first_visible_row) &&
38183985c8c6SSaleem Abdulrasool                 ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows())))
381944d93782SGreg Clayton             {
382044d93782SGreg Clayton                 row.x = m_min_x;
382144d93782SGreg Clayton                 row.y = m_num_rows - m_first_visible_row + 1;
382244d93782SGreg Clayton                 if (DisplayRowObject (window,
382344d93782SGreg Clayton                                       row,
382444d93782SGreg Clayton                                       options,
382544d93782SGreg Clayton                                       window_is_active && m_num_rows == m_selected_row_idx,
382644d93782SGreg Clayton                                       last_child))
382744d93782SGreg Clayton                 {
382844d93782SGreg Clayton                     ++m_num_rows;
382944d93782SGreg Clayton                 }
383044d93782SGreg Clayton                 else
383144d93782SGreg Clayton                 {
383244d93782SGreg Clayton                     row.x = 0;
383344d93782SGreg Clayton                     row.y = 0;
383444d93782SGreg Clayton                 }
383544d93782SGreg Clayton             }
383644d93782SGreg Clayton             else
383744d93782SGreg Clayton             {
383844d93782SGreg Clayton                 row.x = 0;
383944d93782SGreg Clayton                 row.y = 0;
384044d93782SGreg Clayton                 ++m_num_rows;
384144d93782SGreg Clayton             }
384244d93782SGreg Clayton 
384344d93782SGreg Clayton             if (row.expanded && !row.children.empty())
384444d93782SGreg Clayton             {
384544d93782SGreg Clayton                 DisplayRows (window,
384644d93782SGreg Clayton                              row.children,
384744d93782SGreg Clayton                              options);
384844d93782SGreg Clayton             }
384944d93782SGreg Clayton         }
385044d93782SGreg Clayton     }
385144d93782SGreg Clayton 
385244d93782SGreg Clayton     int
385344d93782SGreg Clayton     CalculateTotalNumberRows (const std::vector<Row> &rows)
385444d93782SGreg Clayton     {
385544d93782SGreg Clayton         int row_count = 0;
385644d93782SGreg Clayton         for (const auto &row : rows)
385744d93782SGreg Clayton         {
385844d93782SGreg Clayton             ++row_count;
385944d93782SGreg Clayton             if (row.expanded)
386044d93782SGreg Clayton                 row_count += CalculateTotalNumberRows(row.children);
386144d93782SGreg Clayton         }
386244d93782SGreg Clayton         return row_count;
386344d93782SGreg Clayton     }
386444d93782SGreg Clayton     static Row *
386544d93782SGreg Clayton     GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index)
386644d93782SGreg Clayton     {
386744d93782SGreg Clayton         for (auto &row : rows)
386844d93782SGreg Clayton         {
386944d93782SGreg Clayton             if (row_index == 0)
387044d93782SGreg Clayton                 return &row;
387144d93782SGreg Clayton             else
387244d93782SGreg Clayton             {
387344d93782SGreg Clayton                 --row_index;
387444d93782SGreg Clayton                 if (row.expanded && !row.children.empty())
387544d93782SGreg Clayton                 {
387644d93782SGreg Clayton                     Row *result = GetRowForRowIndexImpl (row.children, row_index);
387744d93782SGreg Clayton                     if (result)
387844d93782SGreg Clayton                         return result;
387944d93782SGreg Clayton                 }
388044d93782SGreg Clayton             }
388144d93782SGreg Clayton         }
388244d93782SGreg Clayton         return NULL;
388344d93782SGreg Clayton     }
388444d93782SGreg Clayton 
388544d93782SGreg Clayton     Row *
388644d93782SGreg Clayton     GetRowForRowIndex (size_t row_index)
388744d93782SGreg Clayton     {
388844d93782SGreg Clayton         return GetRowForRowIndexImpl (m_rows, row_index);
388944d93782SGreg Clayton     }
389044d93782SGreg Clayton 
389144d93782SGreg Clayton     int
389244d93782SGreg Clayton     NumVisibleRows () const
389344d93782SGreg Clayton     {
389444d93782SGreg Clayton         return m_max_y - m_min_y;
389544d93782SGreg Clayton     }
389644d93782SGreg Clayton 
389744d93782SGreg Clayton     static DisplayOptions g_options;
389844d93782SGreg Clayton };
389944d93782SGreg Clayton 
390044d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate
390144d93782SGreg Clayton {
390244d93782SGreg Clayton public:
390344d93782SGreg Clayton     FrameVariablesWindowDelegate (Debugger &debugger) :
390444d93782SGreg Clayton         ValueObjectListDelegate (),
390544d93782SGreg Clayton         m_debugger (debugger),
390644d93782SGreg Clayton         m_frame_block (NULL)
390744d93782SGreg Clayton     {
390844d93782SGreg Clayton     }
390944d93782SGreg Clayton 
3910bd5ae6b4SGreg Clayton     ~FrameVariablesWindowDelegate() override
391144d93782SGreg Clayton     {
391244d93782SGreg Clayton     }
391344d93782SGreg Clayton 
3914bd5ae6b4SGreg Clayton     const char *
3915bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
391644d93782SGreg Clayton     {
391744d93782SGreg Clayton         return "Frame variable window keyboard shortcuts:";
391844d93782SGreg Clayton     }
391944d93782SGreg Clayton 
3920bd5ae6b4SGreg Clayton     bool
3921bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
392244d93782SGreg Clayton     {
392344d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
392444d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
392544d93782SGreg Clayton         Block *frame_block = NULL;
392644d93782SGreg Clayton         StackFrame *frame = NULL;
392744d93782SGreg Clayton 
392844d93782SGreg Clayton         if (process)
392944d93782SGreg Clayton         {
393044d93782SGreg Clayton             StateType state = process->GetState();
393144d93782SGreg Clayton             if (StateIsStoppedState(state, true))
393244d93782SGreg Clayton             {
393344d93782SGreg Clayton                 frame = exe_ctx.GetFramePtr();
393444d93782SGreg Clayton                 if (frame)
393544d93782SGreg Clayton                     frame_block = frame->GetFrameBlock ();
393644d93782SGreg Clayton             }
393744d93782SGreg Clayton             else if (StateIsRunningState(state))
393844d93782SGreg Clayton             {
393944d93782SGreg Clayton                 return true; // Don't do any updating when we are running
394044d93782SGreg Clayton             }
394144d93782SGreg Clayton         }
394244d93782SGreg Clayton 
3943eb72dc7dSGreg Clayton 
394444d93782SGreg Clayton         ValueObjectList local_values;
394544d93782SGreg Clayton         if (frame_block)
394644d93782SGreg Clayton         {
394744d93782SGreg Clayton             // Only update the variables if they have changed
394844d93782SGreg Clayton             if (m_frame_block != frame_block)
394944d93782SGreg Clayton             {
395044d93782SGreg Clayton                 m_frame_block = frame_block;
395144d93782SGreg Clayton 
395244d93782SGreg Clayton                 VariableList *locals = frame->GetVariableList(true);
395344d93782SGreg Clayton                 if (locals)
395444d93782SGreg Clayton                 {
395544d93782SGreg Clayton                     const DynamicValueType use_dynamic = eDynamicDontRunTarget;
395644d93782SGreg Clayton                     const size_t num_locals = locals->GetSize();
395744d93782SGreg Clayton                     for (size_t i=0; i<num_locals; ++i)
3958eb72dc7dSGreg Clayton                     {
3959eb72dc7dSGreg Clayton                         ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic);
3960eb72dc7dSGreg Clayton                         if (value_sp)
3961eb72dc7dSGreg Clayton                         {
3962eb72dc7dSGreg Clayton                             ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3963eb72dc7dSGreg Clayton                             if (synthetic_value_sp)
3964eb72dc7dSGreg Clayton                                 local_values.Append(synthetic_value_sp);
3965eb72dc7dSGreg Clayton                             else
3966eb72dc7dSGreg Clayton                                 local_values.Append(value_sp);
3967eb72dc7dSGreg Clayton 
3968eb72dc7dSGreg Clayton                         }
3969eb72dc7dSGreg Clayton                     }
397044d93782SGreg Clayton                     // Update the values
397144d93782SGreg Clayton                     SetValues(local_values);
397244d93782SGreg Clayton                 }
397344d93782SGreg Clayton             }
397444d93782SGreg Clayton         }
397544d93782SGreg Clayton         else
397644d93782SGreg Clayton         {
397744d93782SGreg Clayton             m_frame_block = NULL;
397844d93782SGreg Clayton             // Update the values with an empty list if there is no frame
397944d93782SGreg Clayton             SetValues(local_values);
398044d93782SGreg Clayton         }
398144d93782SGreg Clayton 
398244d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
398344d93782SGreg Clayton 
398444d93782SGreg Clayton     }
398544d93782SGreg Clayton 
398644d93782SGreg Clayton protected:
398744d93782SGreg Clayton     Debugger &m_debugger;
398844d93782SGreg Clayton     Block *m_frame_block;
398944d93782SGreg Clayton };
399044d93782SGreg Clayton 
399144d93782SGreg Clayton 
399244d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate
399344d93782SGreg Clayton {
399444d93782SGreg Clayton public:
399544d93782SGreg Clayton     RegistersWindowDelegate (Debugger &debugger) :
399644d93782SGreg Clayton         ValueObjectListDelegate (),
399744d93782SGreg Clayton         m_debugger (debugger)
399844d93782SGreg Clayton     {
399944d93782SGreg Clayton     }
400044d93782SGreg Clayton 
400144d93782SGreg Clayton     ~RegistersWindowDelegate()
400244d93782SGreg Clayton     {
400344d93782SGreg Clayton     }
400444d93782SGreg Clayton 
4005bd5ae6b4SGreg Clayton     const char *
4006bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
400744d93782SGreg Clayton     {
400844d93782SGreg Clayton         return "Register window keyboard shortcuts:";
400944d93782SGreg Clayton     }
401044d93782SGreg Clayton 
4011bd5ae6b4SGreg Clayton     bool
4012bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
401344d93782SGreg Clayton     {
401444d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
401544d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
401644d93782SGreg Clayton 
401744d93782SGreg Clayton         ValueObjectList value_list;
401844d93782SGreg Clayton         if (frame)
401944d93782SGreg Clayton         {
402044d93782SGreg Clayton             if (frame->GetStackID() != m_stack_id)
402144d93782SGreg Clayton             {
402244d93782SGreg Clayton                 m_stack_id = frame->GetStackID();
402344d93782SGreg Clayton                 RegisterContextSP reg_ctx (frame->GetRegisterContext());
402444d93782SGreg Clayton                 if (reg_ctx)
402544d93782SGreg Clayton                 {
402644d93782SGreg Clayton                     const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
402744d93782SGreg Clayton                     for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx)
402844d93782SGreg Clayton                     {
402944d93782SGreg Clayton                         value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx));
403044d93782SGreg Clayton                     }
403144d93782SGreg Clayton                 }
403244d93782SGreg Clayton                 SetValues(value_list);
403344d93782SGreg Clayton             }
403444d93782SGreg Clayton         }
403544d93782SGreg Clayton         else
403644d93782SGreg Clayton         {
403744d93782SGreg Clayton             Process *process = exe_ctx.GetProcessPtr();
403844d93782SGreg Clayton             if (process && process->IsAlive())
403944d93782SGreg Clayton                 return true; // Don't do any updating if we are running
404044d93782SGreg Clayton             else
404144d93782SGreg Clayton             {
404244d93782SGreg Clayton                 // Update the values with an empty list if there
404344d93782SGreg Clayton                 // is no process or the process isn't alive anymore
404444d93782SGreg Clayton                 SetValues(value_list);
404544d93782SGreg Clayton             }
404644d93782SGreg Clayton         }
404744d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
404844d93782SGreg Clayton     }
404944d93782SGreg Clayton 
405044d93782SGreg Clayton protected:
405144d93782SGreg Clayton     Debugger &m_debugger;
405244d93782SGreg Clayton     StackID m_stack_id;
405344d93782SGreg Clayton };
405444d93782SGreg Clayton 
405544d93782SGreg Clayton static const char *
405644d93782SGreg Clayton CursesKeyToCString (int ch)
405744d93782SGreg Clayton {
405844d93782SGreg Clayton     static char g_desc[32];
405944d93782SGreg Clayton     if (ch >= KEY_F0 && ch < KEY_F0 + 64)
406044d93782SGreg Clayton     {
406144d93782SGreg Clayton         snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
406244d93782SGreg Clayton         return g_desc;
406344d93782SGreg Clayton     }
406444d93782SGreg Clayton     switch (ch)
406544d93782SGreg Clayton     {
406644d93782SGreg Clayton         case KEY_DOWN:  return "down";
406744d93782SGreg Clayton         case KEY_UP:    return "up";
406844d93782SGreg Clayton         case KEY_LEFT:  return "left";
406944d93782SGreg Clayton         case KEY_RIGHT: return "right";
407044d93782SGreg Clayton         case KEY_HOME:  return "home";
407144d93782SGreg Clayton         case KEY_BACKSPACE: return "backspace";
407244d93782SGreg Clayton         case KEY_DL:        return "delete-line";
407344d93782SGreg Clayton         case KEY_IL:        return "insert-line";
407444d93782SGreg Clayton         case KEY_DC:        return "delete-char";
407544d93782SGreg Clayton         case KEY_IC:        return "insert-char";
407644d93782SGreg Clayton         case KEY_CLEAR:     return "clear";
407744d93782SGreg Clayton         case KEY_EOS:       return "clear-to-eos";
407844d93782SGreg Clayton         case KEY_EOL:       return "clear-to-eol";
407944d93782SGreg Clayton         case KEY_SF:        return "scroll-forward";
408044d93782SGreg Clayton         case KEY_SR:        return "scroll-backward";
408144d93782SGreg Clayton         case KEY_NPAGE:     return "page-down";
408244d93782SGreg Clayton         case KEY_PPAGE:     return "page-up";
408344d93782SGreg Clayton         case KEY_STAB:      return "set-tab";
408444d93782SGreg Clayton         case KEY_CTAB:      return "clear-tab";
408544d93782SGreg Clayton         case KEY_CATAB:     return "clear-all-tabs";
408644d93782SGreg Clayton         case KEY_ENTER:     return "enter";
408744d93782SGreg Clayton         case KEY_PRINT:     return "print";
408844d93782SGreg Clayton         case KEY_LL:        return "lower-left key";
408944d93782SGreg Clayton         case KEY_A1:        return "upper left of keypad";
409044d93782SGreg Clayton         case KEY_A3:        return "upper right of keypad";
409144d93782SGreg Clayton         case KEY_B2:        return "center of keypad";
409244d93782SGreg Clayton         case KEY_C1:        return "lower left of keypad";
409344d93782SGreg Clayton         case KEY_C3:        return "lower right of keypad";
409444d93782SGreg Clayton         case KEY_BTAB:      return "back-tab key";
409544d93782SGreg Clayton         case KEY_BEG:       return "begin key";
409644d93782SGreg Clayton         case KEY_CANCEL:    return "cancel key";
409744d93782SGreg Clayton         case KEY_CLOSE:     return "close key";
409844d93782SGreg Clayton         case KEY_COMMAND:   return "command key";
409944d93782SGreg Clayton         case KEY_COPY:      return "copy key";
410044d93782SGreg Clayton         case KEY_CREATE:    return "create key";
410144d93782SGreg Clayton         case KEY_END:       return "end key";
410244d93782SGreg Clayton         case KEY_EXIT:      return "exit key";
410344d93782SGreg Clayton         case KEY_FIND:      return "find key";
410444d93782SGreg Clayton         case KEY_HELP:      return "help key";
410544d93782SGreg Clayton         case KEY_MARK:      return "mark key";
410644d93782SGreg Clayton         case KEY_MESSAGE:   return "message key";
410744d93782SGreg Clayton         case KEY_MOVE:      return "move key";
410844d93782SGreg Clayton         case KEY_NEXT:      return "next key";
410944d93782SGreg Clayton         case KEY_OPEN:      return "open key";
411044d93782SGreg Clayton         case KEY_OPTIONS:   return "options key";
411144d93782SGreg Clayton         case KEY_PREVIOUS:  return "previous key";
411244d93782SGreg Clayton         case KEY_REDO:      return "redo key";
411344d93782SGreg Clayton         case KEY_REFERENCE: return "reference key";
411444d93782SGreg Clayton         case KEY_REFRESH:   return "refresh key";
411544d93782SGreg Clayton         case KEY_REPLACE:   return "replace key";
411644d93782SGreg Clayton         case KEY_RESTART:   return "restart key";
411744d93782SGreg Clayton         case KEY_RESUME:    return "resume key";
411844d93782SGreg Clayton         case KEY_SAVE:      return "save key";
411944d93782SGreg Clayton         case KEY_SBEG:      return "shifted begin key";
412044d93782SGreg Clayton         case KEY_SCANCEL:   return "shifted cancel key";
412144d93782SGreg Clayton         case KEY_SCOMMAND:  return "shifted command key";
412244d93782SGreg Clayton         case KEY_SCOPY:     return "shifted copy key";
412344d93782SGreg Clayton         case KEY_SCREATE:   return "shifted create key";
412444d93782SGreg Clayton         case KEY_SDC:       return "shifted delete-character key";
412544d93782SGreg Clayton         case KEY_SDL:       return "shifted delete-line key";
412644d93782SGreg Clayton         case KEY_SELECT:    return "select key";
412744d93782SGreg Clayton         case KEY_SEND:      return "shifted end key";
412844d93782SGreg Clayton         case KEY_SEOL:      return "shifted clear-to-end-of-line key";
412944d93782SGreg Clayton         case KEY_SEXIT:     return "shifted exit key";
413044d93782SGreg Clayton         case KEY_SFIND:     return "shifted find key";
413144d93782SGreg Clayton         case KEY_SHELP:     return "shifted help key";
413244d93782SGreg Clayton         case KEY_SHOME:     return "shifted home key";
413344d93782SGreg Clayton         case KEY_SIC:       return "shifted insert-character key";
413444d93782SGreg Clayton         case KEY_SLEFT:     return "shifted left-arrow key";
413544d93782SGreg Clayton         case KEY_SMESSAGE:  return "shifted message key";
413644d93782SGreg Clayton         case KEY_SMOVE:     return "shifted move key";
413744d93782SGreg Clayton         case KEY_SNEXT:     return "shifted next key";
413844d93782SGreg Clayton         case KEY_SOPTIONS:  return "shifted options key";
413944d93782SGreg Clayton         case KEY_SPREVIOUS: return "shifted previous key";
414044d93782SGreg Clayton         case KEY_SPRINT:    return "shifted print key";
414144d93782SGreg Clayton         case KEY_SREDO:     return "shifted redo key";
414244d93782SGreg Clayton         case KEY_SREPLACE:  return "shifted replace key";
414344d93782SGreg Clayton         case KEY_SRIGHT:    return "shifted right-arrow key";
414444d93782SGreg Clayton         case KEY_SRSUME:    return "shifted resume key";
414544d93782SGreg Clayton         case KEY_SSAVE:     return "shifted save key";
414644d93782SGreg Clayton         case KEY_SSUSPEND:  return "shifted suspend key";
414744d93782SGreg Clayton         case KEY_SUNDO:     return "shifted undo key";
414844d93782SGreg Clayton         case KEY_SUSPEND:   return "suspend key";
414944d93782SGreg Clayton         case KEY_UNDO:      return "undo key";
415044d93782SGreg Clayton         case KEY_MOUSE:     return "Mouse event has occurred";
415144d93782SGreg Clayton         case KEY_RESIZE:    return "Terminal resize event";
415244d93782SGreg Clayton         case KEY_EVENT:     return "We were interrupted by an event";
415344d93782SGreg Clayton         case KEY_RETURN:    return "return";
415444d93782SGreg Clayton         case ' ':           return "space";
41555fdb09bbSGreg Clayton         case '\t':          return "tab";
415644d93782SGreg Clayton         case KEY_ESCAPE:    return "escape";
415744d93782SGreg Clayton         default:
415844d93782SGreg Clayton             if (isprint(ch))
415944d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "%c", ch);
416044d93782SGreg Clayton             else
416144d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
416244d93782SGreg Clayton             return g_desc;
416344d93782SGreg Clayton     }
416444d93782SGreg Clayton     return NULL;
416544d93782SGreg Clayton }
416644d93782SGreg Clayton 
416744d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) :
416844d93782SGreg Clayton     m_text (),
416944d93782SGreg Clayton     m_first_visible_line (0)
417044d93782SGreg Clayton {
417144d93782SGreg Clayton     if (text && text[0])
417244d93782SGreg Clayton     {
417344d93782SGreg Clayton         m_text.SplitIntoLines(text);
417444d93782SGreg Clayton         m_text.AppendString("");
417544d93782SGreg Clayton     }
417644d93782SGreg Clayton     if (key_help_array)
417744d93782SGreg Clayton     {
417844d93782SGreg Clayton         for (KeyHelp *key = key_help_array; key->ch; ++key)
417944d93782SGreg Clayton         {
418044d93782SGreg Clayton             StreamString key_description;
418144d93782SGreg Clayton             key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description);
418244d93782SGreg Clayton             m_text.AppendString(std::move(key_description.GetString()));
418344d93782SGreg Clayton         }
418444d93782SGreg Clayton     }
418544d93782SGreg Clayton }
418644d93782SGreg Clayton 
418744d93782SGreg Clayton HelpDialogDelegate::~HelpDialogDelegate()
418844d93782SGreg Clayton {
418944d93782SGreg Clayton }
419044d93782SGreg Clayton 
419144d93782SGreg Clayton bool
419244d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force)
419344d93782SGreg Clayton {
419444d93782SGreg Clayton     window.Erase();
419544d93782SGreg Clayton     const int window_height = window.GetHeight();
419644d93782SGreg Clayton     int x = 2;
419744d93782SGreg Clayton     int y = 1;
419844d93782SGreg Clayton     const int min_y = y;
419944d93782SGreg Clayton     const int max_y = window_height - 1 - y;
42003985c8c6SSaleem Abdulrasool     const size_t num_visible_lines = max_y - min_y + 1;
420144d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
420244d93782SGreg Clayton     const char *bottom_message;
420344d93782SGreg Clayton     if (num_lines <= num_visible_lines)
420444d93782SGreg Clayton         bottom_message = "Press any key to exit";
420544d93782SGreg Clayton     else
420644d93782SGreg Clayton         bottom_message = "Use arrows to scroll, any other key to exit";
420744d93782SGreg Clayton     window.DrawTitleBox(window.GetName(), bottom_message);
420844d93782SGreg Clayton     while (y <= max_y)
420944d93782SGreg Clayton     {
421044d93782SGreg Clayton         window.MoveCursor(x, y);
421144d93782SGreg Clayton         window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
421244d93782SGreg Clayton         ++y;
421344d93782SGreg Clayton     }
421444d93782SGreg Clayton     return true;
421544d93782SGreg Clayton }
421644d93782SGreg Clayton 
421744d93782SGreg Clayton HandleCharResult
421844d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key)
421944d93782SGreg Clayton {
422044d93782SGreg Clayton     bool done = false;
422144d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
422244d93782SGreg Clayton     const size_t num_visible_lines = window.GetHeight() - 2;
422344d93782SGreg Clayton 
422444d93782SGreg Clayton     if (num_lines <= num_visible_lines)
422544d93782SGreg Clayton     {
422644d93782SGreg Clayton         done = true;
422744d93782SGreg Clayton         // If we have all lines visible and don't need scrolling, then any
422844d93782SGreg Clayton         // key press will cause us to exit
422944d93782SGreg Clayton     }
423044d93782SGreg Clayton     else
423144d93782SGreg Clayton     {
423244d93782SGreg Clayton         switch (key)
423344d93782SGreg Clayton         {
423444d93782SGreg Clayton             case KEY_UP:
423544d93782SGreg Clayton                 if (m_first_visible_line > 0)
423644d93782SGreg Clayton                     --m_first_visible_line;
423744d93782SGreg Clayton                 break;
423844d93782SGreg Clayton 
423944d93782SGreg Clayton             case KEY_DOWN:
424044d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
424144d93782SGreg Clayton                     ++m_first_visible_line;
424244d93782SGreg Clayton                 break;
424344d93782SGreg Clayton 
424444d93782SGreg Clayton             case KEY_PPAGE:
424544d93782SGreg Clayton             case ',':
424644d93782SGreg Clayton                 if (m_first_visible_line > 0)
424744d93782SGreg Clayton                 {
42483985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
424944d93782SGreg Clayton                         m_first_visible_line -= num_visible_lines;
425044d93782SGreg Clayton                     else
425144d93782SGreg Clayton                         m_first_visible_line = 0;
425244d93782SGreg Clayton                 }
425344d93782SGreg Clayton                 break;
425444d93782SGreg Clayton             case KEY_NPAGE:
425544d93782SGreg Clayton             case '.':
425644d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
425744d93782SGreg Clayton                 {
425844d93782SGreg Clayton                     m_first_visible_line += num_visible_lines;
42593985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > num_lines)
426044d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
426144d93782SGreg Clayton                 }
426244d93782SGreg Clayton                 break;
426344d93782SGreg Clayton             default:
426444d93782SGreg Clayton                 done = true;
426544d93782SGreg Clayton                 break;
426644d93782SGreg Clayton         }
426744d93782SGreg Clayton     }
426844d93782SGreg Clayton     if (done)
426944d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
427044d93782SGreg Clayton     return eKeyHandled;
427144d93782SGreg Clayton }
427244d93782SGreg Clayton 
427344d93782SGreg Clayton class ApplicationDelegate :
427444d93782SGreg Clayton     public WindowDelegate,
427544d93782SGreg Clayton     public MenuDelegate
427644d93782SGreg Clayton {
427744d93782SGreg Clayton public:
427844d93782SGreg Clayton     enum {
427944d93782SGreg Clayton         eMenuID_LLDB = 1,
428044d93782SGreg Clayton         eMenuID_LLDBAbout,
428144d93782SGreg Clayton         eMenuID_LLDBExit,
428244d93782SGreg Clayton 
428344d93782SGreg Clayton         eMenuID_Target,
428444d93782SGreg Clayton         eMenuID_TargetCreate,
428544d93782SGreg Clayton         eMenuID_TargetDelete,
428644d93782SGreg Clayton 
428744d93782SGreg Clayton         eMenuID_Process,
428844d93782SGreg Clayton         eMenuID_ProcessAttach,
428944d93782SGreg Clayton         eMenuID_ProcessDetach,
429044d93782SGreg Clayton         eMenuID_ProcessLaunch,
429144d93782SGreg Clayton         eMenuID_ProcessContinue,
429244d93782SGreg Clayton         eMenuID_ProcessHalt,
429344d93782SGreg Clayton         eMenuID_ProcessKill,
429444d93782SGreg Clayton 
429544d93782SGreg Clayton         eMenuID_Thread,
429644d93782SGreg Clayton         eMenuID_ThreadStepIn,
429744d93782SGreg Clayton         eMenuID_ThreadStepOver,
429844d93782SGreg Clayton         eMenuID_ThreadStepOut,
429944d93782SGreg Clayton 
430044d93782SGreg Clayton         eMenuID_View,
430144d93782SGreg Clayton         eMenuID_ViewBacktrace,
430244d93782SGreg Clayton         eMenuID_ViewRegisters,
430344d93782SGreg Clayton         eMenuID_ViewSource,
430444d93782SGreg Clayton         eMenuID_ViewVariables,
430544d93782SGreg Clayton 
430644d93782SGreg Clayton         eMenuID_Help,
430744d93782SGreg Clayton         eMenuID_HelpGUIHelp
430844d93782SGreg Clayton     };
430944d93782SGreg Clayton 
431044d93782SGreg Clayton     ApplicationDelegate (Application &app, Debugger &debugger) :
431144d93782SGreg Clayton         WindowDelegate (),
431244d93782SGreg Clayton         MenuDelegate (),
431344d93782SGreg Clayton         m_app (app),
431444d93782SGreg Clayton         m_debugger (debugger)
431544d93782SGreg Clayton     {
431644d93782SGreg Clayton     }
431744d93782SGreg Clayton 
431844d93782SGreg Clayton     ~ApplicationDelegate ()
431944d93782SGreg Clayton     {
432044d93782SGreg Clayton     }
4321bd5ae6b4SGreg Clayton 
4322bd5ae6b4SGreg Clayton     bool
4323bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
432444d93782SGreg Clayton     {
432544d93782SGreg Clayton         return false; // Drawing not handled, let standard window drawing happen
432644d93782SGreg Clayton     }
432744d93782SGreg Clayton 
4328bd5ae6b4SGreg Clayton     HandleCharResult
4329bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int key) override
433044d93782SGreg Clayton     {
43315fdb09bbSGreg Clayton         switch (key)
433244d93782SGreg Clayton         {
43335fdb09bbSGreg Clayton             case '\t':
433444d93782SGreg Clayton                 window.SelectNextWindowAsActive();
433544d93782SGreg Clayton                 return eKeyHandled;
43365fdb09bbSGreg Clayton 
43375fdb09bbSGreg Clayton             case 'h':
43385fdb09bbSGreg Clayton                 window.CreateHelpSubwindow();
43395fdb09bbSGreg Clayton                 return eKeyHandled;
43405fdb09bbSGreg Clayton 
43415fdb09bbSGreg Clayton             case KEY_ESCAPE:
43425fdb09bbSGreg Clayton                 return eQuitApplication;
43435fdb09bbSGreg Clayton 
43445fdb09bbSGreg Clayton             default:
43455fdb09bbSGreg Clayton                 break;
434644d93782SGreg Clayton         }
434744d93782SGreg Clayton         return eKeyNotHandled;
434844d93782SGreg Clayton     }
434944d93782SGreg Clayton 
43505fdb09bbSGreg Clayton 
4351bd5ae6b4SGreg Clayton     const char *
4352bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
43535fdb09bbSGreg Clayton     {
43545fdb09bbSGreg Clayton         return "Welcome to the LLDB curses GUI.\n\n"
43555fdb09bbSGreg Clayton         "Press the TAB key to change the selected view.\n"
43565fdb09bbSGreg Clayton         "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n"
43575fdb09bbSGreg Clayton         "Common key bindings for all views:";
43585fdb09bbSGreg Clayton     }
43595fdb09bbSGreg Clayton 
4360bd5ae6b4SGreg Clayton     KeyHelp *
4361bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
43625fdb09bbSGreg Clayton     {
43635fdb09bbSGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
43645fdb09bbSGreg Clayton             { '\t', "Select next view" },
43655fdb09bbSGreg Clayton             { 'h', "Show help dialog with view specific key bindings" },
43665fdb09bbSGreg Clayton             { ',', "Page up" },
43675fdb09bbSGreg Clayton             { '.', "Page down" },
43685fdb09bbSGreg Clayton             { KEY_UP, "Select previous" },
43695fdb09bbSGreg Clayton             { KEY_DOWN, "Select next" },
43705fdb09bbSGreg Clayton             { KEY_LEFT, "Unexpand or select parent" },
43715fdb09bbSGreg Clayton             { KEY_RIGHT, "Expand" },
43725fdb09bbSGreg Clayton             { KEY_PPAGE, "Page up" },
43735fdb09bbSGreg Clayton             { KEY_NPAGE, "Page down" },
43745fdb09bbSGreg Clayton             { '\0', NULL }
43755fdb09bbSGreg Clayton         };
43765fdb09bbSGreg Clayton         return g_source_view_key_help;
43775fdb09bbSGreg Clayton     }
43785fdb09bbSGreg Clayton 
4379bd5ae6b4SGreg Clayton     MenuActionResult
4380bd5ae6b4SGreg Clayton     MenuDelegateAction (Menu &menu) override
438144d93782SGreg Clayton     {
438244d93782SGreg Clayton         switch (menu.GetIdentifier())
438344d93782SGreg Clayton         {
438444d93782SGreg Clayton             case eMenuID_ThreadStepIn:
438544d93782SGreg Clayton                 {
438644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
438744d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
438844d93782SGreg Clayton                     {
438944d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
439044d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
43914b4b2478SJim Ingham                             exe_ctx.GetThreadRef().StepIn(true);
439244d93782SGreg Clayton                     }
439344d93782SGreg Clayton                 }
439444d93782SGreg Clayton                 return MenuActionResult::Handled;
439544d93782SGreg Clayton 
439644d93782SGreg Clayton             case eMenuID_ThreadStepOut:
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))
440344d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOut();
440444d93782SGreg Clayton                     }
440544d93782SGreg Clayton                 }
440644d93782SGreg Clayton                 return MenuActionResult::Handled;
440744d93782SGreg Clayton 
440844d93782SGreg Clayton             case eMenuID_ThreadStepOver:
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().StepOver(true);
441644d93782SGreg Clayton                     }
441744d93782SGreg Clayton                 }
441844d93782SGreg Clayton                 return MenuActionResult::Handled;
441944d93782SGreg Clayton 
442044d93782SGreg Clayton             case eMenuID_ProcessContinue:
442144d93782SGreg Clayton                 {
442244d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
442344d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
442444d93782SGreg Clayton                     {
442544d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
442644d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
442744d93782SGreg Clayton                             process->Resume();
442844d93782SGreg Clayton                     }
442944d93782SGreg Clayton                 }
443044d93782SGreg Clayton                 return MenuActionResult::Handled;
443144d93782SGreg Clayton 
443244d93782SGreg Clayton             case eMenuID_ProcessKill:
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())
4439ede3193bSJason Molenda                             process->Destroy(false);
444044d93782SGreg Clayton                     }
444144d93782SGreg Clayton                 }
444244d93782SGreg Clayton                 return MenuActionResult::Handled;
444344d93782SGreg Clayton 
444444d93782SGreg Clayton             case eMenuID_ProcessHalt:
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->Halt();
445244d93782SGreg Clayton                     }
445344d93782SGreg Clayton                 }
445444d93782SGreg Clayton                 return MenuActionResult::Handled;
445544d93782SGreg Clayton 
445644d93782SGreg Clayton             case eMenuID_ProcessDetach:
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->Detach(false);
446444d93782SGreg Clayton                     }
446544d93782SGreg Clayton                 }
446644d93782SGreg Clayton                 return MenuActionResult::Handled;
446744d93782SGreg Clayton 
446844d93782SGreg Clayton             case eMenuID_Process:
446944d93782SGreg Clayton                 {
447044d93782SGreg Clayton                     // Populate the menu with all of the threads if the process is stopped when
447144d93782SGreg Clayton                     // the Process menu gets selected and is about to display its submenu.
447244d93782SGreg Clayton                     Menus &submenus = menu.GetSubmenus();
447344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
447444d93782SGreg Clayton                     Process *process = exe_ctx.GetProcessPtr();
447544d93782SGreg Clayton                     if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
447644d93782SGreg Clayton                     {
447744d93782SGreg Clayton                         if (submenus.size() == 7)
447844d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
447944d93782SGreg Clayton                         else if (submenus.size() > 8)
448044d93782SGreg Clayton                             submenus.erase (submenus.begin() + 8, submenus.end());
448144d93782SGreg Clayton 
448244d93782SGreg Clayton                         ThreadList &threads = process->GetThreadList();
448344d93782SGreg Clayton                         Mutex::Locker locker (threads.GetMutex());
448444d93782SGreg Clayton                         size_t num_threads = threads.GetSize();
448544d93782SGreg Clayton                         for (size_t i=0; i<num_threads; ++i)
448644d93782SGreg Clayton                         {
448744d93782SGreg Clayton                             ThreadSP thread_sp = threads.GetThreadAtIndex(i);
448844d93782SGreg Clayton                             char menu_char = '\0';
448944d93782SGreg Clayton                             if (i < 9)
449044d93782SGreg Clayton                                 menu_char = '1' + i;
449144d93782SGreg Clayton                             StreamString thread_menu_title;
449244d93782SGreg Clayton                             thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
449344d93782SGreg Clayton                             const char *thread_name = thread_sp->GetName();
449444d93782SGreg Clayton                             if (thread_name && thread_name[0])
449544d93782SGreg Clayton                                 thread_menu_title.Printf (" %s", thread_name);
449644d93782SGreg Clayton                             else
449744d93782SGreg Clayton                             {
449844d93782SGreg Clayton                                 const char *queue_name = thread_sp->GetQueueName();
449944d93782SGreg Clayton                                 if (queue_name && queue_name[0])
450044d93782SGreg Clayton                                     thread_menu_title.Printf (" %s", queue_name);
450144d93782SGreg Clayton                             }
450244d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(thread_menu_title.GetString().c_str(), NULL, menu_char, thread_sp->GetID())));
450344d93782SGreg Clayton                         }
450444d93782SGreg Clayton                     }
450544d93782SGreg Clayton                     else if (submenus.size() > 7)
450644d93782SGreg Clayton                     {
450744d93782SGreg Clayton                         // Remove the separator and any other thread submenu items
450844d93782SGreg Clayton                         // that were previously added
450944d93782SGreg Clayton                         submenus.erase (submenus.begin() + 7, submenus.end());
451044d93782SGreg Clayton                     }
451144d93782SGreg Clayton                     // Since we are adding and removing items we need to recalculate the name lengths
451244d93782SGreg Clayton                     menu.RecalculateNameLengths();
451344d93782SGreg Clayton                 }
451444d93782SGreg Clayton                 return MenuActionResult::Handled;
451544d93782SGreg Clayton 
451644d93782SGreg Clayton             case eMenuID_ViewVariables:
451744d93782SGreg Clayton                 {
451844d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
451944d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
452044d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
452144d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
452244d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
452344d93782SGreg Clayton 
452444d93782SGreg Clayton                     if (variables_window_sp)
452544d93782SGreg Clayton                     {
452644d93782SGreg Clayton                         const Rect variables_bounds = variables_window_sp->GetBounds();
452744d93782SGreg Clayton 
452844d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(variables_window_sp.get());
452944d93782SGreg Clayton 
453044d93782SGreg Clayton                         if (registers_window_sp)
453144d93782SGreg Clayton                         {
453244d93782SGreg Clayton                             // We have a registers window, so give all the area back to the registers window
453344d93782SGreg Clayton                             Rect registers_bounds = variables_bounds;
453444d93782SGreg Clayton                             registers_bounds.size.width = source_bounds.size.width;
453544d93782SGreg Clayton                             registers_window_sp->SetBounds(registers_bounds);
453644d93782SGreg Clayton                         }
453744d93782SGreg Clayton                         else
453844d93782SGreg Clayton                         {
453944d93782SGreg Clayton                             // We have no registers window showing so give the bottom
454044d93782SGreg Clayton                             // area back to the source view
454144d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
454244d93782SGreg Clayton                                                       source_bounds.size.height + variables_bounds.size.height);
454344d93782SGreg Clayton                         }
454444d93782SGreg Clayton                     }
454544d93782SGreg Clayton                     else
454644d93782SGreg Clayton                     {
454744d93782SGreg Clayton                         Rect new_variables_rect;
454844d93782SGreg Clayton                         if (registers_window_sp)
454944d93782SGreg Clayton                         {
455044d93782SGreg Clayton                             // We have a registers window so split the area of the registers
455144d93782SGreg Clayton                             // window into two columns where the left hand side will be the
455244d93782SGreg Clayton                             // variables and the right hand side will be the registers
455344d93782SGreg Clayton                             const Rect variables_bounds = registers_window_sp->GetBounds();
455444d93782SGreg Clayton                             Rect new_registers_rect;
455544d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect);
455644d93782SGreg Clayton                             registers_window_sp->SetBounds (new_registers_rect);
455744d93782SGreg Clayton                         }
455844d93782SGreg Clayton                         else
455944d93782SGreg Clayton                         {
456044d93782SGreg Clayton                             // No variables window, grab the bottom part of the source window
456144d93782SGreg Clayton                             Rect new_source_rect;
456244d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect);
456344d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
456444d93782SGreg Clayton                         }
456544d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables",
456644d93782SGreg Clayton                                                                                   new_variables_rect,
456744d93782SGreg Clayton                                                                                   false);
456844d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
456944d93782SGreg Clayton                     }
457044d93782SGreg Clayton                     touchwin(stdscr);
457144d93782SGreg Clayton                 }
457244d93782SGreg Clayton                 return MenuActionResult::Handled;
457344d93782SGreg Clayton 
457444d93782SGreg Clayton             case eMenuID_ViewRegisters:
457544d93782SGreg Clayton                 {
457644d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
457744d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
457844d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
457944d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
458044d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
458144d93782SGreg Clayton 
458244d93782SGreg Clayton                     if (registers_window_sp)
458344d93782SGreg Clayton                     {
458444d93782SGreg Clayton                         if (variables_window_sp)
458544d93782SGreg Clayton                         {
458644d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
458744d93782SGreg Clayton 
458844d93782SGreg Clayton                             // We have a variables window, so give all the area back to the variables window
458944d93782SGreg Clayton                             variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(),
459044d93782SGreg Clayton                                                          variables_bounds.size.height);
459144d93782SGreg Clayton                         }
459244d93782SGreg Clayton                         else
459344d93782SGreg Clayton                         {
459444d93782SGreg Clayton                             // We have no variables window showing so give the bottom
459544d93782SGreg Clayton                             // area back to the source view
459644d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
459744d93782SGreg Clayton                                                       source_bounds.size.height + registers_window_sp->GetHeight());
459844d93782SGreg Clayton                         }
459944d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(registers_window_sp.get());
460044d93782SGreg Clayton                     }
460144d93782SGreg Clayton                     else
460244d93782SGreg Clayton                     {
460344d93782SGreg Clayton                         Rect new_regs_rect;
460444d93782SGreg Clayton                         if (variables_window_sp)
460544d93782SGreg Clayton                         {
460644d93782SGreg Clayton                             // We have a variables window, split it into two columns
460744d93782SGreg Clayton                             // where the left hand side will be the variables and the
460844d93782SGreg Clayton                             // right hand side will be the registers
460944d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
461044d93782SGreg Clayton                             Rect new_vars_rect;
461144d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect);
461244d93782SGreg Clayton                             variables_window_sp->SetBounds (new_vars_rect);
461344d93782SGreg Clayton                         }
461444d93782SGreg Clayton                         else
461544d93782SGreg Clayton                         {
461644d93782SGreg Clayton                             // No registers window, grab the bottom part of the source window
461744d93782SGreg Clayton                             Rect new_source_rect;
461844d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect);
461944d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
462044d93782SGreg Clayton                         }
462144d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers",
462244d93782SGreg Clayton                                                                                   new_regs_rect,
462344d93782SGreg Clayton                                                                                   false);
462444d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
462544d93782SGreg Clayton                     }
462644d93782SGreg Clayton                     touchwin(stdscr);
462744d93782SGreg Clayton                 }
462844d93782SGreg Clayton                 return MenuActionResult::Handled;
462944d93782SGreg Clayton 
463044d93782SGreg Clayton             case eMenuID_HelpGUIHelp:
46315fdb09bbSGreg Clayton                 m_app.GetMainWindow ()->CreateHelpSubwindow();
463244d93782SGreg Clayton                 return MenuActionResult::Handled;
463344d93782SGreg Clayton 
463444d93782SGreg Clayton             default:
463544d93782SGreg Clayton                 break;
463644d93782SGreg Clayton         }
463744d93782SGreg Clayton 
463844d93782SGreg Clayton         return MenuActionResult::NotHandled;
463944d93782SGreg Clayton     }
464044d93782SGreg Clayton protected:
464144d93782SGreg Clayton     Application &m_app;
464244d93782SGreg Clayton     Debugger &m_debugger;
464344d93782SGreg Clayton };
464444d93782SGreg Clayton 
464544d93782SGreg Clayton 
464644d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate
464744d93782SGreg Clayton {
464844d93782SGreg Clayton public:
464944d93782SGreg Clayton     StatusBarWindowDelegate (Debugger &debugger) :
465044d93782SGreg Clayton         m_debugger (debugger)
465144d93782SGreg Clayton     {
4652554f68d3SGreg Clayton         FormatEntity::Parse("Thread: ${thread.id%tid}",
4653554f68d3SGreg Clayton                             m_format);
465444d93782SGreg Clayton     }
465544d93782SGreg Clayton 
465644d93782SGreg Clayton     ~StatusBarWindowDelegate ()
465744d93782SGreg Clayton     {
465844d93782SGreg Clayton     }
4659bd5ae6b4SGreg Clayton 
4660bd5ae6b4SGreg Clayton     bool
4661bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
466244d93782SGreg Clayton     {
466344d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
466444d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
466544d93782SGreg Clayton         Thread *thread = exe_ctx.GetThreadPtr();
466644d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
466744d93782SGreg Clayton         window.Erase();
466844d93782SGreg Clayton         window.SetBackground(2);
466944d93782SGreg Clayton         window.MoveCursor (0, 0);
467044d93782SGreg Clayton         if (process)
467144d93782SGreg Clayton         {
467244d93782SGreg Clayton             const StateType state = process->GetState();
467344d93782SGreg Clayton             window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state));
467444d93782SGreg Clayton 
467544d93782SGreg Clayton             if (StateIsStoppedState(state, true))
467644d93782SGreg Clayton             {
46775b031ebcSEd Maste                 StreamString strm;
4678554f68d3SGreg Clayton                 if (thread && FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
46795b031ebcSEd Maste                 {
468044d93782SGreg Clayton                     window.MoveCursor (40, 0);
46815b031ebcSEd Maste                     window.PutCStringTruncated(strm.GetString().c_str(), 1);
46825b031ebcSEd Maste                 }
468344d93782SGreg Clayton 
468444d93782SGreg Clayton                 window.MoveCursor (60, 0);
468544d93782SGreg Clayton                 if (frame)
468644d93782SGreg Clayton                     window.Printf ("Frame: %3u  PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr()));
468744d93782SGreg Clayton             }
468844d93782SGreg Clayton             else if (state == eStateExited)
468944d93782SGreg Clayton             {
469044d93782SGreg Clayton                 const char *exit_desc = process->GetExitDescription();
469144d93782SGreg Clayton                 const int exit_status = process->GetExitStatus();
469244d93782SGreg Clayton                 if (exit_desc && exit_desc[0])
469344d93782SGreg Clayton                     window.Printf (" with status = %i (%s)", exit_status, exit_desc);
469444d93782SGreg Clayton                 else
469544d93782SGreg Clayton                     window.Printf (" with status = %i", exit_status);
469644d93782SGreg Clayton             }
469744d93782SGreg Clayton         }
469844d93782SGreg Clayton         window.DeferredRefresh();
469944d93782SGreg Clayton         return true;
470044d93782SGreg Clayton     }
470144d93782SGreg Clayton 
470244d93782SGreg Clayton protected:
470344d93782SGreg Clayton     Debugger &m_debugger;
4704554f68d3SGreg Clayton     FormatEntity::Entry m_format;
470544d93782SGreg Clayton };
470644d93782SGreg Clayton 
470744d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate
470844d93782SGreg Clayton {
470944d93782SGreg Clayton public:
471044d93782SGreg Clayton     SourceFileWindowDelegate (Debugger &debugger) :
471144d93782SGreg Clayton         WindowDelegate (),
471244d93782SGreg Clayton         m_debugger (debugger),
471344d93782SGreg Clayton         m_sc (),
471444d93782SGreg Clayton         m_file_sp (),
471544d93782SGreg Clayton         m_disassembly_scope (NULL),
471644d93782SGreg Clayton         m_disassembly_sp (),
471744d93782SGreg Clayton         m_disassembly_range (),
4718ec990867SGreg Clayton         m_title (),
471944d93782SGreg Clayton         m_line_width (4),
472044d93782SGreg Clayton         m_selected_line (0),
472144d93782SGreg Clayton         m_pc_line (0),
472244d93782SGreg Clayton         m_stop_id (0),
472344d93782SGreg Clayton         m_frame_idx (UINT32_MAX),
472444d93782SGreg Clayton         m_first_visible_line (0),
472544d93782SGreg Clayton         m_min_x (0),
472644d93782SGreg Clayton         m_min_y (0),
472744d93782SGreg Clayton         m_max_x (0),
472844d93782SGreg Clayton         m_max_y (0)
472944d93782SGreg Clayton     {
473044d93782SGreg Clayton     }
473144d93782SGreg Clayton 
4732bd5ae6b4SGreg Clayton     ~SourceFileWindowDelegate() override
473344d93782SGreg Clayton     {
473444d93782SGreg Clayton     }
473544d93782SGreg Clayton 
473644d93782SGreg Clayton     void
473744d93782SGreg Clayton     Update (const SymbolContext &sc)
473844d93782SGreg Clayton     {
473944d93782SGreg Clayton         m_sc = sc;
474044d93782SGreg Clayton     }
474144d93782SGreg Clayton 
474244d93782SGreg Clayton     uint32_t
474344d93782SGreg Clayton     NumVisibleLines () const
474444d93782SGreg Clayton     {
474544d93782SGreg Clayton         return m_max_y - m_min_y;
474644d93782SGreg Clayton     }
474744d93782SGreg Clayton 
4748bd5ae6b4SGreg Clayton     const char *
4749bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
475044d93782SGreg Clayton     {
475144d93782SGreg Clayton         return "Source/Disassembly window keyboard shortcuts:";
475244d93782SGreg Clayton     }
475344d93782SGreg Clayton 
4754bd5ae6b4SGreg Clayton     KeyHelp *
4755bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
475644d93782SGreg Clayton     {
475744d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
475844d93782SGreg Clayton             { KEY_RETURN, "Run to selected line with one shot breakpoint" },
475944d93782SGreg Clayton             { KEY_UP, "Select previous source line" },
476044d93782SGreg Clayton             { KEY_DOWN, "Select next source line" },
476144d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
476244d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
476344d93782SGreg Clayton             { 'b', "Set breakpoint on selected source/disassembly line" },
476444d93782SGreg Clayton             { 'c', "Continue process" },
476544d93782SGreg Clayton             { 'd', "Detach and resume process" },
476644d93782SGreg Clayton             { 'D', "Detach with process suspended" },
476744d93782SGreg Clayton             { 'h', "Show help dialog" },
476844d93782SGreg Clayton             { 'k', "Kill process" },
476944d93782SGreg Clayton             { 'n', "Step over (source line)" },
477044d93782SGreg Clayton             { 'N', "Step over (single instruction)" },
477144d93782SGreg Clayton             { 'o', "Step out" },
477244d93782SGreg Clayton             { 's', "Step in (source line)" },
477344d93782SGreg Clayton             { 'S', "Step in (single instruction)" },
477444d93782SGreg Clayton             { ',', "Page up" },
477544d93782SGreg Clayton             { '.', "Page down" },
477644d93782SGreg Clayton             { '\0', NULL }
477744d93782SGreg Clayton         };
477844d93782SGreg Clayton         return g_source_view_key_help;
477944d93782SGreg Clayton     }
478044d93782SGreg Clayton 
4781bd5ae6b4SGreg Clayton     bool
4782bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
478344d93782SGreg Clayton     {
478444d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
478544d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
478644d93782SGreg Clayton         Thread *thread = NULL;
478744d93782SGreg Clayton 
478844d93782SGreg Clayton         bool update_location = false;
478944d93782SGreg Clayton         if (process)
479044d93782SGreg Clayton         {
479144d93782SGreg Clayton             StateType state = process->GetState();
479244d93782SGreg Clayton             if (StateIsStoppedState(state, true))
479344d93782SGreg Clayton             {
479444d93782SGreg Clayton                 // We are stopped, so it is ok to
479544d93782SGreg Clayton                 update_location = true;
479644d93782SGreg Clayton             }
479744d93782SGreg Clayton         }
479844d93782SGreg Clayton 
479944d93782SGreg Clayton         m_min_x = 1;
4800ec990867SGreg Clayton         m_min_y = 2;
480144d93782SGreg Clayton         m_max_x = window.GetMaxX()-1;
480244d93782SGreg Clayton         m_max_y = window.GetMaxY()-1;
480344d93782SGreg Clayton 
480444d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
480544d93782SGreg Clayton         StackFrameSP frame_sp;
480644d93782SGreg Clayton         bool set_selected_line_to_pc = false;
480744d93782SGreg Clayton 
480844d93782SGreg Clayton         if (update_location)
480944d93782SGreg Clayton         {
481044d93782SGreg Clayton             const bool process_alive = process ? process->IsAlive() : false;
481144d93782SGreg Clayton             bool thread_changed = false;
481244d93782SGreg Clayton             if (process_alive)
481344d93782SGreg Clayton             {
481444d93782SGreg Clayton                 thread = exe_ctx.GetThreadPtr();
481544d93782SGreg Clayton                 if (thread)
481644d93782SGreg Clayton                 {
481744d93782SGreg Clayton                     frame_sp = thread->GetSelectedFrame();
481844d93782SGreg Clayton                     auto tid = thread->GetID();
481944d93782SGreg Clayton                     thread_changed = tid != m_tid;
482044d93782SGreg Clayton                     m_tid = tid;
482144d93782SGreg Clayton                 }
482244d93782SGreg Clayton                 else
482344d93782SGreg Clayton                 {
482444d93782SGreg Clayton                     if (m_tid != LLDB_INVALID_THREAD_ID)
482544d93782SGreg Clayton                     {
482644d93782SGreg Clayton                         thread_changed = true;
482744d93782SGreg Clayton                         m_tid = LLDB_INVALID_THREAD_ID;
482844d93782SGreg Clayton                     }
482944d93782SGreg Clayton                 }
483044d93782SGreg Clayton             }
483144d93782SGreg Clayton             const uint32_t stop_id = process ? process->GetStopID() : 0;
483244d93782SGreg Clayton             const bool stop_id_changed = stop_id != m_stop_id;
483344d93782SGreg Clayton             bool frame_changed = false;
483444d93782SGreg Clayton             m_stop_id = stop_id;
4835ec990867SGreg Clayton             m_title.Clear();
483644d93782SGreg Clayton             if (frame_sp)
483744d93782SGreg Clayton             {
483844d93782SGreg Clayton                 m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
4839ec990867SGreg Clayton                 if (m_sc.module_sp)
4840ec990867SGreg Clayton                 {
4841ec990867SGreg Clayton                     m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
4842ec990867SGreg Clayton                     ConstString func_name = m_sc.GetFunctionName();
4843ec990867SGreg Clayton                     if (func_name)
4844ec990867SGreg Clayton                         m_title.Printf("`%s", func_name.GetCString());
4845ec990867SGreg Clayton                 }
484644d93782SGreg Clayton                 const uint32_t frame_idx = frame_sp->GetFrameIndex();
484744d93782SGreg Clayton                 frame_changed = frame_idx != m_frame_idx;
484844d93782SGreg Clayton                 m_frame_idx = frame_idx;
484944d93782SGreg Clayton             }
485044d93782SGreg Clayton             else
485144d93782SGreg Clayton             {
485244d93782SGreg Clayton                 m_sc.Clear(true);
485344d93782SGreg Clayton                 frame_changed = m_frame_idx != UINT32_MAX;
485444d93782SGreg Clayton                 m_frame_idx = UINT32_MAX;
485544d93782SGreg Clayton             }
485644d93782SGreg Clayton 
485744d93782SGreg Clayton             const bool context_changed = thread_changed || frame_changed || stop_id_changed;
485844d93782SGreg Clayton 
485944d93782SGreg Clayton             if (process_alive)
486044d93782SGreg Clayton             {
486144d93782SGreg Clayton                 if (m_sc.line_entry.IsValid())
486244d93782SGreg Clayton                 {
486344d93782SGreg Clayton                     m_pc_line = m_sc.line_entry.line;
486444d93782SGreg Clayton                     if (m_pc_line != UINT32_MAX)
486544d93782SGreg Clayton                         --m_pc_line; // Convert to zero based line number...
486644d93782SGreg Clayton                     // Update the selected line if the stop ID changed...
486744d93782SGreg Clayton                     if (context_changed)
486844d93782SGreg Clayton                         m_selected_line = m_pc_line;
486944d93782SGreg Clayton 
487044d93782SGreg Clayton                     if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file))
487144d93782SGreg Clayton                     {
487244d93782SGreg Clayton                         // Same file, nothing to do, we should either have the
487344d93782SGreg Clayton                         // lines or not (source file missing)
48743985c8c6SSaleem Abdulrasool                         if (m_selected_line >= static_cast<size_t>(m_first_visible_line))
487544d93782SGreg Clayton                         {
487644d93782SGreg Clayton                             if (m_selected_line >= m_first_visible_line + num_visible_lines)
487744d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
487844d93782SGreg Clayton                         }
487944d93782SGreg Clayton                         else
488044d93782SGreg Clayton                         {
488144d93782SGreg Clayton                             if (m_selected_line > 10)
488244d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
488344d93782SGreg Clayton                             else
488444d93782SGreg Clayton                                 m_first_visible_line = 0;
488544d93782SGreg Clayton                         }
488644d93782SGreg Clayton                     }
488744d93782SGreg Clayton                     else
488844d93782SGreg Clayton                     {
488944d93782SGreg Clayton                         // File changed, set selected line to the line with the PC
489044d93782SGreg Clayton                         m_selected_line = m_pc_line;
489144d93782SGreg Clayton                         m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
489244d93782SGreg Clayton                         if (m_file_sp)
489344d93782SGreg Clayton                         {
489444d93782SGreg Clayton                             const size_t num_lines = m_file_sp->GetNumLines();
489544d93782SGreg Clayton                             int m_line_width = 1;
489644d93782SGreg Clayton                             for (size_t n = num_lines; n >= 10; n = n / 10)
489744d93782SGreg Clayton                                 ++m_line_width;
489844d93782SGreg Clayton 
489944d93782SGreg Clayton                             snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width);
490044d93782SGreg Clayton                             if (num_lines < num_visible_lines || m_selected_line < num_visible_lines)
490144d93782SGreg Clayton                                 m_first_visible_line = 0;
490244d93782SGreg Clayton                             else
490344d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
490444d93782SGreg Clayton                         }
490544d93782SGreg Clayton                     }
490644d93782SGreg Clayton                 }
490744d93782SGreg Clayton                 else
490844d93782SGreg Clayton                 {
490944d93782SGreg Clayton                     m_file_sp.reset();
491044d93782SGreg Clayton                 }
491144d93782SGreg Clayton 
491244d93782SGreg Clayton                 if (!m_file_sp || m_file_sp->GetNumLines() == 0)
491344d93782SGreg Clayton                 {
491444d93782SGreg Clayton                     // Show disassembly
491544d93782SGreg Clayton                     bool prefer_file_cache = false;
491644d93782SGreg Clayton                     if (m_sc.function)
491744d93782SGreg Clayton                     {
491844d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.function)
491944d93782SGreg Clayton                         {
492044d93782SGreg Clayton                             m_disassembly_scope = m_sc.function;
492144d93782SGreg Clayton                             m_disassembly_sp = m_sc.function->GetInstructions (exe_ctx, NULL, prefer_file_cache);
492244d93782SGreg Clayton                             if (m_disassembly_sp)
492344d93782SGreg Clayton                             {
492444d93782SGreg Clayton                                 set_selected_line_to_pc = true;
492544d93782SGreg Clayton                                 m_disassembly_range = m_sc.function->GetAddressRange();
492644d93782SGreg Clayton                             }
492744d93782SGreg Clayton                             else
492844d93782SGreg Clayton                             {
492944d93782SGreg Clayton                                 m_disassembly_range.Clear();
493044d93782SGreg Clayton                             }
493144d93782SGreg Clayton                         }
493244d93782SGreg Clayton                         else
493344d93782SGreg Clayton                         {
493444d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
493544d93782SGreg Clayton                         }
493644d93782SGreg Clayton                     }
493744d93782SGreg Clayton                     else if (m_sc.symbol)
493844d93782SGreg Clayton                     {
493944d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.symbol)
494044d93782SGreg Clayton                         {
494144d93782SGreg Clayton                             m_disassembly_scope = m_sc.symbol;
494244d93782SGreg Clayton                             m_disassembly_sp = m_sc.symbol->GetInstructions (exe_ctx, NULL, prefer_file_cache);
494344d93782SGreg Clayton                             if (m_disassembly_sp)
494444d93782SGreg Clayton                             {
494544d93782SGreg Clayton                                 set_selected_line_to_pc = true;
494644d93782SGreg Clayton                                 m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress();
494744d93782SGreg Clayton                                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
494844d93782SGreg Clayton                             }
494944d93782SGreg Clayton                             else
495044d93782SGreg Clayton                             {
495144d93782SGreg Clayton                                 m_disassembly_range.Clear();
495244d93782SGreg Clayton                             }
495344d93782SGreg Clayton                         }
495444d93782SGreg Clayton                         else
495544d93782SGreg Clayton                         {
495644d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
495744d93782SGreg Clayton                         }
495844d93782SGreg Clayton                     }
495944d93782SGreg Clayton                 }
496044d93782SGreg Clayton             }
496144d93782SGreg Clayton             else
496244d93782SGreg Clayton             {
496344d93782SGreg Clayton                 m_pc_line = UINT32_MAX;
496444d93782SGreg Clayton             }
496544d93782SGreg Clayton         }
496644d93782SGreg Clayton 
4967ec990867SGreg Clayton         const int window_width = window.GetWidth();
496844d93782SGreg Clayton         window.Erase();
496944d93782SGreg Clayton         window.DrawTitleBox ("Sources");
4970ec990867SGreg Clayton         if (!m_title.GetString().empty())
4971ec990867SGreg Clayton         {
4972ec990867SGreg Clayton             window.AttributeOn(A_REVERSE);
4973ec990867SGreg Clayton             window.MoveCursor(1, 1);
4974ec990867SGreg Clayton             window.PutChar(' ');
4975ec990867SGreg Clayton             window.PutCStringTruncated(m_title.GetString().c_str(), 1);
4976ec990867SGreg Clayton             int x = window.GetCursorX();
4977ec990867SGreg Clayton             if (x < window_width - 1)
4978ec990867SGreg Clayton             {
4979ec990867SGreg Clayton                 window.Printf ("%*s", window_width - x - 1, "");
4980ec990867SGreg Clayton             }
4981ec990867SGreg Clayton             window.AttributeOff(A_REVERSE);
4982ec990867SGreg Clayton         }
498344d93782SGreg Clayton 
498444d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
498544d93782SGreg Clayton         const size_t num_source_lines = GetNumSourceLines();
498644d93782SGreg Clayton         if (num_source_lines > 0)
498744d93782SGreg Clayton         {
498844d93782SGreg Clayton             // Display source
498944d93782SGreg Clayton             BreakpointLines bp_lines;
499044d93782SGreg Clayton             if (target)
499144d93782SGreg Clayton             {
499244d93782SGreg Clayton                 BreakpointList &bp_list = target->GetBreakpointList();
499344d93782SGreg Clayton                 const size_t num_bps = bp_list.GetSize();
499444d93782SGreg Clayton                 for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
499544d93782SGreg Clayton                 {
499644d93782SGreg Clayton                     BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
499744d93782SGreg Clayton                     const size_t num_bps_locs = bp_sp->GetNumLocations();
499844d93782SGreg Clayton                     for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
499944d93782SGreg Clayton                     {
500044d93782SGreg Clayton                         BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
500144d93782SGreg Clayton                         LineEntry bp_loc_line_entry;
500244d93782SGreg Clayton                         if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry))
500344d93782SGreg Clayton                         {
500444d93782SGreg Clayton                             if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file)
500544d93782SGreg Clayton                             {
500644d93782SGreg Clayton                                 bp_lines.insert(bp_loc_line_entry.line);
500744d93782SGreg Clayton                             }
500844d93782SGreg Clayton                         }
500944d93782SGreg Clayton                     }
501044d93782SGreg Clayton                 }
501144d93782SGreg Clayton             }
501244d93782SGreg Clayton 
501344d93782SGreg Clayton             const attr_t selected_highlight_attr = A_REVERSE;
501444d93782SGreg Clayton             const attr_t pc_highlight_attr = COLOR_PAIR(1);
501544d93782SGreg Clayton 
50163985c8c6SSaleem Abdulrasool             for (size_t i=0; i<num_visible_lines; ++i)
501744d93782SGreg Clayton             {
501844d93782SGreg Clayton                 const uint32_t curr_line = m_first_visible_line + i;
501944d93782SGreg Clayton                 if (curr_line < num_source_lines)
502044d93782SGreg Clayton                 {
5021ec990867SGreg Clayton                     const int line_y = m_min_y+i;
502244d93782SGreg Clayton                     window.MoveCursor(1, line_y);
502344d93782SGreg Clayton                     const bool is_pc_line = curr_line == m_pc_line;
502444d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == curr_line;
502544d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
502644d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
502744d93782SGreg Clayton                     attr_t highlight_attr = 0;
502844d93782SGreg Clayton                     attr_t bp_attr = 0;
502944d93782SGreg Clayton                     if (is_pc_line)
503044d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
503144d93782SGreg Clayton                     else if (line_is_selected)
503244d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
503344d93782SGreg Clayton 
503444d93782SGreg Clayton                     if (bp_lines.find(curr_line+1) != bp_lines.end())
503544d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
503644d93782SGreg Clayton 
503744d93782SGreg Clayton                     if (bp_attr)
503844d93782SGreg Clayton                         window.AttributeOn(bp_attr);
503944d93782SGreg Clayton 
504044d93782SGreg Clayton                     window.Printf (m_line_format, curr_line + 1);
504144d93782SGreg Clayton 
504244d93782SGreg Clayton                     if (bp_attr)
504344d93782SGreg Clayton                         window.AttributeOff(bp_attr);
504444d93782SGreg Clayton 
504544d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
504644d93782SGreg Clayton                     // Mark the line with the PC with a diamond
504744d93782SGreg Clayton                     if (is_pc_line)
504844d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
504944d93782SGreg Clayton                     else
505044d93782SGreg Clayton                         window.PutChar(' ');
505144d93782SGreg Clayton 
505244d93782SGreg Clayton                     if (highlight_attr)
505344d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
505444d93782SGreg Clayton                     const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false);
505544d93782SGreg Clayton                     if (line_len > 0)
505644d93782SGreg Clayton                         window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
505744d93782SGreg Clayton 
505844d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
505944d93782SGreg Clayton                     {
506044d93782SGreg Clayton                         StopInfoSP stop_info_sp;
506144d93782SGreg Clayton                         if (thread)
506244d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
506344d93782SGreg Clayton                         if (stop_info_sp)
506444d93782SGreg Clayton                         {
506544d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
506644d93782SGreg Clayton                             if (stop_description && stop_description[0])
506744d93782SGreg Clayton                             {
506844d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5069ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
507044d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5071ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
507244d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
507344d93782SGreg Clayton                             }
507444d93782SGreg Clayton                         }
507544d93782SGreg Clayton                         else
507644d93782SGreg Clayton                         {
5077ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
507844d93782SGreg Clayton                         }
507944d93782SGreg Clayton                     }
508044d93782SGreg Clayton                     if (highlight_attr)
508144d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
508244d93782SGreg Clayton 
508344d93782SGreg Clayton                 }
508444d93782SGreg Clayton                 else
508544d93782SGreg Clayton                 {
508644d93782SGreg Clayton                     break;
508744d93782SGreg Clayton                 }
508844d93782SGreg Clayton             }
508944d93782SGreg Clayton         }
509044d93782SGreg Clayton         else
509144d93782SGreg Clayton         {
509244d93782SGreg Clayton             size_t num_disassembly_lines = GetNumDisassemblyLines();
509344d93782SGreg Clayton             if (num_disassembly_lines > 0)
509444d93782SGreg Clayton             {
509544d93782SGreg Clayton                 // Display disassembly
509644d93782SGreg Clayton                 BreakpointAddrs bp_file_addrs;
509744d93782SGreg Clayton                 Target *target = exe_ctx.GetTargetPtr();
509844d93782SGreg Clayton                 if (target)
509944d93782SGreg Clayton                 {
510044d93782SGreg Clayton                     BreakpointList &bp_list = target->GetBreakpointList();
510144d93782SGreg Clayton                     const size_t num_bps = bp_list.GetSize();
510244d93782SGreg Clayton                     for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
510344d93782SGreg Clayton                     {
510444d93782SGreg Clayton                         BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
510544d93782SGreg Clayton                         const size_t num_bps_locs = bp_sp->GetNumLocations();
510644d93782SGreg Clayton                         for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
510744d93782SGreg Clayton                         {
510844d93782SGreg Clayton                             BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
510944d93782SGreg Clayton                             LineEntry bp_loc_line_entry;
511044d93782SGreg Clayton                             const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress();
511144d93782SGreg Clayton                             if (file_addr != LLDB_INVALID_ADDRESS)
511244d93782SGreg Clayton                             {
511344d93782SGreg Clayton                                 if (m_disassembly_range.ContainsFileAddress(file_addr))
511444d93782SGreg Clayton                                     bp_file_addrs.insert(file_addr);
511544d93782SGreg Clayton                             }
511644d93782SGreg Clayton                         }
511744d93782SGreg Clayton                     }
511844d93782SGreg Clayton                 }
511944d93782SGreg Clayton 
512044d93782SGreg Clayton                 const attr_t selected_highlight_attr = A_REVERSE;
512144d93782SGreg Clayton                 const attr_t pc_highlight_attr = COLOR_PAIR(1);
512244d93782SGreg Clayton 
512344d93782SGreg Clayton                 StreamString strm;
512444d93782SGreg Clayton 
512544d93782SGreg Clayton                 InstructionList &insts = m_disassembly_sp->GetInstructionList();
512644d93782SGreg Clayton                 Address pc_address;
512744d93782SGreg Clayton 
512844d93782SGreg Clayton                 if (frame_sp)
512944d93782SGreg Clayton                     pc_address = frame_sp->GetFrameCodeAddress();
513044d93782SGreg Clayton                 const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX;
513144d93782SGreg Clayton                 if (set_selected_line_to_pc)
513244d93782SGreg Clayton                 {
513344d93782SGreg Clayton                     m_selected_line = pc_idx;
513444d93782SGreg Clayton                 }
513544d93782SGreg Clayton 
513644d93782SGreg Clayton                 const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
51373985c8c6SSaleem Abdulrasool                 if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
513844d93782SGreg Clayton                     m_first_visible_line = 0;
513944d93782SGreg Clayton 
514044d93782SGreg Clayton                 if (pc_idx < num_disassembly_lines)
514144d93782SGreg Clayton                 {
51423985c8c6SSaleem Abdulrasool                     if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
514344d93782SGreg Clayton                         pc_idx >= m_first_visible_line + num_visible_lines)
514444d93782SGreg Clayton                         m_first_visible_line = pc_idx - non_visible_pc_offset;
514544d93782SGreg Clayton                 }
514644d93782SGreg Clayton 
514744d93782SGreg Clayton                 for (size_t i=0; i<num_visible_lines; ++i)
514844d93782SGreg Clayton                 {
514944d93782SGreg Clayton                     const uint32_t inst_idx = m_first_visible_line + i;
515044d93782SGreg Clayton                     Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
515144d93782SGreg Clayton                     if (!inst)
515244d93782SGreg Clayton                         break;
515344d93782SGreg Clayton 
5154ec990867SGreg Clayton                     const int line_y = m_min_y+i;
5155ec990867SGreg Clayton                     window.MoveCursor(1, line_y);
515644d93782SGreg Clayton                     const bool is_pc_line = frame_sp && inst_idx == pc_idx;
515744d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == inst_idx;
515844d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
515944d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
516044d93782SGreg Clayton                     attr_t highlight_attr = 0;
516144d93782SGreg Clayton                     attr_t bp_attr = 0;
516244d93782SGreg Clayton                     if (is_pc_line)
516344d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
516444d93782SGreg Clayton                     else if (line_is_selected)
516544d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
516644d93782SGreg Clayton 
516744d93782SGreg Clayton                     if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end())
516844d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
516944d93782SGreg Clayton 
517044d93782SGreg Clayton                     if (bp_attr)
517144d93782SGreg Clayton                         window.AttributeOn(bp_attr);
517244d93782SGreg Clayton 
5173324a1036SSaleem Abdulrasool                     window.Printf (" 0x%16.16llx ",
5174324a1036SSaleem Abdulrasool                                    static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target)));
517544d93782SGreg Clayton 
517644d93782SGreg Clayton                     if (bp_attr)
517744d93782SGreg Clayton                         window.AttributeOff(bp_attr);
517844d93782SGreg Clayton 
517944d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
518044d93782SGreg Clayton                     // Mark the line with the PC with a diamond
518144d93782SGreg Clayton                     if (is_pc_line)
518244d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
518344d93782SGreg Clayton                     else
518444d93782SGreg Clayton                         window.PutChar(' ');
518544d93782SGreg Clayton 
518644d93782SGreg Clayton                     if (highlight_attr)
518744d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
518844d93782SGreg Clayton 
518944d93782SGreg Clayton                     const char *mnemonic = inst->GetMnemonic(&exe_ctx);
519044d93782SGreg Clayton                     const char *operands = inst->GetOperands(&exe_ctx);
519144d93782SGreg Clayton                     const char *comment = inst->GetComment(&exe_ctx);
519244d93782SGreg Clayton 
519344d93782SGreg Clayton                     if (mnemonic && mnemonic[0] == '\0')
519444d93782SGreg Clayton                         mnemonic = NULL;
519544d93782SGreg Clayton                     if (operands && operands[0] == '\0')
519644d93782SGreg Clayton                         operands = NULL;
519744d93782SGreg Clayton                     if (comment && comment[0] == '\0')
519844d93782SGreg Clayton                         comment = NULL;
519944d93782SGreg Clayton 
520044d93782SGreg Clayton                     strm.Clear();
520144d93782SGreg Clayton 
520244d93782SGreg Clayton                     if (mnemonic && operands && comment)
520344d93782SGreg Clayton                         strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment);
520444d93782SGreg Clayton                     else if (mnemonic && operands)
520544d93782SGreg Clayton                         strm.Printf ("%-8s %s", mnemonic, operands);
520644d93782SGreg Clayton                     else if (mnemonic)
520744d93782SGreg Clayton                         strm.Printf ("%s", mnemonic);
520844d93782SGreg Clayton 
520944d93782SGreg Clayton                     int right_pad = 1;
521044d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
521144d93782SGreg Clayton 
521244d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
521344d93782SGreg Clayton                     {
521444d93782SGreg Clayton                         StopInfoSP stop_info_sp;
521544d93782SGreg Clayton                         if (thread)
521644d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
521744d93782SGreg Clayton                         if (stop_info_sp)
521844d93782SGreg Clayton                         {
521944d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
522044d93782SGreg Clayton                             if (stop_description && stop_description[0])
522144d93782SGreg Clayton                             {
522244d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5223ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
522444d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5225ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
522644d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
522744d93782SGreg Clayton                             }
522844d93782SGreg Clayton                         }
522944d93782SGreg Clayton                         else
523044d93782SGreg Clayton                         {
5231ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
523244d93782SGreg Clayton                         }
523344d93782SGreg Clayton                     }
523444d93782SGreg Clayton                     if (highlight_attr)
523544d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
523644d93782SGreg Clayton                 }
523744d93782SGreg Clayton             }
523844d93782SGreg Clayton         }
523944d93782SGreg Clayton         window.DeferredRefresh();
524044d93782SGreg Clayton         return true; // Drawing handled
524144d93782SGreg Clayton     }
524244d93782SGreg Clayton 
524344d93782SGreg Clayton     size_t
524444d93782SGreg Clayton     GetNumLines ()
524544d93782SGreg Clayton     {
524644d93782SGreg Clayton         size_t num_lines = GetNumSourceLines();
524744d93782SGreg Clayton         if (num_lines == 0)
524844d93782SGreg Clayton             num_lines = GetNumDisassemblyLines();
524944d93782SGreg Clayton         return num_lines;
525044d93782SGreg Clayton     }
525144d93782SGreg Clayton 
525244d93782SGreg Clayton     size_t
525344d93782SGreg Clayton     GetNumSourceLines () const
525444d93782SGreg Clayton     {
525544d93782SGreg Clayton         if (m_file_sp)
525644d93782SGreg Clayton             return m_file_sp->GetNumLines();
525744d93782SGreg Clayton         return 0;
525844d93782SGreg Clayton     }
525944d93782SGreg Clayton     size_t
526044d93782SGreg Clayton     GetNumDisassemblyLines () const
526144d93782SGreg Clayton     {
526244d93782SGreg Clayton         if (m_disassembly_sp)
526344d93782SGreg Clayton             return m_disassembly_sp->GetInstructionList().GetSize();
526444d93782SGreg Clayton         return 0;
526544d93782SGreg Clayton     }
526644d93782SGreg Clayton 
5267bd5ae6b4SGreg Clayton     HandleCharResult
5268bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
526944d93782SGreg Clayton     {
527044d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
527144d93782SGreg Clayton         const size_t num_lines = GetNumLines ();
527244d93782SGreg Clayton 
527344d93782SGreg Clayton         switch (c)
527444d93782SGreg Clayton         {
527544d93782SGreg Clayton             case ',':
527644d93782SGreg Clayton             case KEY_PPAGE:
527744d93782SGreg Clayton                 // Page up key
52783985c8c6SSaleem Abdulrasool                 if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
527944d93782SGreg Clayton                     m_first_visible_line -= num_visible_lines;
528044d93782SGreg Clayton                 else
528144d93782SGreg Clayton                     m_first_visible_line = 0;
528244d93782SGreg Clayton                 m_selected_line = m_first_visible_line;
528344d93782SGreg Clayton                 return eKeyHandled;
528444d93782SGreg Clayton 
528544d93782SGreg Clayton             case '.':
528644d93782SGreg Clayton             case KEY_NPAGE:
528744d93782SGreg Clayton                 // Page down key
528844d93782SGreg Clayton                 {
528944d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < num_lines)
529044d93782SGreg Clayton                         m_first_visible_line += num_visible_lines;
529144d93782SGreg Clayton                     else if (num_lines < num_visible_lines)
529244d93782SGreg Clayton                         m_first_visible_line = 0;
529344d93782SGreg Clayton                     else
529444d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
529544d93782SGreg Clayton                     m_selected_line = m_first_visible_line;
529644d93782SGreg Clayton                 }
529744d93782SGreg Clayton                 return eKeyHandled;
529844d93782SGreg Clayton 
529944d93782SGreg Clayton             case KEY_UP:
530044d93782SGreg Clayton                 if (m_selected_line > 0)
530144d93782SGreg Clayton                 {
530244d93782SGreg Clayton                     m_selected_line--;
53033985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
530444d93782SGreg Clayton                         m_first_visible_line = m_selected_line;
530544d93782SGreg Clayton                 }
530644d93782SGreg Clayton                 return eKeyHandled;
530744d93782SGreg Clayton 
530844d93782SGreg Clayton             case KEY_DOWN:
530944d93782SGreg Clayton                 if (m_selected_line + 1 < num_lines)
531044d93782SGreg Clayton                 {
531144d93782SGreg Clayton                     m_selected_line++;
531244d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < m_selected_line)
531344d93782SGreg Clayton                         m_first_visible_line++;
531444d93782SGreg Clayton                 }
531544d93782SGreg Clayton                 return eKeyHandled;
531644d93782SGreg Clayton 
531744d93782SGreg Clayton             case '\r':
531844d93782SGreg Clayton             case '\n':
531944d93782SGreg Clayton             case KEY_ENTER:
532044d93782SGreg Clayton                 // Set a breakpoint and run to the line using a one shot breakpoint
532144d93782SGreg Clayton                 if (GetNumSourceLines() > 0)
532244d93782SGreg Clayton                 {
532344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
532444d93782SGreg Clayton                     if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive())
532544d93782SGreg Clayton                     {
532644d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
532744d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
532844d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
532944d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
533044d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
533144d93782SGreg Clayton                                                                                       false,                     // internal
5332055ad9beSIlia K                                                                                       false,                     // request_hardware
5333055ad9beSIlia K                                                                                       eLazyBoolCalculate);       // move_to_nearest_code
533444d93782SGreg Clayton                         // Make breakpoint one shot
533544d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
533644d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
533744d93782SGreg Clayton                     }
533844d93782SGreg Clayton                 }
533944d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
534044d93782SGreg Clayton                 {
534144d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
534244d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
534344d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
534444d93782SGreg Clayton                     {
534544d93782SGreg Clayton                         Address addr = inst->GetAddress();
534644d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
534744d93782SGreg Clayton                                                                                       false,    // internal
534844d93782SGreg Clayton                                                                                       false);   // request_hardware
534944d93782SGreg Clayton                         // Make breakpoint one shot
535044d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
535144d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
535244d93782SGreg Clayton                     }
535344d93782SGreg Clayton                 }
535444d93782SGreg Clayton                 return eKeyHandled;
535544d93782SGreg Clayton 
535644d93782SGreg Clayton             case 'b':   // 'b' == toggle breakpoint on currently selected line
535744d93782SGreg Clayton                 if (m_selected_line < GetNumSourceLines())
535844d93782SGreg Clayton                 {
535944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
536044d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
536144d93782SGreg Clayton                     {
536244d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
536344d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
536444d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
536544d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
536644d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
536744d93782SGreg Clayton                                                                                       false,                     // internal
5368055ad9beSIlia K                                                                                       false,                     // request_hardware
5369055ad9beSIlia K                                                                                       eLazyBoolCalculate);       // move_to_nearest_code
537044d93782SGreg Clayton                     }
537144d93782SGreg Clayton                 }
537244d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
537344d93782SGreg Clayton                 {
537444d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
537544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
537644d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
537744d93782SGreg Clayton                     {
537844d93782SGreg Clayton                         Address addr = inst->GetAddress();
537944d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
538044d93782SGreg Clayton                                                                                       false,    // internal
538144d93782SGreg Clayton                                                                                       false);   // request_hardware
538244d93782SGreg Clayton                     }
538344d93782SGreg Clayton                 }
538444d93782SGreg Clayton                 return eKeyHandled;
538544d93782SGreg Clayton 
538644d93782SGreg Clayton             case 'd':   // 'd' == detach and let run
538744d93782SGreg Clayton             case 'D':   // 'D' == detach and keep stopped
538844d93782SGreg Clayton                 {
538944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
539044d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
539144d93782SGreg Clayton                         exe_ctx.GetProcessRef().Detach(c == 'D');
539244d93782SGreg Clayton                 }
539344d93782SGreg Clayton                 return eKeyHandled;
539444d93782SGreg Clayton 
539544d93782SGreg Clayton             case 'k':
539644d93782SGreg Clayton                 // 'k' == kill
539744d93782SGreg Clayton                 {
539844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
539944d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
5400ede3193bSJason Molenda                         exe_ctx.GetProcessRef().Destroy(false);
540144d93782SGreg Clayton                 }
540244d93782SGreg Clayton                 return eKeyHandled;
540344d93782SGreg Clayton 
540444d93782SGreg Clayton             case 'c':
540544d93782SGreg Clayton                 // 'c' == continue
540644d93782SGreg Clayton                 {
540744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
540844d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
540944d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
541044d93782SGreg Clayton                 }
541144d93782SGreg Clayton                 return eKeyHandled;
541244d93782SGreg Clayton 
541344d93782SGreg Clayton             case 'o':
541444d93782SGreg Clayton                 // 'o' == step out
541544d93782SGreg Clayton                 {
541644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
541744d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
541844d93782SGreg Clayton                     {
541944d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOut();
542044d93782SGreg Clayton                     }
542144d93782SGreg Clayton                 }
542244d93782SGreg Clayton                 return eKeyHandled;
542344d93782SGreg Clayton             case 'n':   // 'n' == step over
542444d93782SGreg Clayton             case 'N':   // 'N' == step over instruction
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                         bool source_step = (c == 'n');
543044d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOver(source_step);
543144d93782SGreg Clayton                     }
543244d93782SGreg Clayton                 }
543344d93782SGreg Clayton                 return eKeyHandled;
543444d93782SGreg Clayton             case 's':   // 's' == step into
543544d93782SGreg Clayton             case 'S':   // 'S' == step into instruction
543644d93782SGreg Clayton                 {
543744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
543844d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
543944d93782SGreg Clayton                     {
544044d93782SGreg Clayton                         bool source_step = (c == 's');
54414b4b2478SJim Ingham                         exe_ctx.GetThreadRef().StepIn(source_step);
544244d93782SGreg Clayton                     }
544344d93782SGreg Clayton                 }
544444d93782SGreg Clayton                 return eKeyHandled;
544544d93782SGreg Clayton 
544644d93782SGreg Clayton             case 'h':
544744d93782SGreg Clayton                 window.CreateHelpSubwindow ();
544844d93782SGreg Clayton                 return eKeyHandled;
544944d93782SGreg Clayton 
545044d93782SGreg Clayton             default:
545144d93782SGreg Clayton                 break;
545244d93782SGreg Clayton         }
545344d93782SGreg Clayton         return eKeyNotHandled;
545444d93782SGreg Clayton     }
545544d93782SGreg Clayton 
545644d93782SGreg Clayton protected:
545744d93782SGreg Clayton     typedef std::set<uint32_t> BreakpointLines;
545844d93782SGreg Clayton     typedef std::set<lldb::addr_t> BreakpointAddrs;
545944d93782SGreg Clayton 
546044d93782SGreg Clayton     Debugger &m_debugger;
546144d93782SGreg Clayton     SymbolContext m_sc;
546244d93782SGreg Clayton     SourceManager::FileSP m_file_sp;
546344d93782SGreg Clayton     SymbolContextScope *m_disassembly_scope;
546444d93782SGreg Clayton     lldb::DisassemblerSP m_disassembly_sp;
546544d93782SGreg Clayton     AddressRange m_disassembly_range;
5466ec990867SGreg Clayton     StreamString m_title;
546744d93782SGreg Clayton     lldb::user_id_t m_tid;
546844d93782SGreg Clayton     char m_line_format[8];
546944d93782SGreg Clayton     int m_line_width;
547044d93782SGreg Clayton     uint32_t m_selected_line;       // The selected line
547144d93782SGreg Clayton     uint32_t m_pc_line;             // The line with the PC
547244d93782SGreg Clayton     uint32_t m_stop_id;
547344d93782SGreg Clayton     uint32_t m_frame_idx;
547444d93782SGreg Clayton     int m_first_visible_line;
547544d93782SGreg Clayton     int m_min_x;
547644d93782SGreg Clayton     int m_min_y;
547744d93782SGreg Clayton     int m_max_x;
547844d93782SGreg Clayton     int m_max_y;
547944d93782SGreg Clayton 
548044d93782SGreg Clayton };
548144d93782SGreg Clayton 
548244d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true };
548344d93782SGreg Clayton 
548444d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) :
5485e30f11d9SKate Stone     IOHandler (debugger, IOHandler::Type::Curses)
548644d93782SGreg Clayton {
548744d93782SGreg Clayton }
548844d93782SGreg Clayton 
548944d93782SGreg Clayton void
549044d93782SGreg Clayton IOHandlerCursesGUI::Activate ()
549144d93782SGreg Clayton {
549244d93782SGreg Clayton     IOHandler::Activate();
549344d93782SGreg Clayton     if (!m_app_ap)
549444d93782SGreg Clayton     {
549544d93782SGreg Clayton         m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE()));
549644d93782SGreg Clayton 
549744d93782SGreg Clayton 
549844d93782SGreg Clayton         // This is both a window and a menu delegate
549944d93782SGreg Clayton         std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger));
550044d93782SGreg Clayton 
550144d93782SGreg Clayton         MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
550244d93782SGreg Clayton         MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
550344d93782SGreg Clayton         MenuSP exit_menuitem_sp(new Menu("Exit", NULL, 'x', ApplicationDelegate::eMenuID_LLDBExit));
550444d93782SGreg Clayton         exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
550544d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", NULL, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
550644d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
550744d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (exit_menuitem_sp);
550844d93782SGreg Clayton 
550944d93782SGreg Clayton         MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target));
551044d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", NULL, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
551144d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", NULL, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
551244d93782SGreg Clayton 
551344d93782SGreg Clayton         MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process));
551444d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach"  , NULL, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
551544d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach"  , NULL, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
551644d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch"  , NULL, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
551744d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
551844d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", NULL, 'c', ApplicationDelegate::eMenuID_ProcessContinue)));
551944d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt"    , NULL, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
552044d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill"    , NULL, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
552144d93782SGreg Clayton 
552244d93782SGreg Clayton         MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread));
552344d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In"  , NULL, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
552444d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", NULL, 'v', ApplicationDelegate::eMenuID_ThreadStepOver)));
552544d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , NULL, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
552644d93782SGreg Clayton 
552744d93782SGreg Clayton         MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
552844d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", NULL, 'b', ApplicationDelegate::eMenuID_ViewBacktrace)));
552944d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", NULL, 'r', ApplicationDelegate::eMenuID_ViewRegisters)));
553044d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Source"   , NULL, 's', ApplicationDelegate::eMenuID_ViewSource)));
553144d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", NULL, 'v', ApplicationDelegate::eMenuID_ViewVariables)));
553244d93782SGreg Clayton 
553344d93782SGreg Clayton         MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
553444d93782SGreg Clayton         help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", NULL, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
553544d93782SGreg Clayton 
553644d93782SGreg Clayton         m_app_ap->Initialize();
553744d93782SGreg Clayton         WindowSP &main_window_sp = m_app_ap->GetMainWindow();
553844d93782SGreg Clayton 
553944d93782SGreg Clayton         MenuSP menubar_sp(new Menu(Menu::Type::Bar));
554044d93782SGreg Clayton         menubar_sp->AddSubmenu (lldb_menu_sp);
554144d93782SGreg Clayton         menubar_sp->AddSubmenu (target_menu_sp);
554244d93782SGreg Clayton         menubar_sp->AddSubmenu (process_menu_sp);
554344d93782SGreg Clayton         menubar_sp->AddSubmenu (thread_menu_sp);
554444d93782SGreg Clayton         menubar_sp->AddSubmenu (view_menu_sp);
554544d93782SGreg Clayton         menubar_sp->AddSubmenu (help_menu_sp);
554644d93782SGreg Clayton         menubar_sp->SetDelegate(app_menu_delegate_sp);
554744d93782SGreg Clayton 
554844d93782SGreg Clayton         Rect content_bounds = main_window_sp->GetFrame();
554944d93782SGreg Clayton         Rect menubar_bounds = content_bounds.MakeMenuBar();
555044d93782SGreg Clayton         Rect status_bounds = content_bounds.MakeStatusBar();
555144d93782SGreg Clayton         Rect source_bounds;
555244d93782SGreg Clayton         Rect variables_bounds;
555344d93782SGreg Clayton         Rect threads_bounds;
555444d93782SGreg Clayton         Rect source_variables_bounds;
555544d93782SGreg Clayton         content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds);
555644d93782SGreg Clayton         source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds);
555744d93782SGreg Clayton 
555844d93782SGreg Clayton         WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
555944d93782SGreg Clayton         // Let the menubar get keys if the active window doesn't handle the
556044d93782SGreg Clayton         // keys that are typed so it can respond to menubar key presses.
556144d93782SGreg Clayton         menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window
556244d93782SGreg Clayton         menubar_window_sp->SetDelegate(menubar_sp);
556344d93782SGreg Clayton 
556444d93782SGreg Clayton         WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source",
556544d93782SGreg Clayton                                                                    source_bounds,
556644d93782SGreg Clayton                                                                    true));
556744d93782SGreg Clayton         WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables",
556844d93782SGreg Clayton                                                                       variables_bounds,
556944d93782SGreg Clayton                                                                       false));
557044d93782SGreg Clayton         WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads",
557144d93782SGreg Clayton                                                                       threads_bounds,
557244d93782SGreg Clayton                                                                       false));
557344d93782SGreg Clayton         WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status",
557444d93782SGreg Clayton                                                                    status_bounds,
557544d93782SGreg Clayton                                                                    false));
557644d93782SGreg Clayton         status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window
557744d93782SGreg Clayton         main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
557844d93782SGreg Clayton         source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
557944d93782SGreg Clayton         variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
5580ec990867SGreg Clayton         TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger));
558144d93782SGreg Clayton         threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
558244d93782SGreg Clayton         status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
558344d93782SGreg Clayton 
55845fdb09bbSGreg Clayton         // Show the main help window once the first time the curses GUI is launched
55855fdb09bbSGreg Clayton         static bool g_showed_help = false;
55865fdb09bbSGreg Clayton         if (!g_showed_help)
55875fdb09bbSGreg Clayton         {
55885fdb09bbSGreg Clayton             g_showed_help = true;
55895fdb09bbSGreg Clayton             main_window_sp->CreateHelpSubwindow();
55905fdb09bbSGreg Clayton         }
55915fdb09bbSGreg Clayton 
559244d93782SGreg Clayton         init_pair (1, COLOR_WHITE   , COLOR_BLUE  );
559344d93782SGreg Clayton         init_pair (2, COLOR_BLACK   , COLOR_WHITE );
559444d93782SGreg Clayton         init_pair (3, COLOR_MAGENTA , COLOR_WHITE );
559544d93782SGreg Clayton         init_pair (4, COLOR_MAGENTA , COLOR_BLACK );
559644d93782SGreg Clayton         init_pair (5, COLOR_RED     , COLOR_BLACK );
559744d93782SGreg Clayton 
559844d93782SGreg Clayton     }
559944d93782SGreg Clayton }
560044d93782SGreg Clayton 
560144d93782SGreg Clayton void
560244d93782SGreg Clayton IOHandlerCursesGUI::Deactivate ()
560344d93782SGreg Clayton {
560444d93782SGreg Clayton     m_app_ap->Terminate();
560544d93782SGreg Clayton }
560644d93782SGreg Clayton 
560744d93782SGreg Clayton void
560844d93782SGreg Clayton IOHandlerCursesGUI::Run ()
560944d93782SGreg Clayton {
561044d93782SGreg Clayton     m_app_ap->Run(m_debugger);
561144d93782SGreg Clayton     SetIsDone(true);
561244d93782SGreg Clayton }
561344d93782SGreg Clayton 
561444d93782SGreg Clayton 
561544d93782SGreg Clayton IOHandlerCursesGUI::~IOHandlerCursesGUI ()
561644d93782SGreg Clayton {
561744d93782SGreg Clayton 
561844d93782SGreg Clayton }
561944d93782SGreg Clayton 
562044d93782SGreg Clayton void
562144d93782SGreg Clayton IOHandlerCursesGUI::Hide ()
562244d93782SGreg Clayton {
562344d93782SGreg Clayton }
562444d93782SGreg Clayton 
562544d93782SGreg Clayton 
562644d93782SGreg Clayton void
562744d93782SGreg Clayton IOHandlerCursesGUI::Refresh ()
562844d93782SGreg Clayton {
562944d93782SGreg Clayton }
563044d93782SGreg Clayton 
5631e68f5d6bSGreg Clayton void
5632e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel ()
5633e68f5d6bSGreg Clayton {
5634e68f5d6bSGreg Clayton }
563544d93782SGreg Clayton 
5636f0066ad0SGreg Clayton bool
563744d93782SGreg Clayton IOHandlerCursesGUI::Interrupt ()
563844d93782SGreg Clayton {
5639f0066ad0SGreg Clayton     return false;
564044d93782SGreg Clayton }
564144d93782SGreg Clayton 
564244d93782SGreg Clayton 
564344d93782SGreg Clayton void
564444d93782SGreg Clayton IOHandlerCursesGUI::GotEOF()
564544d93782SGreg Clayton {
564644d93782SGreg Clayton }
564744d93782SGreg Clayton 
5648914b8d98SDeepak Panickal #endif // #ifndef LLDB_DISABLE_CURSES
5649