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"
22*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
2344d93782SGreg Clayton #include "lldb/Host/Editline.h"
24*cacde7dfSTodd 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 
4144d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger) :
4244d93782SGreg Clayton     IOHandler (debugger,
4344d93782SGreg Clayton                StreamFileSP(),  // Adopt STDIN from top input reader
4444d93782SGreg Clayton                StreamFileSP(),  // Adopt STDOUT from top input reader
45340b0309SGreg Clayton                StreamFileSP(),  // Adopt STDERR from top input reader
46340b0309SGreg Clayton                0)               // Flags
4744d93782SGreg Clayton {
4844d93782SGreg Clayton }
4944d93782SGreg Clayton 
5044d93782SGreg Clayton 
5144d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger,
5244d93782SGreg Clayton                       const lldb::StreamFileSP &input_sp,
5344d93782SGreg Clayton                       const lldb::StreamFileSP &output_sp,
54340b0309SGreg Clayton                       const lldb::StreamFileSP &error_sp,
55340b0309SGreg Clayton                       uint32_t flags) :
5644d93782SGreg Clayton     m_debugger (debugger),
5744d93782SGreg Clayton     m_input_sp (input_sp),
5844d93782SGreg Clayton     m_output_sp (output_sp),
5944d93782SGreg Clayton     m_error_sp (error_sp),
60340b0309SGreg Clayton     m_flags (flags),
6144d93782SGreg Clayton     m_user_data (NULL),
6244d93782SGreg Clayton     m_done (false),
6344d93782SGreg Clayton     m_active (false)
6444d93782SGreg Clayton {
6544d93782SGreg Clayton     // If any files are not specified, then adopt them from the top input reader.
6644d93782SGreg Clayton     if (!m_input_sp || !m_output_sp || !m_error_sp)
6744d93782SGreg Clayton         debugger.AdoptTopIOHandlerFilesIfInvalid (m_input_sp,
6844d93782SGreg Clayton                                                   m_output_sp,
6944d93782SGreg Clayton                                                   m_error_sp);
7044d93782SGreg Clayton }
7144d93782SGreg Clayton 
7244d93782SGreg Clayton IOHandler::~IOHandler()
7344d93782SGreg Clayton {
7444d93782SGreg Clayton }
7544d93782SGreg Clayton 
7644d93782SGreg Clayton 
7744d93782SGreg Clayton int
7844d93782SGreg Clayton IOHandler::GetInputFD()
7944d93782SGreg Clayton {
8044d93782SGreg Clayton     if (m_input_sp)
8144d93782SGreg Clayton         return m_input_sp->GetFile().GetDescriptor();
8244d93782SGreg Clayton     return -1;
8344d93782SGreg Clayton }
8444d93782SGreg Clayton 
8544d93782SGreg Clayton int
8644d93782SGreg Clayton IOHandler::GetOutputFD()
8744d93782SGreg Clayton {
8844d93782SGreg Clayton     if (m_output_sp)
8944d93782SGreg Clayton         return m_output_sp->GetFile().GetDescriptor();
9044d93782SGreg Clayton     return -1;
9144d93782SGreg Clayton }
9244d93782SGreg Clayton 
9344d93782SGreg Clayton int
9444d93782SGreg Clayton IOHandler::GetErrorFD()
9544d93782SGreg Clayton {
9644d93782SGreg Clayton     if (m_error_sp)
9744d93782SGreg Clayton         return m_error_sp->GetFile().GetDescriptor();
9844d93782SGreg Clayton     return -1;
9944d93782SGreg Clayton }
10044d93782SGreg Clayton 
10144d93782SGreg Clayton FILE *
10244d93782SGreg Clayton IOHandler::GetInputFILE()
10344d93782SGreg Clayton {
10444d93782SGreg Clayton     if (m_input_sp)
10544d93782SGreg Clayton         return m_input_sp->GetFile().GetStream();
10644d93782SGreg Clayton     return NULL;
10744d93782SGreg Clayton }
10844d93782SGreg Clayton 
10944d93782SGreg Clayton FILE *
11044d93782SGreg Clayton IOHandler::GetOutputFILE()
11144d93782SGreg Clayton {
11244d93782SGreg Clayton     if (m_output_sp)
11344d93782SGreg Clayton         return m_output_sp->GetFile().GetStream();
11444d93782SGreg Clayton     return NULL;
11544d93782SGreg Clayton }
11644d93782SGreg Clayton 
11744d93782SGreg Clayton FILE *
11844d93782SGreg Clayton IOHandler::GetErrorFILE()
11944d93782SGreg Clayton {
12044d93782SGreg Clayton     if (m_error_sp)
12144d93782SGreg Clayton         return m_error_sp->GetFile().GetStream();
12244d93782SGreg Clayton     return NULL;
12344d93782SGreg Clayton }
12444d93782SGreg Clayton 
12544d93782SGreg Clayton StreamFileSP &
12644d93782SGreg Clayton IOHandler::GetInputStreamFile()
12744d93782SGreg Clayton {
12844d93782SGreg Clayton     return m_input_sp;
12944d93782SGreg Clayton }
13044d93782SGreg Clayton 
13144d93782SGreg Clayton StreamFileSP &
13244d93782SGreg Clayton IOHandler::GetOutputStreamFile()
13344d93782SGreg Clayton {
13444d93782SGreg Clayton     return m_output_sp;
13544d93782SGreg Clayton }
13644d93782SGreg Clayton 
13744d93782SGreg Clayton 
13844d93782SGreg Clayton StreamFileSP &
13944d93782SGreg Clayton IOHandler::GetErrorStreamFile()
14044d93782SGreg Clayton {
14144d93782SGreg Clayton     return m_error_sp;
14244d93782SGreg Clayton }
14344d93782SGreg Clayton 
144340b0309SGreg Clayton bool
145340b0309SGreg Clayton IOHandler::GetIsInteractive ()
146340b0309SGreg Clayton {
147340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsInteractive ();
148340b0309SGreg Clayton }
149340b0309SGreg Clayton 
150340b0309SGreg Clayton bool
151340b0309SGreg Clayton IOHandler::GetIsRealTerminal ()
152340b0309SGreg Clayton {
153340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsRealTerminal();
154340b0309SGreg Clayton }
15544d93782SGreg Clayton 
15644d93782SGreg Clayton IOHandlerConfirm::IOHandlerConfirm (Debugger &debugger,
15744d93782SGreg Clayton                                     const char *prompt,
15844d93782SGreg Clayton                                     bool default_response) :
15944d93782SGreg Clayton     IOHandlerEditline(debugger,
16044d93782SGreg Clayton                       NULL,     // NULL editline_name means no history loaded/saved
16144d93782SGreg Clayton                       NULL,
16244d93782SGreg Clayton                       false,    // Multi-line
163f6913cd7SGreg Clayton                       0,
16444d93782SGreg Clayton                       *this),
16544d93782SGreg Clayton     m_default_response (default_response),
16644d93782SGreg Clayton     m_user_response (default_response)
16744d93782SGreg Clayton {
16844d93782SGreg Clayton     StreamString prompt_stream;
16944d93782SGreg Clayton     prompt_stream.PutCString(prompt);
17044d93782SGreg Clayton     if (m_default_response)
17144d93782SGreg Clayton         prompt_stream.Printf(": [Y/n] ");
17244d93782SGreg Clayton     else
17344d93782SGreg Clayton         prompt_stream.Printf(": [y/N] ");
17444d93782SGreg Clayton 
17544d93782SGreg Clayton     SetPrompt (prompt_stream.GetString().c_str());
17644d93782SGreg Clayton 
17744d93782SGreg Clayton }
17844d93782SGreg Clayton 
17944d93782SGreg Clayton 
18044d93782SGreg Clayton IOHandlerConfirm::~IOHandlerConfirm ()
18144d93782SGreg Clayton {
18244d93782SGreg Clayton }
18344d93782SGreg Clayton 
18444d93782SGreg Clayton int
18544d93782SGreg Clayton IOHandlerConfirm::IOHandlerComplete (IOHandler &io_handler,
18644d93782SGreg Clayton                                      const char *current_line,
18744d93782SGreg Clayton                                      const char *cursor,
18844d93782SGreg Clayton                                      const char *last_char,
18944d93782SGreg Clayton                                      int skip_first_n_matches,
19044d93782SGreg Clayton                                      int max_matches,
19144d93782SGreg Clayton                                      StringList &matches)
19244d93782SGreg Clayton {
19344d93782SGreg Clayton     if (current_line == cursor)
19444d93782SGreg Clayton     {
19544d93782SGreg Clayton         if (m_default_response)
19644d93782SGreg Clayton         {
19744d93782SGreg Clayton             matches.AppendString("y");
19844d93782SGreg Clayton         }
19944d93782SGreg Clayton         else
20044d93782SGreg Clayton         {
20144d93782SGreg Clayton             matches.AppendString("n");
20244d93782SGreg Clayton         }
20344d93782SGreg Clayton     }
20444d93782SGreg Clayton     return matches.GetSize();
20544d93782SGreg Clayton }
20644d93782SGreg Clayton 
20744d93782SGreg Clayton void
20844d93782SGreg Clayton IOHandlerConfirm::IOHandlerInputComplete (IOHandler &io_handler, std::string &line)
20944d93782SGreg Clayton {
21044d93782SGreg Clayton     if (line.empty())
21144d93782SGreg Clayton     {
21244d93782SGreg Clayton         // User just hit enter, set the response to the default
21344d93782SGreg Clayton         m_user_response = m_default_response;
21444d93782SGreg Clayton         io_handler.SetIsDone(true);
21544d93782SGreg Clayton         return;
21644d93782SGreg Clayton     }
21744d93782SGreg Clayton 
21844d93782SGreg Clayton     if (line.size() == 1)
21944d93782SGreg Clayton     {
22044d93782SGreg Clayton         switch (line[0])
22144d93782SGreg Clayton         {
22244d93782SGreg Clayton             case 'y':
22344d93782SGreg Clayton             case 'Y':
22444d93782SGreg Clayton                 m_user_response = true;
22544d93782SGreg Clayton                 io_handler.SetIsDone(true);
22644d93782SGreg Clayton                 return;
22744d93782SGreg Clayton             case 'n':
22844d93782SGreg Clayton             case 'N':
22944d93782SGreg Clayton                 m_user_response = false;
23044d93782SGreg Clayton                 io_handler.SetIsDone(true);
23144d93782SGreg Clayton                 return;
23244d93782SGreg Clayton             default:
23344d93782SGreg Clayton                 break;
23444d93782SGreg Clayton         }
23544d93782SGreg Clayton     }
23644d93782SGreg Clayton 
23744d93782SGreg Clayton     if (line == "yes" || line == "YES" || line == "Yes")
23844d93782SGreg Clayton     {
23944d93782SGreg Clayton         m_user_response = true;
24044d93782SGreg Clayton         io_handler.SetIsDone(true);
24144d93782SGreg Clayton     }
24244d93782SGreg Clayton     else if (line == "no" || line == "NO" || line == "No")
24344d93782SGreg Clayton     {
24444d93782SGreg Clayton         m_user_response = false;
24544d93782SGreg Clayton         io_handler.SetIsDone(true);
24644d93782SGreg Clayton     }
24744d93782SGreg Clayton }
24844d93782SGreg Clayton 
24944d93782SGreg Clayton int
25044d93782SGreg Clayton IOHandlerDelegate::IOHandlerComplete (IOHandler &io_handler,
25144d93782SGreg Clayton                                       const char *current_line,
25244d93782SGreg Clayton                                       const char *cursor,
25344d93782SGreg Clayton                                       const char *last_char,
25444d93782SGreg Clayton                                       int skip_first_n_matches,
25544d93782SGreg Clayton                                       int max_matches,
25644d93782SGreg Clayton                                       StringList &matches)
25744d93782SGreg Clayton {
25844d93782SGreg Clayton     switch (m_completion)
25944d93782SGreg Clayton     {
26044d93782SGreg Clayton     case Completion::None:
26144d93782SGreg Clayton         break;
26244d93782SGreg Clayton 
26344d93782SGreg Clayton     case Completion::LLDBCommand:
26444d93782SGreg Clayton         return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion (current_line,
26544d93782SGreg Clayton                                                                                   cursor,
26644d93782SGreg Clayton                                                                                   last_char,
26744d93782SGreg Clayton                                                                                   skip_first_n_matches,
26844d93782SGreg Clayton                                                                                   max_matches,
26944d93782SGreg Clayton                                                                                   matches);
27044d93782SGreg Clayton 
27144d93782SGreg Clayton     case Completion::Expression:
27244d93782SGreg Clayton         {
27344d93782SGreg Clayton             bool word_complete = false;
27444d93782SGreg Clayton             const char *word_start = cursor;
27544d93782SGreg Clayton             if (cursor > current_line)
27644d93782SGreg Clayton                 --word_start;
27744d93782SGreg Clayton             while (word_start > current_line && !isspace(*word_start))
27844d93782SGreg Clayton                 --word_start;
27944d93782SGreg Clayton             CommandCompletions::InvokeCommonCompletionCallbacks (io_handler.GetDebugger().GetCommandInterpreter(),
28044d93782SGreg Clayton                                                                  CommandCompletions::eVariablePathCompletion,
28144d93782SGreg Clayton                                                                  word_start,
28244d93782SGreg Clayton                                                                  skip_first_n_matches,
28344d93782SGreg Clayton                                                                  max_matches,
28444d93782SGreg Clayton                                                                  NULL,
28544d93782SGreg Clayton                                                                  word_complete,
28644d93782SGreg Clayton                                                                  matches);
28744d93782SGreg Clayton 
28844d93782SGreg Clayton             size_t num_matches = matches.GetSize();
28944d93782SGreg Clayton             if (num_matches > 0)
29044d93782SGreg Clayton             {
29144d93782SGreg Clayton                 std::string common_prefix;
29244d93782SGreg Clayton                 matches.LongestCommonPrefix (common_prefix);
29344d93782SGreg Clayton                 const size_t partial_name_len = strlen(word_start);
29444d93782SGreg Clayton 
29544d93782SGreg Clayton                 // If we matched a unique single command, add a space...
29644d93782SGreg Clayton                 // Only do this if the completer told us this was a complete word, however...
29744d93782SGreg Clayton                 if (num_matches == 1 && word_complete)
29844d93782SGreg Clayton                 {
29944d93782SGreg Clayton                     common_prefix.push_back(' ');
30044d93782SGreg Clayton                 }
30144d93782SGreg Clayton                 common_prefix.erase (0, partial_name_len);
30244d93782SGreg Clayton                 matches.InsertStringAtIndex(0, std::move(common_prefix));
30344d93782SGreg Clayton             }
30444d93782SGreg Clayton             return num_matches;
30544d93782SGreg Clayton         }
30644d93782SGreg Clayton         break;
30744d93782SGreg Clayton     }
30844d93782SGreg Clayton 
30944d93782SGreg Clayton 
31044d93782SGreg Clayton     return 0;
31144d93782SGreg Clayton }
31244d93782SGreg Clayton 
31344d93782SGreg Clayton 
31444d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
31544d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
31644d93782SGreg Clayton                                       const char *prompt,
31744d93782SGreg Clayton                                       bool multi_line,
318f6913cd7SGreg Clayton                                       uint32_t line_number_start,
31944d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
32044d93782SGreg Clayton     IOHandlerEditline(debugger,
32144d93782SGreg Clayton                       StreamFileSP(), // Inherit input from top input reader
32244d93782SGreg Clayton                       StreamFileSP(), // Inherit output from top input reader
32344d93782SGreg Clayton                       StreamFileSP(), // Inherit error from top input reader
324340b0309SGreg Clayton                       0,              // Flags
32544d93782SGreg Clayton                       editline_name,  // Used for saving history files
32644d93782SGreg Clayton                       prompt,
32744d93782SGreg Clayton                       multi_line,
328f6913cd7SGreg Clayton                       line_number_start,
32944d93782SGreg Clayton                       delegate)
33044d93782SGreg Clayton {
33144d93782SGreg Clayton }
33244d93782SGreg Clayton 
33344d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
33444d93782SGreg Clayton                                       const lldb::StreamFileSP &input_sp,
33544d93782SGreg Clayton                                       const lldb::StreamFileSP &output_sp,
33644d93782SGreg Clayton                                       const lldb::StreamFileSP &error_sp,
337340b0309SGreg Clayton                                       uint32_t flags,
33844d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
33944d93782SGreg Clayton                                       const char *prompt,
34044d93782SGreg Clayton                                       bool multi_line,
341f6913cd7SGreg Clayton                                       uint32_t line_number_start,
34244d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
343340b0309SGreg Clayton     IOHandler (debugger, input_sp, output_sp, error_sp, flags),
344*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
34544d93782SGreg Clayton     m_editline_ap (),
346*cacde7dfSTodd Fiala #endif
34744d93782SGreg Clayton     m_delegate (delegate),
34844d93782SGreg Clayton     m_prompt (),
349f6913cd7SGreg Clayton     m_base_line_number (line_number_start),
350340b0309SGreg Clayton     m_multi_line (multi_line)
35144d93782SGreg Clayton {
35244d93782SGreg Clayton     SetPrompt(prompt);
35344d93782SGreg Clayton 
354*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
355914b8d98SDeepak Panickal     bool use_editline = false;
356340b0309SGreg Clayton 
357914b8d98SDeepak Panickal #ifndef _MSC_VER
358340b0309SGreg Clayton     use_editline = m_input_sp->GetFile().GetIsRealTerminal();
359914b8d98SDeepak Panickal #else
36028e7ed12SColin Riley     // Editline is causing issues on Windows, so use the fallback.
36128e7ed12SColin Riley     use_editline = false;
362914b8d98SDeepak Panickal #endif
36344d93782SGreg Clayton 
36444d93782SGreg Clayton     if (use_editline)
36544d93782SGreg Clayton     {
36644d93782SGreg Clayton         m_editline_ap.reset(new Editline (editline_name,
36744d93782SGreg Clayton                                           prompt ? prompt : "",
36890e9692dSGreg Clayton                                           multi_line,
36944d93782SGreg Clayton                                           GetInputFILE (),
37044d93782SGreg Clayton                                           GetOutputFILE (),
37144d93782SGreg Clayton                                           GetErrorFILE ()));
372f6913cd7SGreg Clayton         if (m_base_line_number > 0)
373f6913cd7SGreg Clayton             m_editline_ap->ShowLineNumbers(true, m_base_line_number);
37444d93782SGreg Clayton         m_editline_ap->SetLineCompleteCallback (LineCompletedCallback, this);
37544d93782SGreg Clayton         m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this);
37644d93782SGreg Clayton     }
377*cacde7dfSTodd Fiala #endif
37844d93782SGreg Clayton }
37944d93782SGreg Clayton 
38044d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline ()
38144d93782SGreg Clayton {
382*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
38344d93782SGreg Clayton     m_editline_ap.reset();
384*cacde7dfSTodd Fiala #endif
38544d93782SGreg Clayton }
38644d93782SGreg Clayton 
38744d93782SGreg Clayton 
38844d93782SGreg Clayton bool
389f0066ad0SGreg Clayton IOHandlerEditline::GetLine (std::string &line, bool &interrupted)
39044d93782SGreg Clayton {
391*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
39244d93782SGreg Clayton     if (m_editline_ap)
39344d93782SGreg Clayton     {
394f0066ad0SGreg Clayton         return m_editline_ap->GetLine(line, interrupted).Success();
39544d93782SGreg Clayton     }
39644d93782SGreg Clayton     else
39744d93782SGreg Clayton     {
398*cacde7dfSTodd Fiala #endif
39944d93782SGreg Clayton         line.clear();
40044d93782SGreg Clayton 
40144d93782SGreg Clayton         FILE *in = GetInputFILE();
40244d93782SGreg Clayton         if (in)
40344d93782SGreg Clayton         {
404340b0309SGreg Clayton             if (GetIsInteractive())
40544d93782SGreg Clayton             {
40644d93782SGreg Clayton                 const char *prompt = GetPrompt();
40744d93782SGreg Clayton                 if (prompt && prompt[0])
40844d93782SGreg Clayton                 {
40944d93782SGreg Clayton                     FILE *out = GetOutputFILE();
41044d93782SGreg Clayton                     if (out)
41144d93782SGreg Clayton                     {
41244d93782SGreg Clayton                         ::fprintf(out, "%s", prompt);
41344d93782SGreg Clayton                         ::fflush(out);
41444d93782SGreg Clayton                     }
41544d93782SGreg Clayton                 }
41644d93782SGreg Clayton             }
41744d93782SGreg Clayton             char buffer[256];
41844d93782SGreg Clayton             bool done = false;
4190f86e6e7SGreg Clayton             bool got_line = false;
42044d93782SGreg Clayton             while (!done)
42144d93782SGreg Clayton             {
42244d93782SGreg Clayton                 if (fgets(buffer, sizeof(buffer), in) == NULL)
423c9cf5798SGreg Clayton                 {
424c7797accSGreg Clayton                     const int saved_errno = errno;
425c9cf5798SGreg Clayton                     if (feof(in))
42644d93782SGreg Clayton                         done = true;
427c7797accSGreg Clayton                     else if (ferror(in))
428c7797accSGreg Clayton                     {
429c7797accSGreg Clayton                         if (saved_errno != EINTR)
430c7797accSGreg Clayton                             done = true;
431c7797accSGreg Clayton                     }
432c9cf5798SGreg Clayton                 }
43344d93782SGreg Clayton                 else
43444d93782SGreg Clayton                 {
4350f86e6e7SGreg Clayton                     got_line = true;
43644d93782SGreg Clayton                     size_t buffer_len = strlen(buffer);
43744d93782SGreg Clayton                     assert (buffer[buffer_len] == '\0');
43844d93782SGreg Clayton                     char last_char = buffer[buffer_len-1];
43944d93782SGreg Clayton                     if (last_char == '\r' || last_char == '\n')
44044d93782SGreg Clayton                     {
44144d93782SGreg Clayton                         done = true;
44244d93782SGreg Clayton                         // Strip trailing newlines
44344d93782SGreg Clayton                         while (last_char == '\r' || last_char == '\n')
44444d93782SGreg Clayton                         {
44544d93782SGreg Clayton                             --buffer_len;
44644d93782SGreg Clayton                             if (buffer_len == 0)
44744d93782SGreg Clayton                                 break;
44844d93782SGreg Clayton                             last_char = buffer[buffer_len-1];
44944d93782SGreg Clayton                         }
45044d93782SGreg Clayton                     }
45144d93782SGreg Clayton                     line.append(buffer, buffer_len);
45244d93782SGreg Clayton                 }
45344d93782SGreg Clayton             }
4540f86e6e7SGreg Clayton             // We might have gotten a newline on a line by itself
4550f86e6e7SGreg Clayton             // make sure to return true in this case.
4560f86e6e7SGreg Clayton             return got_line;
45744d93782SGreg Clayton         }
45844d93782SGreg Clayton         else
45944d93782SGreg Clayton         {
46044d93782SGreg Clayton             // No more input file, we are done...
46144d93782SGreg Clayton             SetIsDone(true);
46244d93782SGreg Clayton         }
463340b0309SGreg Clayton         return false;
464*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
46544d93782SGreg Clayton     }
466*cacde7dfSTodd Fiala #endif
46744d93782SGreg Clayton }
46844d93782SGreg Clayton 
46944d93782SGreg Clayton 
470*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
47144d93782SGreg Clayton LineStatus
47244d93782SGreg Clayton IOHandlerEditline::LineCompletedCallback (Editline *editline,
47344d93782SGreg Clayton                                           StringList &lines,
47444d93782SGreg Clayton                                           uint32_t line_idx,
47544d93782SGreg Clayton                                           Error &error,
47644d93782SGreg Clayton                                           void *baton)
47744d93782SGreg Clayton {
47844d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
47944d93782SGreg Clayton     return editline_reader->m_delegate.IOHandlerLinesUpdated(*editline_reader, lines, line_idx, error);
48044d93782SGreg Clayton }
48144d93782SGreg Clayton 
48244d93782SGreg Clayton int
48344d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line,
48444d93782SGreg Clayton                                          const char *cursor,
48544d93782SGreg Clayton                                          const char *last_char,
48644d93782SGreg Clayton                                          int skip_first_n_matches,
48744d93782SGreg Clayton                                          int max_matches,
48844d93782SGreg Clayton                                          StringList &matches,
48944d93782SGreg Clayton                                          void *baton)
49044d93782SGreg Clayton {
49144d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
49244d93782SGreg Clayton     if (editline_reader)
49344d93782SGreg Clayton         return editline_reader->m_delegate.IOHandlerComplete (*editline_reader,
49444d93782SGreg Clayton                                                               current_line,
49544d93782SGreg Clayton                                                               cursor,
49644d93782SGreg Clayton                                                               last_char,
49744d93782SGreg Clayton                                                               skip_first_n_matches,
49844d93782SGreg Clayton                                                               max_matches,
49944d93782SGreg Clayton                                                               matches);
50044d93782SGreg Clayton     return 0;
50144d93782SGreg Clayton }
502*cacde7dfSTodd Fiala #endif
50344d93782SGreg Clayton 
50444d93782SGreg Clayton const char *
50544d93782SGreg Clayton IOHandlerEditline::GetPrompt ()
50644d93782SGreg Clayton {
507*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
50844d93782SGreg Clayton     if (m_editline_ap)
509*cacde7dfSTodd Fiala     {
51044d93782SGreg Clayton         return m_editline_ap->GetPrompt ();
511*cacde7dfSTodd Fiala     }
512*cacde7dfSTodd Fiala     else
513*cacde7dfSTodd Fiala     {
514*cacde7dfSTodd Fiala #endif
515*cacde7dfSTodd Fiala         if (m_prompt.empty())
51644d93782SGreg Clayton             return NULL;
517*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
518*cacde7dfSTodd Fiala     }
519*cacde7dfSTodd Fiala #endif
52044d93782SGreg Clayton     return m_prompt.c_str();
52144d93782SGreg Clayton }
52244d93782SGreg Clayton 
52344d93782SGreg Clayton bool
52444d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p)
52544d93782SGreg Clayton {
52644d93782SGreg Clayton     if (p && p[0])
52744d93782SGreg Clayton         m_prompt = p;
52844d93782SGreg Clayton     else
52944d93782SGreg Clayton         m_prompt.clear();
530*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
53144d93782SGreg Clayton     if (m_editline_ap)
53244d93782SGreg Clayton         m_editline_ap->SetPrompt (m_prompt.empty() ? NULL : m_prompt.c_str());
533*cacde7dfSTodd Fiala #endif
53444d93782SGreg Clayton     return true;
53544d93782SGreg Clayton }
53644d93782SGreg Clayton 
537f6913cd7SGreg Clayton void
538f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line)
539f6913cd7SGreg Clayton {
540f6913cd7SGreg Clayton     m_base_line_number = line;
541*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
542f6913cd7SGreg Clayton     if (m_editline_ap)
543f6913cd7SGreg Clayton         m_editline_ap->ShowLineNumbers (true, line);
544*cacde7dfSTodd Fiala #endif
545f6913cd7SGreg Clayton 
546f6913cd7SGreg Clayton }
54744d93782SGreg Clayton bool
548f0066ad0SGreg Clayton IOHandlerEditline::GetLines (StringList &lines, bool &interrupted)
54944d93782SGreg Clayton {
55044d93782SGreg Clayton     bool success = false;
551*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
55244d93782SGreg Clayton     if (m_editline_ap)
55344d93782SGreg Clayton     {
55444d93782SGreg Clayton         std::string end_token;
555f0066ad0SGreg Clayton         success = m_editline_ap->GetLines(end_token, lines, interrupted).Success();
55644d93782SGreg Clayton     }
55744d93782SGreg Clayton     else
55844d93782SGreg Clayton     {
559*cacde7dfSTodd Fiala #endif
56044d93782SGreg Clayton         LineStatus lines_status = LineStatus::Success;
561c3d874a5SGreg Clayton         Error error;
56244d93782SGreg Clayton 
56344d93782SGreg Clayton         while (lines_status == LineStatus::Success)
56444d93782SGreg Clayton         {
565f6913cd7SGreg Clayton             // Show line numbers if we are asked to
56644d93782SGreg Clayton             std::string line;
567f6913cd7SGreg Clayton             if (m_base_line_number > 0 && GetIsInteractive())
568f6913cd7SGreg Clayton             {
569f6913cd7SGreg Clayton                 FILE *out = GetOutputFILE();
570f6913cd7SGreg Clayton                 if (out)
571bc88d938SGreg Clayton                     ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(), GetPrompt() == NULL ? " " : "");
572f6913cd7SGreg Clayton             }
573f6913cd7SGreg Clayton 
574f0066ad0SGreg Clayton             bool interrupted = false;
575f0066ad0SGreg Clayton             if (GetLine(line, interrupted))
576f0066ad0SGreg Clayton             {
577f0066ad0SGreg Clayton                 if (interrupted)
578f0066ad0SGreg Clayton                 {
579f0066ad0SGreg Clayton                     lines_status = LineStatus::Done;
580f0066ad0SGreg Clayton                 }
581f0066ad0SGreg Clayton                 else
58244d93782SGreg Clayton                 {
58344d93782SGreg Clayton                     lines.AppendString(line);
58444d93782SGreg Clayton                     lines_status = m_delegate.IOHandlerLinesUpdated(*this, lines, lines.GetSize() - 1, error);
58544d93782SGreg Clayton                 }
586f0066ad0SGreg Clayton             }
58744d93782SGreg Clayton             else
58844d93782SGreg Clayton             {
58944d93782SGreg Clayton                 lines_status = LineStatus::Done;
59044d93782SGreg Clayton             }
59144d93782SGreg Clayton         }
592c3d874a5SGreg Clayton 
593c3d874a5SGreg Clayton         // Call the IOHandlerLinesUpdated function with UINT32_MAX as the line
594c3d874a5SGreg Clayton         // number to indicate all lines are complete
595c3d874a5SGreg Clayton         m_delegate.IOHandlerLinesUpdated(*this, lines, UINT32_MAX, error);
596c3d874a5SGreg Clayton 
59744d93782SGreg Clayton         success = lines.GetSize() > 0;
598*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
59944d93782SGreg Clayton     }
600*cacde7dfSTodd Fiala #endif
60144d93782SGreg Clayton     return success;
60244d93782SGreg Clayton }
60344d93782SGreg Clayton 
60444d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data
60544d93782SGreg Clayton // from the "in" and place output into "out" and "err and return
60644d93782SGreg Clayton // when done.
60744d93782SGreg Clayton void
60844d93782SGreg Clayton IOHandlerEditline::Run ()
60944d93782SGreg Clayton {
61044d93782SGreg Clayton     std::string line;
61144d93782SGreg Clayton     while (IsActive())
61244d93782SGreg Clayton     {
613f0066ad0SGreg Clayton         bool interrupted = false;
61444d93782SGreg Clayton         if (m_multi_line)
61544d93782SGreg Clayton         {
61644d93782SGreg Clayton             StringList lines;
617f0066ad0SGreg Clayton             if (GetLines (lines, interrupted))
618f0066ad0SGreg Clayton             {
619f0066ad0SGreg Clayton                 if (interrupted)
620f0066ad0SGreg Clayton                 {
621f0066ad0SGreg Clayton                     m_done = true;
622f0066ad0SGreg Clayton                 }
623f0066ad0SGreg Clayton                 else
62444d93782SGreg Clayton                 {
62544d93782SGreg Clayton                     line = lines.CopyList();
62644d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete(*this, line);
62744d93782SGreg Clayton                 }
628f0066ad0SGreg Clayton             }
62944d93782SGreg Clayton             else
63044d93782SGreg Clayton             {
63144d93782SGreg Clayton                 m_done = true;
63244d93782SGreg Clayton             }
63344d93782SGreg Clayton         }
63444d93782SGreg Clayton         else
63544d93782SGreg Clayton         {
636f0066ad0SGreg Clayton             if (GetLine(line, interrupted))
63744d93782SGreg Clayton             {
638f0066ad0SGreg Clayton                 if (!interrupted)
63944d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete(*this, line);
64044d93782SGreg Clayton             }
64144d93782SGreg Clayton             else
64244d93782SGreg Clayton             {
64344d93782SGreg Clayton                 m_done = true;
64444d93782SGreg Clayton             }
64544d93782SGreg Clayton         }
64644d93782SGreg Clayton     }
64744d93782SGreg Clayton }
64844d93782SGreg Clayton 
64944d93782SGreg Clayton void
65044d93782SGreg Clayton IOHandlerEditline::Hide ()
65144d93782SGreg Clayton {
652*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
653b89b7496SGreg Clayton     if (m_editline_ap)
65444d93782SGreg Clayton         m_editline_ap->Hide();
655*cacde7dfSTodd Fiala #endif
65644d93782SGreg Clayton }
65744d93782SGreg Clayton 
65844d93782SGreg Clayton 
65944d93782SGreg Clayton void
66044d93782SGreg Clayton IOHandlerEditline::Refresh ()
66144d93782SGreg Clayton {
662*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
663b89b7496SGreg Clayton     if (m_editline_ap)
664b89b7496SGreg Clayton     {
66544d93782SGreg Clayton         m_editline_ap->Refresh();
666b89b7496SGreg Clayton     }
66744d93782SGreg Clayton     else
66844d93782SGreg Clayton     {
669*cacde7dfSTodd Fiala #endif
67044d93782SGreg Clayton         const char *prompt = GetPrompt();
67144d93782SGreg Clayton         if (prompt && prompt[0])
67244d93782SGreg Clayton         {
67344d93782SGreg Clayton             FILE *out = GetOutputFILE();
67444d93782SGreg Clayton             if (out)
67544d93782SGreg Clayton             {
67644d93782SGreg Clayton                 ::fprintf(out, "%s", prompt);
67744d93782SGreg Clayton                 ::fflush(out);
67844d93782SGreg Clayton             }
67944d93782SGreg Clayton         }
680*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
68144d93782SGreg Clayton     }
682*cacde7dfSTodd Fiala #endif
68344d93782SGreg Clayton }
68444d93782SGreg Clayton 
68544d93782SGreg Clayton void
686e68f5d6bSGreg Clayton IOHandlerEditline::Cancel ()
687e68f5d6bSGreg Clayton {
688*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
689e68f5d6bSGreg Clayton     if (m_editline_ap)
690e68f5d6bSGreg Clayton         m_editline_ap->Interrupt ();
691*cacde7dfSTodd Fiala #endif
692e68f5d6bSGreg Clayton }
693e68f5d6bSGreg Clayton 
694f0066ad0SGreg Clayton bool
69544d93782SGreg Clayton IOHandlerEditline::Interrupt ()
69644d93782SGreg Clayton {
697f0066ad0SGreg Clayton     // Let the delgate handle it first
698f0066ad0SGreg Clayton     if (m_delegate.IOHandlerInterrupt(*this))
699f0066ad0SGreg Clayton         return true;
700f0066ad0SGreg Clayton 
701*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
70244d93782SGreg Clayton     if (m_editline_ap)
703f0066ad0SGreg Clayton         return m_editline_ap->Interrupt();
704*cacde7dfSTodd Fiala #endif
705f0066ad0SGreg Clayton     return false;
70644d93782SGreg Clayton }
70744d93782SGreg Clayton 
70844d93782SGreg Clayton void
70944d93782SGreg Clayton IOHandlerEditline::GotEOF()
71044d93782SGreg Clayton {
711*cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
71244d93782SGreg Clayton     if (m_editline_ap)
71344d93782SGreg Clayton         m_editline_ap->Interrupt();
714*cacde7dfSTodd Fiala #endif
71544d93782SGreg Clayton }
71644d93782SGreg Clayton 
717914b8d98SDeepak Panickal // we may want curses to be disabled for some builds
718914b8d98SDeepak Panickal // for instance, windows
719914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
720914b8d98SDeepak Panickal 
72144d93782SGreg Clayton #include "lldb/Core/ValueObject.h"
72244d93782SGreg Clayton #include "lldb/Symbol/VariableList.h"
72344d93782SGreg Clayton #include "lldb/Target/Target.h"
72444d93782SGreg Clayton #include "lldb/Target/Process.h"
72544d93782SGreg Clayton #include "lldb/Target/Thread.h"
72644d93782SGreg Clayton #include "lldb/Target/StackFrame.h"
72744d93782SGreg Clayton 
72844d93782SGreg Clayton #define KEY_RETURN   10
72944d93782SGreg Clayton #define KEY_ESCAPE  27
73044d93782SGreg Clayton 
73144d93782SGreg Clayton namespace curses
73244d93782SGreg Clayton {
73344d93782SGreg Clayton     class Menu;
73444d93782SGreg Clayton     class MenuDelegate;
73544d93782SGreg Clayton     class Window;
73644d93782SGreg Clayton     class WindowDelegate;
73744d93782SGreg Clayton     typedef std::shared_ptr<Menu> MenuSP;
73844d93782SGreg Clayton     typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
73944d93782SGreg Clayton     typedef std::shared_ptr<Window> WindowSP;
74044d93782SGreg Clayton     typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
74144d93782SGreg Clayton     typedef std::vector<MenuSP> Menus;
74244d93782SGreg Clayton     typedef std::vector<WindowSP> Windows;
74344d93782SGreg Clayton     typedef std::vector<WindowDelegateSP> WindowDelegates;
74444d93782SGreg Clayton 
74544d93782SGreg Clayton #if 0
74644d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
74744d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
74844d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
74944d93782SGreg Clayton #endif
75044d93782SGreg Clayton     struct Point
75144d93782SGreg Clayton     {
75244d93782SGreg Clayton         int x;
75344d93782SGreg Clayton         int y;
75444d93782SGreg Clayton 
75544d93782SGreg Clayton         Point (int _x = 0, int _y = 0) :
75644d93782SGreg Clayton             x(_x),
75744d93782SGreg Clayton             y(_y)
75844d93782SGreg Clayton         {
75944d93782SGreg Clayton         }
76044d93782SGreg Clayton 
76144d93782SGreg Clayton         void
76244d93782SGreg Clayton         Clear ()
76344d93782SGreg Clayton         {
76444d93782SGreg Clayton             x = 0;
76544d93782SGreg Clayton             y = 0;
76644d93782SGreg Clayton         }
76744d93782SGreg Clayton 
76844d93782SGreg Clayton         Point &
76944d93782SGreg Clayton         operator += (const Point &rhs)
77044d93782SGreg Clayton         {
77144d93782SGreg Clayton             x += rhs.x;
77244d93782SGreg Clayton             y += rhs.y;
77344d93782SGreg Clayton             return *this;
77444d93782SGreg Clayton         }
77544d93782SGreg Clayton 
77644d93782SGreg Clayton         void
77744d93782SGreg Clayton         Dump ()
77844d93782SGreg Clayton         {
77944d93782SGreg Clayton             printf ("(x=%i, y=%i)\n", x, y);
78044d93782SGreg Clayton         }
78144d93782SGreg Clayton 
78244d93782SGreg Clayton     };
78344d93782SGreg Clayton 
78444d93782SGreg Clayton     bool operator == (const Point &lhs, const Point &rhs)
78544d93782SGreg Clayton     {
78644d93782SGreg Clayton         return lhs.x == rhs.x && lhs.y == rhs.y;
78744d93782SGreg Clayton     }
78844d93782SGreg Clayton     bool operator != (const Point &lhs, const Point &rhs)
78944d93782SGreg Clayton     {
79044d93782SGreg Clayton         return lhs.x != rhs.x || lhs.y != rhs.y;
79144d93782SGreg Clayton     }
79244d93782SGreg Clayton 
79344d93782SGreg Clayton     struct Size
79444d93782SGreg Clayton     {
79544d93782SGreg Clayton         int width;
79644d93782SGreg Clayton         int height;
79744d93782SGreg Clayton         Size (int w = 0, int h = 0) :
79844d93782SGreg Clayton             width (w),
79944d93782SGreg Clayton             height (h)
80044d93782SGreg Clayton         {
80144d93782SGreg Clayton         }
80244d93782SGreg Clayton 
80344d93782SGreg Clayton         void
80444d93782SGreg Clayton         Clear ()
80544d93782SGreg Clayton         {
80644d93782SGreg Clayton             width = 0;
80744d93782SGreg Clayton             height = 0;
80844d93782SGreg Clayton         }
80944d93782SGreg Clayton 
81044d93782SGreg Clayton         void
81144d93782SGreg Clayton         Dump ()
81244d93782SGreg Clayton         {
81344d93782SGreg Clayton             printf ("(w=%i, h=%i)\n", width, height);
81444d93782SGreg Clayton         }
81544d93782SGreg Clayton 
81644d93782SGreg Clayton     };
81744d93782SGreg Clayton 
81844d93782SGreg Clayton     bool operator == (const Size &lhs, const Size &rhs)
81944d93782SGreg Clayton     {
82044d93782SGreg Clayton         return lhs.width == rhs.width && lhs.height == rhs.height;
82144d93782SGreg Clayton     }
82244d93782SGreg Clayton     bool operator != (const Size &lhs, const Size &rhs)
82344d93782SGreg Clayton     {
82444d93782SGreg Clayton         return lhs.width != rhs.width || lhs.height != rhs.height;
82544d93782SGreg Clayton     }
82644d93782SGreg Clayton 
82744d93782SGreg Clayton     struct Rect
82844d93782SGreg Clayton     {
82944d93782SGreg Clayton         Point origin;
83044d93782SGreg Clayton         Size size;
83144d93782SGreg Clayton 
83244d93782SGreg Clayton         Rect () :
83344d93782SGreg Clayton             origin(),
83444d93782SGreg Clayton             size()
83544d93782SGreg Clayton         {
83644d93782SGreg Clayton         }
83744d93782SGreg Clayton 
83844d93782SGreg Clayton         Rect (const Point &p, const Size &s) :
83944d93782SGreg Clayton             origin (p),
84044d93782SGreg Clayton             size (s)
84144d93782SGreg Clayton         {
84244d93782SGreg Clayton         }
84344d93782SGreg Clayton 
84444d93782SGreg Clayton         void
84544d93782SGreg Clayton         Clear ()
84644d93782SGreg Clayton         {
84744d93782SGreg Clayton             origin.Clear();
84844d93782SGreg Clayton             size.Clear();
84944d93782SGreg Clayton         }
85044d93782SGreg Clayton 
85144d93782SGreg Clayton         void
85244d93782SGreg Clayton         Dump ()
85344d93782SGreg Clayton         {
85444d93782SGreg Clayton             printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height);
85544d93782SGreg Clayton         }
85644d93782SGreg Clayton 
85744d93782SGreg Clayton         void
85844d93782SGreg Clayton         Inset (int w, int h)
85944d93782SGreg Clayton         {
86044d93782SGreg Clayton             if (size.width > w*2)
86144d93782SGreg Clayton                 size.width -= w*2;
86244d93782SGreg Clayton             origin.x += w;
86344d93782SGreg Clayton 
86444d93782SGreg Clayton             if (size.height > h*2)
86544d93782SGreg Clayton                 size.height -= h*2;
86644d93782SGreg Clayton             origin.y += h;
86744d93782SGreg Clayton         }
86844d93782SGreg Clayton         // Return a status bar rectangle which is the last line of
86944d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
87044d93782SGreg Clayton         // include the status bar area.
87144d93782SGreg Clayton         Rect
87244d93782SGreg Clayton         MakeStatusBar ()
87344d93782SGreg Clayton         {
87444d93782SGreg Clayton             Rect status_bar;
87544d93782SGreg Clayton             if (size.height > 1)
87644d93782SGreg Clayton             {
87744d93782SGreg Clayton                 status_bar.origin.x = origin.x;
87844d93782SGreg Clayton                 status_bar.origin.y = size.height;
87944d93782SGreg Clayton                 status_bar.size.width = size.width;
88044d93782SGreg Clayton                 status_bar.size.height = 1;
88144d93782SGreg Clayton                 --size.height;
88244d93782SGreg Clayton             }
88344d93782SGreg Clayton             return status_bar;
88444d93782SGreg Clayton         }
88544d93782SGreg Clayton 
88644d93782SGreg Clayton         // Return a menubar rectangle which is the first line of
88744d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
88844d93782SGreg Clayton         // include the menubar area.
88944d93782SGreg Clayton         Rect
89044d93782SGreg Clayton         MakeMenuBar ()
89144d93782SGreg Clayton         {
89244d93782SGreg Clayton             Rect menubar;
89344d93782SGreg Clayton             if (size.height > 1)
89444d93782SGreg Clayton             {
89544d93782SGreg Clayton                 menubar.origin.x = origin.x;
89644d93782SGreg Clayton                 menubar.origin.y = origin.y;
89744d93782SGreg Clayton                 menubar.size.width = size.width;
89844d93782SGreg Clayton                 menubar.size.height = 1;
89944d93782SGreg Clayton                 ++origin.y;
90044d93782SGreg Clayton                 --size.height;
90144d93782SGreg Clayton             }
90244d93782SGreg Clayton             return menubar;
90344d93782SGreg Clayton         }
90444d93782SGreg Clayton 
90544d93782SGreg Clayton         void
90644d93782SGreg Clayton         HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const
90744d93782SGreg Clayton         {
90844d93782SGreg Clayton             float top_height = top_percentage * size.height;
90944d93782SGreg Clayton             HorizontalSplit (top_height, top, bottom);
91044d93782SGreg Clayton         }
91144d93782SGreg Clayton 
91244d93782SGreg Clayton         void
91344d93782SGreg Clayton         HorizontalSplit (int top_height, Rect &top, Rect &bottom) const
91444d93782SGreg Clayton         {
91544d93782SGreg Clayton             top = *this;
91644d93782SGreg Clayton             if (top_height < size.height)
91744d93782SGreg Clayton             {
91844d93782SGreg Clayton                 top.size.height = top_height;
91944d93782SGreg Clayton                 bottom.origin.x = origin.x;
92044d93782SGreg Clayton                 bottom.origin.y = origin.y + top.size.height;
92144d93782SGreg Clayton                 bottom.size.width = size.width;
92244d93782SGreg Clayton                 bottom.size.height = size.height - top.size.height;
92344d93782SGreg Clayton             }
92444d93782SGreg Clayton             else
92544d93782SGreg Clayton             {
92644d93782SGreg Clayton                 bottom.Clear();
92744d93782SGreg Clayton             }
92844d93782SGreg Clayton         }
92944d93782SGreg Clayton 
93044d93782SGreg Clayton         void
93144d93782SGreg Clayton         VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const
93244d93782SGreg Clayton         {
93344d93782SGreg Clayton             float left_width = left_percentage * size.width;
93444d93782SGreg Clayton             VerticalSplit (left_width, left, right);
93544d93782SGreg Clayton         }
93644d93782SGreg Clayton 
93744d93782SGreg Clayton 
93844d93782SGreg Clayton         void
93944d93782SGreg Clayton         VerticalSplit (int left_width, Rect &left, Rect &right) const
94044d93782SGreg Clayton         {
94144d93782SGreg Clayton             left = *this;
94244d93782SGreg Clayton             if (left_width < size.width)
94344d93782SGreg Clayton             {
94444d93782SGreg Clayton                 left.size.width = left_width;
94544d93782SGreg Clayton                 right.origin.x = origin.x + left.size.width;
94644d93782SGreg Clayton                 right.origin.y = origin.y;
94744d93782SGreg Clayton                 right.size.width = size.width - left.size.width;
94844d93782SGreg Clayton                 right.size.height = size.height;
94944d93782SGreg Clayton             }
95044d93782SGreg Clayton             else
95144d93782SGreg Clayton             {
95244d93782SGreg Clayton                 right.Clear();
95344d93782SGreg Clayton             }
95444d93782SGreg Clayton         }
95544d93782SGreg Clayton     };
95644d93782SGreg Clayton 
95744d93782SGreg Clayton     bool operator == (const Rect &lhs, const Rect &rhs)
95844d93782SGreg Clayton     {
95944d93782SGreg Clayton         return lhs.origin == rhs.origin && lhs.size == rhs.size;
96044d93782SGreg Clayton     }
96144d93782SGreg Clayton     bool operator != (const Rect &lhs, const Rect &rhs)
96244d93782SGreg Clayton     {
96344d93782SGreg Clayton         return lhs.origin != rhs.origin || lhs.size != rhs.size;
96444d93782SGreg Clayton     }
96544d93782SGreg Clayton 
96644d93782SGreg Clayton     enum HandleCharResult
96744d93782SGreg Clayton     {
96844d93782SGreg Clayton         eKeyNotHandled      = 0,
96944d93782SGreg Clayton         eKeyHandled         = 1,
97044d93782SGreg Clayton         eQuitApplication    = 2
97144d93782SGreg Clayton     };
97244d93782SGreg Clayton 
97344d93782SGreg Clayton     enum class MenuActionResult
97444d93782SGreg Clayton     {
97544d93782SGreg Clayton         Handled,
97644d93782SGreg Clayton         NotHandled,
97744d93782SGreg Clayton         Quit    // Exit all menus and quit
97844d93782SGreg Clayton     };
97944d93782SGreg Clayton 
98044d93782SGreg Clayton     struct KeyHelp
98144d93782SGreg Clayton     {
98244d93782SGreg Clayton         int ch;
98344d93782SGreg Clayton         const char *description;
98444d93782SGreg Clayton     };
98544d93782SGreg Clayton 
98644d93782SGreg Clayton     class WindowDelegate
98744d93782SGreg Clayton     {
98844d93782SGreg Clayton     public:
98944d93782SGreg Clayton         virtual
99044d93782SGreg Clayton         ~WindowDelegate()
99144d93782SGreg Clayton         {
99244d93782SGreg Clayton         }
99344d93782SGreg Clayton 
99444d93782SGreg Clayton         virtual bool
99544d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force)
99644d93782SGreg Clayton         {
99744d93782SGreg Clayton             return false; // Drawing not handled
99844d93782SGreg Clayton         }
99944d93782SGreg Clayton 
100044d93782SGreg Clayton         virtual HandleCharResult
100144d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key)
100244d93782SGreg Clayton         {
100344d93782SGreg Clayton             return eKeyNotHandled;
100444d93782SGreg Clayton         }
100544d93782SGreg Clayton 
100644d93782SGreg Clayton         virtual const char *
100744d93782SGreg Clayton         WindowDelegateGetHelpText ()
100844d93782SGreg Clayton         {
100944d93782SGreg Clayton             return NULL;
101044d93782SGreg Clayton         }
101144d93782SGreg Clayton 
101244d93782SGreg Clayton         virtual KeyHelp *
101344d93782SGreg Clayton         WindowDelegateGetKeyHelp ()
101444d93782SGreg Clayton         {
101544d93782SGreg Clayton             return NULL;
101644d93782SGreg Clayton         }
101744d93782SGreg Clayton     };
101844d93782SGreg Clayton 
101944d93782SGreg Clayton     class HelpDialogDelegate :
102044d93782SGreg Clayton         public WindowDelegate
102144d93782SGreg Clayton     {
102244d93782SGreg Clayton     public:
102344d93782SGreg Clayton         HelpDialogDelegate (const char *text, KeyHelp *key_help_array);
102444d93782SGreg Clayton 
102544d93782SGreg Clayton         virtual
102644d93782SGreg Clayton         ~HelpDialogDelegate();
102744d93782SGreg Clayton 
102844d93782SGreg Clayton         virtual bool
102944d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force);
103044d93782SGreg Clayton 
103144d93782SGreg Clayton         virtual HandleCharResult
103244d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key);
103344d93782SGreg Clayton 
103444d93782SGreg Clayton         size_t
103544d93782SGreg Clayton         GetNumLines() const
103644d93782SGreg Clayton         {
103744d93782SGreg Clayton             return m_text.GetSize();
103844d93782SGreg Clayton         }
103944d93782SGreg Clayton 
104044d93782SGreg Clayton         size_t
104144d93782SGreg Clayton         GetMaxLineLength () const
104244d93782SGreg Clayton         {
104344d93782SGreg Clayton             return m_text.GetMaxStringLength();
104444d93782SGreg Clayton         }
104544d93782SGreg Clayton 
104644d93782SGreg Clayton     protected:
104744d93782SGreg Clayton         StringList m_text;
104844d93782SGreg Clayton         int m_first_visible_line;
104944d93782SGreg Clayton     };
105044d93782SGreg Clayton 
105144d93782SGreg Clayton 
105244d93782SGreg Clayton     class Window
105344d93782SGreg Clayton     {
105444d93782SGreg Clayton     public:
105544d93782SGreg Clayton 
105644d93782SGreg Clayton         Window (const char *name) :
105744d93782SGreg Clayton             m_name (name),
105844d93782SGreg Clayton             m_window (NULL),
105944d93782SGreg Clayton             m_panel (NULL),
106044d93782SGreg Clayton             m_parent (NULL),
106144d93782SGreg Clayton             m_subwindows (),
106244d93782SGreg Clayton             m_delegate_sp (),
106344d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
106444d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
106544d93782SGreg Clayton             m_delete (false),
106644d93782SGreg Clayton             m_needs_update (true),
106744d93782SGreg Clayton             m_can_activate (true),
106844d93782SGreg Clayton             m_is_subwin (false)
106944d93782SGreg Clayton         {
107044d93782SGreg Clayton         }
107144d93782SGreg Clayton 
107244d93782SGreg Clayton         Window (const char *name, WINDOW *w, bool del = true) :
107344d93782SGreg Clayton             m_name (name),
107444d93782SGreg Clayton             m_window (NULL),
107544d93782SGreg Clayton             m_panel (NULL),
107644d93782SGreg Clayton             m_parent (NULL),
107744d93782SGreg Clayton             m_subwindows (),
107844d93782SGreg Clayton             m_delegate_sp (),
107944d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
108044d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
108144d93782SGreg Clayton             m_delete (del),
108244d93782SGreg Clayton             m_needs_update (true),
108344d93782SGreg Clayton             m_can_activate (true),
108444d93782SGreg Clayton             m_is_subwin (false)
108544d93782SGreg Clayton         {
108644d93782SGreg Clayton             if (w)
108744d93782SGreg Clayton                 Reset(w);
108844d93782SGreg Clayton         }
108944d93782SGreg Clayton 
109044d93782SGreg Clayton         Window (const char *name, const Rect &bounds) :
109144d93782SGreg Clayton             m_name (name),
109244d93782SGreg Clayton             m_window (NULL),
109344d93782SGreg Clayton             m_parent (NULL),
109444d93782SGreg Clayton             m_subwindows (),
109544d93782SGreg Clayton             m_delegate_sp (),
109644d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
109744d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
109844d93782SGreg Clayton             m_delete (true),
109944d93782SGreg Clayton             m_needs_update (true),
110044d93782SGreg Clayton             m_can_activate (true),
110144d93782SGreg Clayton             m_is_subwin (false)
110244d93782SGreg Clayton         {
110344d93782SGreg Clayton             Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y));
110444d93782SGreg Clayton         }
110544d93782SGreg Clayton 
110644d93782SGreg Clayton         virtual
110744d93782SGreg Clayton         ~Window ()
110844d93782SGreg Clayton         {
110944d93782SGreg Clayton             RemoveSubWindows ();
111044d93782SGreg Clayton             Reset ();
111144d93782SGreg Clayton         }
111244d93782SGreg Clayton 
111344d93782SGreg Clayton         void
111444d93782SGreg Clayton         Reset (WINDOW *w = NULL, bool del = true)
111544d93782SGreg Clayton         {
111644d93782SGreg Clayton             if (m_window == w)
111744d93782SGreg Clayton                 return;
111844d93782SGreg Clayton 
111944d93782SGreg Clayton             if (m_panel)
112044d93782SGreg Clayton             {
112144d93782SGreg Clayton                 ::del_panel (m_panel);
112244d93782SGreg Clayton                 m_panel = NULL;
112344d93782SGreg Clayton             }
112444d93782SGreg Clayton             if (m_window && m_delete)
112544d93782SGreg Clayton             {
112644d93782SGreg Clayton                 ::delwin (m_window);
112744d93782SGreg Clayton                 m_window = NULL;
112844d93782SGreg Clayton                 m_delete = false;
112944d93782SGreg Clayton             }
113044d93782SGreg Clayton             if (w)
113144d93782SGreg Clayton             {
113244d93782SGreg Clayton                 m_window = w;
113344d93782SGreg Clayton                 m_panel = ::new_panel (m_window);
113444d93782SGreg Clayton                 m_delete = del;
113544d93782SGreg Clayton             }
113644d93782SGreg Clayton         }
113744d93782SGreg Clayton 
113844d93782SGreg Clayton         void    AttributeOn (attr_t attr)   { ::wattron (m_window, attr); }
113944d93782SGreg Clayton         void    AttributeOff (attr_t attr)  { ::wattroff (m_window, attr); }
114044d93782SGreg Clayton         void    Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); }
114144d93782SGreg Clayton         void    Clear ()    { ::wclear (m_window); }
114244d93782SGreg Clayton         void    Erase ()    { ::werase (m_window); }
114344d93782SGreg Clayton         Rect    GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window
114444d93782SGreg Clayton         int     GetChar ()  { return ::wgetch (m_window); }
114544d93782SGreg Clayton         int     GetCursorX ()     { return getcurx (m_window); }
114644d93782SGreg Clayton         int     GetCursorY ()     { return getcury (m_window); }
114744d93782SGreg Clayton         Rect    GetFrame ()    { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system
114844d93782SGreg Clayton         Point   GetParentOrigin() { return Point (GetParentX(), GetParentY()); }
114944d93782SGreg Clayton         Size    GetSize()         { return Size (GetWidth(), GetHeight()); }
115044d93782SGreg Clayton         int     GetParentX ()     { return getparx (m_window); }
115144d93782SGreg Clayton         int     GetParentY ()     { return getpary (m_window); }
115244d93782SGreg Clayton         int     GetMaxX()   { return getmaxx (m_window); }
115344d93782SGreg Clayton         int     GetMaxY()   { return getmaxy (m_window); }
115444d93782SGreg Clayton         int     GetWidth()  { return GetMaxX(); }
115544d93782SGreg Clayton         int     GetHeight() { return GetMaxY(); }
115644d93782SGreg Clayton         void    MoveCursor (int x, int y) {  ::wmove (m_window, y, x); }
115744d93782SGreg Clayton         void    MoveWindow (int x, int y) {  MoveWindow(Point(x,y)); }
115844d93782SGreg Clayton         void    Resize (int w, int h) { ::wresize(m_window, h, w); }
115944d93782SGreg Clayton         void    Resize (const Size &size) { ::wresize(m_window, size.height, size.width); }
116044d93782SGreg Clayton         void    PutChar (int ch)    { ::waddch (m_window, ch); }
116144d93782SGreg Clayton         void    PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); }
116244d93782SGreg Clayton         void    Refresh ()  { ::wrefresh (m_window); }
116344d93782SGreg Clayton         void    DeferredRefresh ()
116444d93782SGreg Clayton         {
116544d93782SGreg Clayton             // We are using panels, so we don't need to call this...
116644d93782SGreg Clayton             //::wnoutrefresh(m_window);
116744d93782SGreg Clayton         }
116844d93782SGreg Clayton         void    SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); }
116944d93782SGreg Clayton         void    UnderlineOn ()  { AttributeOn(A_UNDERLINE); }
117044d93782SGreg Clayton         void    UnderlineOff () { AttributeOff(A_UNDERLINE); }
117144d93782SGreg Clayton 
117244d93782SGreg Clayton         void    PutCStringTruncated (const char *s, int right_pad)
117344d93782SGreg Clayton         {
117444d93782SGreg Clayton             int bytes_left = GetWidth() - GetCursorX();
117544d93782SGreg Clayton             if (bytes_left > right_pad)
117644d93782SGreg Clayton             {
117744d93782SGreg Clayton                 bytes_left -= right_pad;
117844d93782SGreg Clayton                 ::waddnstr (m_window, s, bytes_left);
117944d93782SGreg Clayton             }
118044d93782SGreg Clayton         }
118144d93782SGreg Clayton 
118244d93782SGreg Clayton         void
118344d93782SGreg Clayton         MoveWindow (const Point &origin)
118444d93782SGreg Clayton         {
118544d93782SGreg Clayton             const bool moving_window = origin != GetParentOrigin();
118644d93782SGreg Clayton             if (m_is_subwin && moving_window)
118744d93782SGreg Clayton             {
118844d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
118944d93782SGreg Clayton                 Size size = GetSize();
119044d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
119144d93782SGreg Clayton                                  size.height,
119244d93782SGreg Clayton                                  size.width,
119344d93782SGreg Clayton                                  origin.y,
119444d93782SGreg Clayton                                  origin.x), true);
119544d93782SGreg Clayton             }
119644d93782SGreg Clayton             else
119744d93782SGreg Clayton             {
119844d93782SGreg Clayton                 ::mvwin (m_window, origin.y, origin.x);
119944d93782SGreg Clayton             }
120044d93782SGreg Clayton         }
120144d93782SGreg Clayton 
120244d93782SGreg Clayton         void
120344d93782SGreg Clayton         SetBounds (const Rect &bounds)
120444d93782SGreg Clayton         {
120544d93782SGreg Clayton             const bool moving_window = bounds.origin != GetParentOrigin();
120644d93782SGreg Clayton             if (m_is_subwin && moving_window)
120744d93782SGreg Clayton             {
120844d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
120944d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
121044d93782SGreg Clayton                                  bounds.size.height,
121144d93782SGreg Clayton                                  bounds.size.width,
121244d93782SGreg Clayton                                  bounds.origin.y,
121344d93782SGreg Clayton                                  bounds.origin.x), true);
121444d93782SGreg Clayton             }
121544d93782SGreg Clayton             else
121644d93782SGreg Clayton             {
121744d93782SGreg Clayton                 if (moving_window)
121844d93782SGreg Clayton                     MoveWindow(bounds.origin);
121944d93782SGreg Clayton                 Resize (bounds.size);
122044d93782SGreg Clayton             }
122144d93782SGreg Clayton         }
122244d93782SGreg Clayton 
122344d93782SGreg Clayton         void
122444d93782SGreg Clayton         Printf (const char *format, ...)  __attribute__ ((format (printf, 2, 3)))
122544d93782SGreg Clayton         {
122644d93782SGreg Clayton             va_list args;
122744d93782SGreg Clayton             va_start (args, format);
122844d93782SGreg Clayton             vwprintw(m_window, format, args);
122944d93782SGreg Clayton             va_end (args);
123044d93782SGreg Clayton         }
123144d93782SGreg Clayton 
123244d93782SGreg Clayton         void
123344d93782SGreg Clayton         Touch ()
123444d93782SGreg Clayton         {
123544d93782SGreg Clayton             ::touchwin (m_window);
123644d93782SGreg Clayton             if (m_parent)
123744d93782SGreg Clayton                 m_parent->Touch();
123844d93782SGreg Clayton         }
123944d93782SGreg Clayton 
124044d93782SGreg Clayton         WindowSP
124144d93782SGreg Clayton         CreateSubWindow (const char *name, const Rect &bounds, bool make_active)
124244d93782SGreg Clayton         {
124344d93782SGreg Clayton             WindowSP subwindow_sp;
124444d93782SGreg Clayton             if (m_window)
124544d93782SGreg Clayton             {
124644d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::subwin (m_window,
124744d93782SGreg Clayton                                                               bounds.size.height,
124844d93782SGreg Clayton                                                               bounds.size.width,
124944d93782SGreg Clayton                                                               bounds.origin.y,
125044d93782SGreg Clayton                                                               bounds.origin.x), true));
125144d93782SGreg Clayton                 subwindow_sp->m_is_subwin = true;
125244d93782SGreg Clayton             }
125344d93782SGreg Clayton             else
125444d93782SGreg Clayton             {
125544d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height,
125644d93782SGreg Clayton                                                               bounds.size.width,
125744d93782SGreg Clayton                                                               bounds.origin.y,
125844d93782SGreg Clayton                                                               bounds.origin.x), true));
125944d93782SGreg Clayton                 subwindow_sp->m_is_subwin = false;
126044d93782SGreg Clayton             }
126144d93782SGreg Clayton             subwindow_sp->m_parent = this;
126244d93782SGreg Clayton             if (make_active)
126344d93782SGreg Clayton             {
126444d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
126544d93782SGreg Clayton                 m_curr_active_window_idx = m_subwindows.size();
126644d93782SGreg Clayton             }
126744d93782SGreg Clayton             m_subwindows.push_back(subwindow_sp);
126844d93782SGreg Clayton             ::top_panel (subwindow_sp->m_panel);
126944d93782SGreg Clayton             m_needs_update = true;
127044d93782SGreg Clayton             return subwindow_sp;
127144d93782SGreg Clayton         }
127244d93782SGreg Clayton 
127344d93782SGreg Clayton         bool
127444d93782SGreg Clayton         RemoveSubWindow (Window *window)
127544d93782SGreg Clayton         {
127644d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
127744d93782SGreg Clayton             size_t i = 0;
127844d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
127944d93782SGreg Clayton             {
128044d93782SGreg Clayton                 if ((*pos).get() == window)
128144d93782SGreg Clayton                 {
128244d93782SGreg Clayton                     if (m_prev_active_window_idx == i)
128344d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
128444d93782SGreg Clayton                     else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i)
128544d93782SGreg Clayton                         --m_prev_active_window_idx;
128644d93782SGreg Clayton 
128744d93782SGreg Clayton                     if (m_curr_active_window_idx == i)
128844d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
128944d93782SGreg Clayton                     else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i)
129044d93782SGreg Clayton                         --m_curr_active_window_idx;
129144d93782SGreg Clayton                     window->Erase();
129244d93782SGreg Clayton                     m_subwindows.erase(pos);
129344d93782SGreg Clayton                     m_needs_update = true;
129444d93782SGreg Clayton                     if (m_parent)
129544d93782SGreg Clayton                         m_parent->Touch();
129644d93782SGreg Clayton                     else
129744d93782SGreg Clayton                         ::touchwin (stdscr);
129844d93782SGreg Clayton                     return true;
129944d93782SGreg Clayton                 }
130044d93782SGreg Clayton             }
130144d93782SGreg Clayton             return false;
130244d93782SGreg Clayton         }
130344d93782SGreg Clayton 
130444d93782SGreg Clayton         WindowSP
130544d93782SGreg Clayton         FindSubWindow (const char *name)
130644d93782SGreg Clayton         {
130744d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
130844d93782SGreg Clayton             size_t i = 0;
130944d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
131044d93782SGreg Clayton             {
131144d93782SGreg Clayton                 if ((*pos)->m_name.compare(name) == 0)
131244d93782SGreg Clayton                     return *pos;
131344d93782SGreg Clayton             }
131444d93782SGreg Clayton             return WindowSP();
131544d93782SGreg Clayton         }
131644d93782SGreg Clayton 
131744d93782SGreg Clayton         void
131844d93782SGreg Clayton         RemoveSubWindows ()
131944d93782SGreg Clayton         {
132044d93782SGreg Clayton             m_curr_active_window_idx = UINT32_MAX;
132144d93782SGreg Clayton             m_prev_active_window_idx = UINT32_MAX;
132244d93782SGreg Clayton             for (Windows::iterator pos = m_subwindows.begin();
132344d93782SGreg Clayton                  pos != m_subwindows.end();
132444d93782SGreg Clayton                  pos = m_subwindows.erase(pos))
132544d93782SGreg Clayton             {
132644d93782SGreg Clayton                 (*pos)->Erase();
132744d93782SGreg Clayton             }
132844d93782SGreg Clayton             if (m_parent)
132944d93782SGreg Clayton                 m_parent->Touch();
133044d93782SGreg Clayton             else
133144d93782SGreg Clayton                 ::touchwin (stdscr);
133244d93782SGreg Clayton         }
133344d93782SGreg Clayton 
133444d93782SGreg Clayton         WINDOW *
133544d93782SGreg Clayton         get()
133644d93782SGreg Clayton         {
133744d93782SGreg Clayton             return m_window;
133844d93782SGreg Clayton         }
133944d93782SGreg Clayton 
134044d93782SGreg Clayton         operator WINDOW *()
134144d93782SGreg Clayton         {
134244d93782SGreg Clayton             return m_window;
134344d93782SGreg Clayton         }
134444d93782SGreg Clayton 
134544d93782SGreg Clayton         //----------------------------------------------------------------------
134644d93782SGreg Clayton         // Window drawing utilities
134744d93782SGreg Clayton         //----------------------------------------------------------------------
134844d93782SGreg Clayton         void
134944d93782SGreg Clayton         DrawTitleBox (const char *title, const char *bottom_message = NULL)
135044d93782SGreg Clayton         {
135144d93782SGreg Clayton             attr_t attr = 0;
135244d93782SGreg Clayton             if (IsActive())
135344d93782SGreg Clayton                 attr = A_BOLD | COLOR_PAIR(2);
135444d93782SGreg Clayton             else
135544d93782SGreg Clayton                 attr = 0;
135644d93782SGreg Clayton             if (attr)
135744d93782SGreg Clayton                 AttributeOn(attr);
135844d93782SGreg Clayton 
135944d93782SGreg Clayton             Box();
136044d93782SGreg Clayton             MoveCursor(3, 0);
136144d93782SGreg Clayton 
136244d93782SGreg Clayton             if (title && title[0])
136344d93782SGreg Clayton             {
136444d93782SGreg Clayton                 PutChar ('<');
136544d93782SGreg Clayton                 PutCString (title);
136644d93782SGreg Clayton                 PutChar ('>');
136744d93782SGreg Clayton             }
136844d93782SGreg Clayton 
136944d93782SGreg Clayton             if (bottom_message && bottom_message[0])
137044d93782SGreg Clayton             {
137144d93782SGreg Clayton                 int bottom_message_length = strlen(bottom_message);
137244d93782SGreg Clayton                 int x = GetWidth() - 3 - (bottom_message_length + 2);
137344d93782SGreg Clayton 
137444d93782SGreg Clayton                 if (x > 0)
137544d93782SGreg Clayton                 {
137644d93782SGreg Clayton                     MoveCursor (x, GetHeight() - 1);
137744d93782SGreg Clayton                     PutChar ('[');
137844d93782SGreg Clayton                     PutCString(bottom_message);
137944d93782SGreg Clayton                     PutChar (']');
138044d93782SGreg Clayton                 }
138144d93782SGreg Clayton                 else
138244d93782SGreg Clayton                 {
138344d93782SGreg Clayton                     MoveCursor (1, GetHeight() - 1);
138444d93782SGreg Clayton                     PutChar ('[');
138544d93782SGreg Clayton                     PutCStringTruncated (bottom_message, 1);
138644d93782SGreg Clayton                 }
138744d93782SGreg Clayton             }
138844d93782SGreg Clayton             if (attr)
138944d93782SGreg Clayton                 AttributeOff(attr);
139044d93782SGreg Clayton 
139144d93782SGreg Clayton         }
139244d93782SGreg Clayton 
139344d93782SGreg Clayton         virtual void
139444d93782SGreg Clayton         Draw (bool force)
139544d93782SGreg Clayton         {
139644d93782SGreg Clayton             if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force))
139744d93782SGreg Clayton                 return;
139844d93782SGreg Clayton 
139944d93782SGreg Clayton             for (auto &subwindow_sp : m_subwindows)
140044d93782SGreg Clayton                 subwindow_sp->Draw(force);
140144d93782SGreg Clayton         }
140244d93782SGreg Clayton 
140344d93782SGreg Clayton         bool
140444d93782SGreg Clayton         CreateHelpSubwindow ()
140544d93782SGreg Clayton         {
140644d93782SGreg Clayton             if (m_delegate_sp)
140744d93782SGreg Clayton             {
140844d93782SGreg Clayton                 const char *text = m_delegate_sp->WindowDelegateGetHelpText ();
140944d93782SGreg Clayton                 KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp ();
141044d93782SGreg Clayton                 if ((text && text[0]) || key_help)
141144d93782SGreg Clayton                 {
141244d93782SGreg Clayton                     std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help));
141344d93782SGreg Clayton                     const size_t num_lines = help_delegate_ap->GetNumLines();
141444d93782SGreg Clayton                     const size_t max_length = help_delegate_ap->GetMaxLineLength();
141544d93782SGreg Clayton                     Rect bounds = GetBounds();
141644d93782SGreg Clayton                     bounds.Inset(1, 1);
14173985c8c6SSaleem Abdulrasool                     if (max_length + 4 < static_cast<size_t>(bounds.size.width))
141844d93782SGreg Clayton                     {
141944d93782SGreg Clayton                         bounds.origin.x += (bounds.size.width - max_length + 4)/2;
142044d93782SGreg Clayton                         bounds.size.width = max_length + 4;
142144d93782SGreg Clayton                     }
142244d93782SGreg Clayton                     else
142344d93782SGreg Clayton                     {
142444d93782SGreg Clayton                         if (bounds.size.width > 100)
142544d93782SGreg Clayton                         {
142644d93782SGreg Clayton                             const int inset_w = bounds.size.width / 4;
142744d93782SGreg Clayton                             bounds.origin.x += inset_w;
142844d93782SGreg Clayton                             bounds.size.width -= 2*inset_w;
142944d93782SGreg Clayton                         }
143044d93782SGreg Clayton                     }
143144d93782SGreg Clayton 
14323985c8c6SSaleem Abdulrasool                     if (num_lines + 2 < static_cast<size_t>(bounds.size.height))
143344d93782SGreg Clayton                     {
143444d93782SGreg Clayton                         bounds.origin.y += (bounds.size.height - num_lines + 2)/2;
143544d93782SGreg Clayton                         bounds.size.height = num_lines + 2;
143644d93782SGreg Clayton                     }
143744d93782SGreg Clayton                     else
143844d93782SGreg Clayton                     {
143944d93782SGreg Clayton                         if (bounds.size.height > 100)
144044d93782SGreg Clayton                         {
144144d93782SGreg Clayton                             const int inset_h = bounds.size.height / 4;
144244d93782SGreg Clayton                             bounds.origin.y += inset_h;
144344d93782SGreg Clayton                             bounds.size.height -= 2*inset_h;
144444d93782SGreg Clayton                         }
144544d93782SGreg Clayton                     }
14465fdb09bbSGreg Clayton                     WindowSP help_window_sp;
14475fdb09bbSGreg Clayton                     Window *parent_window = GetParent();
14485fdb09bbSGreg Clayton                     if (parent_window)
14495fdb09bbSGreg Clayton                         help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
14505fdb09bbSGreg Clayton                     else
14515fdb09bbSGreg Clayton                         help_window_sp = CreateSubWindow("Help", bounds, true);
145244d93782SGreg Clayton                     help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release()));
145344d93782SGreg Clayton                     return true;
145444d93782SGreg Clayton                 }
145544d93782SGreg Clayton             }
145644d93782SGreg Clayton             return false;
145744d93782SGreg Clayton         }
145844d93782SGreg Clayton 
145944d93782SGreg Clayton         virtual HandleCharResult
146044d93782SGreg Clayton         HandleChar (int key)
146144d93782SGreg Clayton         {
146244d93782SGreg Clayton             // Always check the active window first
146344d93782SGreg Clayton             HandleCharResult result = eKeyNotHandled;
146444d93782SGreg Clayton             WindowSP active_window_sp = GetActiveWindow ();
146544d93782SGreg Clayton             if (active_window_sp)
146644d93782SGreg Clayton             {
146744d93782SGreg Clayton                 result = active_window_sp->HandleChar (key);
146844d93782SGreg Clayton                 if (result != eKeyNotHandled)
146944d93782SGreg Clayton                     return result;
147044d93782SGreg Clayton             }
147144d93782SGreg Clayton 
147244d93782SGreg Clayton             if (m_delegate_sp)
147344d93782SGreg Clayton             {
147444d93782SGreg Clayton                 result = m_delegate_sp->WindowDelegateHandleChar (*this, key);
147544d93782SGreg Clayton                 if (result != eKeyNotHandled)
147644d93782SGreg Clayton                     return result;
147744d93782SGreg Clayton             }
147844d93782SGreg Clayton 
147944d93782SGreg Clayton             // Then check for any windows that want any keys
148044d93782SGreg Clayton             // that weren't handled. This is typically only
148144d93782SGreg Clayton             // for a menubar.
148244d93782SGreg Clayton             // Make a copy of the subwindows in case any HandleChar()
148344d93782SGreg Clayton             // functions muck with the subwindows. If we don't do this,
148444d93782SGreg Clayton             // we can crash when iterating over the subwindows.
148544d93782SGreg Clayton             Windows subwindows (m_subwindows);
148644d93782SGreg Clayton             for (auto subwindow_sp : subwindows)
148744d93782SGreg Clayton             {
148844d93782SGreg Clayton                 if (subwindow_sp->m_can_activate == false)
148944d93782SGreg Clayton                 {
149044d93782SGreg Clayton                     HandleCharResult result = subwindow_sp->HandleChar(key);
149144d93782SGreg Clayton                     if (result != eKeyNotHandled)
149244d93782SGreg Clayton                         return result;
149344d93782SGreg Clayton                 }
149444d93782SGreg Clayton             }
149544d93782SGreg Clayton 
149644d93782SGreg Clayton             return eKeyNotHandled;
149744d93782SGreg Clayton         }
149844d93782SGreg Clayton 
149944d93782SGreg Clayton         bool
150044d93782SGreg Clayton         SetActiveWindow (Window *window)
150144d93782SGreg Clayton         {
150244d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
150344d93782SGreg Clayton             for (size_t i=0; i<num_subwindows; ++i)
150444d93782SGreg Clayton             {
150544d93782SGreg Clayton                 if (m_subwindows[i].get() == window)
150644d93782SGreg Clayton                 {
150744d93782SGreg Clayton                     m_prev_active_window_idx = m_curr_active_window_idx;
150844d93782SGreg Clayton                     ::top_panel (window->m_panel);
150944d93782SGreg Clayton                     m_curr_active_window_idx = i;
151044d93782SGreg Clayton                     return true;
151144d93782SGreg Clayton                 }
151244d93782SGreg Clayton             }
151344d93782SGreg Clayton             return false;
151444d93782SGreg Clayton         }
151544d93782SGreg Clayton 
151644d93782SGreg Clayton         WindowSP
151744d93782SGreg Clayton         GetActiveWindow ()
151844d93782SGreg Clayton         {
151944d93782SGreg Clayton             if (!m_subwindows.empty())
152044d93782SGreg Clayton             {
152144d93782SGreg Clayton                 if (m_curr_active_window_idx >= m_subwindows.size())
152244d93782SGreg Clayton                 {
152344d93782SGreg Clayton                     if (m_prev_active_window_idx < m_subwindows.size())
152444d93782SGreg Clayton                     {
152544d93782SGreg Clayton                         m_curr_active_window_idx = m_prev_active_window_idx;
152644d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
152744d93782SGreg Clayton                     }
152844d93782SGreg Clayton                     else if (IsActive())
152944d93782SGreg Clayton                     {
153044d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
153144d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
153244d93782SGreg Clayton 
153344d93782SGreg Clayton                         // Find first window that wants to be active if this window is active
153444d93782SGreg Clayton                         const size_t num_subwindows = m_subwindows.size();
153544d93782SGreg Clayton                         for (size_t i=0; i<num_subwindows; ++i)
153644d93782SGreg Clayton                         {
153744d93782SGreg Clayton                             if (m_subwindows[i]->GetCanBeActive())
153844d93782SGreg Clayton                             {
153944d93782SGreg Clayton                                 m_curr_active_window_idx = i;
154044d93782SGreg Clayton                                 break;
154144d93782SGreg Clayton                             }
154244d93782SGreg Clayton                         }
154344d93782SGreg Clayton                     }
154444d93782SGreg Clayton                 }
154544d93782SGreg Clayton 
154644d93782SGreg Clayton                 if (m_curr_active_window_idx < m_subwindows.size())
154744d93782SGreg Clayton                     return m_subwindows[m_curr_active_window_idx];
154844d93782SGreg Clayton             }
154944d93782SGreg Clayton             return WindowSP();
155044d93782SGreg Clayton         }
155144d93782SGreg Clayton 
155244d93782SGreg Clayton         bool
155344d93782SGreg Clayton         GetCanBeActive () const
155444d93782SGreg Clayton         {
155544d93782SGreg Clayton             return m_can_activate;
155644d93782SGreg Clayton         }
155744d93782SGreg Clayton 
155844d93782SGreg Clayton         void
155944d93782SGreg Clayton         SetCanBeActive (bool b)
156044d93782SGreg Clayton         {
156144d93782SGreg Clayton             m_can_activate = b;
156244d93782SGreg Clayton         }
156344d93782SGreg Clayton 
156444d93782SGreg Clayton         const WindowDelegateSP &
156544d93782SGreg Clayton         GetDelegate () const
156644d93782SGreg Clayton         {
156744d93782SGreg Clayton             return m_delegate_sp;
156844d93782SGreg Clayton         }
156944d93782SGreg Clayton 
157044d93782SGreg Clayton         void
157144d93782SGreg Clayton         SetDelegate (const WindowDelegateSP &delegate_sp)
157244d93782SGreg Clayton         {
157344d93782SGreg Clayton             m_delegate_sp = delegate_sp;
157444d93782SGreg Clayton         }
157544d93782SGreg Clayton 
157644d93782SGreg Clayton         Window *
157744d93782SGreg Clayton         GetParent () const
157844d93782SGreg Clayton         {
157944d93782SGreg Clayton             return m_parent;
158044d93782SGreg Clayton         }
158144d93782SGreg Clayton 
158244d93782SGreg Clayton         bool
158344d93782SGreg Clayton         IsActive () const
158444d93782SGreg Clayton         {
158544d93782SGreg Clayton             if (m_parent)
158644d93782SGreg Clayton                 return m_parent->GetActiveWindow().get() == this;
158744d93782SGreg Clayton             else
158844d93782SGreg Clayton                 return true; // Top level window is always active
158944d93782SGreg Clayton         }
159044d93782SGreg Clayton 
159144d93782SGreg Clayton         void
159244d93782SGreg Clayton         SelectNextWindowAsActive ()
159344d93782SGreg Clayton         {
159444d93782SGreg Clayton             // Move active focus to next window
159544d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
159644d93782SGreg Clayton             if (m_curr_active_window_idx == UINT32_MAX)
159744d93782SGreg Clayton             {
159844d93782SGreg Clayton                 uint32_t idx = 0;
159944d93782SGreg Clayton                 for (auto subwindow_sp : m_subwindows)
160044d93782SGreg Clayton                 {
160144d93782SGreg Clayton                     if (subwindow_sp->GetCanBeActive())
160244d93782SGreg Clayton                     {
160344d93782SGreg Clayton                         m_curr_active_window_idx = idx;
160444d93782SGreg Clayton                         break;
160544d93782SGreg Clayton                     }
160644d93782SGreg Clayton                     ++idx;
160744d93782SGreg Clayton                 }
160844d93782SGreg Clayton             }
160944d93782SGreg Clayton             else if (m_curr_active_window_idx + 1 < num_subwindows)
161044d93782SGreg Clayton             {
161144d93782SGreg Clayton                 bool handled = false;
161244d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
161344d93782SGreg Clayton                 for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx)
161444d93782SGreg Clayton                 {
161544d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
161644d93782SGreg Clayton                     {
161744d93782SGreg Clayton                         m_curr_active_window_idx = idx;
161844d93782SGreg Clayton                         handled = true;
161944d93782SGreg Clayton                         break;
162044d93782SGreg Clayton                     }
162144d93782SGreg Clayton                 }
162244d93782SGreg Clayton                 if (!handled)
162344d93782SGreg Clayton                 {
162444d93782SGreg Clayton                     for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx)
162544d93782SGreg Clayton                     {
162644d93782SGreg Clayton                         if (m_subwindows[idx]->GetCanBeActive())
162744d93782SGreg Clayton                         {
162844d93782SGreg Clayton                             m_curr_active_window_idx = idx;
162944d93782SGreg Clayton                             break;
163044d93782SGreg Clayton                         }
163144d93782SGreg Clayton                     }
163244d93782SGreg Clayton                 }
163344d93782SGreg Clayton             }
163444d93782SGreg Clayton             else
163544d93782SGreg Clayton             {
163644d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
163744d93782SGreg Clayton                 for (size_t idx=0; idx<num_subwindows; ++idx)
163844d93782SGreg Clayton                 {
163944d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
164044d93782SGreg Clayton                     {
164144d93782SGreg Clayton                         m_curr_active_window_idx = idx;
164244d93782SGreg Clayton                         break;
164344d93782SGreg Clayton                     }
164444d93782SGreg Clayton                 }
164544d93782SGreg Clayton             }
164644d93782SGreg Clayton         }
164744d93782SGreg Clayton 
164844d93782SGreg Clayton         const char *
164944d93782SGreg Clayton         GetName () const
165044d93782SGreg Clayton         {
165144d93782SGreg Clayton             return m_name.c_str();
165244d93782SGreg Clayton         }
165344d93782SGreg Clayton     protected:
165444d93782SGreg Clayton         std::string m_name;
165544d93782SGreg Clayton         WINDOW *m_window;
165644d93782SGreg Clayton         PANEL *m_panel;
165744d93782SGreg Clayton         Window *m_parent;
165844d93782SGreg Clayton         Windows m_subwindows;
165944d93782SGreg Clayton         WindowDelegateSP m_delegate_sp;
166044d93782SGreg Clayton         uint32_t m_curr_active_window_idx;
166144d93782SGreg Clayton         uint32_t m_prev_active_window_idx;
166244d93782SGreg Clayton         bool m_delete;
166344d93782SGreg Clayton         bool m_needs_update;
166444d93782SGreg Clayton         bool m_can_activate;
166544d93782SGreg Clayton         bool m_is_subwin;
166644d93782SGreg Clayton 
166744d93782SGreg Clayton     private:
166844d93782SGreg Clayton         DISALLOW_COPY_AND_ASSIGN(Window);
166944d93782SGreg Clayton     };
167044d93782SGreg Clayton 
167144d93782SGreg Clayton     class MenuDelegate
167244d93782SGreg Clayton     {
167344d93782SGreg Clayton     public:
167444d93782SGreg Clayton         virtual ~MenuDelegate() {}
167544d93782SGreg Clayton 
167644d93782SGreg Clayton         virtual MenuActionResult
167744d93782SGreg Clayton         MenuDelegateAction (Menu &menu) = 0;
167844d93782SGreg Clayton     };
167944d93782SGreg Clayton 
168044d93782SGreg Clayton     class Menu : public WindowDelegate
168144d93782SGreg Clayton     {
168244d93782SGreg Clayton     public:
168344d93782SGreg Clayton         enum class Type
168444d93782SGreg Clayton         {
168544d93782SGreg Clayton             Invalid,
168644d93782SGreg Clayton             Bar,
168744d93782SGreg Clayton             Item,
168844d93782SGreg Clayton             Separator
168944d93782SGreg Clayton         };
169044d93782SGreg Clayton 
169144d93782SGreg Clayton         // Menubar or separator constructor
169244d93782SGreg Clayton         Menu (Type type);
169344d93782SGreg Clayton 
169444d93782SGreg Clayton         // Menuitem constructor
169544d93782SGreg Clayton         Menu (const char *name,
169644d93782SGreg Clayton               const char *key_name,
169744d93782SGreg Clayton               int key_value,
169844d93782SGreg Clayton               uint64_t identifier);
169944d93782SGreg Clayton 
170044d93782SGreg Clayton         virtual ~
170144d93782SGreg Clayton         Menu ()
170244d93782SGreg Clayton         {
170344d93782SGreg Clayton         }
170444d93782SGreg Clayton 
170544d93782SGreg Clayton         const MenuDelegateSP &
170644d93782SGreg Clayton         GetDelegate () const
170744d93782SGreg Clayton         {
170844d93782SGreg Clayton             return m_delegate_sp;
170944d93782SGreg Clayton         }
171044d93782SGreg Clayton 
171144d93782SGreg Clayton         void
171244d93782SGreg Clayton         SetDelegate (const MenuDelegateSP &delegate_sp)
171344d93782SGreg Clayton         {
171444d93782SGreg Clayton             m_delegate_sp = delegate_sp;
171544d93782SGreg Clayton         }
171644d93782SGreg Clayton 
171744d93782SGreg Clayton         void
171844d93782SGreg Clayton         RecalculateNameLengths();
171944d93782SGreg Clayton 
172044d93782SGreg Clayton         void
172144d93782SGreg Clayton         AddSubmenu (const MenuSP &menu_sp);
172244d93782SGreg Clayton 
172344d93782SGreg Clayton         int
172444d93782SGreg Clayton         DrawAndRunMenu (Window &window);
172544d93782SGreg Clayton 
172644d93782SGreg Clayton         void
172744d93782SGreg Clayton         DrawMenuTitle (Window &window, bool highlight);
172844d93782SGreg Clayton 
172944d93782SGreg Clayton         virtual bool
173044d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force);
173144d93782SGreg Clayton 
173244d93782SGreg Clayton         virtual HandleCharResult
173344d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key);
173444d93782SGreg Clayton 
173544d93782SGreg Clayton         MenuActionResult
173644d93782SGreg Clayton         ActionPrivate (Menu &menu)
173744d93782SGreg Clayton         {
173844d93782SGreg Clayton             MenuActionResult result = MenuActionResult::NotHandled;
173944d93782SGreg Clayton             if (m_delegate_sp)
174044d93782SGreg Clayton             {
174144d93782SGreg Clayton                 result = m_delegate_sp->MenuDelegateAction (menu);
174244d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
174344d93782SGreg Clayton                     return result;
174444d93782SGreg Clayton             }
174544d93782SGreg Clayton             else if (m_parent)
174644d93782SGreg Clayton             {
174744d93782SGreg Clayton                 result = m_parent->ActionPrivate(menu);
174844d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
174944d93782SGreg Clayton                     return result;
175044d93782SGreg Clayton             }
175144d93782SGreg Clayton             return m_canned_result;
175244d93782SGreg Clayton         }
175344d93782SGreg Clayton 
175444d93782SGreg Clayton         MenuActionResult
175544d93782SGreg Clayton         Action ()
175644d93782SGreg Clayton         {
175744d93782SGreg Clayton             // Call the recursive action so it can try to handle it
175844d93782SGreg Clayton             // with the menu delegate, and if not, try our parent menu
175944d93782SGreg Clayton             return ActionPrivate (*this);
176044d93782SGreg Clayton         }
176144d93782SGreg Clayton 
176244d93782SGreg Clayton         void
176344d93782SGreg Clayton         SetCannedResult (MenuActionResult result)
176444d93782SGreg Clayton         {
176544d93782SGreg Clayton             m_canned_result = result;
176644d93782SGreg Clayton         }
176744d93782SGreg Clayton 
176844d93782SGreg Clayton         Menus &
176944d93782SGreg Clayton         GetSubmenus()
177044d93782SGreg Clayton         {
177144d93782SGreg Clayton             return m_submenus;
177244d93782SGreg Clayton         }
177344d93782SGreg Clayton 
177444d93782SGreg Clayton         const Menus &
177544d93782SGreg Clayton         GetSubmenus() const
177644d93782SGreg Clayton         {
177744d93782SGreg Clayton             return m_submenus;
177844d93782SGreg Clayton         }
177944d93782SGreg Clayton 
178044d93782SGreg Clayton         int
178144d93782SGreg Clayton         GetSelectedSubmenuIndex () const
178244d93782SGreg Clayton         {
178344d93782SGreg Clayton             return m_selected;
178444d93782SGreg Clayton         }
178544d93782SGreg Clayton 
178644d93782SGreg Clayton         void
178744d93782SGreg Clayton         SetSelectedSubmenuIndex (int idx)
178844d93782SGreg Clayton         {
178944d93782SGreg Clayton             m_selected = idx;
179044d93782SGreg Clayton         }
179144d93782SGreg Clayton 
179244d93782SGreg Clayton         Type
179344d93782SGreg Clayton         GetType () const
179444d93782SGreg Clayton         {
179544d93782SGreg Clayton             return m_type;
179644d93782SGreg Clayton         }
179744d93782SGreg Clayton 
179844d93782SGreg Clayton         int
179944d93782SGreg Clayton         GetStartingColumn() const
180044d93782SGreg Clayton         {
180144d93782SGreg Clayton             return m_start_col;
180244d93782SGreg Clayton         }
180344d93782SGreg Clayton 
180444d93782SGreg Clayton         void
180544d93782SGreg Clayton         SetStartingColumn(int col)
180644d93782SGreg Clayton         {
180744d93782SGreg Clayton             m_start_col = col;
180844d93782SGreg Clayton         }
180944d93782SGreg Clayton 
181044d93782SGreg Clayton         int
181144d93782SGreg Clayton         GetKeyValue() const
181244d93782SGreg Clayton         {
181344d93782SGreg Clayton             return m_key_value;
181444d93782SGreg Clayton         }
181544d93782SGreg Clayton 
181644d93782SGreg Clayton         void
181744d93782SGreg Clayton         SetKeyValue(int key_value)
181844d93782SGreg Clayton         {
181944d93782SGreg Clayton             m_key_value = key_value;
182044d93782SGreg Clayton         }
182144d93782SGreg Clayton 
182244d93782SGreg Clayton         std::string &
182344d93782SGreg Clayton         GetName()
182444d93782SGreg Clayton         {
182544d93782SGreg Clayton             return m_name;
182644d93782SGreg Clayton         }
182744d93782SGreg Clayton 
182844d93782SGreg Clayton         std::string &
182944d93782SGreg Clayton         GetKeyName()
183044d93782SGreg Clayton         {
183144d93782SGreg Clayton             return m_key_name;
183244d93782SGreg Clayton         }
183344d93782SGreg Clayton 
183444d93782SGreg Clayton         int
183544d93782SGreg Clayton         GetDrawWidth () const
183644d93782SGreg Clayton         {
183744d93782SGreg Clayton             return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
183844d93782SGreg Clayton         }
183944d93782SGreg Clayton 
184044d93782SGreg Clayton 
184144d93782SGreg Clayton         uint64_t
184244d93782SGreg Clayton         GetIdentifier() const
184344d93782SGreg Clayton         {
184444d93782SGreg Clayton             return m_identifier;
184544d93782SGreg Clayton         }
184644d93782SGreg Clayton 
184744d93782SGreg Clayton         void
184844d93782SGreg Clayton         SetIdentifier (uint64_t identifier)
184944d93782SGreg Clayton         {
185044d93782SGreg Clayton             m_identifier = identifier;
185144d93782SGreg Clayton         }
185244d93782SGreg Clayton 
185344d93782SGreg Clayton     protected:
185444d93782SGreg Clayton         std::string m_name;
185544d93782SGreg Clayton         std::string m_key_name;
185644d93782SGreg Clayton         uint64_t m_identifier;
185744d93782SGreg Clayton         Type m_type;
185844d93782SGreg Clayton         int m_key_value;
185944d93782SGreg Clayton         int m_start_col;
186044d93782SGreg Clayton         int m_max_submenu_name_length;
186144d93782SGreg Clayton         int m_max_submenu_key_name_length;
186244d93782SGreg Clayton         int m_selected;
186344d93782SGreg Clayton         Menu *m_parent;
186444d93782SGreg Clayton         Menus m_submenus;
186544d93782SGreg Clayton         WindowSP m_menu_window_sp;
186644d93782SGreg Clayton         MenuActionResult m_canned_result;
186744d93782SGreg Clayton         MenuDelegateSP m_delegate_sp;
186844d93782SGreg Clayton     };
186944d93782SGreg Clayton 
187044d93782SGreg Clayton     // Menubar or separator constructor
187144d93782SGreg Clayton     Menu::Menu (Type type) :
187244d93782SGreg Clayton         m_name (),
187344d93782SGreg Clayton         m_key_name (),
187444d93782SGreg Clayton         m_identifier (0),
187544d93782SGreg Clayton         m_type (type),
187644d93782SGreg Clayton         m_key_value (0),
187744d93782SGreg Clayton         m_start_col (0),
187844d93782SGreg Clayton         m_max_submenu_name_length (0),
187944d93782SGreg Clayton         m_max_submenu_key_name_length (0),
188044d93782SGreg Clayton         m_selected (0),
188144d93782SGreg Clayton         m_parent (NULL),
188244d93782SGreg Clayton         m_submenus (),
188344d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
188444d93782SGreg Clayton         m_delegate_sp()
188544d93782SGreg Clayton     {
188644d93782SGreg Clayton     }
188744d93782SGreg Clayton 
188844d93782SGreg Clayton     // Menuitem constructor
188944d93782SGreg Clayton     Menu::Menu (const char *name,
189044d93782SGreg Clayton                 const char *key_name,
189144d93782SGreg Clayton                 int key_value,
189244d93782SGreg Clayton                 uint64_t identifier) :
189344d93782SGreg Clayton         m_name (),
189444d93782SGreg Clayton         m_key_name (),
189544d93782SGreg Clayton         m_identifier (identifier),
189644d93782SGreg Clayton         m_type (Type::Invalid),
189744d93782SGreg Clayton         m_key_value (key_value),
189844d93782SGreg Clayton         m_start_col (0),
189944d93782SGreg Clayton         m_max_submenu_name_length (0),
190044d93782SGreg Clayton         m_max_submenu_key_name_length (0),
190144d93782SGreg Clayton         m_selected (0),
190244d93782SGreg Clayton         m_parent (NULL),
190344d93782SGreg Clayton         m_submenus (),
190444d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
190544d93782SGreg Clayton         m_delegate_sp()
190644d93782SGreg Clayton     {
190744d93782SGreg Clayton         if (name && name[0])
190844d93782SGreg Clayton         {
190944d93782SGreg Clayton             m_name = name;
191044d93782SGreg Clayton             m_type = Type::Item;
191144d93782SGreg Clayton             if (key_name && key_name[0])
191244d93782SGreg Clayton                 m_key_name = key_name;
191344d93782SGreg Clayton         }
191444d93782SGreg Clayton         else
191544d93782SGreg Clayton         {
191644d93782SGreg Clayton             m_type = Type::Separator;
191744d93782SGreg Clayton         }
191844d93782SGreg Clayton     }
191944d93782SGreg Clayton 
192044d93782SGreg Clayton     void
192144d93782SGreg Clayton     Menu::RecalculateNameLengths()
192244d93782SGreg Clayton     {
192344d93782SGreg Clayton         m_max_submenu_name_length = 0;
192444d93782SGreg Clayton         m_max_submenu_key_name_length = 0;
192544d93782SGreg Clayton         Menus &submenus = GetSubmenus();
192644d93782SGreg Clayton         const size_t num_submenus = submenus.size();
192744d93782SGreg Clayton         for (size_t i=0; i<num_submenus; ++i)
192844d93782SGreg Clayton         {
192944d93782SGreg Clayton             Menu *submenu = submenus[i].get();
19303985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
193144d93782SGreg Clayton                 m_max_submenu_name_length = submenu->m_name.size();
19323985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size())
193344d93782SGreg Clayton                 m_max_submenu_key_name_length = submenu->m_key_name.size();
193444d93782SGreg Clayton         }
193544d93782SGreg Clayton     }
193644d93782SGreg Clayton 
193744d93782SGreg Clayton     void
193844d93782SGreg Clayton     Menu::AddSubmenu (const MenuSP &menu_sp)
193944d93782SGreg Clayton     {
194044d93782SGreg Clayton         menu_sp->m_parent = this;
19413985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
194244d93782SGreg Clayton             m_max_submenu_name_length = menu_sp->m_name.size();
19433985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size())
194444d93782SGreg Clayton             m_max_submenu_key_name_length = menu_sp->m_key_name.size();
194544d93782SGreg Clayton         m_submenus.push_back(menu_sp);
194644d93782SGreg Clayton     }
194744d93782SGreg Clayton 
194844d93782SGreg Clayton     void
194944d93782SGreg Clayton     Menu::DrawMenuTitle (Window &window, bool highlight)
195044d93782SGreg Clayton     {
195144d93782SGreg Clayton         if (m_type == Type::Separator)
195244d93782SGreg Clayton         {
195344d93782SGreg Clayton             window.MoveCursor(0, window.GetCursorY());
195444d93782SGreg Clayton             window.PutChar(ACS_LTEE);
195544d93782SGreg Clayton             int width = window.GetWidth();
195644d93782SGreg Clayton             if (width > 2)
195744d93782SGreg Clayton             {
195844d93782SGreg Clayton                 width -= 2;
19593985c8c6SSaleem Abdulrasool                 for (int i=0; i< width; ++i)
196044d93782SGreg Clayton                     window.PutChar(ACS_HLINE);
196144d93782SGreg Clayton             }
196244d93782SGreg Clayton             window.PutChar(ACS_RTEE);
196344d93782SGreg Clayton         }
196444d93782SGreg Clayton         else
196544d93782SGreg Clayton         {
196644d93782SGreg Clayton             const int shortcut_key = m_key_value;
196744d93782SGreg Clayton             bool underlined_shortcut = false;
196844d93782SGreg Clayton             const attr_t hilgight_attr = A_REVERSE;
196944d93782SGreg Clayton             if (highlight)
197044d93782SGreg Clayton                 window.AttributeOn(hilgight_attr);
197144d93782SGreg Clayton             if (isprint(shortcut_key))
197244d93782SGreg Clayton             {
197344d93782SGreg Clayton                 size_t lower_pos = m_name.find(tolower(shortcut_key));
197444d93782SGreg Clayton                 size_t upper_pos = m_name.find(toupper(shortcut_key));
197544d93782SGreg Clayton                 const char *name = m_name.c_str();
197644d93782SGreg Clayton                 size_t pos = std::min<size_t>(lower_pos, upper_pos);
197744d93782SGreg Clayton                 if (pos != std::string::npos)
197844d93782SGreg Clayton                 {
197944d93782SGreg Clayton                     underlined_shortcut = true;
198044d93782SGreg Clayton                     if (pos > 0)
198144d93782SGreg Clayton                     {
198244d93782SGreg Clayton                         window.PutCString(name, pos);
198344d93782SGreg Clayton                         name += pos;
198444d93782SGreg Clayton                     }
198544d93782SGreg Clayton                     const attr_t shortcut_attr = A_UNDERLINE|A_BOLD;
198644d93782SGreg Clayton                     window.AttributeOn (shortcut_attr);
198744d93782SGreg Clayton                     window.PutChar(name[0]);
198844d93782SGreg Clayton                     window.AttributeOff(shortcut_attr);
198944d93782SGreg Clayton                     name++;
199044d93782SGreg Clayton                     if (name[0])
199144d93782SGreg Clayton                         window.PutCString(name);
199244d93782SGreg Clayton                 }
199344d93782SGreg Clayton             }
199444d93782SGreg Clayton 
199544d93782SGreg Clayton             if (!underlined_shortcut)
199644d93782SGreg Clayton             {
199744d93782SGreg Clayton                 window.PutCString(m_name.c_str());
199844d93782SGreg Clayton             }
199944d93782SGreg Clayton 
200044d93782SGreg Clayton             if (highlight)
200144d93782SGreg Clayton                 window.AttributeOff(hilgight_attr);
200244d93782SGreg Clayton 
200344d93782SGreg Clayton             if (m_key_name.empty())
200444d93782SGreg Clayton             {
200544d93782SGreg Clayton                 if (!underlined_shortcut && isprint(m_key_value))
200644d93782SGreg Clayton                 {
200744d93782SGreg Clayton                     window.AttributeOn (COLOR_PAIR(3));
200844d93782SGreg Clayton                     window.Printf (" (%c)", m_key_value);
200944d93782SGreg Clayton                     window.AttributeOff (COLOR_PAIR(3));
201044d93782SGreg Clayton                 }
201144d93782SGreg Clayton             }
201244d93782SGreg Clayton             else
201344d93782SGreg Clayton             {
201444d93782SGreg Clayton                 window.AttributeOn (COLOR_PAIR(3));
201544d93782SGreg Clayton                 window.Printf (" (%s)", m_key_name.c_str());
201644d93782SGreg Clayton                 window.AttributeOff (COLOR_PAIR(3));
201744d93782SGreg Clayton             }
201844d93782SGreg Clayton         }
201944d93782SGreg Clayton     }
202044d93782SGreg Clayton 
202144d93782SGreg Clayton     bool
202244d93782SGreg Clayton     Menu::WindowDelegateDraw (Window &window, bool force)
202344d93782SGreg Clayton     {
202444d93782SGreg Clayton         Menus &submenus = GetSubmenus();
202544d93782SGreg Clayton         const size_t num_submenus = submenus.size();
202644d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
202744d93782SGreg Clayton         Menu::Type menu_type = GetType ();
202844d93782SGreg Clayton         switch (menu_type)
202944d93782SGreg Clayton         {
203044d93782SGreg Clayton         case  Menu::Type::Bar:
203144d93782SGreg Clayton             {
203244d93782SGreg Clayton                 window.SetBackground(2);
203344d93782SGreg Clayton                 window.MoveCursor(0, 0);
203444d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
203544d93782SGreg Clayton                 {
203644d93782SGreg Clayton                     Menu *menu = submenus[i].get();
203744d93782SGreg Clayton                     if (i > 0)
203844d93782SGreg Clayton                         window.PutChar(' ');
203944d93782SGreg Clayton                     menu->SetStartingColumn (window.GetCursorX());
204044d93782SGreg Clayton                     window.PutCString("| ");
204144d93782SGreg Clayton                     menu->DrawMenuTitle (window, false);
204244d93782SGreg Clayton                 }
204344d93782SGreg Clayton                 window.PutCString(" |");
204444d93782SGreg Clayton                 window.DeferredRefresh();
204544d93782SGreg Clayton             }
204644d93782SGreg Clayton             break;
204744d93782SGreg Clayton 
204844d93782SGreg Clayton         case Menu::Type::Item:
204944d93782SGreg Clayton             {
205044d93782SGreg Clayton                 int y = 1;
205144d93782SGreg Clayton                 int x = 3;
205244d93782SGreg Clayton                 // Draw the menu
205344d93782SGreg Clayton                 int cursor_x = 0;
205444d93782SGreg Clayton                 int cursor_y = 0;
205544d93782SGreg Clayton                 window.Erase();
205644d93782SGreg Clayton                 window.SetBackground(2);
205744d93782SGreg Clayton                 window.Box();
205844d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
205944d93782SGreg Clayton                 {
20603985c8c6SSaleem Abdulrasool                     const bool is_selected =
20613985c8c6SSaleem Abdulrasool                       (i == static_cast<size_t>(selected_idx));
206244d93782SGreg Clayton                     window.MoveCursor(x, y + i);
206344d93782SGreg Clayton                     if (is_selected)
206444d93782SGreg Clayton                     {
206544d93782SGreg Clayton                         // Remember where we want the cursor to be
206644d93782SGreg Clayton                         cursor_x = x-1;
206744d93782SGreg Clayton                         cursor_y = y+i;
206844d93782SGreg Clayton                     }
206944d93782SGreg Clayton                     submenus[i]->DrawMenuTitle (window, is_selected);
207044d93782SGreg Clayton                 }
207144d93782SGreg Clayton                 window.MoveCursor(cursor_x, cursor_y);
207244d93782SGreg Clayton                 window.DeferredRefresh();
207344d93782SGreg Clayton             }
207444d93782SGreg Clayton             break;
207544d93782SGreg Clayton 
207644d93782SGreg Clayton         default:
207744d93782SGreg Clayton         case Menu::Type::Separator:
207844d93782SGreg Clayton             break;
207944d93782SGreg Clayton         }
208044d93782SGreg Clayton         return true; // Drawing handled...
208144d93782SGreg Clayton     }
208244d93782SGreg Clayton 
208344d93782SGreg Clayton     HandleCharResult
208444d93782SGreg Clayton     Menu::WindowDelegateHandleChar (Window &window, int key)
208544d93782SGreg Clayton     {
208644d93782SGreg Clayton         HandleCharResult result = eKeyNotHandled;
208744d93782SGreg Clayton 
208844d93782SGreg Clayton         Menus &submenus = GetSubmenus();
208944d93782SGreg Clayton         const size_t num_submenus = submenus.size();
209044d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
209144d93782SGreg Clayton         Menu::Type menu_type = GetType ();
209244d93782SGreg Clayton         if (menu_type == Menu::Type::Bar)
209344d93782SGreg Clayton         {
209444d93782SGreg Clayton             MenuSP run_menu_sp;
209544d93782SGreg Clayton             switch (key)
209644d93782SGreg Clayton             {
209744d93782SGreg Clayton                 case KEY_DOWN:
209844d93782SGreg Clayton                 case KEY_UP:
209944d93782SGreg Clayton                     // Show last menu or first menu
21003985c8c6SSaleem Abdulrasool                     if (selected_idx < static_cast<int>(num_submenus))
210144d93782SGreg Clayton                         run_menu_sp = submenus[selected_idx];
210244d93782SGreg Clayton                     else if (!submenus.empty())
210344d93782SGreg Clayton                         run_menu_sp = submenus.front();
210444d93782SGreg Clayton                     result = eKeyHandled;
210544d93782SGreg Clayton                     break;
210644d93782SGreg Clayton 
210744d93782SGreg Clayton                 case KEY_RIGHT:
210844d93782SGreg Clayton                 {
210944d93782SGreg Clayton                     ++m_selected;
21103985c8c6SSaleem Abdulrasool                     if (m_selected >= static_cast<int>(num_submenus))
211144d93782SGreg Clayton                         m_selected = 0;
21123985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
211344d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
211444d93782SGreg Clayton                     else if (!submenus.empty())
211544d93782SGreg Clayton                         run_menu_sp = submenus.front();
211644d93782SGreg Clayton                     result = eKeyHandled;
211744d93782SGreg Clayton                 }
211844d93782SGreg Clayton                     break;
211944d93782SGreg Clayton 
212044d93782SGreg Clayton                 case KEY_LEFT:
212144d93782SGreg Clayton                 {
212244d93782SGreg Clayton                     --m_selected;
212344d93782SGreg Clayton                     if (m_selected < 0)
212444d93782SGreg Clayton                         m_selected = num_submenus - 1;
21253985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
212644d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
212744d93782SGreg Clayton                     else if (!submenus.empty())
212844d93782SGreg Clayton                         run_menu_sp = submenus.front();
212944d93782SGreg Clayton                     result = eKeyHandled;
213044d93782SGreg Clayton                 }
213144d93782SGreg Clayton                     break;
213244d93782SGreg Clayton 
213344d93782SGreg Clayton                 default:
213444d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
213544d93782SGreg Clayton                     {
213644d93782SGreg Clayton                         if (submenus[i]->GetKeyValue() == key)
213744d93782SGreg Clayton                         {
213844d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
213944d93782SGreg Clayton                             run_menu_sp = submenus[i];
214044d93782SGreg Clayton                             result = eKeyHandled;
214144d93782SGreg Clayton                             break;
214244d93782SGreg Clayton                         }
214344d93782SGreg Clayton                     }
214444d93782SGreg Clayton                     break;
214544d93782SGreg Clayton             }
214644d93782SGreg Clayton 
214744d93782SGreg Clayton             if (run_menu_sp)
214844d93782SGreg Clayton             {
214944d93782SGreg Clayton                 // Run the action on this menu in case we need to populate the
215044d93782SGreg Clayton                 // menu with dynamic content and also in case check marks, and
215144d93782SGreg Clayton                 // any other menu decorations need to be caclulated
215244d93782SGreg Clayton                 if (run_menu_sp->Action() == MenuActionResult::Quit)
215344d93782SGreg Clayton                     return eQuitApplication;
215444d93782SGreg Clayton 
215544d93782SGreg Clayton                 Rect menu_bounds;
215644d93782SGreg Clayton                 menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
215744d93782SGreg Clayton                 menu_bounds.origin.y = 1;
215844d93782SGreg Clayton                 menu_bounds.size.width = run_menu_sp->GetDrawWidth();
215944d93782SGreg Clayton                 menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
216044d93782SGreg Clayton                 if (m_menu_window_sp)
216144d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
216244d93782SGreg Clayton 
216344d93782SGreg Clayton                 m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(),
216444d93782SGreg Clayton                                                                         menu_bounds,
216544d93782SGreg Clayton                                                                         true);
216644d93782SGreg Clayton                 m_menu_window_sp->SetDelegate (run_menu_sp);
216744d93782SGreg Clayton             }
216844d93782SGreg Clayton         }
216944d93782SGreg Clayton         else if (menu_type == Menu::Type::Item)
217044d93782SGreg Clayton         {
217144d93782SGreg Clayton             switch (key)
217244d93782SGreg Clayton             {
217344d93782SGreg Clayton                 case KEY_DOWN:
217444d93782SGreg Clayton                     if (m_submenus.size() > 1)
217544d93782SGreg Clayton                     {
217644d93782SGreg Clayton                         const int start_select = m_selected;
217744d93782SGreg Clayton                         while (++m_selected != start_select)
217844d93782SGreg Clayton                         {
21793985c8c6SSaleem Abdulrasool                             if (static_cast<size_t>(m_selected) >= num_submenus)
218044d93782SGreg Clayton                                 m_selected = 0;
218144d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
218244d93782SGreg Clayton                                 continue;
218344d93782SGreg Clayton                             else
218444d93782SGreg Clayton                                 break;
218544d93782SGreg Clayton                         }
218644d93782SGreg Clayton                         return eKeyHandled;
218744d93782SGreg Clayton                     }
218844d93782SGreg Clayton                     break;
218944d93782SGreg Clayton 
219044d93782SGreg Clayton                 case KEY_UP:
219144d93782SGreg Clayton                     if (m_submenus.size() > 1)
219244d93782SGreg Clayton                     {
219344d93782SGreg Clayton                         const int start_select = m_selected;
219444d93782SGreg Clayton                         while (--m_selected != start_select)
219544d93782SGreg Clayton                         {
21963985c8c6SSaleem Abdulrasool                             if (m_selected < static_cast<int>(0))
219744d93782SGreg Clayton                                 m_selected = num_submenus - 1;
219844d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
219944d93782SGreg Clayton                                 continue;
220044d93782SGreg Clayton                             else
220144d93782SGreg Clayton                                 break;
220244d93782SGreg Clayton                         }
220344d93782SGreg Clayton                         return eKeyHandled;
220444d93782SGreg Clayton                     }
220544d93782SGreg Clayton                     break;
220644d93782SGreg Clayton 
220744d93782SGreg Clayton                 case KEY_RETURN:
22083985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(selected_idx) < num_submenus)
220944d93782SGreg Clayton                     {
221044d93782SGreg Clayton                         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
221144d93782SGreg Clayton                             return eQuitApplication;
221244d93782SGreg Clayton                         window.GetParent()->RemoveSubWindow(&window);
221344d93782SGreg Clayton                         return eKeyHandled;
221444d93782SGreg Clayton                     }
221544d93782SGreg Clayton                     break;
221644d93782SGreg Clayton 
221744d93782SGreg Clayton                 case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences
221844d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(&window);
221944d93782SGreg Clayton                     return eKeyHandled;
222044d93782SGreg Clayton 
222144d93782SGreg Clayton                 default:
222244d93782SGreg Clayton                 {
222344d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
222444d93782SGreg Clayton                     {
222544d93782SGreg Clayton                         Menu *menu = submenus[i].get();
222644d93782SGreg Clayton                         if (menu->GetKeyValue() == key)
222744d93782SGreg Clayton                         {
222844d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
222944d93782SGreg Clayton                             window.GetParent()->RemoveSubWindow(&window);
223044d93782SGreg Clayton                             if (menu->Action() == MenuActionResult::Quit)
223144d93782SGreg Clayton                                 return eQuitApplication;
223244d93782SGreg Clayton                             return eKeyHandled;
223344d93782SGreg Clayton                         }
223444d93782SGreg Clayton                     }
223544d93782SGreg Clayton                 }
223644d93782SGreg Clayton                     break;
223744d93782SGreg Clayton 
223844d93782SGreg Clayton             }
223944d93782SGreg Clayton         }
224044d93782SGreg Clayton         else if (menu_type == Menu::Type::Separator)
224144d93782SGreg Clayton         {
224244d93782SGreg Clayton 
224344d93782SGreg Clayton         }
224444d93782SGreg Clayton         return result;
224544d93782SGreg Clayton     }
224644d93782SGreg Clayton 
224744d93782SGreg Clayton 
224844d93782SGreg Clayton     class Application
224944d93782SGreg Clayton     {
225044d93782SGreg Clayton     public:
225144d93782SGreg Clayton         Application (FILE *in, FILE *out) :
225244d93782SGreg Clayton             m_window_sp(),
225344d93782SGreg Clayton             m_screen (NULL),
225444d93782SGreg Clayton             m_in (in),
225544d93782SGreg Clayton             m_out (out)
225644d93782SGreg Clayton         {
225744d93782SGreg Clayton 
225844d93782SGreg Clayton         }
225944d93782SGreg Clayton 
226044d93782SGreg Clayton         ~Application ()
226144d93782SGreg Clayton         {
226244d93782SGreg Clayton             m_window_delegates.clear();
226344d93782SGreg Clayton             m_window_sp.reset();
226444d93782SGreg Clayton             if (m_screen)
226544d93782SGreg Clayton             {
226644d93782SGreg Clayton                 ::delscreen(m_screen);
226744d93782SGreg Clayton                 m_screen = NULL;
226844d93782SGreg Clayton             }
226944d93782SGreg Clayton         }
227044d93782SGreg Clayton 
227144d93782SGreg Clayton         void
227244d93782SGreg Clayton         Initialize ()
227344d93782SGreg Clayton         {
227444d93782SGreg Clayton             ::setlocale(LC_ALL, "");
227544d93782SGreg Clayton             ::setlocale(LC_CTYPE, "");
227644d93782SGreg Clayton #if 0
227744d93782SGreg Clayton             ::initscr();
227844d93782SGreg Clayton #else
227944d93782SGreg Clayton             m_screen = ::newterm(NULL, m_out, m_in);
228044d93782SGreg Clayton #endif
228144d93782SGreg Clayton             ::start_color();
228244d93782SGreg Clayton             ::curs_set(0);
228344d93782SGreg Clayton             ::noecho();
228444d93782SGreg Clayton             ::keypad(stdscr,TRUE);
228544d93782SGreg Clayton         }
228644d93782SGreg Clayton 
228744d93782SGreg Clayton         void
228844d93782SGreg Clayton         Terminate ()
228944d93782SGreg Clayton         {
229044d93782SGreg Clayton             ::endwin();
229144d93782SGreg Clayton         }
229244d93782SGreg Clayton 
229344d93782SGreg Clayton         void
229444d93782SGreg Clayton         Run (Debugger &debugger)
229544d93782SGreg Clayton         {
229644d93782SGreg Clayton             bool done = false;
229744d93782SGreg Clayton             int delay_in_tenths_of_a_second = 1;
229844d93782SGreg Clayton 
229944d93782SGreg Clayton             // Alas the threading model in curses is a bit lame so we need to
230044d93782SGreg Clayton             // resort to polling every 0.5 seconds. We could poll for stdin
230144d93782SGreg Clayton             // ourselves and then pass the keys down but then we need to
230244d93782SGreg Clayton             // translate all of the escape sequences ourselves. So we resort to
230344d93782SGreg Clayton             // polling for input because we need to receive async process events
230444d93782SGreg Clayton             // while in this loop.
230544d93782SGreg Clayton 
230644d93782SGreg Clayton             halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar()
230744d93782SGreg Clayton 
230844d93782SGreg Clayton             ListenerSP listener_sp (new Listener ("lldb.IOHandler.curses.Application"));
230944d93782SGreg Clayton             ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
231044d93782SGreg Clayton             ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
231144d93782SGreg Clayton             ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
231244d93782SGreg Clayton             debugger.EnableForwardEvents (listener_sp);
231344d93782SGreg Clayton 
231444d93782SGreg Clayton             bool update = true;
231544d93782SGreg Clayton #if defined(__APPLE__)
231644d93782SGreg Clayton             std::deque<int> escape_chars;
231744d93782SGreg Clayton #endif
231844d93782SGreg Clayton 
231944d93782SGreg Clayton             while (!done)
232044d93782SGreg Clayton             {
232144d93782SGreg Clayton                 if (update)
232244d93782SGreg Clayton                 {
232344d93782SGreg Clayton                     m_window_sp->Draw(false);
232444d93782SGreg Clayton                     // All windows should be calling Window::DeferredRefresh() instead
232544d93782SGreg Clayton                     // of Window::Refresh() so we can do a single update and avoid
232644d93782SGreg Clayton                     // any screen blinking
232744d93782SGreg Clayton                     update_panels();
232844d93782SGreg Clayton 
232944d93782SGreg Clayton                     // Cursor hiding isn't working on MacOSX, so hide it in the top left corner
233044d93782SGreg Clayton                     m_window_sp->MoveCursor(0, 0);
233144d93782SGreg Clayton 
233244d93782SGreg Clayton                     doupdate();
233344d93782SGreg Clayton                     update = false;
233444d93782SGreg Clayton                 }
233544d93782SGreg Clayton 
233644d93782SGreg Clayton #if defined(__APPLE__)
233744d93782SGreg Clayton                 // Terminal.app doesn't map its function keys correctly, F1-F4 default to:
233844d93782SGreg Clayton                 // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible
233944d93782SGreg Clayton                 int ch;
234044d93782SGreg Clayton                 if (escape_chars.empty())
234144d93782SGreg Clayton                     ch = m_window_sp->GetChar();
234244d93782SGreg Clayton                 else
234344d93782SGreg Clayton                 {
234444d93782SGreg Clayton                     ch = escape_chars.front();
234544d93782SGreg Clayton                     escape_chars.pop_front();
234644d93782SGreg Clayton                 }
234744d93782SGreg Clayton                 if (ch == KEY_ESCAPE)
234844d93782SGreg Clayton                 {
234944d93782SGreg Clayton                     int ch2 = m_window_sp->GetChar();
235044d93782SGreg Clayton                     if (ch2 == 'O')
235144d93782SGreg Clayton                     {
235244d93782SGreg Clayton                         int ch3 = m_window_sp->GetChar();
235344d93782SGreg Clayton                         switch (ch3)
235444d93782SGreg Clayton                         {
235544d93782SGreg Clayton                             case 'P': ch = KEY_F(1); break;
235644d93782SGreg Clayton                             case 'Q': ch = KEY_F(2); break;
235744d93782SGreg Clayton                             case 'R': ch = KEY_F(3); break;
235844d93782SGreg Clayton                             case 'S': ch = KEY_F(4); break;
235944d93782SGreg Clayton                             default:
236044d93782SGreg Clayton                                 escape_chars.push_back(ch2);
236144d93782SGreg Clayton                                 if (ch3 != -1)
236244d93782SGreg Clayton                                     escape_chars.push_back(ch3);
236344d93782SGreg Clayton                                 break;
236444d93782SGreg Clayton                         }
236544d93782SGreg Clayton                     }
236644d93782SGreg Clayton                     else if (ch2 != -1)
236744d93782SGreg Clayton                         escape_chars.push_back(ch2);
236844d93782SGreg Clayton                 }
236944d93782SGreg Clayton #else
237044d93782SGreg Clayton                 int ch = m_window_sp->GetChar();
237144d93782SGreg Clayton 
237244d93782SGreg Clayton #endif
237344d93782SGreg Clayton                 if (ch == -1)
237444d93782SGreg Clayton                 {
237544d93782SGreg Clayton                     if (feof(m_in) || ferror(m_in))
237644d93782SGreg Clayton                     {
237744d93782SGreg Clayton                         done = true;
237844d93782SGreg Clayton                     }
237944d93782SGreg Clayton                     else
238044d93782SGreg Clayton                     {
238144d93782SGreg Clayton                         // Just a timeout from using halfdelay(), check for events
238244d93782SGreg Clayton                         EventSP event_sp;
238344d93782SGreg Clayton                         while (listener_sp->PeekAtNextEvent())
238444d93782SGreg Clayton                         {
238544d93782SGreg Clayton                             listener_sp->GetNextEvent(event_sp);
238644d93782SGreg Clayton 
238744d93782SGreg Clayton                             if (event_sp)
238844d93782SGreg Clayton                             {
238944d93782SGreg Clayton                                 Broadcaster *broadcaster = event_sp->GetBroadcaster();
239044d93782SGreg Clayton                                 if (broadcaster)
239144d93782SGreg Clayton                                 {
239244d93782SGreg Clayton                                     //uint32_t event_type = event_sp->GetType();
239344d93782SGreg Clayton                                     ConstString broadcaster_class (broadcaster->GetBroadcasterClass());
239444d93782SGreg Clayton                                     if (broadcaster_class == broadcaster_class_process)
239544d93782SGreg Clayton                                     {
2396ec990867SGreg Clayton                                         debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
239744d93782SGreg Clayton                                         update = true;
239844d93782SGreg Clayton                                         continue; // Don't get any key, just update our view
239944d93782SGreg Clayton                                     }
240044d93782SGreg Clayton                                 }
240144d93782SGreg Clayton                             }
240244d93782SGreg Clayton                         }
240344d93782SGreg Clayton                     }
240444d93782SGreg Clayton                 }
240544d93782SGreg Clayton                 else
240644d93782SGreg Clayton                 {
240744d93782SGreg Clayton                     HandleCharResult key_result = m_window_sp->HandleChar(ch);
240844d93782SGreg Clayton                     switch (key_result)
240944d93782SGreg Clayton                     {
241044d93782SGreg Clayton                         case eKeyHandled:
2411ec990867SGreg Clayton                             debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
241244d93782SGreg Clayton                             update = true;
241344d93782SGreg Clayton                             break;
241444d93782SGreg Clayton                         case eKeyNotHandled:
241544d93782SGreg Clayton                             break;
241644d93782SGreg Clayton                         case eQuitApplication:
241744d93782SGreg Clayton                             done = true;
241844d93782SGreg Clayton                             break;
241944d93782SGreg Clayton                     }
242044d93782SGreg Clayton                 }
242144d93782SGreg Clayton             }
242244d93782SGreg Clayton 
242344d93782SGreg Clayton             debugger.CancelForwardEvents (listener_sp);
242444d93782SGreg Clayton 
242544d93782SGreg Clayton         }
242644d93782SGreg Clayton 
242744d93782SGreg Clayton         WindowSP &
242844d93782SGreg Clayton         GetMainWindow ()
242944d93782SGreg Clayton         {
243044d93782SGreg Clayton             if (!m_window_sp)
243144d93782SGreg Clayton                 m_window_sp.reset (new Window ("main", stdscr, false));
243244d93782SGreg Clayton             return m_window_sp;
243344d93782SGreg Clayton         }
243444d93782SGreg Clayton 
243544d93782SGreg Clayton         WindowDelegates &
243644d93782SGreg Clayton         GetWindowDelegates ()
243744d93782SGreg Clayton         {
243844d93782SGreg Clayton             return m_window_delegates;
243944d93782SGreg Clayton         }
244044d93782SGreg Clayton 
244144d93782SGreg Clayton     protected:
244244d93782SGreg Clayton         WindowSP m_window_sp;
244344d93782SGreg Clayton         WindowDelegates m_window_delegates;
244444d93782SGreg Clayton         SCREEN *m_screen;
244544d93782SGreg Clayton         FILE *m_in;
244644d93782SGreg Clayton         FILE *m_out;
244744d93782SGreg Clayton     };
244844d93782SGreg Clayton 
244944d93782SGreg Clayton 
245044d93782SGreg Clayton } // namespace curses
245144d93782SGreg Clayton 
245244d93782SGreg Clayton 
245344d93782SGreg Clayton using namespace curses;
245444d93782SGreg Clayton 
245544d93782SGreg Clayton struct Row
245644d93782SGreg Clayton {
245744d93782SGreg Clayton     ValueObjectSP valobj;
245844d93782SGreg Clayton     Row *parent;
245944d93782SGreg Clayton     int row_idx;
246044d93782SGreg Clayton     int x;
246144d93782SGreg Clayton     int y;
246244d93782SGreg Clayton     bool might_have_children;
246344d93782SGreg Clayton     bool expanded;
246444d93782SGreg Clayton     bool calculated_children;
246544d93782SGreg Clayton     std::vector<Row> children;
246644d93782SGreg Clayton 
246744d93782SGreg Clayton     Row (const ValueObjectSP &v, Row *p) :
246844d93782SGreg Clayton     valobj (v),
246944d93782SGreg Clayton     parent (p),
247044d93782SGreg Clayton     row_idx(0),
247144d93782SGreg Clayton     x(1),
247244d93782SGreg Clayton     y(1),
247344d93782SGreg Clayton     might_have_children (v ? v->MightHaveChildren() : false),
247444d93782SGreg Clayton     expanded (false),
247544d93782SGreg Clayton     calculated_children (false),
247644d93782SGreg Clayton     children()
247744d93782SGreg Clayton     {
247844d93782SGreg Clayton     }
247944d93782SGreg Clayton 
248044d93782SGreg Clayton     size_t
248144d93782SGreg Clayton     GetDepth () const
248244d93782SGreg Clayton     {
248344d93782SGreg Clayton         if (parent)
248444d93782SGreg Clayton             return 1 + parent->GetDepth();
248544d93782SGreg Clayton         return 0;
248644d93782SGreg Clayton     }
248744d93782SGreg Clayton 
248844d93782SGreg Clayton     void
248944d93782SGreg Clayton     Expand()
249044d93782SGreg Clayton     {
249144d93782SGreg Clayton         expanded = true;
249244d93782SGreg Clayton         if (!calculated_children)
249344d93782SGreg Clayton         {
249444d93782SGreg Clayton             calculated_children = true;
249544d93782SGreg Clayton             if (valobj)
249644d93782SGreg Clayton             {
249744d93782SGreg Clayton                 const size_t num_children = valobj->GetNumChildren();
249844d93782SGreg Clayton                 for (size_t i=0; i<num_children; ++i)
249944d93782SGreg Clayton                 {
250044d93782SGreg Clayton                     children.push_back(Row (valobj->GetChildAtIndex(i, true), this));
250144d93782SGreg Clayton                 }
250244d93782SGreg Clayton             }
250344d93782SGreg Clayton         }
250444d93782SGreg Clayton     }
250544d93782SGreg Clayton 
250644d93782SGreg Clayton     void
250744d93782SGreg Clayton     Unexpand ()
250844d93782SGreg Clayton     {
250944d93782SGreg Clayton         expanded = false;
251044d93782SGreg Clayton     }
251144d93782SGreg Clayton 
251244d93782SGreg Clayton     void
251344d93782SGreg Clayton     DrawTree (Window &window)
251444d93782SGreg Clayton     {
251544d93782SGreg Clayton         if (parent)
251644d93782SGreg Clayton             parent->DrawTreeForChild (window, this, 0);
251744d93782SGreg Clayton 
251844d93782SGreg Clayton         if (might_have_children)
251944d93782SGreg Clayton         {
252044d93782SGreg Clayton             // It we can get UTF8 characters to work we should try to use the "symbol"
252144d93782SGreg Clayton             // UTF8 string below
252244d93782SGreg Clayton //            const char *symbol = "";
252344d93782SGreg Clayton //            if (row.expanded)
252444d93782SGreg Clayton //                symbol = "\xe2\x96\xbd ";
252544d93782SGreg Clayton //            else
252644d93782SGreg Clayton //                symbol = "\xe2\x96\xb7 ";
252744d93782SGreg Clayton //            window.PutCString (symbol);
252844d93782SGreg Clayton 
252944d93782SGreg Clayton             // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
253044d93782SGreg Clayton             // 'v' or '>' character...
253144d93782SGreg Clayton //            if (expanded)
253244d93782SGreg Clayton //                window.PutChar (ACS_DARROW);
253344d93782SGreg Clayton //            else
253444d93782SGreg Clayton //                window.PutChar (ACS_RARROW);
253544d93782SGreg Clayton             // Since we can't find any good looking right arrow/down arrow
253644d93782SGreg Clayton             // symbols, just use a diamond...
253744d93782SGreg Clayton             window.PutChar (ACS_DIAMOND);
253844d93782SGreg Clayton             window.PutChar (ACS_HLINE);
253944d93782SGreg Clayton         }
254044d93782SGreg Clayton     }
254144d93782SGreg Clayton 
254244d93782SGreg Clayton     void
254344d93782SGreg Clayton     DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth)
254444d93782SGreg Clayton     {
254544d93782SGreg Clayton         if (parent)
254644d93782SGreg Clayton             parent->DrawTreeForChild (window, this, reverse_depth + 1);
254744d93782SGreg Clayton 
254844d93782SGreg Clayton         if (&children.back() == child)
254944d93782SGreg Clayton         {
255044d93782SGreg Clayton             // Last child
255144d93782SGreg Clayton             if (reverse_depth == 0)
255244d93782SGreg Clayton             {
255344d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
255444d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
255544d93782SGreg Clayton             }
255644d93782SGreg Clayton             else
255744d93782SGreg Clayton             {
255844d93782SGreg Clayton                 window.PutChar (' ');
255944d93782SGreg Clayton                 window.PutChar (' ');
256044d93782SGreg Clayton             }
256144d93782SGreg Clayton         }
256244d93782SGreg Clayton         else
256344d93782SGreg Clayton         {
256444d93782SGreg Clayton             if (reverse_depth == 0)
256544d93782SGreg Clayton             {
256644d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
256744d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
256844d93782SGreg Clayton             }
256944d93782SGreg Clayton             else
257044d93782SGreg Clayton             {
257144d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
257244d93782SGreg Clayton                 window.PutChar (' ');
257344d93782SGreg Clayton             }
257444d93782SGreg Clayton         }
257544d93782SGreg Clayton     }
257644d93782SGreg Clayton };
257744d93782SGreg Clayton 
257844d93782SGreg Clayton struct DisplayOptions
257944d93782SGreg Clayton {
258044d93782SGreg Clayton     bool show_types;
258144d93782SGreg Clayton };
258244d93782SGreg Clayton 
258344d93782SGreg Clayton class TreeItem;
258444d93782SGreg Clayton 
258544d93782SGreg Clayton class TreeDelegate
258644d93782SGreg Clayton {
258744d93782SGreg Clayton public:
258844d93782SGreg Clayton     TreeDelegate() {}
258944d93782SGreg Clayton     virtual ~TreeDelegate() {}
259044d93782SGreg Clayton     virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0;
259144d93782SGreg Clayton     virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0;
259244d93782SGreg Clayton     virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views
259344d93782SGreg Clayton };
259444d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
259544d93782SGreg Clayton 
259644d93782SGreg Clayton class TreeItem
259744d93782SGreg Clayton {
259844d93782SGreg Clayton public:
259944d93782SGreg Clayton 
260044d93782SGreg Clayton     TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) :
260144d93782SGreg Clayton         m_parent (parent),
260244d93782SGreg Clayton         m_delegate (delegate),
2603ec990867SGreg Clayton         m_user_data (NULL),
260444d93782SGreg Clayton         m_identifier (0),
260544d93782SGreg Clayton         m_row_idx (-1),
260644d93782SGreg Clayton         m_children (),
260744d93782SGreg Clayton         m_might_have_children (might_have_children),
260844d93782SGreg Clayton         m_is_expanded (false)
260944d93782SGreg Clayton     {
261044d93782SGreg Clayton     }
261144d93782SGreg Clayton 
261244d93782SGreg Clayton     TreeItem &
261344d93782SGreg Clayton     operator=(const TreeItem &rhs)
261444d93782SGreg Clayton     {
261544d93782SGreg Clayton         if (this != &rhs)
261644d93782SGreg Clayton         {
261744d93782SGreg Clayton             m_parent = rhs.m_parent;
261844d93782SGreg Clayton             m_delegate = rhs.m_delegate;
2619ec990867SGreg Clayton             m_user_data = rhs.m_user_data;
262044d93782SGreg Clayton             m_identifier = rhs.m_identifier;
262144d93782SGreg Clayton             m_row_idx = rhs.m_row_idx;
262244d93782SGreg Clayton             m_children = rhs.m_children;
262344d93782SGreg Clayton             m_might_have_children = rhs.m_might_have_children;
262444d93782SGreg Clayton             m_is_expanded = rhs.m_is_expanded;
262544d93782SGreg Clayton         }
262644d93782SGreg Clayton         return *this;
262744d93782SGreg Clayton     }
262844d93782SGreg Clayton 
262944d93782SGreg Clayton     size_t
263044d93782SGreg Clayton     GetDepth () const
263144d93782SGreg Clayton     {
263244d93782SGreg Clayton         if (m_parent)
263344d93782SGreg Clayton             return 1 + m_parent->GetDepth();
263444d93782SGreg Clayton         return 0;
263544d93782SGreg Clayton     }
263644d93782SGreg Clayton 
263744d93782SGreg Clayton     int
263844d93782SGreg Clayton     GetRowIndex () const
263944d93782SGreg Clayton     {
264044d93782SGreg Clayton         return m_row_idx;
264144d93782SGreg Clayton     }
264244d93782SGreg Clayton 
264344d93782SGreg Clayton     void
264444d93782SGreg Clayton     ClearChildren ()
264544d93782SGreg Clayton     {
264644d93782SGreg Clayton         m_children.clear();
264744d93782SGreg Clayton     }
264844d93782SGreg Clayton 
264944d93782SGreg Clayton     void
265044d93782SGreg Clayton     Resize (size_t n, const TreeItem &t)
265144d93782SGreg Clayton     {
265244d93782SGreg Clayton         m_children.resize(n, t);
265344d93782SGreg Clayton     }
265444d93782SGreg Clayton 
265544d93782SGreg Clayton     TreeItem &
265644d93782SGreg Clayton     operator [](size_t i)
265744d93782SGreg Clayton     {
265844d93782SGreg Clayton         return m_children[i];
265944d93782SGreg Clayton     }
266044d93782SGreg Clayton 
266144d93782SGreg Clayton     void
266244d93782SGreg Clayton     SetRowIndex (int row_idx)
266344d93782SGreg Clayton     {
266444d93782SGreg Clayton         m_row_idx = row_idx;
266544d93782SGreg Clayton     }
266644d93782SGreg Clayton 
266744d93782SGreg Clayton     size_t
266844d93782SGreg Clayton     GetNumChildren ()
266944d93782SGreg Clayton     {
267044d93782SGreg Clayton         m_delegate.TreeDelegateGenerateChildren (*this);
267144d93782SGreg Clayton         return m_children.size();
267244d93782SGreg Clayton     }
267344d93782SGreg Clayton 
267444d93782SGreg Clayton     void
267544d93782SGreg Clayton     ItemWasSelected ()
267644d93782SGreg Clayton     {
267744d93782SGreg Clayton         m_delegate.TreeDelegateItemSelected(*this);
267844d93782SGreg Clayton     }
267944d93782SGreg Clayton     void
268044d93782SGreg Clayton     CalculateRowIndexes (int &row_idx)
268144d93782SGreg Clayton     {
268244d93782SGreg Clayton         SetRowIndex(row_idx);
268344d93782SGreg Clayton         ++row_idx;
268444d93782SGreg Clayton 
2685ec990867SGreg Clayton         const bool expanded = IsExpanded();
2686ec990867SGreg Clayton 
2687ec990867SGreg Clayton         // The root item must calculate its children,
2688ec990867SGreg Clayton         // or we must calculate the number of children
2689ec990867SGreg Clayton         // if the item is expanded
2690ec990867SGreg Clayton         if (m_parent == NULL || expanded)
269144d93782SGreg Clayton             GetNumChildren();
269244d93782SGreg Clayton 
269344d93782SGreg Clayton         for (auto &item : m_children)
269444d93782SGreg Clayton         {
269544d93782SGreg Clayton             if (expanded)
269644d93782SGreg Clayton                 item.CalculateRowIndexes(row_idx);
269744d93782SGreg Clayton             else
269844d93782SGreg Clayton                 item.SetRowIndex(-1);
269944d93782SGreg Clayton         }
270044d93782SGreg Clayton     }
270144d93782SGreg Clayton 
270244d93782SGreg Clayton     TreeItem *
270344d93782SGreg Clayton     GetParent ()
270444d93782SGreg Clayton     {
270544d93782SGreg Clayton         return m_parent;
270644d93782SGreg Clayton     }
270744d93782SGreg Clayton 
270844d93782SGreg Clayton     bool
270944d93782SGreg Clayton     IsExpanded () const
271044d93782SGreg Clayton     {
271144d93782SGreg Clayton         return m_is_expanded;
271244d93782SGreg Clayton     }
271344d93782SGreg Clayton 
271444d93782SGreg Clayton     void
271544d93782SGreg Clayton     Expand()
271644d93782SGreg Clayton     {
271744d93782SGreg Clayton         m_is_expanded = true;
271844d93782SGreg Clayton     }
271944d93782SGreg Clayton 
272044d93782SGreg Clayton     void
272144d93782SGreg Clayton     Unexpand ()
272244d93782SGreg Clayton     {
272344d93782SGreg Clayton         m_is_expanded = false;
272444d93782SGreg Clayton     }
272544d93782SGreg Clayton 
272644d93782SGreg Clayton     bool
272744d93782SGreg Clayton     Draw (Window &window,
272844d93782SGreg Clayton           const int first_visible_row,
272944d93782SGreg Clayton           const uint32_t selected_row_idx,
273044d93782SGreg Clayton           int &row_idx,
273144d93782SGreg Clayton           int &num_rows_left)
273244d93782SGreg Clayton     {
273344d93782SGreg Clayton         if (num_rows_left <= 0)
273444d93782SGreg Clayton             return false;
273544d93782SGreg Clayton 
273644d93782SGreg Clayton         if (m_row_idx >= first_visible_row)
273744d93782SGreg Clayton         {
273844d93782SGreg Clayton             window.MoveCursor(2, row_idx + 1);
273944d93782SGreg Clayton 
274044d93782SGreg Clayton             if (m_parent)
274144d93782SGreg Clayton                 m_parent->DrawTreeForChild (window, this, 0);
274244d93782SGreg Clayton 
274344d93782SGreg Clayton             if (m_might_have_children)
274444d93782SGreg Clayton             {
274544d93782SGreg Clayton                 // It we can get UTF8 characters to work we should try to use the "symbol"
274644d93782SGreg Clayton                 // UTF8 string below
274744d93782SGreg Clayton                 //            const char *symbol = "";
274844d93782SGreg Clayton                 //            if (row.expanded)
274944d93782SGreg Clayton                 //                symbol = "\xe2\x96\xbd ";
275044d93782SGreg Clayton                 //            else
275144d93782SGreg Clayton                 //                symbol = "\xe2\x96\xb7 ";
275244d93782SGreg Clayton                 //            window.PutCString (symbol);
275344d93782SGreg Clayton 
275444d93782SGreg Clayton                 // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
275544d93782SGreg Clayton                 // 'v' or '>' character...
275644d93782SGreg Clayton                 //            if (expanded)
275744d93782SGreg Clayton                 //                window.PutChar (ACS_DARROW);
275844d93782SGreg Clayton                 //            else
275944d93782SGreg Clayton                 //                window.PutChar (ACS_RARROW);
276044d93782SGreg Clayton                 // Since we can't find any good looking right arrow/down arrow
276144d93782SGreg Clayton                 // symbols, just use a diamond...
276244d93782SGreg Clayton                 window.PutChar (ACS_DIAMOND);
276344d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
276444d93782SGreg Clayton             }
27653985c8c6SSaleem Abdulrasool             bool highlight =
27663985c8c6SSaleem Abdulrasool               (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive();
276744d93782SGreg Clayton 
276844d93782SGreg Clayton             if (highlight)
276944d93782SGreg Clayton                 window.AttributeOn(A_REVERSE);
277044d93782SGreg Clayton 
277144d93782SGreg Clayton             m_delegate.TreeDelegateDrawTreeItem(*this, window);
277244d93782SGreg Clayton 
277344d93782SGreg Clayton             if (highlight)
277444d93782SGreg Clayton                 window.AttributeOff(A_REVERSE);
277544d93782SGreg Clayton             ++row_idx;
277644d93782SGreg Clayton             --num_rows_left;
277744d93782SGreg Clayton         }
277844d93782SGreg Clayton 
277944d93782SGreg Clayton         if (num_rows_left <= 0)
278044d93782SGreg Clayton             return false; // We are done drawing...
278144d93782SGreg Clayton 
278244d93782SGreg Clayton         if (IsExpanded())
278344d93782SGreg Clayton         {
278444d93782SGreg Clayton             for (auto &item : m_children)
278544d93782SGreg Clayton             {
278644d93782SGreg Clayton                 // If we displayed all the rows and item.Draw() returns
278744d93782SGreg Clayton                 // false we are done drawing and can exit this for loop
278844d93782SGreg Clayton                 if (item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left) == false)
278944d93782SGreg Clayton                     break;
279044d93782SGreg Clayton             }
279144d93782SGreg Clayton         }
279244d93782SGreg Clayton         return num_rows_left >= 0; // Return true if not done drawing yet
279344d93782SGreg Clayton     }
279444d93782SGreg Clayton 
279544d93782SGreg Clayton     void
279644d93782SGreg Clayton     DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth)
279744d93782SGreg Clayton     {
279844d93782SGreg Clayton         if (m_parent)
279944d93782SGreg Clayton             m_parent->DrawTreeForChild (window, this, reverse_depth + 1);
280044d93782SGreg Clayton 
280144d93782SGreg Clayton         if (&m_children.back() == child)
280244d93782SGreg Clayton         {
280344d93782SGreg Clayton             // Last child
280444d93782SGreg Clayton             if (reverse_depth == 0)
280544d93782SGreg Clayton             {
280644d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
280744d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
280844d93782SGreg Clayton             }
280944d93782SGreg Clayton             else
281044d93782SGreg Clayton             {
281144d93782SGreg Clayton                 window.PutChar (' ');
281244d93782SGreg Clayton                 window.PutChar (' ');
281344d93782SGreg Clayton             }
281444d93782SGreg Clayton         }
281544d93782SGreg Clayton         else
281644d93782SGreg Clayton         {
281744d93782SGreg Clayton             if (reverse_depth == 0)
281844d93782SGreg Clayton             {
281944d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
282044d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
282144d93782SGreg Clayton             }
282244d93782SGreg Clayton             else
282344d93782SGreg Clayton             {
282444d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
282544d93782SGreg Clayton                 window.PutChar (' ');
282644d93782SGreg Clayton             }
282744d93782SGreg Clayton         }
282844d93782SGreg Clayton     }
282944d93782SGreg Clayton 
283044d93782SGreg Clayton     TreeItem *
283144d93782SGreg Clayton     GetItemForRowIndex (uint32_t row_idx)
283244d93782SGreg Clayton     {
28333985c8c6SSaleem Abdulrasool         if (static_cast<uint32_t>(m_row_idx) == row_idx)
283444d93782SGreg Clayton             return this;
283544d93782SGreg Clayton         if (m_children.empty())
283644d93782SGreg Clayton             return NULL;
28373985c8c6SSaleem Abdulrasool         if (static_cast<uint32_t>(m_children.back().m_row_idx) < row_idx)
283844d93782SGreg Clayton             return NULL;
283944d93782SGreg Clayton         if (IsExpanded())
284044d93782SGreg Clayton         {
284144d93782SGreg Clayton             for (auto &item : m_children)
284244d93782SGreg Clayton             {
284344d93782SGreg Clayton                 TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
284444d93782SGreg Clayton                 if (selected_item_ptr)
284544d93782SGreg Clayton                     return selected_item_ptr;
284644d93782SGreg Clayton             }
284744d93782SGreg Clayton         }
284844d93782SGreg Clayton         return NULL;
284944d93782SGreg Clayton     }
285044d93782SGreg Clayton 
2851ec990867SGreg Clayton     void *
2852ec990867SGreg Clayton     GetUserData() const
2853ec990867SGreg Clayton     {
2854ec990867SGreg Clayton         return m_user_data;
2855ec990867SGreg Clayton     }
2856ec990867SGreg Clayton 
2857ec990867SGreg Clayton     void
2858ec990867SGreg Clayton     SetUserData (void *user_data)
2859ec990867SGreg Clayton     {
2860ec990867SGreg Clayton         m_user_data = user_data;
2861ec990867SGreg Clayton     }
2862ec990867SGreg Clayton 
286344d93782SGreg Clayton     uint64_t
286444d93782SGreg Clayton     GetIdentifier() const
286544d93782SGreg Clayton     {
286644d93782SGreg Clayton         return m_identifier;
286744d93782SGreg Clayton     }
286844d93782SGreg Clayton 
286944d93782SGreg Clayton     void
287044d93782SGreg Clayton     SetIdentifier (uint64_t identifier)
287144d93782SGreg Clayton     {
287244d93782SGreg Clayton         m_identifier = identifier;
287344d93782SGreg Clayton     }
287444d93782SGreg Clayton 
287544d93782SGreg Clayton 
2876ec990867SGreg Clayton     void
2877ec990867SGreg Clayton     SetMightHaveChildren (bool b)
2878ec990867SGreg Clayton     {
2879ec990867SGreg Clayton         m_might_have_children = b;
2880ec990867SGreg Clayton     }
2881ec990867SGreg Clayton 
288244d93782SGreg Clayton protected:
288344d93782SGreg Clayton     TreeItem *m_parent;
288444d93782SGreg Clayton     TreeDelegate &m_delegate;
2885ec990867SGreg Clayton     void *m_user_data;
288644d93782SGreg Clayton     uint64_t m_identifier;
288744d93782SGreg Clayton     int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item
288844d93782SGreg Clayton     std::vector<TreeItem> m_children;
288944d93782SGreg Clayton     bool m_might_have_children;
289044d93782SGreg Clayton     bool m_is_expanded;
289144d93782SGreg Clayton 
289244d93782SGreg Clayton };
289344d93782SGreg Clayton 
289444d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate
289544d93782SGreg Clayton {
289644d93782SGreg Clayton public:
289744d93782SGreg Clayton     TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) :
289844d93782SGreg Clayton         m_debugger (debugger),
289944d93782SGreg Clayton         m_delegate_sp (delegate_sp),
290044d93782SGreg Clayton         m_root (NULL, *delegate_sp, true),
290144d93782SGreg Clayton         m_selected_item (NULL),
290244d93782SGreg Clayton         m_num_rows (0),
290344d93782SGreg Clayton         m_selected_row_idx (0),
290444d93782SGreg Clayton         m_first_visible_row (0),
290544d93782SGreg Clayton         m_min_x (0),
290644d93782SGreg Clayton         m_min_y (0),
290744d93782SGreg Clayton         m_max_x (0),
290844d93782SGreg Clayton         m_max_y (0)
290944d93782SGreg Clayton     {
291044d93782SGreg Clayton     }
291144d93782SGreg Clayton 
291244d93782SGreg Clayton     int
291344d93782SGreg Clayton     NumVisibleRows () const
291444d93782SGreg Clayton     {
291544d93782SGreg Clayton         return m_max_y - m_min_y;
291644d93782SGreg Clayton     }
291744d93782SGreg Clayton 
291844d93782SGreg Clayton     virtual bool
291944d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
292044d93782SGreg Clayton     {
292144d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
292244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
292344d93782SGreg Clayton 
292444d93782SGreg Clayton         bool display_content = false;
292544d93782SGreg Clayton         if (process)
292644d93782SGreg Clayton         {
292744d93782SGreg Clayton             StateType state = process->GetState();
292844d93782SGreg Clayton             if (StateIsStoppedState(state, true))
292944d93782SGreg Clayton             {
293044d93782SGreg Clayton                 // We are stopped, so it is ok to
293144d93782SGreg Clayton                 display_content = true;
293244d93782SGreg Clayton             }
293344d93782SGreg Clayton             else if (StateIsRunningState(state))
293444d93782SGreg Clayton             {
293544d93782SGreg Clayton                 return true; // Don't do any updating when we are running
293644d93782SGreg Clayton             }
293744d93782SGreg Clayton         }
293844d93782SGreg Clayton 
293944d93782SGreg Clayton         m_min_x = 2;
294044d93782SGreg Clayton         m_min_y = 1;
294144d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
294244d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
294344d93782SGreg Clayton 
294444d93782SGreg Clayton         window.Erase();
294544d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
294644d93782SGreg Clayton 
294744d93782SGreg Clayton         if (display_content)
294844d93782SGreg Clayton         {
294944d93782SGreg Clayton             const int num_visible_rows = NumVisibleRows();
295044d93782SGreg Clayton             m_num_rows = 0;
295144d93782SGreg Clayton             m_root.CalculateRowIndexes(m_num_rows);
295244d93782SGreg Clayton 
295344d93782SGreg Clayton             // If we unexpanded while having something selected our
295444d93782SGreg Clayton             // total number of rows is less than the num visible rows,
295544d93782SGreg Clayton             // then make sure we show all the rows by setting the first
295644d93782SGreg Clayton             // visible row accordingly.
295744d93782SGreg Clayton             if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
295844d93782SGreg Clayton                 m_first_visible_row = 0;
295944d93782SGreg Clayton 
296044d93782SGreg Clayton             // Make sure the selected row is always visible
296144d93782SGreg Clayton             if (m_selected_row_idx < m_first_visible_row)
296244d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx;
296344d93782SGreg Clayton             else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
296444d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
296544d93782SGreg Clayton 
296644d93782SGreg Clayton             int row_idx = 0;
296744d93782SGreg Clayton             int num_rows_left = num_visible_rows;
296844d93782SGreg Clayton             m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left);
296944d93782SGreg Clayton             // Get the selected row
297044d93782SGreg Clayton             m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx);
297144d93782SGreg Clayton         }
297244d93782SGreg Clayton         else
297344d93782SGreg Clayton         {
297444d93782SGreg Clayton             m_selected_item = NULL;
297544d93782SGreg Clayton         }
297644d93782SGreg Clayton 
297744d93782SGreg Clayton         window.DeferredRefresh();
297844d93782SGreg Clayton 
297944d93782SGreg Clayton 
298044d93782SGreg Clayton         return true; // Drawing handled
298144d93782SGreg Clayton     }
298244d93782SGreg Clayton 
298344d93782SGreg Clayton 
298444d93782SGreg Clayton     virtual const char *
298544d93782SGreg Clayton     WindowDelegateGetHelpText ()
298644d93782SGreg Clayton     {
298744d93782SGreg Clayton         return "Thread window keyboard shortcuts:";
298844d93782SGreg Clayton     }
298944d93782SGreg Clayton 
299044d93782SGreg Clayton     virtual KeyHelp *
299144d93782SGreg Clayton     WindowDelegateGetKeyHelp ()
299244d93782SGreg Clayton     {
299344d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
299444d93782SGreg Clayton             { KEY_UP, "Select previous item" },
299544d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
299644d93782SGreg Clayton             { KEY_RIGHT, "Expand the selected item" },
299744d93782SGreg Clayton             { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" },
299844d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
299944d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
300044d93782SGreg Clayton             { 'h', "Show help dialog" },
300144d93782SGreg Clayton             { ' ', "Toggle item expansion" },
300244d93782SGreg Clayton             { ',', "Page up" },
300344d93782SGreg Clayton             { '.', "Page down" },
300444d93782SGreg Clayton             { '\0', NULL }
300544d93782SGreg Clayton         };
300644d93782SGreg Clayton         return g_source_view_key_help;
300744d93782SGreg Clayton     }
300844d93782SGreg Clayton 
300944d93782SGreg Clayton     virtual HandleCharResult
301044d93782SGreg Clayton     WindowDelegateHandleChar (Window &window, int c)
301144d93782SGreg Clayton     {
301244d93782SGreg Clayton         switch(c)
301344d93782SGreg Clayton         {
301444d93782SGreg Clayton             case ',':
301544d93782SGreg Clayton             case KEY_PPAGE:
301644d93782SGreg Clayton                 // Page up key
301744d93782SGreg Clayton                 if (m_first_visible_row > 0)
301844d93782SGreg Clayton                 {
301944d93782SGreg Clayton                     if (m_first_visible_row > m_max_y)
302044d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
302144d93782SGreg Clayton                     else
302244d93782SGreg Clayton                         m_first_visible_row = 0;
302344d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
302444d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
302544d93782SGreg Clayton                     if (m_selected_item)
302644d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
302744d93782SGreg Clayton                 }
302844d93782SGreg Clayton                 return eKeyHandled;
302944d93782SGreg Clayton 
303044d93782SGreg Clayton             case '.':
303144d93782SGreg Clayton             case KEY_NPAGE:
303244d93782SGreg Clayton                 // Page down key
303344d93782SGreg Clayton                 if (m_num_rows > m_max_y)
303444d93782SGreg Clayton                 {
303544d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
303644d93782SGreg Clayton                     {
303744d93782SGreg Clayton                         m_first_visible_row += m_max_y;
303844d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
303944d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
304044d93782SGreg Clayton                         if (m_selected_item)
304144d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
304244d93782SGreg Clayton                     }
304344d93782SGreg Clayton                 }
304444d93782SGreg Clayton                 return eKeyHandled;
304544d93782SGreg Clayton 
304644d93782SGreg Clayton             case KEY_UP:
304744d93782SGreg Clayton                 if (m_selected_row_idx > 0)
304844d93782SGreg Clayton                 {
304944d93782SGreg Clayton                     --m_selected_row_idx;
305044d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
305144d93782SGreg Clayton                     if (m_selected_item)
305244d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
305344d93782SGreg Clayton                 }
305444d93782SGreg Clayton                 return eKeyHandled;
305544d93782SGreg Clayton             case KEY_DOWN:
305644d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
305744d93782SGreg Clayton                 {
305844d93782SGreg Clayton                     ++m_selected_row_idx;
305944d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
306044d93782SGreg Clayton                     if (m_selected_item)
306144d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
306244d93782SGreg Clayton                 }
306344d93782SGreg Clayton                 return eKeyHandled;
306444d93782SGreg Clayton 
306544d93782SGreg Clayton             case KEY_RIGHT:
306644d93782SGreg Clayton                 if (m_selected_item)
306744d93782SGreg Clayton                 {
306844d93782SGreg Clayton                     if (!m_selected_item->IsExpanded())
306944d93782SGreg Clayton                         m_selected_item->Expand();
307044d93782SGreg Clayton                 }
307144d93782SGreg Clayton                 return eKeyHandled;
307244d93782SGreg Clayton 
307344d93782SGreg Clayton             case KEY_LEFT:
307444d93782SGreg Clayton                 if (m_selected_item)
307544d93782SGreg Clayton                 {
307644d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
307744d93782SGreg Clayton                         m_selected_item->Unexpand();
307844d93782SGreg Clayton                     else if (m_selected_item->GetParent())
307944d93782SGreg Clayton                     {
308044d93782SGreg Clayton                         m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
308144d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
308244d93782SGreg Clayton                         if (m_selected_item)
308344d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
308444d93782SGreg Clayton                     }
308544d93782SGreg Clayton                 }
308644d93782SGreg Clayton                 return eKeyHandled;
308744d93782SGreg Clayton 
308844d93782SGreg Clayton             case ' ':
308944d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
309044d93782SGreg Clayton                 if (m_selected_item)
309144d93782SGreg Clayton                 {
309244d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
309344d93782SGreg Clayton                         m_selected_item->Unexpand();
309444d93782SGreg Clayton                     else
309544d93782SGreg Clayton                         m_selected_item->Expand();
309644d93782SGreg Clayton                 }
309744d93782SGreg Clayton                 return eKeyHandled;
309844d93782SGreg Clayton 
309944d93782SGreg Clayton             case 'h':
310044d93782SGreg Clayton                 window.CreateHelpSubwindow ();
310144d93782SGreg Clayton                 return eKeyHandled;
310244d93782SGreg Clayton 
310344d93782SGreg Clayton             default:
310444d93782SGreg Clayton                 break;
310544d93782SGreg Clayton         }
310644d93782SGreg Clayton         return eKeyNotHandled;
310744d93782SGreg Clayton     }
310844d93782SGreg Clayton 
310944d93782SGreg Clayton protected:
311044d93782SGreg Clayton     Debugger &m_debugger;
311144d93782SGreg Clayton     TreeDelegateSP m_delegate_sp;
311244d93782SGreg Clayton     TreeItem m_root;
311344d93782SGreg Clayton     TreeItem *m_selected_item;
311444d93782SGreg Clayton     int m_num_rows;
311544d93782SGreg Clayton     int m_selected_row_idx;
311644d93782SGreg Clayton     int m_first_visible_row;
311744d93782SGreg Clayton     int m_min_x;
311844d93782SGreg Clayton     int m_min_y;
311944d93782SGreg Clayton     int m_max_x;
312044d93782SGreg Clayton     int m_max_y;
312144d93782SGreg Clayton 
312244d93782SGreg Clayton };
312344d93782SGreg Clayton 
312444d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate
312544d93782SGreg Clayton {
312644d93782SGreg Clayton public:
3127ec990867SGreg Clayton     FrameTreeDelegate () :
3128ec990867SGreg Clayton         TreeDelegate()
312944d93782SGreg Clayton     {
313044d93782SGreg Clayton     }
313144d93782SGreg Clayton 
313244d93782SGreg Clayton     virtual ~FrameTreeDelegate()
313344d93782SGreg Clayton     {
313444d93782SGreg Clayton     }
313544d93782SGreg Clayton 
313644d93782SGreg Clayton     virtual void
313744d93782SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window)
313844d93782SGreg Clayton     {
3139ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3140ec990867SGreg Clayton         if (thread)
314144d93782SGreg Clayton         {
314244d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3143ec990867SGreg Clayton             StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
314444d93782SGreg Clayton             if (frame_sp)
314544d93782SGreg Clayton             {
314644d93782SGreg Clayton                 StreamString strm;
314744d93782SGreg Clayton                 const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
314844d93782SGreg Clayton                 ExecutionContext exe_ctx (frame_sp);
314944d93782SGreg Clayton                 //const char *frame_format = "frame #${frame.index}: ${module.file.basename}{`${function.name}${function.pc-offset}}}";
315044d93782SGreg Clayton                 const char *frame_format = "frame #${frame.index}: {${function.name}${function.pc-offset}}}";
315144d93782SGreg Clayton                 if (Debugger::FormatPrompt (frame_format, &sc, &exe_ctx, NULL, strm))
315244d93782SGreg Clayton                 {
315344d93782SGreg Clayton                     int right_pad = 1;
315444d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
315544d93782SGreg Clayton                 }
315644d93782SGreg Clayton             }
315744d93782SGreg Clayton         }
315844d93782SGreg Clayton     }
315944d93782SGreg Clayton     virtual void
316044d93782SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)
316144d93782SGreg Clayton     {
316244d93782SGreg Clayton         // No children for frames yet...
316344d93782SGreg Clayton     }
316444d93782SGreg Clayton 
316544d93782SGreg Clayton     virtual bool
316644d93782SGreg Clayton     TreeDelegateItemSelected (TreeItem &item)
316744d93782SGreg Clayton     {
3168ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3169ec990867SGreg Clayton         if (thread)
317044d93782SGreg Clayton         {
3171ec990867SGreg Clayton             thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID());
317244d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3173ec990867SGreg Clayton             thread->SetSelectedFrameByIndex(frame_idx);
317444d93782SGreg Clayton             return true;
317544d93782SGreg Clayton         }
317644d93782SGreg Clayton         return false;
317744d93782SGreg Clayton     }
317844d93782SGreg Clayton };
317944d93782SGreg Clayton 
318044d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate
318144d93782SGreg Clayton {
318244d93782SGreg Clayton public:
318344d93782SGreg Clayton     ThreadTreeDelegate (Debugger &debugger) :
318444d93782SGreg Clayton         TreeDelegate(),
318544d93782SGreg Clayton         m_debugger (debugger),
318644d93782SGreg Clayton         m_tid (LLDB_INVALID_THREAD_ID),
318744d93782SGreg Clayton         m_stop_id (UINT32_MAX)
318844d93782SGreg Clayton     {
318944d93782SGreg Clayton     }
319044d93782SGreg Clayton 
319144d93782SGreg Clayton     virtual
319244d93782SGreg Clayton     ~ThreadTreeDelegate()
319344d93782SGreg Clayton     {
319444d93782SGreg Clayton     }
319544d93782SGreg Clayton 
3196ec990867SGreg Clayton     ProcessSP
3197ec990867SGreg Clayton     GetProcess ()
3198ec990867SGreg Clayton     {
3199ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3200ec990867SGreg Clayton     }
3201ec990867SGreg Clayton 
3202ec990867SGreg Clayton     ThreadSP
3203ec990867SGreg Clayton     GetThread (const TreeItem &item)
3204ec990867SGreg Clayton     {
3205ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3206ec990867SGreg Clayton         if (process_sp)
3207ec990867SGreg Clayton             return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
3208ec990867SGreg Clayton         return ThreadSP();
3209ec990867SGreg Clayton     }
3210ec990867SGreg Clayton 
321144d93782SGreg Clayton     virtual void
321244d93782SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window)
321344d93782SGreg Clayton     {
3214ec990867SGreg Clayton         ThreadSP thread_sp = GetThread (item);
321544d93782SGreg Clayton         if (thread_sp)
321644d93782SGreg Clayton         {
321744d93782SGreg Clayton             StreamString strm;
321844d93782SGreg Clayton             ExecutionContext exe_ctx (thread_sp);
321944d93782SGreg Clayton             const char *format = "thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}";
322044d93782SGreg Clayton             if (Debugger::FormatPrompt (format, NULL, &exe_ctx, NULL, strm))
322144d93782SGreg Clayton             {
322244d93782SGreg Clayton                 int right_pad = 1;
322344d93782SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
322444d93782SGreg Clayton             }
322544d93782SGreg Clayton         }
322644d93782SGreg Clayton     }
322744d93782SGreg Clayton     virtual void
322844d93782SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)
322944d93782SGreg Clayton     {
3230ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
323144d93782SGreg Clayton         if (process_sp && process_sp->IsAlive())
323244d93782SGreg Clayton         {
323344d93782SGreg Clayton             StateType state = process_sp->GetState();
323444d93782SGreg Clayton             if (StateIsStoppedState(state, true))
323544d93782SGreg Clayton             {
3236ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
323744d93782SGreg Clayton                 if (thread_sp)
323844d93782SGreg Clayton                 {
323944d93782SGreg Clayton                     if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid)
324044d93782SGreg Clayton                         return; // Children are already up to date
3241ec990867SGreg Clayton                     if (!m_frame_delegate_sp)
324244d93782SGreg Clayton                     {
324344d93782SGreg Clayton                         // Always expand the thread item the first time we show it
3244ec990867SGreg Clayton                         m_frame_delegate_sp.reset (new FrameTreeDelegate());
324544d93782SGreg Clayton                     }
324644d93782SGreg Clayton 
324744d93782SGreg Clayton                     m_stop_id = process_sp->GetStopID();
324844d93782SGreg Clayton                     m_tid = thread_sp->GetID();
324944d93782SGreg Clayton 
325044d93782SGreg Clayton                     TreeItem t (&item, *m_frame_delegate_sp, false);
325144d93782SGreg Clayton                     size_t num_frames = thread_sp->GetStackFrameCount();
325244d93782SGreg Clayton                     item.Resize (num_frames, t);
325344d93782SGreg Clayton                     for (size_t i=0; i<num_frames; ++i)
325444d93782SGreg Clayton                     {
3255ec990867SGreg Clayton                         item[i].SetUserData(thread_sp.get());
325644d93782SGreg Clayton                         item[i].SetIdentifier(i);
325744d93782SGreg Clayton                     }
325844d93782SGreg Clayton                 }
325944d93782SGreg Clayton                 return;
326044d93782SGreg Clayton             }
326144d93782SGreg Clayton         }
326244d93782SGreg Clayton         item.ClearChildren();
326344d93782SGreg Clayton     }
326444d93782SGreg Clayton 
326544d93782SGreg Clayton     virtual bool
326644d93782SGreg Clayton     TreeDelegateItemSelected (TreeItem &item)
326744d93782SGreg Clayton     {
3268ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3269ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3270ec990867SGreg Clayton         {
3271ec990867SGreg Clayton             StateType state = process_sp->GetState();
3272ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3273ec990867SGreg Clayton             {
3274ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
327544d93782SGreg Clayton                 if (thread_sp)
327644d93782SGreg Clayton                 {
327744d93782SGreg Clayton                     ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
327844d93782SGreg Clayton                     Mutex::Locker locker (thread_list.GetMutex());
327944d93782SGreg Clayton                     ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
328044d93782SGreg Clayton                     if (selected_thread_sp->GetID() != thread_sp->GetID())
328144d93782SGreg Clayton                     {
328244d93782SGreg Clayton                         thread_list.SetSelectedThreadByID(thread_sp->GetID());
328344d93782SGreg Clayton                         return true;
328444d93782SGreg Clayton                     }
328544d93782SGreg Clayton                 }
3286ec990867SGreg Clayton             }
3287ec990867SGreg Clayton         }
328844d93782SGreg Clayton         return false;
328944d93782SGreg Clayton     }
329044d93782SGreg Clayton 
329144d93782SGreg Clayton protected:
329244d93782SGreg Clayton     Debugger &m_debugger;
329344d93782SGreg Clayton     std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
329444d93782SGreg Clayton     lldb::user_id_t m_tid;
329544d93782SGreg Clayton     uint32_t m_stop_id;
329644d93782SGreg Clayton };
329744d93782SGreg Clayton 
3298ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate
3299ec990867SGreg Clayton {
3300ec990867SGreg Clayton public:
3301ec990867SGreg Clayton     ThreadsTreeDelegate (Debugger &debugger) :
3302ec990867SGreg Clayton         TreeDelegate(),
3303ec990867SGreg Clayton         m_thread_delegate_sp (),
3304ec990867SGreg Clayton         m_debugger (debugger),
3305ec990867SGreg Clayton         m_stop_id (UINT32_MAX)
3306ec990867SGreg Clayton     {
3307ec990867SGreg Clayton     }
3308ec990867SGreg Clayton 
3309ec990867SGreg Clayton     virtual
3310ec990867SGreg Clayton     ~ThreadsTreeDelegate()
3311ec990867SGreg Clayton     {
3312ec990867SGreg Clayton     }
3313ec990867SGreg Clayton 
3314ec990867SGreg Clayton     ProcessSP
3315ec990867SGreg Clayton     GetProcess ()
3316ec990867SGreg Clayton     {
3317ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3318ec990867SGreg Clayton     }
3319ec990867SGreg Clayton 
3320ec990867SGreg Clayton     virtual void
3321ec990867SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window)
3322ec990867SGreg Clayton     {
3323ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3324ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3325ec990867SGreg Clayton         {
3326ec990867SGreg Clayton             StreamString strm;
3327ec990867SGreg Clayton             ExecutionContext exe_ctx (process_sp);
3328ec990867SGreg Clayton             const char *format = "process ${process.id}{, name = ${process.name}}";
3329ec990867SGreg Clayton             if (Debugger::FormatPrompt (format, NULL, &exe_ctx, NULL, strm))
3330ec990867SGreg Clayton             {
3331ec990867SGreg Clayton                 int right_pad = 1;
3332ec990867SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
3333ec990867SGreg Clayton             }
3334ec990867SGreg Clayton         }
3335ec990867SGreg Clayton     }
3336ec990867SGreg Clayton 
3337ec990867SGreg Clayton     virtual void
3338ec990867SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)
3339ec990867SGreg Clayton     {
3340ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3341ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3342ec990867SGreg Clayton         {
3343ec990867SGreg Clayton             StateType state = process_sp->GetState();
3344ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3345ec990867SGreg Clayton             {
3346ec990867SGreg Clayton                 const uint32_t stop_id = process_sp->GetStopID();
3347ec990867SGreg Clayton                 if (m_stop_id == stop_id)
3348ec990867SGreg Clayton                     return; // Children are already up to date
3349ec990867SGreg Clayton 
3350ec990867SGreg Clayton                 m_stop_id = stop_id;
3351ec990867SGreg Clayton 
3352ec990867SGreg Clayton                 if (!m_thread_delegate_sp)
3353ec990867SGreg Clayton                 {
3354ec990867SGreg Clayton                     // Always expand the thread item the first time we show it
3355ec990867SGreg Clayton                     //item.Expand();
3356ec990867SGreg Clayton                     m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger));
3357ec990867SGreg Clayton                 }
3358ec990867SGreg Clayton 
3359ec990867SGreg Clayton                 TreeItem t (&item, *m_thread_delegate_sp, false);
3360ec990867SGreg Clayton                 ThreadList &threads = process_sp->GetThreadList();
3361ec990867SGreg Clayton                 Mutex::Locker locker (threads.GetMutex());
3362ec990867SGreg Clayton                 size_t num_threads = threads.GetSize();
3363ec990867SGreg Clayton                 item.Resize (num_threads, t);
3364ec990867SGreg Clayton                 for (size_t i=0; i<num_threads; ++i)
3365ec990867SGreg Clayton                 {
3366ec990867SGreg Clayton                     item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
3367ec990867SGreg Clayton                     item[i].SetMightHaveChildren(true);
3368ec990867SGreg Clayton                 }
3369ec990867SGreg Clayton                 return;
3370ec990867SGreg Clayton             }
3371ec990867SGreg Clayton         }
3372ec990867SGreg Clayton         item.ClearChildren();
3373ec990867SGreg Clayton     }
3374ec990867SGreg Clayton 
3375ec990867SGreg Clayton     virtual bool
3376ec990867SGreg Clayton     TreeDelegateItemSelected (TreeItem &item)
3377ec990867SGreg Clayton     {
3378ec990867SGreg Clayton         return false;
3379ec990867SGreg Clayton     }
3380ec990867SGreg Clayton 
3381ec990867SGreg Clayton protected:
3382ec990867SGreg Clayton     std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
3383ec990867SGreg Clayton     Debugger &m_debugger;
3384ec990867SGreg Clayton     uint32_t m_stop_id;
3385ec990867SGreg Clayton };
3386ec990867SGreg Clayton 
338744d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate
338844d93782SGreg Clayton {
338944d93782SGreg Clayton public:
339044d93782SGreg Clayton     ValueObjectListDelegate () :
339144d93782SGreg Clayton         m_valobj_list (),
339244d93782SGreg Clayton         m_rows (),
339344d93782SGreg Clayton         m_selected_row (NULL),
339444d93782SGreg Clayton         m_selected_row_idx (0),
339544d93782SGreg Clayton         m_first_visible_row (0),
339644d93782SGreg Clayton         m_num_rows (0),
339744d93782SGreg Clayton         m_max_x (0),
339844d93782SGreg Clayton         m_max_y (0)
339944d93782SGreg Clayton     {
340044d93782SGreg Clayton     }
340144d93782SGreg Clayton 
340244d93782SGreg Clayton     ValueObjectListDelegate (ValueObjectList &valobj_list) :
340344d93782SGreg Clayton         m_valobj_list (valobj_list),
340444d93782SGreg Clayton         m_rows (),
340544d93782SGreg Clayton         m_selected_row (NULL),
340644d93782SGreg Clayton         m_selected_row_idx (0),
340744d93782SGreg Clayton         m_first_visible_row (0),
340844d93782SGreg Clayton         m_num_rows (0),
340944d93782SGreg Clayton         m_max_x (0),
341044d93782SGreg Clayton         m_max_y (0)
341144d93782SGreg Clayton     {
341244d93782SGreg Clayton         SetValues (valobj_list);
341344d93782SGreg Clayton     }
341444d93782SGreg Clayton 
341544d93782SGreg Clayton     virtual
341644d93782SGreg Clayton     ~ValueObjectListDelegate()
341744d93782SGreg Clayton     {
341844d93782SGreg Clayton     }
341944d93782SGreg Clayton 
342044d93782SGreg Clayton     void
342144d93782SGreg Clayton     SetValues (ValueObjectList &valobj_list)
342244d93782SGreg Clayton     {
342344d93782SGreg Clayton         m_selected_row = NULL;
342444d93782SGreg Clayton         m_selected_row_idx = 0;
342544d93782SGreg Clayton         m_first_visible_row = 0;
342644d93782SGreg Clayton         m_num_rows = 0;
342744d93782SGreg Clayton         m_rows.clear();
342844d93782SGreg Clayton         m_valobj_list = valobj_list;
342944d93782SGreg Clayton         const size_t num_values = m_valobj_list.GetSize();
343044d93782SGreg Clayton         for (size_t i=0; i<num_values; ++i)
343144d93782SGreg Clayton             m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), NULL));
343244d93782SGreg Clayton     }
343344d93782SGreg Clayton 
343444d93782SGreg Clayton     virtual bool
343544d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
343644d93782SGreg Clayton     {
343744d93782SGreg Clayton         m_num_rows = 0;
343844d93782SGreg Clayton         m_min_x = 2;
343944d93782SGreg Clayton         m_min_y = 1;
344044d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
344144d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
344244d93782SGreg Clayton 
344344d93782SGreg Clayton         window.Erase();
344444d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
344544d93782SGreg Clayton 
344644d93782SGreg Clayton         const int num_visible_rows = NumVisibleRows();
344744d93782SGreg Clayton         const int num_rows = CalculateTotalNumberRows (m_rows);
344844d93782SGreg Clayton 
344944d93782SGreg Clayton         // If we unexpanded while having something selected our
345044d93782SGreg Clayton         // total number of rows is less than the num visible rows,
345144d93782SGreg Clayton         // then make sure we show all the rows by setting the first
345244d93782SGreg Clayton         // visible row accordingly.
345344d93782SGreg Clayton         if (m_first_visible_row > 0 && num_rows < num_visible_rows)
345444d93782SGreg Clayton             m_first_visible_row = 0;
345544d93782SGreg Clayton 
345644d93782SGreg Clayton         // Make sure the selected row is always visible
345744d93782SGreg Clayton         if (m_selected_row_idx < m_first_visible_row)
345844d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx;
345944d93782SGreg Clayton         else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
346044d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
346144d93782SGreg Clayton 
346244d93782SGreg Clayton         DisplayRows (window, m_rows, g_options);
346344d93782SGreg Clayton 
346444d93782SGreg Clayton         window.DeferredRefresh();
346544d93782SGreg Clayton 
346644d93782SGreg Clayton         // Get the selected row
346744d93782SGreg Clayton         m_selected_row = GetRowForRowIndex (m_selected_row_idx);
346844d93782SGreg Clayton         // Keep the cursor on the selected row so the highlight and the cursor
346944d93782SGreg Clayton         // are always on the same line
347044d93782SGreg Clayton         if (m_selected_row)
347144d93782SGreg Clayton             window.MoveCursor (m_selected_row->x,
347244d93782SGreg Clayton                                m_selected_row->y);
347344d93782SGreg Clayton 
347444d93782SGreg Clayton         return true; // Drawing handled
347544d93782SGreg Clayton     }
347644d93782SGreg Clayton 
347744d93782SGreg Clayton     virtual KeyHelp *
347844d93782SGreg Clayton     WindowDelegateGetKeyHelp ()
347944d93782SGreg Clayton     {
348044d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
348144d93782SGreg Clayton             { KEY_UP, "Select previous item" },
348244d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
348344d93782SGreg Clayton             { KEY_RIGHT, "Expand selected item" },
348444d93782SGreg Clayton             { KEY_LEFT, "Unexpand selected item or select parent if not expanded" },
348544d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
348644d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
348744d93782SGreg Clayton             { 'A', "Format as annotated address" },
348844d93782SGreg Clayton             { 'b', "Format as binary" },
348944d93782SGreg Clayton             { 'B', "Format as hex bytes with ASCII" },
349044d93782SGreg Clayton             { 'c', "Format as character" },
349144d93782SGreg Clayton             { 'd', "Format as a signed integer" },
349244d93782SGreg Clayton             { 'D', "Format selected value using the default format for the type" },
349344d93782SGreg Clayton             { 'f', "Format as float" },
349444d93782SGreg Clayton             { 'h', "Show help dialog" },
349544d93782SGreg Clayton             { 'i', "Format as instructions" },
349644d93782SGreg Clayton             { 'o', "Format as octal" },
349744d93782SGreg Clayton             { 'p', "Format as pointer" },
349844d93782SGreg Clayton             { 's', "Format as C string" },
349944d93782SGreg Clayton             { 't', "Toggle showing/hiding type names" },
350044d93782SGreg Clayton             { 'u', "Format as an unsigned integer" },
350144d93782SGreg Clayton             { 'x', "Format as hex" },
350244d93782SGreg Clayton             { 'X', "Format as uppercase hex" },
350344d93782SGreg Clayton             { ' ', "Toggle item expansion" },
350444d93782SGreg Clayton             { ',', "Page up" },
350544d93782SGreg Clayton             { '.', "Page down" },
350644d93782SGreg Clayton             { '\0', NULL }
350744d93782SGreg Clayton         };
350844d93782SGreg Clayton         return g_source_view_key_help;
350944d93782SGreg Clayton     }
351044d93782SGreg Clayton 
351144d93782SGreg Clayton 
351244d93782SGreg Clayton     virtual HandleCharResult
351344d93782SGreg Clayton     WindowDelegateHandleChar (Window &window, int c)
351444d93782SGreg Clayton     {
351544d93782SGreg Clayton         switch(c)
351644d93782SGreg Clayton         {
351744d93782SGreg Clayton             case 'x':
351844d93782SGreg Clayton             case 'X':
351944d93782SGreg Clayton             case 'o':
352044d93782SGreg Clayton             case 's':
352144d93782SGreg Clayton             case 'u':
352244d93782SGreg Clayton             case 'd':
352344d93782SGreg Clayton             case 'D':
352444d93782SGreg Clayton             case 'i':
352544d93782SGreg Clayton             case 'A':
352644d93782SGreg Clayton             case 'p':
352744d93782SGreg Clayton             case 'c':
352844d93782SGreg Clayton             case 'b':
352944d93782SGreg Clayton             case 'B':
353044d93782SGreg Clayton             case 'f':
353144d93782SGreg Clayton                 // Change the format for the currently selected item
353244d93782SGreg Clayton                 if (m_selected_row)
353344d93782SGreg Clayton                     m_selected_row->valobj->SetFormat (FormatForChar (c));
353444d93782SGreg Clayton                 return eKeyHandled;
353544d93782SGreg Clayton 
353644d93782SGreg Clayton             case 't':
353744d93782SGreg Clayton                 // Toggle showing type names
353844d93782SGreg Clayton                 g_options.show_types = !g_options.show_types;
353944d93782SGreg Clayton                 return eKeyHandled;
354044d93782SGreg Clayton 
354144d93782SGreg Clayton             case ',':
354244d93782SGreg Clayton             case KEY_PPAGE:
354344d93782SGreg Clayton                 // Page up key
354444d93782SGreg Clayton                 if (m_first_visible_row > 0)
354544d93782SGreg Clayton                 {
35463985c8c6SSaleem Abdulrasool                     if (static_cast<int>(m_first_visible_row) > m_max_y)
354744d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
354844d93782SGreg Clayton                     else
354944d93782SGreg Clayton                         m_first_visible_row = 0;
355044d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
355144d93782SGreg Clayton                 }
355244d93782SGreg Clayton                 return eKeyHandled;
355344d93782SGreg Clayton 
355444d93782SGreg Clayton             case '.':
355544d93782SGreg Clayton             case KEY_NPAGE:
355644d93782SGreg Clayton                 // Page down key
35573985c8c6SSaleem Abdulrasool                 if (m_num_rows > static_cast<size_t>(m_max_y))
355844d93782SGreg Clayton                 {
355944d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
356044d93782SGreg Clayton                     {
356144d93782SGreg Clayton                         m_first_visible_row += m_max_y;
356244d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
356344d93782SGreg Clayton                     }
356444d93782SGreg Clayton                 }
356544d93782SGreg Clayton                 return eKeyHandled;
356644d93782SGreg Clayton 
356744d93782SGreg Clayton             case KEY_UP:
356844d93782SGreg Clayton                 if (m_selected_row_idx > 0)
356944d93782SGreg Clayton                     --m_selected_row_idx;
357044d93782SGreg Clayton                 return eKeyHandled;
357144d93782SGreg Clayton             case KEY_DOWN:
357244d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
357344d93782SGreg Clayton                     ++m_selected_row_idx;
357444d93782SGreg Clayton                 return eKeyHandled;
357544d93782SGreg Clayton 
357644d93782SGreg Clayton             case KEY_RIGHT:
357744d93782SGreg Clayton                 if (m_selected_row)
357844d93782SGreg Clayton                 {
357944d93782SGreg Clayton                     if (!m_selected_row->expanded)
358044d93782SGreg Clayton                         m_selected_row->Expand();
358144d93782SGreg Clayton                 }
358244d93782SGreg Clayton                 return eKeyHandled;
358344d93782SGreg Clayton 
358444d93782SGreg Clayton             case KEY_LEFT:
358544d93782SGreg Clayton                 if (m_selected_row)
358644d93782SGreg Clayton                 {
358744d93782SGreg Clayton                     if (m_selected_row->expanded)
358844d93782SGreg Clayton                         m_selected_row->Unexpand();
358944d93782SGreg Clayton                     else if (m_selected_row->parent)
359044d93782SGreg Clayton                         m_selected_row_idx = m_selected_row->parent->row_idx;
359144d93782SGreg Clayton                 }
359244d93782SGreg Clayton                 return eKeyHandled;
359344d93782SGreg Clayton 
359444d93782SGreg Clayton             case ' ':
359544d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
359644d93782SGreg Clayton                 if (m_selected_row)
359744d93782SGreg Clayton                 {
359844d93782SGreg Clayton                     if (m_selected_row->expanded)
359944d93782SGreg Clayton                         m_selected_row->Unexpand();
360044d93782SGreg Clayton                     else
360144d93782SGreg Clayton                         m_selected_row->Expand();
360244d93782SGreg Clayton                 }
360344d93782SGreg Clayton                 return eKeyHandled;
360444d93782SGreg Clayton 
360544d93782SGreg Clayton             case 'h':
360644d93782SGreg Clayton                 window.CreateHelpSubwindow ();
360744d93782SGreg Clayton                 return eKeyHandled;
360844d93782SGreg Clayton 
360944d93782SGreg Clayton             default:
361044d93782SGreg Clayton                 break;
361144d93782SGreg Clayton         }
361244d93782SGreg Clayton         return eKeyNotHandled;
361344d93782SGreg Clayton     }
361444d93782SGreg Clayton 
361544d93782SGreg Clayton protected:
361644d93782SGreg Clayton     ValueObjectList m_valobj_list;
361744d93782SGreg Clayton     std::vector<Row> m_rows;
361844d93782SGreg Clayton     Row *m_selected_row;
361944d93782SGreg Clayton     uint32_t m_selected_row_idx;
362044d93782SGreg Clayton     uint32_t m_first_visible_row;
362144d93782SGreg Clayton     uint32_t m_num_rows;
362244d93782SGreg Clayton     int m_min_x;
362344d93782SGreg Clayton     int m_min_y;
362444d93782SGreg Clayton     int m_max_x;
362544d93782SGreg Clayton     int m_max_y;
362644d93782SGreg Clayton 
362744d93782SGreg Clayton     static Format
362844d93782SGreg Clayton     FormatForChar (int c)
362944d93782SGreg Clayton     {
363044d93782SGreg Clayton         switch (c)
363144d93782SGreg Clayton         {
363244d93782SGreg Clayton             case 'x': return eFormatHex;
363344d93782SGreg Clayton             case 'X': return eFormatHexUppercase;
363444d93782SGreg Clayton             case 'o': return eFormatOctal;
363544d93782SGreg Clayton             case 's': return eFormatCString;
363644d93782SGreg Clayton             case 'u': return eFormatUnsigned;
363744d93782SGreg Clayton             case 'd': return eFormatDecimal;
363844d93782SGreg Clayton             case 'D': return eFormatDefault;
363944d93782SGreg Clayton             case 'i': return eFormatInstruction;
364044d93782SGreg Clayton             case 'A': return eFormatAddressInfo;
364144d93782SGreg Clayton             case 'p': return eFormatPointer;
364244d93782SGreg Clayton             case 'c': return eFormatChar;
364344d93782SGreg Clayton             case 'b': return eFormatBinary;
364444d93782SGreg Clayton             case 'B': return eFormatBytesWithASCII;
364544d93782SGreg Clayton             case 'f': return eFormatFloat;
364644d93782SGreg Clayton         }
364744d93782SGreg Clayton         return eFormatDefault;
364844d93782SGreg Clayton     }
364944d93782SGreg Clayton 
365044d93782SGreg Clayton     bool
365144d93782SGreg Clayton     DisplayRowObject (Window &window,
365244d93782SGreg Clayton                       Row &row,
365344d93782SGreg Clayton                       DisplayOptions &options,
365444d93782SGreg Clayton                       bool highlight,
365544d93782SGreg Clayton                       bool last_child)
365644d93782SGreg Clayton     {
365744d93782SGreg Clayton         ValueObject *valobj = row.valobj.get();
365844d93782SGreg Clayton 
365944d93782SGreg Clayton         if (valobj == NULL)
366044d93782SGreg Clayton             return false;
366144d93782SGreg Clayton 
366244d93782SGreg Clayton         const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : NULL;
366344d93782SGreg Clayton         const char *name = valobj->GetName().GetCString();
366444d93782SGreg Clayton         const char *value = valobj->GetValueAsCString ();
366544d93782SGreg Clayton         const char *summary = valobj->GetSummaryAsCString ();
366644d93782SGreg Clayton 
366744d93782SGreg Clayton         window.MoveCursor (row.x, row.y);
366844d93782SGreg Clayton 
366944d93782SGreg Clayton         row.DrawTree (window);
367044d93782SGreg Clayton 
367144d93782SGreg Clayton         if (highlight)
367244d93782SGreg Clayton             window.AttributeOn(A_REVERSE);
367344d93782SGreg Clayton 
367444d93782SGreg Clayton         if (type_name && type_name[0])
367544d93782SGreg Clayton             window.Printf ("(%s) ", type_name);
367644d93782SGreg Clayton 
367744d93782SGreg Clayton         if (name && name[0])
367844d93782SGreg Clayton             window.PutCString(name);
367944d93782SGreg Clayton 
368044d93782SGreg Clayton         attr_t changd_attr = 0;
368144d93782SGreg Clayton         if (valobj->GetValueDidChange())
368244d93782SGreg Clayton             changd_attr = COLOR_PAIR(5) | A_BOLD;
368344d93782SGreg Clayton 
368444d93782SGreg Clayton         if (value && value[0])
368544d93782SGreg Clayton         {
368644d93782SGreg Clayton             window.PutCString(" = ");
368744d93782SGreg Clayton             if (changd_attr)
368844d93782SGreg Clayton                 window.AttributeOn(changd_attr);
368944d93782SGreg Clayton             window.PutCString (value);
369044d93782SGreg Clayton             if (changd_attr)
369144d93782SGreg Clayton                 window.AttributeOff(changd_attr);
369244d93782SGreg Clayton         }
369344d93782SGreg Clayton 
369444d93782SGreg Clayton         if (summary && summary[0])
369544d93782SGreg Clayton         {
369644d93782SGreg Clayton             window.PutChar(' ');
369744d93782SGreg Clayton             if (changd_attr)
369844d93782SGreg Clayton                 window.AttributeOn(changd_attr);
369944d93782SGreg Clayton             window.PutCString(summary);
370044d93782SGreg Clayton             if (changd_attr)
370144d93782SGreg Clayton                 window.AttributeOff(changd_attr);
370244d93782SGreg Clayton         }
370344d93782SGreg Clayton 
370444d93782SGreg Clayton         if (highlight)
370544d93782SGreg Clayton             window.AttributeOff (A_REVERSE);
370644d93782SGreg Clayton 
370744d93782SGreg Clayton         return true;
370844d93782SGreg Clayton     }
370944d93782SGreg Clayton     void
371044d93782SGreg Clayton     DisplayRows (Window &window,
371144d93782SGreg Clayton                  std::vector<Row> &rows,
371244d93782SGreg Clayton                  DisplayOptions &options)
371344d93782SGreg Clayton     {
371444d93782SGreg Clayton         // >   0x25B7
371544d93782SGreg Clayton         // \/  0x25BD
371644d93782SGreg Clayton 
371744d93782SGreg Clayton         bool window_is_active = window.IsActive();
371844d93782SGreg Clayton         for (auto &row : rows)
371944d93782SGreg Clayton         {
372044d93782SGreg Clayton             const bool last_child = row.parent && &rows[rows.size()-1] == &row;
372144d93782SGreg Clayton             // Save the row index in each Row structure
372244d93782SGreg Clayton             row.row_idx = m_num_rows;
372344d93782SGreg Clayton             if ((m_num_rows >= m_first_visible_row) &&
37243985c8c6SSaleem Abdulrasool                 ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows())))
372544d93782SGreg Clayton             {
372644d93782SGreg Clayton                 row.x = m_min_x;
372744d93782SGreg Clayton                 row.y = m_num_rows - m_first_visible_row + 1;
372844d93782SGreg Clayton                 if (DisplayRowObject (window,
372944d93782SGreg Clayton                                       row,
373044d93782SGreg Clayton                                       options,
373144d93782SGreg Clayton                                       window_is_active && m_num_rows == m_selected_row_idx,
373244d93782SGreg Clayton                                       last_child))
373344d93782SGreg Clayton                 {
373444d93782SGreg Clayton                     ++m_num_rows;
373544d93782SGreg Clayton                 }
373644d93782SGreg Clayton                 else
373744d93782SGreg Clayton                 {
373844d93782SGreg Clayton                     row.x = 0;
373944d93782SGreg Clayton                     row.y = 0;
374044d93782SGreg Clayton                 }
374144d93782SGreg Clayton             }
374244d93782SGreg Clayton             else
374344d93782SGreg Clayton             {
374444d93782SGreg Clayton                 row.x = 0;
374544d93782SGreg Clayton                 row.y = 0;
374644d93782SGreg Clayton                 ++m_num_rows;
374744d93782SGreg Clayton             }
374844d93782SGreg Clayton 
374944d93782SGreg Clayton             if (row.expanded && !row.children.empty())
375044d93782SGreg Clayton             {
375144d93782SGreg Clayton                 DisplayRows (window,
375244d93782SGreg Clayton                              row.children,
375344d93782SGreg Clayton                              options);
375444d93782SGreg Clayton             }
375544d93782SGreg Clayton         }
375644d93782SGreg Clayton     }
375744d93782SGreg Clayton 
375844d93782SGreg Clayton     int
375944d93782SGreg Clayton     CalculateTotalNumberRows (const std::vector<Row> &rows)
376044d93782SGreg Clayton     {
376144d93782SGreg Clayton         int row_count = 0;
376244d93782SGreg Clayton         for (const auto &row : rows)
376344d93782SGreg Clayton         {
376444d93782SGreg Clayton             ++row_count;
376544d93782SGreg Clayton             if (row.expanded)
376644d93782SGreg Clayton                 row_count += CalculateTotalNumberRows(row.children);
376744d93782SGreg Clayton         }
376844d93782SGreg Clayton         return row_count;
376944d93782SGreg Clayton     }
377044d93782SGreg Clayton     static Row *
377144d93782SGreg Clayton     GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index)
377244d93782SGreg Clayton     {
377344d93782SGreg Clayton         for (auto &row : rows)
377444d93782SGreg Clayton         {
377544d93782SGreg Clayton             if (row_index == 0)
377644d93782SGreg Clayton                 return &row;
377744d93782SGreg Clayton             else
377844d93782SGreg Clayton             {
377944d93782SGreg Clayton                 --row_index;
378044d93782SGreg Clayton                 if (row.expanded && !row.children.empty())
378144d93782SGreg Clayton                 {
378244d93782SGreg Clayton                     Row *result = GetRowForRowIndexImpl (row.children, row_index);
378344d93782SGreg Clayton                     if (result)
378444d93782SGreg Clayton                         return result;
378544d93782SGreg Clayton                 }
378644d93782SGreg Clayton             }
378744d93782SGreg Clayton         }
378844d93782SGreg Clayton         return NULL;
378944d93782SGreg Clayton     }
379044d93782SGreg Clayton 
379144d93782SGreg Clayton     Row *
379244d93782SGreg Clayton     GetRowForRowIndex (size_t row_index)
379344d93782SGreg Clayton     {
379444d93782SGreg Clayton         return GetRowForRowIndexImpl (m_rows, row_index);
379544d93782SGreg Clayton     }
379644d93782SGreg Clayton 
379744d93782SGreg Clayton     int
379844d93782SGreg Clayton     NumVisibleRows () const
379944d93782SGreg Clayton     {
380044d93782SGreg Clayton         return m_max_y - m_min_y;
380144d93782SGreg Clayton     }
380244d93782SGreg Clayton 
380344d93782SGreg Clayton     static DisplayOptions g_options;
380444d93782SGreg Clayton };
380544d93782SGreg Clayton 
380644d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate
380744d93782SGreg Clayton {
380844d93782SGreg Clayton public:
380944d93782SGreg Clayton     FrameVariablesWindowDelegate (Debugger &debugger) :
381044d93782SGreg Clayton         ValueObjectListDelegate (),
381144d93782SGreg Clayton         m_debugger (debugger),
381244d93782SGreg Clayton         m_frame_block (NULL)
381344d93782SGreg Clayton     {
381444d93782SGreg Clayton     }
381544d93782SGreg Clayton 
381644d93782SGreg Clayton     virtual
381744d93782SGreg Clayton     ~FrameVariablesWindowDelegate()
381844d93782SGreg Clayton     {
381944d93782SGreg Clayton     }
382044d93782SGreg Clayton 
382144d93782SGreg Clayton     virtual const char *
382244d93782SGreg Clayton     WindowDelegateGetHelpText ()
382344d93782SGreg Clayton     {
382444d93782SGreg Clayton         return "Frame variable window keyboard shortcuts:";
382544d93782SGreg Clayton     }
382644d93782SGreg Clayton 
382744d93782SGreg Clayton     virtual bool
382844d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
382944d93782SGreg Clayton     {
383044d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
383144d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
383244d93782SGreg Clayton         Block *frame_block = NULL;
383344d93782SGreg Clayton         StackFrame *frame = NULL;
383444d93782SGreg Clayton 
383544d93782SGreg Clayton         if (process)
383644d93782SGreg Clayton         {
383744d93782SGreg Clayton             StateType state = process->GetState();
383844d93782SGreg Clayton             if (StateIsStoppedState(state, true))
383944d93782SGreg Clayton             {
384044d93782SGreg Clayton                 frame = exe_ctx.GetFramePtr();
384144d93782SGreg Clayton                 if (frame)
384244d93782SGreg Clayton                     frame_block = frame->GetFrameBlock ();
384344d93782SGreg Clayton             }
384444d93782SGreg Clayton             else if (StateIsRunningState(state))
384544d93782SGreg Clayton             {
384644d93782SGreg Clayton                 return true; // Don't do any updating when we are running
384744d93782SGreg Clayton             }
384844d93782SGreg Clayton         }
384944d93782SGreg Clayton 
385044d93782SGreg Clayton         ValueObjectList local_values;
385144d93782SGreg Clayton         if (frame_block)
385244d93782SGreg Clayton         {
385344d93782SGreg Clayton             // Only update the variables if they have changed
385444d93782SGreg Clayton             if (m_frame_block != frame_block)
385544d93782SGreg Clayton             {
385644d93782SGreg Clayton                 m_frame_block = frame_block;
385744d93782SGreg Clayton 
385844d93782SGreg Clayton                 VariableList *locals = frame->GetVariableList(true);
385944d93782SGreg Clayton                 if (locals)
386044d93782SGreg Clayton                 {
386144d93782SGreg Clayton                     const DynamicValueType use_dynamic = eDynamicDontRunTarget;
386244d93782SGreg Clayton                     const size_t num_locals = locals->GetSize();
386344d93782SGreg Clayton                     for (size_t i=0; i<num_locals; ++i)
386444d93782SGreg Clayton                         local_values.Append(frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic));
386544d93782SGreg Clayton                     // Update the values
386644d93782SGreg Clayton                     SetValues(local_values);
386744d93782SGreg Clayton                 }
386844d93782SGreg Clayton             }
386944d93782SGreg Clayton         }
387044d93782SGreg Clayton         else
387144d93782SGreg Clayton         {
387244d93782SGreg Clayton             m_frame_block = NULL;
387344d93782SGreg Clayton             // Update the values with an empty list if there is no frame
387444d93782SGreg Clayton             SetValues(local_values);
387544d93782SGreg Clayton         }
387644d93782SGreg Clayton 
387744d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
387844d93782SGreg Clayton 
387944d93782SGreg Clayton     }
388044d93782SGreg Clayton 
388144d93782SGreg Clayton protected:
388244d93782SGreg Clayton     Debugger &m_debugger;
388344d93782SGreg Clayton     Block *m_frame_block;
388444d93782SGreg Clayton };
388544d93782SGreg Clayton 
388644d93782SGreg Clayton 
388744d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate
388844d93782SGreg Clayton {
388944d93782SGreg Clayton public:
389044d93782SGreg Clayton     RegistersWindowDelegate (Debugger &debugger) :
389144d93782SGreg Clayton         ValueObjectListDelegate (),
389244d93782SGreg Clayton         m_debugger (debugger)
389344d93782SGreg Clayton     {
389444d93782SGreg Clayton     }
389544d93782SGreg Clayton 
389644d93782SGreg Clayton     virtual
389744d93782SGreg Clayton     ~RegistersWindowDelegate()
389844d93782SGreg Clayton     {
389944d93782SGreg Clayton     }
390044d93782SGreg Clayton 
390144d93782SGreg Clayton     virtual const char *
390244d93782SGreg Clayton     WindowDelegateGetHelpText ()
390344d93782SGreg Clayton     {
390444d93782SGreg Clayton         return "Register window keyboard shortcuts:";
390544d93782SGreg Clayton     }
390644d93782SGreg Clayton 
390744d93782SGreg Clayton     virtual bool
390844d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
390944d93782SGreg Clayton     {
391044d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
391144d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
391244d93782SGreg Clayton 
391344d93782SGreg Clayton         ValueObjectList value_list;
391444d93782SGreg Clayton         if (frame)
391544d93782SGreg Clayton         {
391644d93782SGreg Clayton             if (frame->GetStackID() != m_stack_id)
391744d93782SGreg Clayton             {
391844d93782SGreg Clayton                 m_stack_id = frame->GetStackID();
391944d93782SGreg Clayton                 RegisterContextSP reg_ctx (frame->GetRegisterContext());
392044d93782SGreg Clayton                 if (reg_ctx)
392144d93782SGreg Clayton                 {
392244d93782SGreg Clayton                     const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
392344d93782SGreg Clayton                     for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx)
392444d93782SGreg Clayton                     {
392544d93782SGreg Clayton                         value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx));
392644d93782SGreg Clayton                     }
392744d93782SGreg Clayton                 }
392844d93782SGreg Clayton                 SetValues(value_list);
392944d93782SGreg Clayton             }
393044d93782SGreg Clayton         }
393144d93782SGreg Clayton         else
393244d93782SGreg Clayton         {
393344d93782SGreg Clayton             Process *process = exe_ctx.GetProcessPtr();
393444d93782SGreg Clayton             if (process && process->IsAlive())
393544d93782SGreg Clayton                 return true; // Don't do any updating if we are running
393644d93782SGreg Clayton             else
393744d93782SGreg Clayton             {
393844d93782SGreg Clayton                 // Update the values with an empty list if there
393944d93782SGreg Clayton                 // is no process or the process isn't alive anymore
394044d93782SGreg Clayton                 SetValues(value_list);
394144d93782SGreg Clayton             }
394244d93782SGreg Clayton         }
394344d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
394444d93782SGreg Clayton     }
394544d93782SGreg Clayton 
394644d93782SGreg Clayton protected:
394744d93782SGreg Clayton     Debugger &m_debugger;
394844d93782SGreg Clayton     StackID m_stack_id;
394944d93782SGreg Clayton };
395044d93782SGreg Clayton 
395144d93782SGreg Clayton static const char *
395244d93782SGreg Clayton CursesKeyToCString (int ch)
395344d93782SGreg Clayton {
395444d93782SGreg Clayton     static char g_desc[32];
395544d93782SGreg Clayton     if (ch >= KEY_F0 && ch < KEY_F0 + 64)
395644d93782SGreg Clayton     {
395744d93782SGreg Clayton         snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
395844d93782SGreg Clayton         return g_desc;
395944d93782SGreg Clayton     }
396044d93782SGreg Clayton     switch (ch)
396144d93782SGreg Clayton     {
396244d93782SGreg Clayton         case KEY_DOWN:  return "down";
396344d93782SGreg Clayton         case KEY_UP:    return "up";
396444d93782SGreg Clayton         case KEY_LEFT:  return "left";
396544d93782SGreg Clayton         case KEY_RIGHT: return "right";
396644d93782SGreg Clayton         case KEY_HOME:  return "home";
396744d93782SGreg Clayton         case KEY_BACKSPACE: return "backspace";
396844d93782SGreg Clayton         case KEY_DL:        return "delete-line";
396944d93782SGreg Clayton         case KEY_IL:        return "insert-line";
397044d93782SGreg Clayton         case KEY_DC:        return "delete-char";
397144d93782SGreg Clayton         case KEY_IC:        return "insert-char";
397244d93782SGreg Clayton         case KEY_CLEAR:     return "clear";
397344d93782SGreg Clayton         case KEY_EOS:       return "clear-to-eos";
397444d93782SGreg Clayton         case KEY_EOL:       return "clear-to-eol";
397544d93782SGreg Clayton         case KEY_SF:        return "scroll-forward";
397644d93782SGreg Clayton         case KEY_SR:        return "scroll-backward";
397744d93782SGreg Clayton         case KEY_NPAGE:     return "page-down";
397844d93782SGreg Clayton         case KEY_PPAGE:     return "page-up";
397944d93782SGreg Clayton         case KEY_STAB:      return "set-tab";
398044d93782SGreg Clayton         case KEY_CTAB:      return "clear-tab";
398144d93782SGreg Clayton         case KEY_CATAB:     return "clear-all-tabs";
398244d93782SGreg Clayton         case KEY_ENTER:     return "enter";
398344d93782SGreg Clayton         case KEY_PRINT:     return "print";
398444d93782SGreg Clayton         case KEY_LL:        return "lower-left key";
398544d93782SGreg Clayton         case KEY_A1:        return "upper left of keypad";
398644d93782SGreg Clayton         case KEY_A3:        return "upper right of keypad";
398744d93782SGreg Clayton         case KEY_B2:        return "center of keypad";
398844d93782SGreg Clayton         case KEY_C1:        return "lower left of keypad";
398944d93782SGreg Clayton         case KEY_C3:        return "lower right of keypad";
399044d93782SGreg Clayton         case KEY_BTAB:      return "back-tab key";
399144d93782SGreg Clayton         case KEY_BEG:       return "begin key";
399244d93782SGreg Clayton         case KEY_CANCEL:    return "cancel key";
399344d93782SGreg Clayton         case KEY_CLOSE:     return "close key";
399444d93782SGreg Clayton         case KEY_COMMAND:   return "command key";
399544d93782SGreg Clayton         case KEY_COPY:      return "copy key";
399644d93782SGreg Clayton         case KEY_CREATE:    return "create key";
399744d93782SGreg Clayton         case KEY_END:       return "end key";
399844d93782SGreg Clayton         case KEY_EXIT:      return "exit key";
399944d93782SGreg Clayton         case KEY_FIND:      return "find key";
400044d93782SGreg Clayton         case KEY_HELP:      return "help key";
400144d93782SGreg Clayton         case KEY_MARK:      return "mark key";
400244d93782SGreg Clayton         case KEY_MESSAGE:   return "message key";
400344d93782SGreg Clayton         case KEY_MOVE:      return "move key";
400444d93782SGreg Clayton         case KEY_NEXT:      return "next key";
400544d93782SGreg Clayton         case KEY_OPEN:      return "open key";
400644d93782SGreg Clayton         case KEY_OPTIONS:   return "options key";
400744d93782SGreg Clayton         case KEY_PREVIOUS:  return "previous key";
400844d93782SGreg Clayton         case KEY_REDO:      return "redo key";
400944d93782SGreg Clayton         case KEY_REFERENCE: return "reference key";
401044d93782SGreg Clayton         case KEY_REFRESH:   return "refresh key";
401144d93782SGreg Clayton         case KEY_REPLACE:   return "replace key";
401244d93782SGreg Clayton         case KEY_RESTART:   return "restart key";
401344d93782SGreg Clayton         case KEY_RESUME:    return "resume key";
401444d93782SGreg Clayton         case KEY_SAVE:      return "save key";
401544d93782SGreg Clayton         case KEY_SBEG:      return "shifted begin key";
401644d93782SGreg Clayton         case KEY_SCANCEL:   return "shifted cancel key";
401744d93782SGreg Clayton         case KEY_SCOMMAND:  return "shifted command key";
401844d93782SGreg Clayton         case KEY_SCOPY:     return "shifted copy key";
401944d93782SGreg Clayton         case KEY_SCREATE:   return "shifted create key";
402044d93782SGreg Clayton         case KEY_SDC:       return "shifted delete-character key";
402144d93782SGreg Clayton         case KEY_SDL:       return "shifted delete-line key";
402244d93782SGreg Clayton         case KEY_SELECT:    return "select key";
402344d93782SGreg Clayton         case KEY_SEND:      return "shifted end key";
402444d93782SGreg Clayton         case KEY_SEOL:      return "shifted clear-to-end-of-line key";
402544d93782SGreg Clayton         case KEY_SEXIT:     return "shifted exit key";
402644d93782SGreg Clayton         case KEY_SFIND:     return "shifted find key";
402744d93782SGreg Clayton         case KEY_SHELP:     return "shifted help key";
402844d93782SGreg Clayton         case KEY_SHOME:     return "shifted home key";
402944d93782SGreg Clayton         case KEY_SIC:       return "shifted insert-character key";
403044d93782SGreg Clayton         case KEY_SLEFT:     return "shifted left-arrow key";
403144d93782SGreg Clayton         case KEY_SMESSAGE:  return "shifted message key";
403244d93782SGreg Clayton         case KEY_SMOVE:     return "shifted move key";
403344d93782SGreg Clayton         case KEY_SNEXT:     return "shifted next key";
403444d93782SGreg Clayton         case KEY_SOPTIONS:  return "shifted options key";
403544d93782SGreg Clayton         case KEY_SPREVIOUS: return "shifted previous key";
403644d93782SGreg Clayton         case KEY_SPRINT:    return "shifted print key";
403744d93782SGreg Clayton         case KEY_SREDO:     return "shifted redo key";
403844d93782SGreg Clayton         case KEY_SREPLACE:  return "shifted replace key";
403944d93782SGreg Clayton         case KEY_SRIGHT:    return "shifted right-arrow key";
404044d93782SGreg Clayton         case KEY_SRSUME:    return "shifted resume key";
404144d93782SGreg Clayton         case KEY_SSAVE:     return "shifted save key";
404244d93782SGreg Clayton         case KEY_SSUSPEND:  return "shifted suspend key";
404344d93782SGreg Clayton         case KEY_SUNDO:     return "shifted undo key";
404444d93782SGreg Clayton         case KEY_SUSPEND:   return "suspend key";
404544d93782SGreg Clayton         case KEY_UNDO:      return "undo key";
404644d93782SGreg Clayton         case KEY_MOUSE:     return "Mouse event has occurred";
404744d93782SGreg Clayton         case KEY_RESIZE:    return "Terminal resize event";
404844d93782SGreg Clayton         case KEY_EVENT:     return "We were interrupted by an event";
404944d93782SGreg Clayton         case KEY_RETURN:    return "return";
405044d93782SGreg Clayton         case ' ':           return "space";
40515fdb09bbSGreg Clayton         case '\t':          return "tab";
405244d93782SGreg Clayton         case KEY_ESCAPE:    return "escape";
405344d93782SGreg Clayton         default:
405444d93782SGreg Clayton             if (isprint(ch))
405544d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "%c", ch);
405644d93782SGreg Clayton             else
405744d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
405844d93782SGreg Clayton             return g_desc;
405944d93782SGreg Clayton     }
406044d93782SGreg Clayton     return NULL;
406144d93782SGreg Clayton }
406244d93782SGreg Clayton 
406344d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) :
406444d93782SGreg Clayton     m_text (),
406544d93782SGreg Clayton     m_first_visible_line (0)
406644d93782SGreg Clayton {
406744d93782SGreg Clayton     if (text && text[0])
406844d93782SGreg Clayton     {
406944d93782SGreg Clayton         m_text.SplitIntoLines(text);
407044d93782SGreg Clayton         m_text.AppendString("");
407144d93782SGreg Clayton     }
407244d93782SGreg Clayton     if (key_help_array)
407344d93782SGreg Clayton     {
407444d93782SGreg Clayton         for (KeyHelp *key = key_help_array; key->ch; ++key)
407544d93782SGreg Clayton         {
407644d93782SGreg Clayton             StreamString key_description;
407744d93782SGreg Clayton             key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description);
407844d93782SGreg Clayton             m_text.AppendString(std::move(key_description.GetString()));
407944d93782SGreg Clayton         }
408044d93782SGreg Clayton     }
408144d93782SGreg Clayton }
408244d93782SGreg Clayton 
408344d93782SGreg Clayton HelpDialogDelegate::~HelpDialogDelegate()
408444d93782SGreg Clayton {
408544d93782SGreg Clayton }
408644d93782SGreg Clayton 
408744d93782SGreg Clayton bool
408844d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force)
408944d93782SGreg Clayton {
409044d93782SGreg Clayton     window.Erase();
409144d93782SGreg Clayton     const int window_height = window.GetHeight();
409244d93782SGreg Clayton     int x = 2;
409344d93782SGreg Clayton     int y = 1;
409444d93782SGreg Clayton     const int min_y = y;
409544d93782SGreg Clayton     const int max_y = window_height - 1 - y;
40963985c8c6SSaleem Abdulrasool     const size_t num_visible_lines = max_y - min_y + 1;
409744d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
409844d93782SGreg Clayton     const char *bottom_message;
409944d93782SGreg Clayton     if (num_lines <= num_visible_lines)
410044d93782SGreg Clayton         bottom_message = "Press any key to exit";
410144d93782SGreg Clayton     else
410244d93782SGreg Clayton         bottom_message = "Use arrows to scroll, any other key to exit";
410344d93782SGreg Clayton     window.DrawTitleBox(window.GetName(), bottom_message);
410444d93782SGreg Clayton     while (y <= max_y)
410544d93782SGreg Clayton     {
410644d93782SGreg Clayton         window.MoveCursor(x, y);
410744d93782SGreg Clayton         window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
410844d93782SGreg Clayton         ++y;
410944d93782SGreg Clayton     }
411044d93782SGreg Clayton     return true;
411144d93782SGreg Clayton }
411244d93782SGreg Clayton 
411344d93782SGreg Clayton HandleCharResult
411444d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key)
411544d93782SGreg Clayton {
411644d93782SGreg Clayton     bool done = false;
411744d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
411844d93782SGreg Clayton     const size_t num_visible_lines = window.GetHeight() - 2;
411944d93782SGreg Clayton 
412044d93782SGreg Clayton     if (num_lines <= num_visible_lines)
412144d93782SGreg Clayton     {
412244d93782SGreg Clayton         done = true;
412344d93782SGreg Clayton         // If we have all lines visible and don't need scrolling, then any
412444d93782SGreg Clayton         // key press will cause us to exit
412544d93782SGreg Clayton     }
412644d93782SGreg Clayton     else
412744d93782SGreg Clayton     {
412844d93782SGreg Clayton         switch (key)
412944d93782SGreg Clayton         {
413044d93782SGreg Clayton             case KEY_UP:
413144d93782SGreg Clayton                 if (m_first_visible_line > 0)
413244d93782SGreg Clayton                     --m_first_visible_line;
413344d93782SGreg Clayton                 break;
413444d93782SGreg Clayton 
413544d93782SGreg Clayton             case KEY_DOWN:
413644d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
413744d93782SGreg Clayton                     ++m_first_visible_line;
413844d93782SGreg Clayton                 break;
413944d93782SGreg Clayton 
414044d93782SGreg Clayton             case KEY_PPAGE:
414144d93782SGreg Clayton             case ',':
414244d93782SGreg Clayton                 if (m_first_visible_line > 0)
414344d93782SGreg Clayton                 {
41443985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
414544d93782SGreg Clayton                         m_first_visible_line -= num_visible_lines;
414644d93782SGreg Clayton                     else
414744d93782SGreg Clayton                         m_first_visible_line = 0;
414844d93782SGreg Clayton                 }
414944d93782SGreg Clayton                 break;
415044d93782SGreg Clayton             case KEY_NPAGE:
415144d93782SGreg Clayton             case '.':
415244d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
415344d93782SGreg Clayton                 {
415444d93782SGreg Clayton                     m_first_visible_line += num_visible_lines;
41553985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > num_lines)
415644d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
415744d93782SGreg Clayton                 }
415844d93782SGreg Clayton                 break;
415944d93782SGreg Clayton             default:
416044d93782SGreg Clayton                 done = true;
416144d93782SGreg Clayton                 break;
416244d93782SGreg Clayton         }
416344d93782SGreg Clayton     }
416444d93782SGreg Clayton     if (done)
416544d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
416644d93782SGreg Clayton     return eKeyHandled;
416744d93782SGreg Clayton }
416844d93782SGreg Clayton 
416944d93782SGreg Clayton class ApplicationDelegate :
417044d93782SGreg Clayton     public WindowDelegate,
417144d93782SGreg Clayton     public MenuDelegate
417244d93782SGreg Clayton {
417344d93782SGreg Clayton public:
417444d93782SGreg Clayton     enum {
417544d93782SGreg Clayton         eMenuID_LLDB = 1,
417644d93782SGreg Clayton         eMenuID_LLDBAbout,
417744d93782SGreg Clayton         eMenuID_LLDBExit,
417844d93782SGreg Clayton 
417944d93782SGreg Clayton         eMenuID_Target,
418044d93782SGreg Clayton         eMenuID_TargetCreate,
418144d93782SGreg Clayton         eMenuID_TargetDelete,
418244d93782SGreg Clayton 
418344d93782SGreg Clayton         eMenuID_Process,
418444d93782SGreg Clayton         eMenuID_ProcessAttach,
418544d93782SGreg Clayton         eMenuID_ProcessDetach,
418644d93782SGreg Clayton         eMenuID_ProcessLaunch,
418744d93782SGreg Clayton         eMenuID_ProcessContinue,
418844d93782SGreg Clayton         eMenuID_ProcessHalt,
418944d93782SGreg Clayton         eMenuID_ProcessKill,
419044d93782SGreg Clayton 
419144d93782SGreg Clayton         eMenuID_Thread,
419244d93782SGreg Clayton         eMenuID_ThreadStepIn,
419344d93782SGreg Clayton         eMenuID_ThreadStepOver,
419444d93782SGreg Clayton         eMenuID_ThreadStepOut,
419544d93782SGreg Clayton 
419644d93782SGreg Clayton         eMenuID_View,
419744d93782SGreg Clayton         eMenuID_ViewBacktrace,
419844d93782SGreg Clayton         eMenuID_ViewRegisters,
419944d93782SGreg Clayton         eMenuID_ViewSource,
420044d93782SGreg Clayton         eMenuID_ViewVariables,
420144d93782SGreg Clayton 
420244d93782SGreg Clayton         eMenuID_Help,
420344d93782SGreg Clayton         eMenuID_HelpGUIHelp
420444d93782SGreg Clayton     };
420544d93782SGreg Clayton 
420644d93782SGreg Clayton     ApplicationDelegate (Application &app, Debugger &debugger) :
420744d93782SGreg Clayton         WindowDelegate (),
420844d93782SGreg Clayton         MenuDelegate (),
420944d93782SGreg Clayton         m_app (app),
421044d93782SGreg Clayton         m_debugger (debugger)
421144d93782SGreg Clayton     {
421244d93782SGreg Clayton     }
421344d93782SGreg Clayton 
421444d93782SGreg Clayton     virtual
421544d93782SGreg Clayton     ~ApplicationDelegate ()
421644d93782SGreg Clayton     {
421744d93782SGreg Clayton     }
421844d93782SGreg Clayton     virtual bool
421944d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
422044d93782SGreg Clayton     {
422144d93782SGreg Clayton         return false; // Drawing not handled, let standard window drawing happen
422244d93782SGreg Clayton     }
422344d93782SGreg Clayton 
422444d93782SGreg Clayton     virtual HandleCharResult
422544d93782SGreg Clayton     WindowDelegateHandleChar (Window &window, int key)
422644d93782SGreg Clayton     {
42275fdb09bbSGreg Clayton         switch (key)
422844d93782SGreg Clayton         {
42295fdb09bbSGreg Clayton             case '\t':
423044d93782SGreg Clayton                 window.SelectNextWindowAsActive();
423144d93782SGreg Clayton                 return eKeyHandled;
42325fdb09bbSGreg Clayton 
42335fdb09bbSGreg Clayton             case 'h':
42345fdb09bbSGreg Clayton                 window.CreateHelpSubwindow();
42355fdb09bbSGreg Clayton                 return eKeyHandled;
42365fdb09bbSGreg Clayton 
42375fdb09bbSGreg Clayton             case KEY_ESCAPE:
42385fdb09bbSGreg Clayton                 return eQuitApplication;
42395fdb09bbSGreg Clayton 
42405fdb09bbSGreg Clayton             default:
42415fdb09bbSGreg Clayton                 break;
424244d93782SGreg Clayton         }
424344d93782SGreg Clayton         return eKeyNotHandled;
424444d93782SGreg Clayton     }
424544d93782SGreg Clayton 
42465fdb09bbSGreg Clayton 
42475fdb09bbSGreg Clayton     virtual const char *
42485fdb09bbSGreg Clayton     WindowDelegateGetHelpText ()
42495fdb09bbSGreg Clayton     {
42505fdb09bbSGreg Clayton         return "Welcome to the LLDB curses GUI.\n\n"
42515fdb09bbSGreg Clayton         "Press the TAB key to change the selected view.\n"
42525fdb09bbSGreg Clayton         "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n"
42535fdb09bbSGreg Clayton         "Common key bindings for all views:";
42545fdb09bbSGreg Clayton     }
42555fdb09bbSGreg Clayton 
42565fdb09bbSGreg Clayton     virtual KeyHelp *
42575fdb09bbSGreg Clayton     WindowDelegateGetKeyHelp ()
42585fdb09bbSGreg Clayton     {
42595fdb09bbSGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
42605fdb09bbSGreg Clayton             { '\t', "Select next view" },
42615fdb09bbSGreg Clayton             { 'h', "Show help dialog with view specific key bindings" },
42625fdb09bbSGreg Clayton             { ',', "Page up" },
42635fdb09bbSGreg Clayton             { '.', "Page down" },
42645fdb09bbSGreg Clayton             { KEY_UP, "Select previous" },
42655fdb09bbSGreg Clayton             { KEY_DOWN, "Select next" },
42665fdb09bbSGreg Clayton             { KEY_LEFT, "Unexpand or select parent" },
42675fdb09bbSGreg Clayton             { KEY_RIGHT, "Expand" },
42685fdb09bbSGreg Clayton             { KEY_PPAGE, "Page up" },
42695fdb09bbSGreg Clayton             { KEY_NPAGE, "Page down" },
42705fdb09bbSGreg Clayton             { '\0', NULL }
42715fdb09bbSGreg Clayton         };
42725fdb09bbSGreg Clayton         return g_source_view_key_help;
42735fdb09bbSGreg Clayton     }
42745fdb09bbSGreg Clayton 
427544d93782SGreg Clayton     virtual MenuActionResult
427644d93782SGreg Clayton     MenuDelegateAction (Menu &menu)
427744d93782SGreg Clayton     {
427844d93782SGreg Clayton         switch (menu.GetIdentifier())
427944d93782SGreg Clayton         {
428044d93782SGreg Clayton             case eMenuID_ThreadStepIn:
428144d93782SGreg Clayton                 {
428244d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
428344d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
428444d93782SGreg Clayton                     {
428544d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
428644d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
42874b4b2478SJim Ingham                             exe_ctx.GetThreadRef().StepIn(true);
428844d93782SGreg Clayton                     }
428944d93782SGreg Clayton                 }
429044d93782SGreg Clayton                 return MenuActionResult::Handled;
429144d93782SGreg Clayton 
429244d93782SGreg Clayton             case eMenuID_ThreadStepOut:
429344d93782SGreg Clayton                 {
429444d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
429544d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
429644d93782SGreg Clayton                     {
429744d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
429844d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
429944d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOut();
430044d93782SGreg Clayton                     }
430144d93782SGreg Clayton                 }
430244d93782SGreg Clayton                 return MenuActionResult::Handled;
430344d93782SGreg Clayton 
430444d93782SGreg Clayton             case eMenuID_ThreadStepOver:
430544d93782SGreg Clayton                 {
430644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
430744d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
430844d93782SGreg Clayton                     {
430944d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
431044d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
431144d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOver(true);
431244d93782SGreg Clayton                     }
431344d93782SGreg Clayton                 }
431444d93782SGreg Clayton                 return MenuActionResult::Handled;
431544d93782SGreg Clayton 
431644d93782SGreg Clayton             case eMenuID_ProcessContinue:
431744d93782SGreg Clayton                 {
431844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
431944d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
432044d93782SGreg Clayton                     {
432144d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
432244d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
432344d93782SGreg Clayton                             process->Resume();
432444d93782SGreg Clayton                     }
432544d93782SGreg Clayton                 }
432644d93782SGreg Clayton                 return MenuActionResult::Handled;
432744d93782SGreg Clayton 
432844d93782SGreg Clayton             case eMenuID_ProcessKill:
432944d93782SGreg Clayton                 {
433044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
433144d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
433244d93782SGreg Clayton                     {
433344d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
433444d93782SGreg Clayton                         if (process && process->IsAlive())
433544d93782SGreg Clayton                             process->Destroy();
433644d93782SGreg Clayton                     }
433744d93782SGreg Clayton                 }
433844d93782SGreg Clayton                 return MenuActionResult::Handled;
433944d93782SGreg Clayton 
434044d93782SGreg Clayton             case eMenuID_ProcessHalt:
434144d93782SGreg Clayton                 {
434244d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
434344d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
434444d93782SGreg Clayton                     {
434544d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
434644d93782SGreg Clayton                         if (process && process->IsAlive())
434744d93782SGreg Clayton                             process->Halt();
434844d93782SGreg Clayton                     }
434944d93782SGreg Clayton                 }
435044d93782SGreg Clayton                 return MenuActionResult::Handled;
435144d93782SGreg Clayton 
435244d93782SGreg Clayton             case eMenuID_ProcessDetach:
435344d93782SGreg Clayton                 {
435444d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
435544d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
435644d93782SGreg Clayton                     {
435744d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
435844d93782SGreg Clayton                         if (process && process->IsAlive())
435944d93782SGreg Clayton                             process->Detach(false);
436044d93782SGreg Clayton                     }
436144d93782SGreg Clayton                 }
436244d93782SGreg Clayton                 return MenuActionResult::Handled;
436344d93782SGreg Clayton 
436444d93782SGreg Clayton             case eMenuID_Process:
436544d93782SGreg Clayton                 {
436644d93782SGreg Clayton                     // Populate the menu with all of the threads if the process is stopped when
436744d93782SGreg Clayton                     // the Process menu gets selected and is about to display its submenu.
436844d93782SGreg Clayton                     Menus &submenus = menu.GetSubmenus();
436944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
437044d93782SGreg Clayton                     Process *process = exe_ctx.GetProcessPtr();
437144d93782SGreg Clayton                     if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
437244d93782SGreg Clayton                     {
437344d93782SGreg Clayton                         if (submenus.size() == 7)
437444d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
437544d93782SGreg Clayton                         else if (submenus.size() > 8)
437644d93782SGreg Clayton                             submenus.erase (submenus.begin() + 8, submenus.end());
437744d93782SGreg Clayton 
437844d93782SGreg Clayton                         ThreadList &threads = process->GetThreadList();
437944d93782SGreg Clayton                         Mutex::Locker locker (threads.GetMutex());
438044d93782SGreg Clayton                         size_t num_threads = threads.GetSize();
438144d93782SGreg Clayton                         for (size_t i=0; i<num_threads; ++i)
438244d93782SGreg Clayton                         {
438344d93782SGreg Clayton                             ThreadSP thread_sp = threads.GetThreadAtIndex(i);
438444d93782SGreg Clayton                             char menu_char = '\0';
438544d93782SGreg Clayton                             if (i < 9)
438644d93782SGreg Clayton                                 menu_char = '1' + i;
438744d93782SGreg Clayton                             StreamString thread_menu_title;
438844d93782SGreg Clayton                             thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
438944d93782SGreg Clayton                             const char *thread_name = thread_sp->GetName();
439044d93782SGreg Clayton                             if (thread_name && thread_name[0])
439144d93782SGreg Clayton                                 thread_menu_title.Printf (" %s", thread_name);
439244d93782SGreg Clayton                             else
439344d93782SGreg Clayton                             {
439444d93782SGreg Clayton                                 const char *queue_name = thread_sp->GetQueueName();
439544d93782SGreg Clayton                                 if (queue_name && queue_name[0])
439644d93782SGreg Clayton                                     thread_menu_title.Printf (" %s", queue_name);
439744d93782SGreg Clayton                             }
439844d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(thread_menu_title.GetString().c_str(), NULL, menu_char, thread_sp->GetID())));
439944d93782SGreg Clayton                         }
440044d93782SGreg Clayton                     }
440144d93782SGreg Clayton                     else if (submenus.size() > 7)
440244d93782SGreg Clayton                     {
440344d93782SGreg Clayton                         // Remove the separator and any other thread submenu items
440444d93782SGreg Clayton                         // that were previously added
440544d93782SGreg Clayton                         submenus.erase (submenus.begin() + 7, submenus.end());
440644d93782SGreg Clayton                     }
440744d93782SGreg Clayton                     // Since we are adding and removing items we need to recalculate the name lengths
440844d93782SGreg Clayton                     menu.RecalculateNameLengths();
440944d93782SGreg Clayton                 }
441044d93782SGreg Clayton                 return MenuActionResult::Handled;
441144d93782SGreg Clayton 
441244d93782SGreg Clayton             case eMenuID_ViewVariables:
441344d93782SGreg Clayton                 {
441444d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
441544d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
441644d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
441744d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
441844d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
441944d93782SGreg Clayton 
442044d93782SGreg Clayton                     if (variables_window_sp)
442144d93782SGreg Clayton                     {
442244d93782SGreg Clayton                         const Rect variables_bounds = variables_window_sp->GetBounds();
442344d93782SGreg Clayton 
442444d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(variables_window_sp.get());
442544d93782SGreg Clayton 
442644d93782SGreg Clayton                         if (registers_window_sp)
442744d93782SGreg Clayton                         {
442844d93782SGreg Clayton                             // We have a registers window, so give all the area back to the registers window
442944d93782SGreg Clayton                             Rect registers_bounds = variables_bounds;
443044d93782SGreg Clayton                             registers_bounds.size.width = source_bounds.size.width;
443144d93782SGreg Clayton                             registers_window_sp->SetBounds(registers_bounds);
443244d93782SGreg Clayton                         }
443344d93782SGreg Clayton                         else
443444d93782SGreg Clayton                         {
443544d93782SGreg Clayton                             // We have no registers window showing so give the bottom
443644d93782SGreg Clayton                             // area back to the source view
443744d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
443844d93782SGreg Clayton                                                       source_bounds.size.height + variables_bounds.size.height);
443944d93782SGreg Clayton                         }
444044d93782SGreg Clayton                     }
444144d93782SGreg Clayton                     else
444244d93782SGreg Clayton                     {
444344d93782SGreg Clayton                         Rect new_variables_rect;
444444d93782SGreg Clayton                         if (registers_window_sp)
444544d93782SGreg Clayton                         {
444644d93782SGreg Clayton                             // We have a registers window so split the area of the registers
444744d93782SGreg Clayton                             // window into two columns where the left hand side will be the
444844d93782SGreg Clayton                             // variables and the right hand side will be the registers
444944d93782SGreg Clayton                             const Rect variables_bounds = registers_window_sp->GetBounds();
445044d93782SGreg Clayton                             Rect new_registers_rect;
445144d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect);
445244d93782SGreg Clayton                             registers_window_sp->SetBounds (new_registers_rect);
445344d93782SGreg Clayton                         }
445444d93782SGreg Clayton                         else
445544d93782SGreg Clayton                         {
445644d93782SGreg Clayton                             // No variables window, grab the bottom part of the source window
445744d93782SGreg Clayton                             Rect new_source_rect;
445844d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect);
445944d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
446044d93782SGreg Clayton                         }
446144d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables",
446244d93782SGreg Clayton                                                                                   new_variables_rect,
446344d93782SGreg Clayton                                                                                   false);
446444d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
446544d93782SGreg Clayton                     }
446644d93782SGreg Clayton                     touchwin(stdscr);
446744d93782SGreg Clayton                 }
446844d93782SGreg Clayton                 return MenuActionResult::Handled;
446944d93782SGreg Clayton 
447044d93782SGreg Clayton             case eMenuID_ViewRegisters:
447144d93782SGreg Clayton                 {
447244d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
447344d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
447444d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
447544d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
447644d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
447744d93782SGreg Clayton 
447844d93782SGreg Clayton                     if (registers_window_sp)
447944d93782SGreg Clayton                     {
448044d93782SGreg Clayton                         if (variables_window_sp)
448144d93782SGreg Clayton                         {
448244d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
448344d93782SGreg Clayton 
448444d93782SGreg Clayton                             // We have a variables window, so give all the area back to the variables window
448544d93782SGreg Clayton                             variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(),
448644d93782SGreg Clayton                                                          variables_bounds.size.height);
448744d93782SGreg Clayton                         }
448844d93782SGreg Clayton                         else
448944d93782SGreg Clayton                         {
449044d93782SGreg Clayton                             // We have no variables window showing so give the bottom
449144d93782SGreg Clayton                             // area back to the source view
449244d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
449344d93782SGreg Clayton                                                       source_bounds.size.height + registers_window_sp->GetHeight());
449444d93782SGreg Clayton                         }
449544d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(registers_window_sp.get());
449644d93782SGreg Clayton                     }
449744d93782SGreg Clayton                     else
449844d93782SGreg Clayton                     {
449944d93782SGreg Clayton                         Rect new_regs_rect;
450044d93782SGreg Clayton                         if (variables_window_sp)
450144d93782SGreg Clayton                         {
450244d93782SGreg Clayton                             // We have a variables window, split it into two columns
450344d93782SGreg Clayton                             // where the left hand side will be the variables and the
450444d93782SGreg Clayton                             // right hand side will be the registers
450544d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
450644d93782SGreg Clayton                             Rect new_vars_rect;
450744d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect);
450844d93782SGreg Clayton                             variables_window_sp->SetBounds (new_vars_rect);
450944d93782SGreg Clayton                         }
451044d93782SGreg Clayton                         else
451144d93782SGreg Clayton                         {
451244d93782SGreg Clayton                             // No registers window, grab the bottom part of the source window
451344d93782SGreg Clayton                             Rect new_source_rect;
451444d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect);
451544d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
451644d93782SGreg Clayton                         }
451744d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers",
451844d93782SGreg Clayton                                                                                   new_regs_rect,
451944d93782SGreg Clayton                                                                                   false);
452044d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
452144d93782SGreg Clayton                     }
452244d93782SGreg Clayton                     touchwin(stdscr);
452344d93782SGreg Clayton                 }
452444d93782SGreg Clayton                 return MenuActionResult::Handled;
452544d93782SGreg Clayton 
452644d93782SGreg Clayton             case eMenuID_HelpGUIHelp:
45275fdb09bbSGreg Clayton                 m_app.GetMainWindow ()->CreateHelpSubwindow();
452844d93782SGreg Clayton                 return MenuActionResult::Handled;
452944d93782SGreg Clayton 
453044d93782SGreg Clayton             default:
453144d93782SGreg Clayton                 break;
453244d93782SGreg Clayton         }
453344d93782SGreg Clayton 
453444d93782SGreg Clayton         return MenuActionResult::NotHandled;
453544d93782SGreg Clayton     }
453644d93782SGreg Clayton protected:
453744d93782SGreg Clayton     Application &m_app;
453844d93782SGreg Clayton     Debugger &m_debugger;
453944d93782SGreg Clayton };
454044d93782SGreg Clayton 
454144d93782SGreg Clayton 
454244d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate
454344d93782SGreg Clayton {
454444d93782SGreg Clayton public:
454544d93782SGreg Clayton     StatusBarWindowDelegate (Debugger &debugger) :
454644d93782SGreg Clayton         m_debugger (debugger)
454744d93782SGreg Clayton     {
454844d93782SGreg Clayton     }
454944d93782SGreg Clayton 
455044d93782SGreg Clayton     virtual
455144d93782SGreg Clayton     ~StatusBarWindowDelegate ()
455244d93782SGreg Clayton     {
455344d93782SGreg Clayton     }
455444d93782SGreg Clayton     virtual bool
455544d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
455644d93782SGreg Clayton     {
455744d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
455844d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
455944d93782SGreg Clayton         Thread *thread = exe_ctx.GetThreadPtr();
456044d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
456144d93782SGreg Clayton         window.Erase();
456244d93782SGreg Clayton         window.SetBackground(2);
456344d93782SGreg Clayton         window.MoveCursor (0, 0);
456444d93782SGreg Clayton         if (process)
456544d93782SGreg Clayton         {
456644d93782SGreg Clayton             const StateType state = process->GetState();
456744d93782SGreg Clayton             window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state));
456844d93782SGreg Clayton 
456944d93782SGreg Clayton             if (StateIsStoppedState(state, true))
457044d93782SGreg Clayton             {
45715b031ebcSEd Maste                 StreamString strm;
45725b031ebcSEd Maste                 const char *format = "Thread: ${thread.id%tid}";
45735b031ebcSEd Maste                 if (thread && Debugger::FormatPrompt (format, NULL, &exe_ctx, NULL, strm))
45745b031ebcSEd Maste                 {
457544d93782SGreg Clayton                     window.MoveCursor (40, 0);
45765b031ebcSEd Maste                     window.PutCStringTruncated(strm.GetString().c_str(), 1);
45775b031ebcSEd Maste                 }
457844d93782SGreg Clayton 
457944d93782SGreg Clayton                 window.MoveCursor (60, 0);
458044d93782SGreg Clayton                 if (frame)
458144d93782SGreg Clayton                     window.Printf ("Frame: %3u  PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr()));
458244d93782SGreg Clayton             }
458344d93782SGreg Clayton             else if (state == eStateExited)
458444d93782SGreg Clayton             {
458544d93782SGreg Clayton                 const char *exit_desc = process->GetExitDescription();
458644d93782SGreg Clayton                 const int exit_status = process->GetExitStatus();
458744d93782SGreg Clayton                 if (exit_desc && exit_desc[0])
458844d93782SGreg Clayton                     window.Printf (" with status = %i (%s)", exit_status, exit_desc);
458944d93782SGreg Clayton                 else
459044d93782SGreg Clayton                     window.Printf (" with status = %i", exit_status);
459144d93782SGreg Clayton             }
459244d93782SGreg Clayton         }
459344d93782SGreg Clayton         window.DeferredRefresh();
459444d93782SGreg Clayton         return true;
459544d93782SGreg Clayton     }
459644d93782SGreg Clayton 
459744d93782SGreg Clayton protected:
459844d93782SGreg Clayton     Debugger &m_debugger;
459944d93782SGreg Clayton };
460044d93782SGreg Clayton 
460144d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate
460244d93782SGreg Clayton {
460344d93782SGreg Clayton public:
460444d93782SGreg Clayton     SourceFileWindowDelegate (Debugger &debugger) :
460544d93782SGreg Clayton         WindowDelegate (),
460644d93782SGreg Clayton         m_debugger (debugger),
460744d93782SGreg Clayton         m_sc (),
460844d93782SGreg Clayton         m_file_sp (),
460944d93782SGreg Clayton         m_disassembly_scope (NULL),
461044d93782SGreg Clayton         m_disassembly_sp (),
461144d93782SGreg Clayton         m_disassembly_range (),
4612ec990867SGreg Clayton         m_title (),
461344d93782SGreg Clayton         m_line_width (4),
461444d93782SGreg Clayton         m_selected_line (0),
461544d93782SGreg Clayton         m_pc_line (0),
461644d93782SGreg Clayton         m_stop_id (0),
461744d93782SGreg Clayton         m_frame_idx (UINT32_MAX),
461844d93782SGreg Clayton         m_first_visible_line (0),
461944d93782SGreg Clayton         m_min_x (0),
462044d93782SGreg Clayton         m_min_y (0),
462144d93782SGreg Clayton         m_max_x (0),
462244d93782SGreg Clayton         m_max_y (0)
462344d93782SGreg Clayton     {
462444d93782SGreg Clayton     }
462544d93782SGreg Clayton 
462644d93782SGreg Clayton     virtual
462744d93782SGreg Clayton     ~SourceFileWindowDelegate()
462844d93782SGreg Clayton     {
462944d93782SGreg Clayton     }
463044d93782SGreg Clayton 
463144d93782SGreg Clayton     void
463244d93782SGreg Clayton     Update (const SymbolContext &sc)
463344d93782SGreg Clayton     {
463444d93782SGreg Clayton         m_sc = sc;
463544d93782SGreg Clayton     }
463644d93782SGreg Clayton 
463744d93782SGreg Clayton     uint32_t
463844d93782SGreg Clayton     NumVisibleLines () const
463944d93782SGreg Clayton     {
464044d93782SGreg Clayton         return m_max_y - m_min_y;
464144d93782SGreg Clayton     }
464244d93782SGreg Clayton 
464344d93782SGreg Clayton     virtual const char *
464444d93782SGreg Clayton     WindowDelegateGetHelpText ()
464544d93782SGreg Clayton     {
464644d93782SGreg Clayton         return "Source/Disassembly window keyboard shortcuts:";
464744d93782SGreg Clayton     }
464844d93782SGreg Clayton 
464944d93782SGreg Clayton     virtual KeyHelp *
465044d93782SGreg Clayton     WindowDelegateGetKeyHelp ()
465144d93782SGreg Clayton     {
465244d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
465344d93782SGreg Clayton             { KEY_RETURN, "Run to selected line with one shot breakpoint" },
465444d93782SGreg Clayton             { KEY_UP, "Select previous source line" },
465544d93782SGreg Clayton             { KEY_DOWN, "Select next source line" },
465644d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
465744d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
465844d93782SGreg Clayton             { 'b', "Set breakpoint on selected source/disassembly line" },
465944d93782SGreg Clayton             { 'c', "Continue process" },
466044d93782SGreg Clayton             { 'd', "Detach and resume process" },
466144d93782SGreg Clayton             { 'D', "Detach with process suspended" },
466244d93782SGreg Clayton             { 'h', "Show help dialog" },
466344d93782SGreg Clayton             { 'k', "Kill process" },
466444d93782SGreg Clayton             { 'n', "Step over (source line)" },
466544d93782SGreg Clayton             { 'N', "Step over (single instruction)" },
466644d93782SGreg Clayton             { 'o', "Step out" },
466744d93782SGreg Clayton             { 's', "Step in (source line)" },
466844d93782SGreg Clayton             { 'S', "Step in (single instruction)" },
466944d93782SGreg Clayton             { ',', "Page up" },
467044d93782SGreg Clayton             { '.', "Page down" },
467144d93782SGreg Clayton             { '\0', NULL }
467244d93782SGreg Clayton         };
467344d93782SGreg Clayton         return g_source_view_key_help;
467444d93782SGreg Clayton     }
467544d93782SGreg Clayton 
467644d93782SGreg Clayton     virtual bool
467744d93782SGreg Clayton     WindowDelegateDraw (Window &window, bool force)
467844d93782SGreg Clayton     {
467944d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
468044d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
468144d93782SGreg Clayton         Thread *thread = NULL;
468244d93782SGreg Clayton 
468344d93782SGreg Clayton         bool update_location = false;
468444d93782SGreg Clayton         if (process)
468544d93782SGreg Clayton         {
468644d93782SGreg Clayton             StateType state = process->GetState();
468744d93782SGreg Clayton             if (StateIsStoppedState(state, true))
468844d93782SGreg Clayton             {
468944d93782SGreg Clayton                 // We are stopped, so it is ok to
469044d93782SGreg Clayton                 update_location = true;
469144d93782SGreg Clayton             }
469244d93782SGreg Clayton         }
469344d93782SGreg Clayton 
469444d93782SGreg Clayton         m_min_x = 1;
4695ec990867SGreg Clayton         m_min_y = 2;
469644d93782SGreg Clayton         m_max_x = window.GetMaxX()-1;
469744d93782SGreg Clayton         m_max_y = window.GetMaxY()-1;
469844d93782SGreg Clayton 
469944d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
470044d93782SGreg Clayton         StackFrameSP frame_sp;
470144d93782SGreg Clayton         bool set_selected_line_to_pc = false;
470244d93782SGreg Clayton 
470344d93782SGreg Clayton         if (update_location)
470444d93782SGreg Clayton         {
470544d93782SGreg Clayton             const bool process_alive = process ? process->IsAlive() : false;
470644d93782SGreg Clayton             bool thread_changed = false;
470744d93782SGreg Clayton             if (process_alive)
470844d93782SGreg Clayton             {
470944d93782SGreg Clayton                 thread = exe_ctx.GetThreadPtr();
471044d93782SGreg Clayton                 if (thread)
471144d93782SGreg Clayton                 {
471244d93782SGreg Clayton                     frame_sp = thread->GetSelectedFrame();
471344d93782SGreg Clayton                     auto tid = thread->GetID();
471444d93782SGreg Clayton                     thread_changed = tid != m_tid;
471544d93782SGreg Clayton                     m_tid = tid;
471644d93782SGreg Clayton                 }
471744d93782SGreg Clayton                 else
471844d93782SGreg Clayton                 {
471944d93782SGreg Clayton                     if (m_tid != LLDB_INVALID_THREAD_ID)
472044d93782SGreg Clayton                     {
472144d93782SGreg Clayton                         thread_changed = true;
472244d93782SGreg Clayton                         m_tid = LLDB_INVALID_THREAD_ID;
472344d93782SGreg Clayton                     }
472444d93782SGreg Clayton                 }
472544d93782SGreg Clayton             }
472644d93782SGreg Clayton             const uint32_t stop_id = process ? process->GetStopID() : 0;
472744d93782SGreg Clayton             const bool stop_id_changed = stop_id != m_stop_id;
472844d93782SGreg Clayton             bool frame_changed = false;
472944d93782SGreg Clayton             m_stop_id = stop_id;
4730ec990867SGreg Clayton             m_title.Clear();
473144d93782SGreg Clayton             if (frame_sp)
473244d93782SGreg Clayton             {
473344d93782SGreg Clayton                 m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
4734ec990867SGreg Clayton                 if (m_sc.module_sp)
4735ec990867SGreg Clayton                 {
4736ec990867SGreg Clayton                     m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
4737ec990867SGreg Clayton                     ConstString func_name = m_sc.GetFunctionName();
4738ec990867SGreg Clayton                     if (func_name)
4739ec990867SGreg Clayton                         m_title.Printf("`%s", func_name.GetCString());
4740ec990867SGreg Clayton                 }
474144d93782SGreg Clayton                 const uint32_t frame_idx = frame_sp->GetFrameIndex();
474244d93782SGreg Clayton                 frame_changed = frame_idx != m_frame_idx;
474344d93782SGreg Clayton                 m_frame_idx = frame_idx;
474444d93782SGreg Clayton             }
474544d93782SGreg Clayton             else
474644d93782SGreg Clayton             {
474744d93782SGreg Clayton                 m_sc.Clear(true);
474844d93782SGreg Clayton                 frame_changed = m_frame_idx != UINT32_MAX;
474944d93782SGreg Clayton                 m_frame_idx = UINT32_MAX;
475044d93782SGreg Clayton             }
475144d93782SGreg Clayton 
475244d93782SGreg Clayton             const bool context_changed = thread_changed || frame_changed || stop_id_changed;
475344d93782SGreg Clayton 
475444d93782SGreg Clayton             if (process_alive)
475544d93782SGreg Clayton             {
475644d93782SGreg Clayton                 if (m_sc.line_entry.IsValid())
475744d93782SGreg Clayton                 {
475844d93782SGreg Clayton                     m_pc_line = m_sc.line_entry.line;
475944d93782SGreg Clayton                     if (m_pc_line != UINT32_MAX)
476044d93782SGreg Clayton                         --m_pc_line; // Convert to zero based line number...
476144d93782SGreg Clayton                     // Update the selected line if the stop ID changed...
476244d93782SGreg Clayton                     if (context_changed)
476344d93782SGreg Clayton                         m_selected_line = m_pc_line;
476444d93782SGreg Clayton 
476544d93782SGreg Clayton                     if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file))
476644d93782SGreg Clayton                     {
476744d93782SGreg Clayton                         // Same file, nothing to do, we should either have the
476844d93782SGreg Clayton                         // lines or not (source file missing)
47693985c8c6SSaleem Abdulrasool                         if (m_selected_line >= static_cast<size_t>(m_first_visible_line))
477044d93782SGreg Clayton                         {
477144d93782SGreg Clayton                             if (m_selected_line >= m_first_visible_line + num_visible_lines)
477244d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
477344d93782SGreg Clayton                         }
477444d93782SGreg Clayton                         else
477544d93782SGreg Clayton                         {
477644d93782SGreg Clayton                             if (m_selected_line > 10)
477744d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
477844d93782SGreg Clayton                             else
477944d93782SGreg Clayton                                 m_first_visible_line = 0;
478044d93782SGreg Clayton                         }
478144d93782SGreg Clayton                     }
478244d93782SGreg Clayton                     else
478344d93782SGreg Clayton                     {
478444d93782SGreg Clayton                         // File changed, set selected line to the line with the PC
478544d93782SGreg Clayton                         m_selected_line = m_pc_line;
478644d93782SGreg Clayton                         m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
478744d93782SGreg Clayton                         if (m_file_sp)
478844d93782SGreg Clayton                         {
478944d93782SGreg Clayton                             const size_t num_lines = m_file_sp->GetNumLines();
479044d93782SGreg Clayton                             int m_line_width = 1;
479144d93782SGreg Clayton                             for (size_t n = num_lines; n >= 10; n = n / 10)
479244d93782SGreg Clayton                                 ++m_line_width;
479344d93782SGreg Clayton 
479444d93782SGreg Clayton                             snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width);
479544d93782SGreg Clayton                             if (num_lines < num_visible_lines || m_selected_line < num_visible_lines)
479644d93782SGreg Clayton                                 m_first_visible_line = 0;
479744d93782SGreg Clayton                             else
479844d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
479944d93782SGreg Clayton                         }
480044d93782SGreg Clayton                     }
480144d93782SGreg Clayton                 }
480244d93782SGreg Clayton                 else
480344d93782SGreg Clayton                 {
480444d93782SGreg Clayton                     m_file_sp.reset();
480544d93782SGreg Clayton                 }
480644d93782SGreg Clayton 
480744d93782SGreg Clayton                 if (!m_file_sp || m_file_sp->GetNumLines() == 0)
480844d93782SGreg Clayton                 {
480944d93782SGreg Clayton                     // Show disassembly
481044d93782SGreg Clayton                     bool prefer_file_cache = false;
481144d93782SGreg Clayton                     if (m_sc.function)
481244d93782SGreg Clayton                     {
481344d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.function)
481444d93782SGreg Clayton                         {
481544d93782SGreg Clayton                             m_disassembly_scope = m_sc.function;
481644d93782SGreg Clayton                             m_disassembly_sp = m_sc.function->GetInstructions (exe_ctx, NULL, prefer_file_cache);
481744d93782SGreg Clayton                             if (m_disassembly_sp)
481844d93782SGreg Clayton                             {
481944d93782SGreg Clayton                                 set_selected_line_to_pc = true;
482044d93782SGreg Clayton                                 m_disassembly_range = m_sc.function->GetAddressRange();
482144d93782SGreg Clayton                             }
482244d93782SGreg Clayton                             else
482344d93782SGreg Clayton                             {
482444d93782SGreg Clayton                                 m_disassembly_range.Clear();
482544d93782SGreg Clayton                             }
482644d93782SGreg Clayton                         }
482744d93782SGreg Clayton                         else
482844d93782SGreg Clayton                         {
482944d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
483044d93782SGreg Clayton                         }
483144d93782SGreg Clayton                     }
483244d93782SGreg Clayton                     else if (m_sc.symbol)
483344d93782SGreg Clayton                     {
483444d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.symbol)
483544d93782SGreg Clayton                         {
483644d93782SGreg Clayton                             m_disassembly_scope = m_sc.symbol;
483744d93782SGreg Clayton                             m_disassembly_sp = m_sc.symbol->GetInstructions (exe_ctx, NULL, prefer_file_cache);
483844d93782SGreg Clayton                             if (m_disassembly_sp)
483944d93782SGreg Clayton                             {
484044d93782SGreg Clayton                                 set_selected_line_to_pc = true;
484144d93782SGreg Clayton                                 m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress();
484244d93782SGreg Clayton                                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
484344d93782SGreg Clayton                             }
484444d93782SGreg Clayton                             else
484544d93782SGreg Clayton                             {
484644d93782SGreg Clayton                                 m_disassembly_range.Clear();
484744d93782SGreg Clayton                             }
484844d93782SGreg Clayton                         }
484944d93782SGreg Clayton                         else
485044d93782SGreg Clayton                         {
485144d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
485244d93782SGreg Clayton                         }
485344d93782SGreg Clayton                     }
485444d93782SGreg Clayton                 }
485544d93782SGreg Clayton             }
485644d93782SGreg Clayton             else
485744d93782SGreg Clayton             {
485844d93782SGreg Clayton                 m_pc_line = UINT32_MAX;
485944d93782SGreg Clayton             }
486044d93782SGreg Clayton         }
486144d93782SGreg Clayton 
4862ec990867SGreg Clayton         const int window_width = window.GetWidth();
486344d93782SGreg Clayton         window.Erase();
486444d93782SGreg Clayton         window.DrawTitleBox ("Sources");
4865ec990867SGreg Clayton         if (!m_title.GetString().empty())
4866ec990867SGreg Clayton         {
4867ec990867SGreg Clayton             window.AttributeOn(A_REVERSE);
4868ec990867SGreg Clayton             window.MoveCursor(1, 1);
4869ec990867SGreg Clayton             window.PutChar(' ');
4870ec990867SGreg Clayton             window.PutCStringTruncated(m_title.GetString().c_str(), 1);
4871ec990867SGreg Clayton             int x = window.GetCursorX();
4872ec990867SGreg Clayton             if (x < window_width - 1)
4873ec990867SGreg Clayton             {
4874ec990867SGreg Clayton                 window.Printf ("%*s", window_width - x - 1, "");
4875ec990867SGreg Clayton             }
4876ec990867SGreg Clayton             window.AttributeOff(A_REVERSE);
4877ec990867SGreg Clayton         }
487844d93782SGreg Clayton 
487944d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
488044d93782SGreg Clayton         const size_t num_source_lines = GetNumSourceLines();
488144d93782SGreg Clayton         if (num_source_lines > 0)
488244d93782SGreg Clayton         {
488344d93782SGreg Clayton             // Display source
488444d93782SGreg Clayton             BreakpointLines bp_lines;
488544d93782SGreg Clayton             if (target)
488644d93782SGreg Clayton             {
488744d93782SGreg Clayton                 BreakpointList &bp_list = target->GetBreakpointList();
488844d93782SGreg Clayton                 const size_t num_bps = bp_list.GetSize();
488944d93782SGreg Clayton                 for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
489044d93782SGreg Clayton                 {
489144d93782SGreg Clayton                     BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
489244d93782SGreg Clayton                     const size_t num_bps_locs = bp_sp->GetNumLocations();
489344d93782SGreg Clayton                     for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
489444d93782SGreg Clayton                     {
489544d93782SGreg Clayton                         BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
489644d93782SGreg Clayton                         LineEntry bp_loc_line_entry;
489744d93782SGreg Clayton                         if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry))
489844d93782SGreg Clayton                         {
489944d93782SGreg Clayton                             if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file)
490044d93782SGreg Clayton                             {
490144d93782SGreg Clayton                                 bp_lines.insert(bp_loc_line_entry.line);
490244d93782SGreg Clayton                             }
490344d93782SGreg Clayton                         }
490444d93782SGreg Clayton                     }
490544d93782SGreg Clayton                 }
490644d93782SGreg Clayton             }
490744d93782SGreg Clayton 
490844d93782SGreg Clayton             const attr_t selected_highlight_attr = A_REVERSE;
490944d93782SGreg Clayton             const attr_t pc_highlight_attr = COLOR_PAIR(1);
491044d93782SGreg Clayton 
49113985c8c6SSaleem Abdulrasool             for (size_t i=0; i<num_visible_lines; ++i)
491244d93782SGreg Clayton             {
491344d93782SGreg Clayton                 const uint32_t curr_line = m_first_visible_line + i;
491444d93782SGreg Clayton                 if (curr_line < num_source_lines)
491544d93782SGreg Clayton                 {
4916ec990867SGreg Clayton                     const int line_y = m_min_y+i;
491744d93782SGreg Clayton                     window.MoveCursor(1, line_y);
491844d93782SGreg Clayton                     const bool is_pc_line = curr_line == m_pc_line;
491944d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == curr_line;
492044d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
492144d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
492244d93782SGreg Clayton                     attr_t highlight_attr = 0;
492344d93782SGreg Clayton                     attr_t bp_attr = 0;
492444d93782SGreg Clayton                     if (is_pc_line)
492544d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
492644d93782SGreg Clayton                     else if (line_is_selected)
492744d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
492844d93782SGreg Clayton 
492944d93782SGreg Clayton                     if (bp_lines.find(curr_line+1) != bp_lines.end())
493044d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
493144d93782SGreg Clayton 
493244d93782SGreg Clayton                     if (bp_attr)
493344d93782SGreg Clayton                         window.AttributeOn(bp_attr);
493444d93782SGreg Clayton 
493544d93782SGreg Clayton                     window.Printf (m_line_format, curr_line + 1);
493644d93782SGreg Clayton 
493744d93782SGreg Clayton                     if (bp_attr)
493844d93782SGreg Clayton                         window.AttributeOff(bp_attr);
493944d93782SGreg Clayton 
494044d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
494144d93782SGreg Clayton                     // Mark the line with the PC with a diamond
494244d93782SGreg Clayton                     if (is_pc_line)
494344d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
494444d93782SGreg Clayton                     else
494544d93782SGreg Clayton                         window.PutChar(' ');
494644d93782SGreg Clayton 
494744d93782SGreg Clayton                     if (highlight_attr)
494844d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
494944d93782SGreg Clayton                     const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false);
495044d93782SGreg Clayton                     if (line_len > 0)
495144d93782SGreg Clayton                         window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
495244d93782SGreg Clayton 
495344d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
495444d93782SGreg Clayton                     {
495544d93782SGreg Clayton                         StopInfoSP stop_info_sp;
495644d93782SGreg Clayton                         if (thread)
495744d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
495844d93782SGreg Clayton                         if (stop_info_sp)
495944d93782SGreg Clayton                         {
496044d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
496144d93782SGreg Clayton                             if (stop_description && stop_description[0])
496244d93782SGreg Clayton                             {
496344d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
4964ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
496544d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
4966ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
496744d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
496844d93782SGreg Clayton                             }
496944d93782SGreg Clayton                         }
497044d93782SGreg Clayton                         else
497144d93782SGreg Clayton                         {
4972ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
497344d93782SGreg Clayton                         }
497444d93782SGreg Clayton                     }
497544d93782SGreg Clayton                     if (highlight_attr)
497644d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
497744d93782SGreg Clayton 
497844d93782SGreg Clayton                 }
497944d93782SGreg Clayton                 else
498044d93782SGreg Clayton                 {
498144d93782SGreg Clayton                     break;
498244d93782SGreg Clayton                 }
498344d93782SGreg Clayton             }
498444d93782SGreg Clayton         }
498544d93782SGreg Clayton         else
498644d93782SGreg Clayton         {
498744d93782SGreg Clayton             size_t num_disassembly_lines = GetNumDisassemblyLines();
498844d93782SGreg Clayton             if (num_disassembly_lines > 0)
498944d93782SGreg Clayton             {
499044d93782SGreg Clayton                 // Display disassembly
499144d93782SGreg Clayton                 BreakpointAddrs bp_file_addrs;
499244d93782SGreg Clayton                 Target *target = exe_ctx.GetTargetPtr();
499344d93782SGreg Clayton                 if (target)
499444d93782SGreg Clayton                 {
499544d93782SGreg Clayton                     BreakpointList &bp_list = target->GetBreakpointList();
499644d93782SGreg Clayton                     const size_t num_bps = bp_list.GetSize();
499744d93782SGreg Clayton                     for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
499844d93782SGreg Clayton                     {
499944d93782SGreg Clayton                         BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
500044d93782SGreg Clayton                         const size_t num_bps_locs = bp_sp->GetNumLocations();
500144d93782SGreg Clayton                         for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
500244d93782SGreg Clayton                         {
500344d93782SGreg Clayton                             BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
500444d93782SGreg Clayton                             LineEntry bp_loc_line_entry;
500544d93782SGreg Clayton                             const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress();
500644d93782SGreg Clayton                             if (file_addr != LLDB_INVALID_ADDRESS)
500744d93782SGreg Clayton                             {
500844d93782SGreg Clayton                                 if (m_disassembly_range.ContainsFileAddress(file_addr))
500944d93782SGreg Clayton                                     bp_file_addrs.insert(file_addr);
501044d93782SGreg Clayton                             }
501144d93782SGreg Clayton                         }
501244d93782SGreg Clayton                     }
501344d93782SGreg Clayton                 }
501444d93782SGreg Clayton 
501544d93782SGreg Clayton                 const attr_t selected_highlight_attr = A_REVERSE;
501644d93782SGreg Clayton                 const attr_t pc_highlight_attr = COLOR_PAIR(1);
501744d93782SGreg Clayton 
501844d93782SGreg Clayton                 StreamString strm;
501944d93782SGreg Clayton 
502044d93782SGreg Clayton                 InstructionList &insts = m_disassembly_sp->GetInstructionList();
502144d93782SGreg Clayton                 Address pc_address;
502244d93782SGreg Clayton 
502344d93782SGreg Clayton                 if (frame_sp)
502444d93782SGreg Clayton                     pc_address = frame_sp->GetFrameCodeAddress();
502544d93782SGreg Clayton                 const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX;
502644d93782SGreg Clayton                 if (set_selected_line_to_pc)
502744d93782SGreg Clayton                 {
502844d93782SGreg Clayton                     m_selected_line = pc_idx;
502944d93782SGreg Clayton                 }
503044d93782SGreg Clayton 
503144d93782SGreg Clayton                 const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
50323985c8c6SSaleem Abdulrasool                 if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
503344d93782SGreg Clayton                     m_first_visible_line = 0;
503444d93782SGreg Clayton 
503544d93782SGreg Clayton                 if (pc_idx < num_disassembly_lines)
503644d93782SGreg Clayton                 {
50373985c8c6SSaleem Abdulrasool                     if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
503844d93782SGreg Clayton                         pc_idx >= m_first_visible_line + num_visible_lines)
503944d93782SGreg Clayton                         m_first_visible_line = pc_idx - non_visible_pc_offset;
504044d93782SGreg Clayton                 }
504144d93782SGreg Clayton 
504244d93782SGreg Clayton                 for (size_t i=0; i<num_visible_lines; ++i)
504344d93782SGreg Clayton                 {
504444d93782SGreg Clayton                     const uint32_t inst_idx = m_first_visible_line + i;
504544d93782SGreg Clayton                     Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
504644d93782SGreg Clayton                     if (!inst)
504744d93782SGreg Clayton                         break;
504844d93782SGreg Clayton 
5049ec990867SGreg Clayton                     const int line_y = m_min_y+i;
5050ec990867SGreg Clayton                     window.MoveCursor(1, line_y);
505144d93782SGreg Clayton                     const bool is_pc_line = frame_sp && inst_idx == pc_idx;
505244d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == inst_idx;
505344d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
505444d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
505544d93782SGreg Clayton                     attr_t highlight_attr = 0;
505644d93782SGreg Clayton                     attr_t bp_attr = 0;
505744d93782SGreg Clayton                     if (is_pc_line)
505844d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
505944d93782SGreg Clayton                     else if (line_is_selected)
506044d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
506144d93782SGreg Clayton 
506244d93782SGreg Clayton                     if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end())
506344d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
506444d93782SGreg Clayton 
506544d93782SGreg Clayton                     if (bp_attr)
506644d93782SGreg Clayton                         window.AttributeOn(bp_attr);
506744d93782SGreg Clayton 
5068324a1036SSaleem Abdulrasool                     window.Printf (" 0x%16.16llx ",
5069324a1036SSaleem Abdulrasool                                    static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target)));
507044d93782SGreg Clayton 
507144d93782SGreg Clayton                     if (bp_attr)
507244d93782SGreg Clayton                         window.AttributeOff(bp_attr);
507344d93782SGreg Clayton 
507444d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
507544d93782SGreg Clayton                     // Mark the line with the PC with a diamond
507644d93782SGreg Clayton                     if (is_pc_line)
507744d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
507844d93782SGreg Clayton                     else
507944d93782SGreg Clayton                         window.PutChar(' ');
508044d93782SGreg Clayton 
508144d93782SGreg Clayton                     if (highlight_attr)
508244d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
508344d93782SGreg Clayton 
508444d93782SGreg Clayton                     const char *mnemonic = inst->GetMnemonic(&exe_ctx);
508544d93782SGreg Clayton                     const char *operands = inst->GetOperands(&exe_ctx);
508644d93782SGreg Clayton                     const char *comment = inst->GetComment(&exe_ctx);
508744d93782SGreg Clayton 
508844d93782SGreg Clayton                     if (mnemonic && mnemonic[0] == '\0')
508944d93782SGreg Clayton                         mnemonic = NULL;
509044d93782SGreg Clayton                     if (operands && operands[0] == '\0')
509144d93782SGreg Clayton                         operands = NULL;
509244d93782SGreg Clayton                     if (comment && comment[0] == '\0')
509344d93782SGreg Clayton                         comment = NULL;
509444d93782SGreg Clayton 
509544d93782SGreg Clayton                     strm.Clear();
509644d93782SGreg Clayton 
509744d93782SGreg Clayton                     if (mnemonic && operands && comment)
509844d93782SGreg Clayton                         strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment);
509944d93782SGreg Clayton                     else if (mnemonic && operands)
510044d93782SGreg Clayton                         strm.Printf ("%-8s %s", mnemonic, operands);
510144d93782SGreg Clayton                     else if (mnemonic)
510244d93782SGreg Clayton                         strm.Printf ("%s", mnemonic);
510344d93782SGreg Clayton 
510444d93782SGreg Clayton                     int right_pad = 1;
510544d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
510644d93782SGreg Clayton 
510744d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
510844d93782SGreg Clayton                     {
510944d93782SGreg Clayton                         StopInfoSP stop_info_sp;
511044d93782SGreg Clayton                         if (thread)
511144d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
511244d93782SGreg Clayton                         if (stop_info_sp)
511344d93782SGreg Clayton                         {
511444d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
511544d93782SGreg Clayton                             if (stop_description && stop_description[0])
511644d93782SGreg Clayton                             {
511744d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5118ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
511944d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5120ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
512144d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
512244d93782SGreg Clayton                             }
512344d93782SGreg Clayton                         }
512444d93782SGreg Clayton                         else
512544d93782SGreg Clayton                         {
5126ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
512744d93782SGreg Clayton                         }
512844d93782SGreg Clayton                     }
512944d93782SGreg Clayton                     if (highlight_attr)
513044d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
513144d93782SGreg Clayton                 }
513244d93782SGreg Clayton             }
513344d93782SGreg Clayton         }
513444d93782SGreg Clayton         window.DeferredRefresh();
513544d93782SGreg Clayton         return true; // Drawing handled
513644d93782SGreg Clayton     }
513744d93782SGreg Clayton 
513844d93782SGreg Clayton     size_t
513944d93782SGreg Clayton     GetNumLines ()
514044d93782SGreg Clayton     {
514144d93782SGreg Clayton         size_t num_lines = GetNumSourceLines();
514244d93782SGreg Clayton         if (num_lines == 0)
514344d93782SGreg Clayton             num_lines = GetNumDisassemblyLines();
514444d93782SGreg Clayton         return num_lines;
514544d93782SGreg Clayton     }
514644d93782SGreg Clayton 
514744d93782SGreg Clayton     size_t
514844d93782SGreg Clayton     GetNumSourceLines () const
514944d93782SGreg Clayton     {
515044d93782SGreg Clayton         if (m_file_sp)
515144d93782SGreg Clayton             return m_file_sp->GetNumLines();
515244d93782SGreg Clayton         return 0;
515344d93782SGreg Clayton     }
515444d93782SGreg Clayton     size_t
515544d93782SGreg Clayton     GetNumDisassemblyLines () const
515644d93782SGreg Clayton     {
515744d93782SGreg Clayton         if (m_disassembly_sp)
515844d93782SGreg Clayton             return m_disassembly_sp->GetInstructionList().GetSize();
515944d93782SGreg Clayton         return 0;
516044d93782SGreg Clayton     }
516144d93782SGreg Clayton 
516244d93782SGreg Clayton     virtual HandleCharResult
516344d93782SGreg Clayton     WindowDelegateHandleChar (Window &window, int c)
516444d93782SGreg Clayton     {
516544d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
516644d93782SGreg Clayton         const size_t num_lines = GetNumLines ();
516744d93782SGreg Clayton 
516844d93782SGreg Clayton         switch (c)
516944d93782SGreg Clayton         {
517044d93782SGreg Clayton             case ',':
517144d93782SGreg Clayton             case KEY_PPAGE:
517244d93782SGreg Clayton                 // Page up key
51733985c8c6SSaleem Abdulrasool                 if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
517444d93782SGreg Clayton                     m_first_visible_line -= num_visible_lines;
517544d93782SGreg Clayton                 else
517644d93782SGreg Clayton                     m_first_visible_line = 0;
517744d93782SGreg Clayton                 m_selected_line = m_first_visible_line;
517844d93782SGreg Clayton                 return eKeyHandled;
517944d93782SGreg Clayton 
518044d93782SGreg Clayton             case '.':
518144d93782SGreg Clayton             case KEY_NPAGE:
518244d93782SGreg Clayton                 // Page down key
518344d93782SGreg Clayton                 {
518444d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < num_lines)
518544d93782SGreg Clayton                         m_first_visible_line += num_visible_lines;
518644d93782SGreg Clayton                     else if (num_lines < num_visible_lines)
518744d93782SGreg Clayton                         m_first_visible_line = 0;
518844d93782SGreg Clayton                     else
518944d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
519044d93782SGreg Clayton                     m_selected_line = m_first_visible_line;
519144d93782SGreg Clayton                 }
519244d93782SGreg Clayton                 return eKeyHandled;
519344d93782SGreg Clayton 
519444d93782SGreg Clayton             case KEY_UP:
519544d93782SGreg Clayton                 if (m_selected_line > 0)
519644d93782SGreg Clayton                 {
519744d93782SGreg Clayton                     m_selected_line--;
51983985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
519944d93782SGreg Clayton                         m_first_visible_line = m_selected_line;
520044d93782SGreg Clayton                 }
520144d93782SGreg Clayton                 return eKeyHandled;
520244d93782SGreg Clayton 
520344d93782SGreg Clayton             case KEY_DOWN:
520444d93782SGreg Clayton                 if (m_selected_line + 1 < num_lines)
520544d93782SGreg Clayton                 {
520644d93782SGreg Clayton                     m_selected_line++;
520744d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < m_selected_line)
520844d93782SGreg Clayton                         m_first_visible_line++;
520944d93782SGreg Clayton                 }
521044d93782SGreg Clayton                 return eKeyHandled;
521144d93782SGreg Clayton 
521244d93782SGreg Clayton             case '\r':
521344d93782SGreg Clayton             case '\n':
521444d93782SGreg Clayton             case KEY_ENTER:
521544d93782SGreg Clayton                 // Set a breakpoint and run to the line using a one shot breakpoint
521644d93782SGreg Clayton                 if (GetNumSourceLines() > 0)
521744d93782SGreg Clayton                 {
521844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
521944d93782SGreg Clayton                     if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive())
522044d93782SGreg Clayton                     {
522144d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
522244d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
522344d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
522444d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
522544d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
522644d93782SGreg Clayton                                                                                       false,                     // internal
522744d93782SGreg Clayton                                                                                       false);                    // request_hardware
522844d93782SGreg Clayton                         // Make breakpoint one shot
522944d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
523044d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
523144d93782SGreg Clayton                     }
523244d93782SGreg Clayton                 }
523344d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
523444d93782SGreg Clayton                 {
523544d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
523644d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
523744d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
523844d93782SGreg Clayton                     {
523944d93782SGreg Clayton                         Address addr = inst->GetAddress();
524044d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
524144d93782SGreg Clayton                                                                                       false,    // internal
524244d93782SGreg Clayton                                                                                       false);   // request_hardware
524344d93782SGreg Clayton                         // Make breakpoint one shot
524444d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
524544d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
524644d93782SGreg Clayton                     }
524744d93782SGreg Clayton                 }
524844d93782SGreg Clayton                 return eKeyHandled;
524944d93782SGreg Clayton 
525044d93782SGreg Clayton             case 'b':   // 'b' == toggle breakpoint on currently selected line
525144d93782SGreg Clayton                 if (m_selected_line < GetNumSourceLines())
525244d93782SGreg Clayton                 {
525344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
525444d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
525544d93782SGreg Clayton                     {
525644d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
525744d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
525844d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
525944d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
526044d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
526144d93782SGreg Clayton                                                                                       false,                     // internal
526244d93782SGreg Clayton                                                                                       false);                    // request_hardware
526344d93782SGreg Clayton                     }
526444d93782SGreg Clayton                 }
526544d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
526644d93782SGreg Clayton                 {
526744d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
526844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
526944d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
527044d93782SGreg Clayton                     {
527144d93782SGreg Clayton                         Address addr = inst->GetAddress();
527244d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
527344d93782SGreg Clayton                                                                                       false,    // internal
527444d93782SGreg Clayton                                                                                       false);   // request_hardware
527544d93782SGreg Clayton                     }
527644d93782SGreg Clayton                 }
527744d93782SGreg Clayton                 return eKeyHandled;
527844d93782SGreg Clayton 
527944d93782SGreg Clayton             case 'd':   // 'd' == detach and let run
528044d93782SGreg Clayton             case 'D':   // 'D' == detach and keep stopped
528144d93782SGreg Clayton                 {
528244d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
528344d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
528444d93782SGreg Clayton                         exe_ctx.GetProcessRef().Detach(c == 'D');
528544d93782SGreg Clayton                 }
528644d93782SGreg Clayton                 return eKeyHandled;
528744d93782SGreg Clayton 
528844d93782SGreg Clayton             case 'k':
528944d93782SGreg Clayton                 // 'k' == kill
529044d93782SGreg Clayton                 {
529144d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
529244d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
529344d93782SGreg Clayton                         exe_ctx.GetProcessRef().Destroy();
529444d93782SGreg Clayton                 }
529544d93782SGreg Clayton                 return eKeyHandled;
529644d93782SGreg Clayton 
529744d93782SGreg Clayton             case 'c':
529844d93782SGreg Clayton                 // 'c' == continue
529944d93782SGreg Clayton                 {
530044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
530144d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
530244d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
530344d93782SGreg Clayton                 }
530444d93782SGreg Clayton                 return eKeyHandled;
530544d93782SGreg Clayton 
530644d93782SGreg Clayton             case 'o':
530744d93782SGreg Clayton                 // 'o' == step out
530844d93782SGreg Clayton                 {
530944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
531044d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
531144d93782SGreg Clayton                     {
531244d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOut();
531344d93782SGreg Clayton                     }
531444d93782SGreg Clayton                 }
531544d93782SGreg Clayton                 return eKeyHandled;
531644d93782SGreg Clayton             case 'n':   // 'n' == step over
531744d93782SGreg Clayton             case 'N':   // 'N' == step over instruction
531844d93782SGreg Clayton                 {
531944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
532044d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
532144d93782SGreg Clayton                     {
532244d93782SGreg Clayton                         bool source_step = (c == 'n');
532344d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOver(source_step);
532444d93782SGreg Clayton                     }
532544d93782SGreg Clayton                 }
532644d93782SGreg Clayton                 return eKeyHandled;
532744d93782SGreg Clayton             case 's':   // 's' == step into
532844d93782SGreg Clayton             case 'S':   // 'S' == step into instruction
532944d93782SGreg Clayton                 {
533044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
533144d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
533244d93782SGreg Clayton                     {
533344d93782SGreg Clayton                         bool source_step = (c == 's');
53344b4b2478SJim Ingham                         exe_ctx.GetThreadRef().StepIn(source_step);
533544d93782SGreg Clayton                     }
533644d93782SGreg Clayton                 }
533744d93782SGreg Clayton                 return eKeyHandled;
533844d93782SGreg Clayton 
533944d93782SGreg Clayton             case 'h':
534044d93782SGreg Clayton                 window.CreateHelpSubwindow ();
534144d93782SGreg Clayton                 return eKeyHandled;
534244d93782SGreg Clayton 
534344d93782SGreg Clayton             default:
534444d93782SGreg Clayton                 break;
534544d93782SGreg Clayton         }
534644d93782SGreg Clayton         return eKeyNotHandled;
534744d93782SGreg Clayton     }
534844d93782SGreg Clayton 
534944d93782SGreg Clayton protected:
535044d93782SGreg Clayton     typedef std::set<uint32_t> BreakpointLines;
535144d93782SGreg Clayton     typedef std::set<lldb::addr_t> BreakpointAddrs;
535244d93782SGreg Clayton 
535344d93782SGreg Clayton     Debugger &m_debugger;
535444d93782SGreg Clayton     SymbolContext m_sc;
535544d93782SGreg Clayton     SourceManager::FileSP m_file_sp;
535644d93782SGreg Clayton     SymbolContextScope *m_disassembly_scope;
535744d93782SGreg Clayton     lldb::DisassemblerSP m_disassembly_sp;
535844d93782SGreg Clayton     AddressRange m_disassembly_range;
5359ec990867SGreg Clayton     StreamString m_title;
536044d93782SGreg Clayton     lldb::user_id_t m_tid;
536144d93782SGreg Clayton     char m_line_format[8];
536244d93782SGreg Clayton     int m_line_width;
536344d93782SGreg Clayton     uint32_t m_selected_line;       // The selected line
536444d93782SGreg Clayton     uint32_t m_pc_line;             // The line with the PC
536544d93782SGreg Clayton     uint32_t m_stop_id;
536644d93782SGreg Clayton     uint32_t m_frame_idx;
536744d93782SGreg Clayton     int m_first_visible_line;
536844d93782SGreg Clayton     int m_min_x;
536944d93782SGreg Clayton     int m_min_y;
537044d93782SGreg Clayton     int m_max_x;
537144d93782SGreg Clayton     int m_max_y;
537244d93782SGreg Clayton 
537344d93782SGreg Clayton };
537444d93782SGreg Clayton 
537544d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true };
537644d93782SGreg Clayton 
537744d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) :
537844d93782SGreg Clayton     IOHandler (debugger)
537944d93782SGreg Clayton {
538044d93782SGreg Clayton }
538144d93782SGreg Clayton 
538244d93782SGreg Clayton void
538344d93782SGreg Clayton IOHandlerCursesGUI::Activate ()
538444d93782SGreg Clayton {
538544d93782SGreg Clayton     IOHandler::Activate();
538644d93782SGreg Clayton     if (!m_app_ap)
538744d93782SGreg Clayton     {
538844d93782SGreg Clayton         m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE()));
538944d93782SGreg Clayton 
539044d93782SGreg Clayton 
539144d93782SGreg Clayton         // This is both a window and a menu delegate
539244d93782SGreg Clayton         std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger));
539344d93782SGreg Clayton 
539444d93782SGreg Clayton         MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
539544d93782SGreg Clayton         MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
539644d93782SGreg Clayton         MenuSP exit_menuitem_sp(new Menu("Exit", NULL, 'x', ApplicationDelegate::eMenuID_LLDBExit));
539744d93782SGreg Clayton         exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
539844d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", NULL, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
539944d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
540044d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (exit_menuitem_sp);
540144d93782SGreg Clayton 
540244d93782SGreg Clayton         MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target));
540344d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", NULL, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
540444d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", NULL, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
540544d93782SGreg Clayton 
540644d93782SGreg Clayton         MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process));
540744d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach"  , NULL, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
540844d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach"  , NULL, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
540944d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch"  , NULL, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
541044d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
541144d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", NULL, 'c', ApplicationDelegate::eMenuID_ProcessContinue)));
541244d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt"    , NULL, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
541344d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill"    , NULL, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
541444d93782SGreg Clayton 
541544d93782SGreg Clayton         MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread));
541644d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In"  , NULL, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
541744d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", NULL, 'v', ApplicationDelegate::eMenuID_ThreadStepOver)));
541844d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , NULL, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
541944d93782SGreg Clayton 
542044d93782SGreg Clayton         MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
542144d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", NULL, 'b', ApplicationDelegate::eMenuID_ViewBacktrace)));
542244d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", NULL, 'r', ApplicationDelegate::eMenuID_ViewRegisters)));
542344d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Source"   , NULL, 's', ApplicationDelegate::eMenuID_ViewSource)));
542444d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", NULL, 'v', ApplicationDelegate::eMenuID_ViewVariables)));
542544d93782SGreg Clayton 
542644d93782SGreg Clayton         MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
542744d93782SGreg Clayton         help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", NULL, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
542844d93782SGreg Clayton 
542944d93782SGreg Clayton         m_app_ap->Initialize();
543044d93782SGreg Clayton         WindowSP &main_window_sp = m_app_ap->GetMainWindow();
543144d93782SGreg Clayton 
543244d93782SGreg Clayton         MenuSP menubar_sp(new Menu(Menu::Type::Bar));
543344d93782SGreg Clayton         menubar_sp->AddSubmenu (lldb_menu_sp);
543444d93782SGreg Clayton         menubar_sp->AddSubmenu (target_menu_sp);
543544d93782SGreg Clayton         menubar_sp->AddSubmenu (process_menu_sp);
543644d93782SGreg Clayton         menubar_sp->AddSubmenu (thread_menu_sp);
543744d93782SGreg Clayton         menubar_sp->AddSubmenu (view_menu_sp);
543844d93782SGreg Clayton         menubar_sp->AddSubmenu (help_menu_sp);
543944d93782SGreg Clayton         menubar_sp->SetDelegate(app_menu_delegate_sp);
544044d93782SGreg Clayton 
544144d93782SGreg Clayton         Rect content_bounds = main_window_sp->GetFrame();
544244d93782SGreg Clayton         Rect menubar_bounds = content_bounds.MakeMenuBar();
544344d93782SGreg Clayton         Rect status_bounds = content_bounds.MakeStatusBar();
544444d93782SGreg Clayton         Rect source_bounds;
544544d93782SGreg Clayton         Rect variables_bounds;
544644d93782SGreg Clayton         Rect threads_bounds;
544744d93782SGreg Clayton         Rect source_variables_bounds;
544844d93782SGreg Clayton         content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds);
544944d93782SGreg Clayton         source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds);
545044d93782SGreg Clayton 
545144d93782SGreg Clayton         WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
545244d93782SGreg Clayton         // Let the menubar get keys if the active window doesn't handle the
545344d93782SGreg Clayton         // keys that are typed so it can respond to menubar key presses.
545444d93782SGreg Clayton         menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window
545544d93782SGreg Clayton         menubar_window_sp->SetDelegate(menubar_sp);
545644d93782SGreg Clayton 
545744d93782SGreg Clayton         WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source",
545844d93782SGreg Clayton                                                                    source_bounds,
545944d93782SGreg Clayton                                                                    true));
546044d93782SGreg Clayton         WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables",
546144d93782SGreg Clayton                                                                       variables_bounds,
546244d93782SGreg Clayton                                                                       false));
546344d93782SGreg Clayton         WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads",
546444d93782SGreg Clayton                                                                       threads_bounds,
546544d93782SGreg Clayton                                                                       false));
546644d93782SGreg Clayton         WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status",
546744d93782SGreg Clayton                                                                    status_bounds,
546844d93782SGreg Clayton                                                                    false));
546944d93782SGreg Clayton         status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window
547044d93782SGreg Clayton         main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
547144d93782SGreg Clayton         source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
547244d93782SGreg Clayton         variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
5473ec990867SGreg Clayton         TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger));
547444d93782SGreg Clayton         threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
547544d93782SGreg Clayton         status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
547644d93782SGreg Clayton 
54775fdb09bbSGreg Clayton         // Show the main help window once the first time the curses GUI is launched
54785fdb09bbSGreg Clayton         static bool g_showed_help = false;
54795fdb09bbSGreg Clayton         if (!g_showed_help)
54805fdb09bbSGreg Clayton         {
54815fdb09bbSGreg Clayton             g_showed_help = true;
54825fdb09bbSGreg Clayton             main_window_sp->CreateHelpSubwindow();
54835fdb09bbSGreg Clayton         }
54845fdb09bbSGreg Clayton 
548544d93782SGreg Clayton         init_pair (1, COLOR_WHITE   , COLOR_BLUE  );
548644d93782SGreg Clayton         init_pair (2, COLOR_BLACK   , COLOR_WHITE );
548744d93782SGreg Clayton         init_pair (3, COLOR_MAGENTA , COLOR_WHITE );
548844d93782SGreg Clayton         init_pair (4, COLOR_MAGENTA , COLOR_BLACK );
548944d93782SGreg Clayton         init_pair (5, COLOR_RED     , COLOR_BLACK );
549044d93782SGreg Clayton 
549144d93782SGreg Clayton     }
549244d93782SGreg Clayton }
549344d93782SGreg Clayton 
549444d93782SGreg Clayton void
549544d93782SGreg Clayton IOHandlerCursesGUI::Deactivate ()
549644d93782SGreg Clayton {
549744d93782SGreg Clayton     m_app_ap->Terminate();
549844d93782SGreg Clayton }
549944d93782SGreg Clayton 
550044d93782SGreg Clayton void
550144d93782SGreg Clayton IOHandlerCursesGUI::Run ()
550244d93782SGreg Clayton {
550344d93782SGreg Clayton     m_app_ap->Run(m_debugger);
550444d93782SGreg Clayton     SetIsDone(true);
550544d93782SGreg Clayton }
550644d93782SGreg Clayton 
550744d93782SGreg Clayton 
550844d93782SGreg Clayton IOHandlerCursesGUI::~IOHandlerCursesGUI ()
550944d93782SGreg Clayton {
551044d93782SGreg Clayton 
551144d93782SGreg Clayton }
551244d93782SGreg Clayton 
551344d93782SGreg Clayton void
551444d93782SGreg Clayton IOHandlerCursesGUI::Hide ()
551544d93782SGreg Clayton {
551644d93782SGreg Clayton }
551744d93782SGreg Clayton 
551844d93782SGreg Clayton 
551944d93782SGreg Clayton void
552044d93782SGreg Clayton IOHandlerCursesGUI::Refresh ()
552144d93782SGreg Clayton {
552244d93782SGreg Clayton }
552344d93782SGreg Clayton 
5524e68f5d6bSGreg Clayton void
5525e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel ()
5526e68f5d6bSGreg Clayton {
5527e68f5d6bSGreg Clayton }
552844d93782SGreg Clayton 
5529f0066ad0SGreg Clayton bool
553044d93782SGreg Clayton IOHandlerCursesGUI::Interrupt ()
553144d93782SGreg Clayton {
5532f0066ad0SGreg Clayton     return false;
553344d93782SGreg Clayton }
553444d93782SGreg Clayton 
553544d93782SGreg Clayton 
553644d93782SGreg Clayton void
553744d93782SGreg Clayton IOHandlerCursesGUI::GotEOF()
553844d93782SGreg Clayton {
553944d93782SGreg Clayton }
554044d93782SGreg Clayton 
5541914b8d98SDeepak Panickal #endif // #ifndef LLDB_DISABLE_CURSES
5542