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