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