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