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 
10315b6884SEugene Zelenko // C Includes
11315b6884SEugene Zelenko #ifndef LLDB_DISABLE_CURSES
1227801f4fSBruce Mitchener #include <curses.h>
13315b6884SEugene Zelenko #include <panel.h>
14315b6884SEugene Zelenko #endif
1544d93782SGreg Clayton 
16315b6884SEugene Zelenko // C++ Includes
177c9aa073STodd Fiala #if defined(__APPLE__)
187c9aa073STodd Fiala #include <deque>
197c9aa073STodd Fiala #endif
2044d93782SGreg Clayton #include <string>
2144d93782SGreg Clayton 
22315b6884SEugene Zelenko // Other libraries and framework includes
23315b6884SEugene Zelenko // Project includes
2444d93782SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h"
2544d93782SGreg Clayton #include "lldb/Core/IOHandler.h"
2644d93782SGreg Clayton #include "lldb/Core/Debugger.h"
27ec990867SGreg Clayton #include "lldb/Core/Module.h"
2844d93782SGreg Clayton #include "lldb/Core/State.h"
2944d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
3044d93782SGreg Clayton #include "lldb/Core/ValueObjectRegister.h"
31cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
3244d93782SGreg Clayton #include "lldb/Host/Editline.h"
33cacde7dfSTodd Fiala #endif
3444d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h"
3544d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h"
3644d93782SGreg Clayton #include "lldb/Symbol/Block.h"
3744d93782SGreg Clayton #include "lldb/Symbol/Function.h"
3844d93782SGreg Clayton #include "lldb/Symbol/Symbol.h"
3944d93782SGreg Clayton #include "lldb/Target/RegisterContext.h"
4044d93782SGreg Clayton #include "lldb/Target/ThreadPlan.h"
4144d93782SGreg Clayton 
427c9aa073STodd Fiala 
437c9aa073STodd Fiala 
4444d93782SGreg Clayton using namespace lldb;
4544d93782SGreg Clayton using namespace lldb_private;
4644d93782SGreg Clayton 
47e30f11d9SKate Stone IOHandler::IOHandler (Debugger &debugger, IOHandler::Type type) :
4844d93782SGreg Clayton     IOHandler (debugger,
49e30f11d9SKate Stone                type,
5044d93782SGreg Clayton                StreamFileSP(),  // Adopt STDIN from top input reader
5144d93782SGreg Clayton                StreamFileSP(),  // Adopt STDOUT from top input reader
52340b0309SGreg Clayton                StreamFileSP(),  // Adopt STDERR from top input reader
53340b0309SGreg Clayton                0)               // Flags
5444d93782SGreg Clayton {
5544d93782SGreg Clayton }
5644d93782SGreg Clayton 
5744d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger,
58e30f11d9SKate Stone                       IOHandler::Type type,
5944d93782SGreg Clayton                       const lldb::StreamFileSP &input_sp,
6044d93782SGreg Clayton                       const lldb::StreamFileSP &output_sp,
61340b0309SGreg Clayton                       const lldb::StreamFileSP &error_sp,
62340b0309SGreg Clayton                       uint32_t flags) :
6344d93782SGreg Clayton     m_debugger (debugger),
6444d93782SGreg Clayton     m_input_sp (input_sp),
6544d93782SGreg Clayton     m_output_sp (output_sp),
6644d93782SGreg Clayton     m_error_sp (error_sp),
67e30f11d9SKate Stone     m_popped (false),
68340b0309SGreg Clayton     m_flags (flags),
69e30f11d9SKate Stone     m_type (type),
7044d93782SGreg Clayton     m_user_data (NULL),
7144d93782SGreg Clayton     m_done (false),
7244d93782SGreg Clayton     m_active (false)
7344d93782SGreg Clayton {
7444d93782SGreg Clayton     // If any files are not specified, then adopt them from the top input reader.
7544d93782SGreg Clayton     if (!m_input_sp || !m_output_sp || !m_error_sp)
7644d93782SGreg Clayton         debugger.AdoptTopIOHandlerFilesIfInvalid (m_input_sp,
7744d93782SGreg Clayton                                                   m_output_sp,
7844d93782SGreg Clayton                                                   m_error_sp);
7944d93782SGreg Clayton }
8044d93782SGreg Clayton 
81315b6884SEugene Zelenko IOHandler::~IOHandler() = default;
8244d93782SGreg Clayton 
8344d93782SGreg Clayton int
8444d93782SGreg Clayton IOHandler::GetInputFD()
8544d93782SGreg Clayton {
8644d93782SGreg Clayton     if (m_input_sp)
8744d93782SGreg Clayton         return m_input_sp->GetFile().GetDescriptor();
8844d93782SGreg Clayton     return -1;
8944d93782SGreg Clayton }
9044d93782SGreg Clayton 
9144d93782SGreg Clayton int
9244d93782SGreg Clayton IOHandler::GetOutputFD()
9344d93782SGreg Clayton {
9444d93782SGreg Clayton     if (m_output_sp)
9544d93782SGreg Clayton         return m_output_sp->GetFile().GetDescriptor();
9644d93782SGreg Clayton     return -1;
9744d93782SGreg Clayton }
9844d93782SGreg Clayton 
9944d93782SGreg Clayton int
10044d93782SGreg Clayton IOHandler::GetErrorFD()
10144d93782SGreg Clayton {
10244d93782SGreg Clayton     if (m_error_sp)
10344d93782SGreg Clayton         return m_error_sp->GetFile().GetDescriptor();
10444d93782SGreg Clayton     return -1;
10544d93782SGreg Clayton }
10644d93782SGreg Clayton 
10744d93782SGreg Clayton FILE *
10844d93782SGreg Clayton IOHandler::GetInputFILE()
10944d93782SGreg Clayton {
11044d93782SGreg Clayton     if (m_input_sp)
11144d93782SGreg Clayton         return m_input_sp->GetFile().GetStream();
11244d93782SGreg Clayton     return NULL;
11344d93782SGreg Clayton }
11444d93782SGreg Clayton 
11544d93782SGreg Clayton FILE *
11644d93782SGreg Clayton IOHandler::GetOutputFILE()
11744d93782SGreg Clayton {
11844d93782SGreg Clayton     if (m_output_sp)
11944d93782SGreg Clayton         return m_output_sp->GetFile().GetStream();
12044d93782SGreg Clayton     return NULL;
12144d93782SGreg Clayton }
12244d93782SGreg Clayton 
12344d93782SGreg Clayton FILE *
12444d93782SGreg Clayton IOHandler::GetErrorFILE()
12544d93782SGreg Clayton {
12644d93782SGreg Clayton     if (m_error_sp)
12744d93782SGreg Clayton         return m_error_sp->GetFile().GetStream();
12844d93782SGreg Clayton     return NULL;
12944d93782SGreg Clayton }
13044d93782SGreg Clayton 
13144d93782SGreg Clayton StreamFileSP &
13244d93782SGreg Clayton IOHandler::GetInputStreamFile()
13344d93782SGreg Clayton {
13444d93782SGreg Clayton     return m_input_sp;
13544d93782SGreg Clayton }
13644d93782SGreg Clayton 
13744d93782SGreg Clayton StreamFileSP &
13844d93782SGreg Clayton IOHandler::GetOutputStreamFile()
13944d93782SGreg Clayton {
14044d93782SGreg Clayton     return m_output_sp;
14144d93782SGreg Clayton }
14244d93782SGreg Clayton 
14344d93782SGreg Clayton StreamFileSP &
14444d93782SGreg Clayton IOHandler::GetErrorStreamFile()
14544d93782SGreg Clayton {
14644d93782SGreg Clayton     return m_error_sp;
14744d93782SGreg Clayton }
14844d93782SGreg Clayton 
149340b0309SGreg Clayton bool
150340b0309SGreg Clayton IOHandler::GetIsInteractive ()
151340b0309SGreg Clayton {
152340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsInteractive ();
153340b0309SGreg Clayton }
154340b0309SGreg Clayton 
155340b0309SGreg Clayton bool
156340b0309SGreg Clayton IOHandler::GetIsRealTerminal ()
157340b0309SGreg Clayton {
158340b0309SGreg Clayton     return GetInputStreamFile()->GetFile().GetIsRealTerminal();
159340b0309SGreg Clayton }
16044d93782SGreg Clayton 
161e30f11d9SKate Stone void
162e30f11d9SKate Stone IOHandler::SetPopped (bool b)
163e30f11d9SKate Stone {
164e30f11d9SKate Stone     m_popped.SetValue(b, eBroadcastOnChange);
165e30f11d9SKate Stone }
166e30f11d9SKate Stone 
167e30f11d9SKate Stone void
168e30f11d9SKate Stone IOHandler::WaitForPop ()
169e30f11d9SKate Stone {
170e30f11d9SKate Stone     m_popped.WaitForValueEqualTo(true);
171e30f11d9SKate Stone }
172e30f11d9SKate Stone 
1734446487dSPavel Labath void
1744446487dSPavel Labath IOHandlerStack::PrintAsync (Stream *stream, const char *s, size_t len)
1754446487dSPavel Labath {
1764446487dSPavel Labath     if (stream)
1774446487dSPavel Labath     {
1784446487dSPavel Labath         Mutex::Locker locker (m_mutex);
1794446487dSPavel Labath         if (m_top)
1804446487dSPavel Labath             m_top->PrintAsync (stream, s, len);
1814446487dSPavel Labath     }
1824446487dSPavel Labath }
1834446487dSPavel Labath 
18444d93782SGreg Clayton IOHandlerConfirm::IOHandlerConfirm (Debugger &debugger,
18544d93782SGreg Clayton                                     const char *prompt,
18644d93782SGreg Clayton                                     bool default_response) :
18744d93782SGreg Clayton     IOHandlerEditline(debugger,
188e30f11d9SKate Stone                       IOHandler::Type::Confirm,
18944d93782SGreg Clayton                       NULL,     // NULL editline_name means no history loaded/saved
190e30f11d9SKate Stone                       NULL,     // No prompt
191e30f11d9SKate Stone                       NULL,     // No continuation prompt
19244d93782SGreg Clayton                       false,    // Multi-line
193e30f11d9SKate Stone                       false,    // Don't colorize the prompt (i.e. the confirm message.)
194f6913cd7SGreg Clayton                       0,
19544d93782SGreg Clayton                       *this),
19644d93782SGreg Clayton     m_default_response (default_response),
19744d93782SGreg Clayton     m_user_response (default_response)
19844d93782SGreg Clayton {
19944d93782SGreg Clayton     StreamString prompt_stream;
20044d93782SGreg Clayton     prompt_stream.PutCString(prompt);
20144d93782SGreg Clayton     if (m_default_response)
20244d93782SGreg Clayton         prompt_stream.Printf(": [Y/n] ");
20344d93782SGreg Clayton     else
20444d93782SGreg Clayton         prompt_stream.Printf(": [y/N] ");
20544d93782SGreg Clayton 
20644d93782SGreg Clayton     SetPrompt (prompt_stream.GetString().c_str());
20744d93782SGreg Clayton 
20844d93782SGreg Clayton }
20944d93782SGreg Clayton 
210315b6884SEugene Zelenko IOHandlerConfirm::~IOHandlerConfirm() = default;
21144d93782SGreg Clayton 
21244d93782SGreg Clayton int
21344d93782SGreg Clayton IOHandlerConfirm::IOHandlerComplete (IOHandler &io_handler,
21444d93782SGreg Clayton                                      const char *current_line,
21544d93782SGreg Clayton                                      const char *cursor,
21644d93782SGreg Clayton                                      const char *last_char,
21744d93782SGreg Clayton                                      int skip_first_n_matches,
21844d93782SGreg Clayton                                      int max_matches,
21944d93782SGreg Clayton                                      StringList &matches)
22044d93782SGreg Clayton {
22144d93782SGreg Clayton     if (current_line == cursor)
22244d93782SGreg Clayton     {
22344d93782SGreg Clayton         if (m_default_response)
22444d93782SGreg Clayton         {
22544d93782SGreg Clayton             matches.AppendString("y");
22644d93782SGreg Clayton         }
22744d93782SGreg Clayton         else
22844d93782SGreg Clayton         {
22944d93782SGreg Clayton             matches.AppendString("n");
23044d93782SGreg Clayton         }
23144d93782SGreg Clayton     }
23244d93782SGreg Clayton     return matches.GetSize();
23344d93782SGreg Clayton }
23444d93782SGreg Clayton 
23544d93782SGreg Clayton void
23644d93782SGreg Clayton IOHandlerConfirm::IOHandlerInputComplete (IOHandler &io_handler, std::string &line)
23744d93782SGreg Clayton {
23844d93782SGreg Clayton     if (line.empty())
23944d93782SGreg Clayton     {
24044d93782SGreg Clayton         // User just hit enter, set the response to the default
24144d93782SGreg Clayton         m_user_response = m_default_response;
24244d93782SGreg Clayton         io_handler.SetIsDone(true);
24344d93782SGreg Clayton         return;
24444d93782SGreg Clayton     }
24544d93782SGreg Clayton 
24644d93782SGreg Clayton     if (line.size() == 1)
24744d93782SGreg Clayton     {
24844d93782SGreg Clayton         switch (line[0])
24944d93782SGreg Clayton         {
25044d93782SGreg Clayton             case 'y':
25144d93782SGreg Clayton             case 'Y':
25244d93782SGreg Clayton                 m_user_response = true;
25344d93782SGreg Clayton                 io_handler.SetIsDone(true);
25444d93782SGreg Clayton                 return;
25544d93782SGreg Clayton             case 'n':
25644d93782SGreg Clayton             case 'N':
25744d93782SGreg Clayton                 m_user_response = false;
25844d93782SGreg Clayton                 io_handler.SetIsDone(true);
25944d93782SGreg Clayton                 return;
26044d93782SGreg Clayton             default:
26144d93782SGreg Clayton                 break;
26244d93782SGreg Clayton         }
26344d93782SGreg Clayton     }
26444d93782SGreg Clayton 
26544d93782SGreg Clayton     if (line == "yes" || line == "YES" || line == "Yes")
26644d93782SGreg Clayton     {
26744d93782SGreg Clayton         m_user_response = true;
26844d93782SGreg Clayton         io_handler.SetIsDone(true);
26944d93782SGreg Clayton     }
27044d93782SGreg Clayton     else if (line == "no" || line == "NO" || line == "No")
27144d93782SGreg Clayton     {
27244d93782SGreg Clayton         m_user_response = false;
27344d93782SGreg Clayton         io_handler.SetIsDone(true);
27444d93782SGreg Clayton     }
27544d93782SGreg Clayton }
27644d93782SGreg Clayton 
27744d93782SGreg Clayton int
27844d93782SGreg Clayton IOHandlerDelegate::IOHandlerComplete (IOHandler &io_handler,
27944d93782SGreg Clayton                                       const char *current_line,
28044d93782SGreg Clayton                                       const char *cursor,
28144d93782SGreg Clayton                                       const char *last_char,
28244d93782SGreg Clayton                                       int skip_first_n_matches,
28344d93782SGreg Clayton                                       int max_matches,
28444d93782SGreg Clayton                                       StringList &matches)
28544d93782SGreg Clayton {
28644d93782SGreg Clayton     switch (m_completion)
28744d93782SGreg Clayton     {
28844d93782SGreg Clayton     case Completion::None:
28944d93782SGreg Clayton         break;
29044d93782SGreg Clayton 
29144d93782SGreg Clayton     case Completion::LLDBCommand:
29244d93782SGreg Clayton         return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion (current_line,
29344d93782SGreg Clayton                                                                                   cursor,
29444d93782SGreg Clayton                                                                                   last_char,
29544d93782SGreg Clayton                                                                                   skip_first_n_matches,
29644d93782SGreg Clayton                                                                                   max_matches,
29744d93782SGreg Clayton                                                                                   matches);
29844d93782SGreg Clayton 
29944d93782SGreg Clayton     case Completion::Expression:
30044d93782SGreg Clayton         {
30144d93782SGreg Clayton             bool word_complete = false;
30244d93782SGreg Clayton             const char *word_start = cursor;
30344d93782SGreg Clayton             if (cursor > current_line)
30444d93782SGreg Clayton                 --word_start;
30544d93782SGreg Clayton             while (word_start > current_line && !isspace(*word_start))
30644d93782SGreg Clayton                 --word_start;
30744d93782SGreg Clayton             CommandCompletions::InvokeCommonCompletionCallbacks (io_handler.GetDebugger().GetCommandInterpreter(),
30844d93782SGreg Clayton                                                                  CommandCompletions::eVariablePathCompletion,
30944d93782SGreg Clayton                                                                  word_start,
31044d93782SGreg Clayton                                                                  skip_first_n_matches,
31144d93782SGreg Clayton                                                                  max_matches,
31244d93782SGreg Clayton                                                                  NULL,
31344d93782SGreg Clayton                                                                  word_complete,
31444d93782SGreg Clayton                                                                  matches);
31544d93782SGreg Clayton 
31644d93782SGreg Clayton             size_t num_matches = matches.GetSize();
31744d93782SGreg Clayton             if (num_matches > 0)
31844d93782SGreg Clayton             {
31944d93782SGreg Clayton                 std::string common_prefix;
32044d93782SGreg Clayton                 matches.LongestCommonPrefix (common_prefix);
32144d93782SGreg Clayton                 const size_t partial_name_len = strlen(word_start);
32244d93782SGreg Clayton 
32344d93782SGreg Clayton                 // If we matched a unique single command, add a space...
32444d93782SGreg Clayton                 // Only do this if the completer told us this was a complete word, however...
32544d93782SGreg Clayton                 if (num_matches == 1 && word_complete)
32644d93782SGreg Clayton                 {
32744d93782SGreg Clayton                     common_prefix.push_back(' ');
32844d93782SGreg Clayton                 }
32944d93782SGreg Clayton                 common_prefix.erase (0, partial_name_len);
33044d93782SGreg Clayton                 matches.InsertStringAtIndex(0, std::move(common_prefix));
33144d93782SGreg Clayton             }
33244d93782SGreg Clayton             return num_matches;
33344d93782SGreg Clayton         }
33444d93782SGreg Clayton         break;
33544d93782SGreg Clayton     }
33644d93782SGreg Clayton 
33744d93782SGreg Clayton     return 0;
33844d93782SGreg Clayton }
33944d93782SGreg Clayton 
34044d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
341e30f11d9SKate Stone                                       IOHandler::Type type,
34244d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
34344d93782SGreg Clayton                                       const char *prompt,
344e30f11d9SKate Stone                                       const char *continuation_prompt,
34544d93782SGreg Clayton                                       bool multi_line,
346e30f11d9SKate Stone                                       bool color_prompts,
347f6913cd7SGreg Clayton                                       uint32_t line_number_start,
34844d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
34944d93782SGreg Clayton     IOHandlerEditline(debugger,
350e30f11d9SKate Stone                       type,
35144d93782SGreg Clayton                       StreamFileSP(), // Inherit input from top input reader
35244d93782SGreg Clayton                       StreamFileSP(), // Inherit output from top input reader
35344d93782SGreg Clayton                       StreamFileSP(), // Inherit error from top input reader
354340b0309SGreg Clayton                       0,              // Flags
35544d93782SGreg Clayton                       editline_name,  // Used for saving history files
35644d93782SGreg Clayton                       prompt,
357e30f11d9SKate Stone                       continuation_prompt,
35844d93782SGreg Clayton                       multi_line,
359e30f11d9SKate Stone                       color_prompts,
360f6913cd7SGreg Clayton                       line_number_start,
36144d93782SGreg Clayton                       delegate)
36244d93782SGreg Clayton {
36344d93782SGreg Clayton }
36444d93782SGreg Clayton 
36544d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger,
366e30f11d9SKate Stone                                       IOHandler::Type type,
36744d93782SGreg Clayton                                       const lldb::StreamFileSP &input_sp,
36844d93782SGreg Clayton                                       const lldb::StreamFileSP &output_sp,
36944d93782SGreg Clayton                                       const lldb::StreamFileSP &error_sp,
370340b0309SGreg Clayton                                       uint32_t flags,
37144d93782SGreg Clayton                                       const char *editline_name, // Used for saving history files
37244d93782SGreg Clayton                                       const char *prompt,
373e30f11d9SKate Stone                                       const char *continuation_prompt,
37444d93782SGreg Clayton                                       bool multi_line,
375e30f11d9SKate Stone                                       bool color_prompts,
376f6913cd7SGreg Clayton                                       uint32_t line_number_start,
37744d93782SGreg Clayton                                       IOHandlerDelegate &delegate) :
378e30f11d9SKate Stone     IOHandler (debugger, type, input_sp, output_sp, error_sp, flags),
379cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
38044d93782SGreg Clayton     m_editline_ap (),
381cacde7dfSTodd Fiala #endif
38244d93782SGreg Clayton     m_delegate (delegate),
38344d93782SGreg Clayton     m_prompt (),
384e30f11d9SKate Stone     m_continuation_prompt(),
385e30f11d9SKate Stone     m_current_lines_ptr (NULL),
386f6913cd7SGreg Clayton     m_base_line_number (line_number_start),
387e30f11d9SKate Stone     m_curr_line_idx (UINT32_MAX),
388e30f11d9SKate Stone     m_multi_line (multi_line),
389e30f11d9SKate Stone     m_color_prompts (color_prompts),
390e034a04eSGreg Clayton     m_interrupt_exits (true),
391e034a04eSGreg Clayton     m_editing (false)
39244d93782SGreg Clayton {
39344d93782SGreg Clayton     SetPrompt(prompt);
39444d93782SGreg Clayton 
395cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
396914b8d98SDeepak Panickal     bool use_editline = false;
397340b0309SGreg Clayton 
398340b0309SGreg Clayton     use_editline = m_input_sp->GetFile().GetIsRealTerminal();
39944d93782SGreg Clayton 
40044d93782SGreg Clayton     if (use_editline)
40144d93782SGreg Clayton     {
40244d93782SGreg Clayton         m_editline_ap.reset(new Editline (editline_name,
40344d93782SGreg Clayton                                           GetInputFILE (),
40444d93782SGreg Clayton                                           GetOutputFILE (),
405e30f11d9SKate Stone                                           GetErrorFILE (),
406e30f11d9SKate Stone                                           m_color_prompts));
407e30f11d9SKate Stone         m_editline_ap->SetIsInputCompleteCallback (IsInputCompleteCallback, this);
40844d93782SGreg Clayton         m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this);
409e30f11d9SKate Stone         // See if the delegate supports fixing indentation
410e30f11d9SKate Stone         const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters();
411e30f11d9SKate Stone         if (indent_chars)
412e30f11d9SKate Stone         {
413e30f11d9SKate Stone             // The delegate does support indentation, hook it up so when any indentation
414e30f11d9SKate Stone             // character is typed, the delegate gets a chance to fix it
415e30f11d9SKate Stone             m_editline_ap->SetFixIndentationCallback (FixIndentationCallback, this, indent_chars);
416e30f11d9SKate Stone         }
41744d93782SGreg Clayton     }
418cacde7dfSTodd Fiala #endif
419e30f11d9SKate Stone     SetBaseLineNumber (m_base_line_number);
420e30f11d9SKate Stone     SetPrompt(prompt ? prompt : "");
421e30f11d9SKate Stone     SetContinuationPrompt(continuation_prompt);
42244d93782SGreg Clayton }
42344d93782SGreg Clayton 
42444d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline ()
42544d93782SGreg Clayton {
426cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
42744d93782SGreg Clayton     m_editline_ap.reset();
428cacde7dfSTodd Fiala #endif
42944d93782SGreg Clayton }
43044d93782SGreg Clayton 
431e30f11d9SKate Stone void
432e30f11d9SKate Stone IOHandlerEditline::Activate ()
433e30f11d9SKate Stone {
434e30f11d9SKate Stone     IOHandler::Activate();
435e30f11d9SKate Stone     m_delegate.IOHandlerActivated(*this);
436e30f11d9SKate Stone }
437e30f11d9SKate Stone 
438e30f11d9SKate Stone void
439e30f11d9SKate Stone IOHandlerEditline::Deactivate ()
440e30f11d9SKate Stone {
441e30f11d9SKate Stone     IOHandler::Deactivate();
442e30f11d9SKate Stone     m_delegate.IOHandlerDeactivated(*this);
443e30f11d9SKate Stone }
444e30f11d9SKate Stone 
44544d93782SGreg Clayton bool
446f0066ad0SGreg Clayton IOHandlerEditline::GetLine (std::string &line, bool &interrupted)
44744d93782SGreg Clayton {
448cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
44944d93782SGreg Clayton     if (m_editline_ap)
45044d93782SGreg Clayton     {
451e30f11d9SKate Stone         return m_editline_ap->GetLine (line, interrupted);
45244d93782SGreg Clayton     }
45344d93782SGreg Clayton     else
45444d93782SGreg Clayton     {
455cacde7dfSTodd Fiala #endif
45644d93782SGreg Clayton         line.clear();
45744d93782SGreg Clayton 
45844d93782SGreg Clayton         FILE *in = GetInputFILE();
45944d93782SGreg Clayton         if (in)
46044d93782SGreg Clayton         {
461340b0309SGreg Clayton             if (GetIsInteractive())
46244d93782SGreg Clayton             {
463e30f11d9SKate Stone                 const char *prompt = NULL;
464e30f11d9SKate Stone 
465e30f11d9SKate Stone                 if (m_multi_line && m_curr_line_idx > 0)
466e30f11d9SKate Stone                     prompt = GetContinuationPrompt();
467e30f11d9SKate Stone 
468e30f11d9SKate Stone                 if (prompt == NULL)
469e30f11d9SKate Stone                     prompt = GetPrompt();
470e30f11d9SKate Stone 
47144d93782SGreg Clayton                 if (prompt && prompt[0])
47244d93782SGreg Clayton                 {
47344d93782SGreg Clayton                     FILE *out = GetOutputFILE();
47444d93782SGreg Clayton                     if (out)
47544d93782SGreg Clayton                     {
47644d93782SGreg Clayton                         ::fprintf(out, "%s", prompt);
47744d93782SGreg Clayton                         ::fflush(out);
47844d93782SGreg Clayton                     }
47944d93782SGreg Clayton                 }
48044d93782SGreg Clayton             }
48144d93782SGreg Clayton             char buffer[256];
48244d93782SGreg Clayton             bool done = false;
4830f86e6e7SGreg Clayton             bool got_line = false;
484e034a04eSGreg Clayton             m_editing = true;
48544d93782SGreg Clayton             while (!done)
48644d93782SGreg Clayton             {
48744d93782SGreg Clayton                 if (fgets(buffer, sizeof(buffer), in) == NULL)
488c9cf5798SGreg Clayton                 {
489c7797accSGreg Clayton                     const int saved_errno = errno;
490c9cf5798SGreg Clayton                     if (feof(in))
49144d93782SGreg Clayton                         done = true;
492c7797accSGreg Clayton                     else if (ferror(in))
493c7797accSGreg Clayton                     {
494c7797accSGreg Clayton                         if (saved_errno != EINTR)
495c7797accSGreg Clayton                             done = true;
496c7797accSGreg Clayton                     }
497c9cf5798SGreg Clayton                 }
49844d93782SGreg Clayton                 else
49944d93782SGreg Clayton                 {
5000f86e6e7SGreg Clayton                     got_line = true;
50144d93782SGreg Clayton                     size_t buffer_len = strlen(buffer);
50244d93782SGreg Clayton                     assert (buffer[buffer_len] == '\0');
50344d93782SGreg Clayton                     char last_char = buffer[buffer_len-1];
50444d93782SGreg Clayton                     if (last_char == '\r' || last_char == '\n')
50544d93782SGreg Clayton                     {
50644d93782SGreg Clayton                         done = true;
50744d93782SGreg Clayton                         // Strip trailing newlines
50844d93782SGreg Clayton                         while (last_char == '\r' || last_char == '\n')
50944d93782SGreg Clayton                         {
51044d93782SGreg Clayton                             --buffer_len;
51144d93782SGreg Clayton                             if (buffer_len == 0)
51244d93782SGreg Clayton                                 break;
51344d93782SGreg Clayton                             last_char = buffer[buffer_len-1];
51444d93782SGreg Clayton                         }
51544d93782SGreg Clayton                     }
51644d93782SGreg Clayton                     line.append(buffer, buffer_len);
51744d93782SGreg Clayton                 }
51844d93782SGreg Clayton             }
519e034a04eSGreg Clayton             m_editing = false;
5200f86e6e7SGreg Clayton             // We might have gotten a newline on a line by itself
5210f86e6e7SGreg Clayton             // make sure to return true in this case.
5220f86e6e7SGreg Clayton             return got_line;
52344d93782SGreg Clayton         }
52444d93782SGreg Clayton         else
52544d93782SGreg Clayton         {
52644d93782SGreg Clayton             // No more input file, we are done...
52744d93782SGreg Clayton             SetIsDone(true);
52844d93782SGreg Clayton         }
529340b0309SGreg Clayton         return false;
530cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
53144d93782SGreg Clayton     }
532cacde7dfSTodd Fiala #endif
53344d93782SGreg Clayton }
53444d93782SGreg Clayton 
53544d93782SGreg Clayton 
536cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
537e30f11d9SKate Stone bool
538e30f11d9SKate Stone IOHandlerEditline::IsInputCompleteCallback (Editline *editline,
53944d93782SGreg Clayton                                           StringList &lines,
54044d93782SGreg Clayton                                           void *baton)
54144d93782SGreg Clayton {
54244d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
543e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader, lines);
544e30f11d9SKate Stone }
545e30f11d9SKate Stone 
546e30f11d9SKate Stone int
547e30f11d9SKate Stone IOHandlerEditline::FixIndentationCallback (Editline *editline,
548e30f11d9SKate Stone                                            const StringList &lines,
549e30f11d9SKate Stone                                            int cursor_position,
550e30f11d9SKate Stone                                            void *baton)
551e30f11d9SKate Stone {
552e30f11d9SKate Stone     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
553e30f11d9SKate Stone     return editline_reader->m_delegate.IOHandlerFixIndentation(*editline_reader, lines, cursor_position);
55444d93782SGreg Clayton }
55544d93782SGreg Clayton 
55644d93782SGreg Clayton int
55744d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line,
55844d93782SGreg Clayton                                          const char *cursor,
55944d93782SGreg Clayton                                          const char *last_char,
56044d93782SGreg Clayton                                          int skip_first_n_matches,
56144d93782SGreg Clayton                                          int max_matches,
56244d93782SGreg Clayton                                          StringList &matches,
56344d93782SGreg Clayton                                          void *baton)
56444d93782SGreg Clayton {
56544d93782SGreg Clayton     IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton;
56644d93782SGreg Clayton     if (editline_reader)
56744d93782SGreg Clayton         return editline_reader->m_delegate.IOHandlerComplete (*editline_reader,
56844d93782SGreg Clayton                                                               current_line,
56944d93782SGreg Clayton                                                               cursor,
57044d93782SGreg Clayton                                                               last_char,
57144d93782SGreg Clayton                                                               skip_first_n_matches,
57244d93782SGreg Clayton                                                               max_matches,
57344d93782SGreg Clayton                                                               matches);
57444d93782SGreg Clayton     return 0;
57544d93782SGreg Clayton }
576cacde7dfSTodd Fiala #endif
57744d93782SGreg Clayton 
57844d93782SGreg Clayton const char *
57944d93782SGreg Clayton IOHandlerEditline::GetPrompt ()
58044d93782SGreg Clayton {
581cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
58244d93782SGreg Clayton     if (m_editline_ap)
583cacde7dfSTodd Fiala     {
58444d93782SGreg Clayton         return m_editline_ap->GetPrompt ();
585cacde7dfSTodd Fiala     }
586cacde7dfSTodd Fiala     else
587cacde7dfSTodd Fiala     {
588cacde7dfSTodd Fiala #endif
589cacde7dfSTodd Fiala         if (m_prompt.empty())
59044d93782SGreg Clayton             return NULL;
591cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
592cacde7dfSTodd Fiala     }
593cacde7dfSTodd Fiala #endif
59444d93782SGreg Clayton     return m_prompt.c_str();
59544d93782SGreg Clayton }
59644d93782SGreg Clayton 
59744d93782SGreg Clayton bool
59844d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p)
59944d93782SGreg Clayton {
60044d93782SGreg Clayton     if (p && p[0])
60144d93782SGreg Clayton         m_prompt = p;
60244d93782SGreg Clayton     else
60344d93782SGreg Clayton         m_prompt.clear();
604cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
60544d93782SGreg Clayton     if (m_editline_ap)
60644d93782SGreg Clayton         m_editline_ap->SetPrompt (m_prompt.empty() ? NULL : m_prompt.c_str());
607cacde7dfSTodd Fiala #endif
60844d93782SGreg Clayton     return true;
60944d93782SGreg Clayton }
61044d93782SGreg Clayton 
611e30f11d9SKate Stone const char *
612e30f11d9SKate Stone IOHandlerEditline::GetContinuationPrompt ()
613e30f11d9SKate Stone {
614e30f11d9SKate Stone     if (m_continuation_prompt.empty())
615e30f11d9SKate Stone         return NULL;
616e30f11d9SKate Stone     return m_continuation_prompt.c_str();
617e30f11d9SKate Stone }
618e30f11d9SKate Stone 
619e30f11d9SKate Stone void
620e30f11d9SKate Stone IOHandlerEditline::SetContinuationPrompt (const char *p)
621e30f11d9SKate Stone {
622e30f11d9SKate Stone     if (p && p[0])
623e30f11d9SKate Stone         m_continuation_prompt = p;
624e30f11d9SKate Stone     else
625e30f11d9SKate Stone         m_continuation_prompt.clear();
626e30f11d9SKate Stone 
627d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
628e30f11d9SKate Stone     if (m_editline_ap)
629e30f11d9SKate Stone         m_editline_ap->SetContinuationPrompt (m_continuation_prompt.empty() ? NULL : m_continuation_prompt.c_str());
630d553d00cSZachary Turner #endif
631e30f11d9SKate Stone }
632e30f11d9SKate Stone 
633f6913cd7SGreg Clayton void
634f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line)
635f6913cd7SGreg Clayton {
636f6913cd7SGreg Clayton     m_base_line_number = line;
637f6913cd7SGreg Clayton }
638e30f11d9SKate Stone 
639e30f11d9SKate Stone uint32_t
640e30f11d9SKate Stone IOHandlerEditline::GetCurrentLineIndex () const
641e30f11d9SKate Stone {
642d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT
643e30f11d9SKate Stone     if (m_editline_ap)
644e30f11d9SKate Stone         return m_editline_ap->GetCurrentLine();
645e30f11d9SKate Stone #endif
646e30f11d9SKate Stone     return m_curr_line_idx;
647e30f11d9SKate Stone }
648e30f11d9SKate Stone 
64944d93782SGreg Clayton bool
650f0066ad0SGreg Clayton IOHandlerEditline::GetLines (StringList &lines, bool &interrupted)
65144d93782SGreg Clayton {
652e30f11d9SKate Stone     m_current_lines_ptr = &lines;
653e30f11d9SKate Stone 
65444d93782SGreg Clayton     bool success = false;
655cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
65644d93782SGreg Clayton     if (m_editline_ap)
65744d93782SGreg Clayton     {
658e30f11d9SKate Stone         return m_editline_ap->GetLines (m_base_line_number, lines, interrupted);
65944d93782SGreg Clayton     }
66044d93782SGreg Clayton     else
66144d93782SGreg Clayton     {
662cacde7dfSTodd Fiala #endif
663e30f11d9SKate Stone         bool done = false;
664c3d874a5SGreg Clayton         Error error;
66544d93782SGreg Clayton 
666e30f11d9SKate Stone         while (!done)
66744d93782SGreg Clayton         {
668f6913cd7SGreg Clayton             // Show line numbers if we are asked to
66944d93782SGreg Clayton             std::string line;
670f6913cd7SGreg Clayton             if (m_base_line_number > 0 && GetIsInteractive())
671f6913cd7SGreg Clayton             {
672f6913cd7SGreg Clayton                 FILE *out = GetOutputFILE();
673f6913cd7SGreg Clayton                 if (out)
674bc88d938SGreg Clayton                     ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(), GetPrompt() == NULL ? " " : "");
675f6913cd7SGreg Clayton             }
676f6913cd7SGreg Clayton 
677e30f11d9SKate Stone             m_curr_line_idx = lines.GetSize();
678e30f11d9SKate Stone 
679f0066ad0SGreg Clayton             bool interrupted = false;
680e30f11d9SKate Stone             if (GetLine(line, interrupted) && !interrupted)
68144d93782SGreg Clayton             {
68244d93782SGreg Clayton                 lines.AppendString(line);
683e30f11d9SKate Stone                 done = m_delegate.IOHandlerIsInputComplete(*this, lines);
684f0066ad0SGreg Clayton             }
68544d93782SGreg Clayton             else
68644d93782SGreg Clayton             {
687e30f11d9SKate Stone                 done = true;
68844d93782SGreg Clayton             }
68944d93782SGreg Clayton         }
69044d93782SGreg Clayton         success = lines.GetSize() > 0;
691cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
69244d93782SGreg Clayton     }
693cacde7dfSTodd Fiala #endif
69444d93782SGreg Clayton     return success;
69544d93782SGreg Clayton }
69644d93782SGreg Clayton 
69744d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data
69844d93782SGreg Clayton // from the "in" and place output into "out" and "err and return
69944d93782SGreg Clayton // when done.
70044d93782SGreg Clayton void
70144d93782SGreg Clayton IOHandlerEditline::Run ()
70244d93782SGreg Clayton {
70344d93782SGreg Clayton     std::string line;
70444d93782SGreg Clayton     while (IsActive())
70544d93782SGreg Clayton     {
706f0066ad0SGreg Clayton         bool interrupted = false;
70744d93782SGreg Clayton         if (m_multi_line)
70844d93782SGreg Clayton         {
70944d93782SGreg Clayton             StringList lines;
710f0066ad0SGreg Clayton             if (GetLines (lines, interrupted))
711f0066ad0SGreg Clayton             {
712f0066ad0SGreg Clayton                 if (interrupted)
713f0066ad0SGreg Clayton                 {
714e30f11d9SKate Stone                     m_done = m_interrupt_exits;
715e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
716e30f11d9SKate Stone 
717f0066ad0SGreg Clayton                 }
718f0066ad0SGreg Clayton                 else
71944d93782SGreg Clayton                 {
72044d93782SGreg Clayton                     line = lines.CopyList();
72144d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
72244d93782SGreg Clayton                 }
723f0066ad0SGreg Clayton             }
72444d93782SGreg Clayton             else
72544d93782SGreg Clayton             {
72644d93782SGreg Clayton                 m_done = true;
72744d93782SGreg Clayton             }
72844d93782SGreg Clayton         }
72944d93782SGreg Clayton         else
73044d93782SGreg Clayton         {
731f0066ad0SGreg Clayton             if (GetLine(line, interrupted))
73244d93782SGreg Clayton             {
733e30f11d9SKate Stone                 if (interrupted)
734e30f11d9SKate Stone                     m_delegate.IOHandlerInputInterrupted (*this, line);
735e30f11d9SKate Stone                 else
73644d93782SGreg Clayton                     m_delegate.IOHandlerInputComplete (*this, line);
73744d93782SGreg Clayton             }
73844d93782SGreg Clayton             else
73944d93782SGreg Clayton             {
74044d93782SGreg Clayton                 m_done = true;
74144d93782SGreg Clayton             }
74244d93782SGreg Clayton         }
74344d93782SGreg Clayton     }
74444d93782SGreg Clayton }
74544d93782SGreg Clayton 
74644d93782SGreg Clayton void
747e68f5d6bSGreg Clayton IOHandlerEditline::Cancel ()
748e68f5d6bSGreg Clayton {
749cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
750e68f5d6bSGreg Clayton     if (m_editline_ap)
7514446487dSPavel Labath         m_editline_ap->Cancel ();
752cacde7dfSTodd Fiala #endif
753e68f5d6bSGreg Clayton }
754e68f5d6bSGreg Clayton 
755f0066ad0SGreg Clayton bool
75644d93782SGreg Clayton IOHandlerEditline::Interrupt ()
75744d93782SGreg Clayton {
758f0066ad0SGreg Clayton     // Let the delgate handle it first
759f0066ad0SGreg Clayton     if (m_delegate.IOHandlerInterrupt(*this))
760f0066ad0SGreg Clayton         return true;
761f0066ad0SGreg Clayton 
762cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
76344d93782SGreg Clayton     if (m_editline_ap)
764f0066ad0SGreg Clayton         return m_editline_ap->Interrupt();
765cacde7dfSTodd Fiala #endif
766f0066ad0SGreg Clayton     return false;
76744d93782SGreg Clayton }
76844d93782SGreg Clayton 
76944d93782SGreg Clayton void
77044d93782SGreg Clayton IOHandlerEditline::GotEOF()
77144d93782SGreg Clayton {
772cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT
77344d93782SGreg Clayton     if (m_editline_ap)
77444d93782SGreg Clayton         m_editline_ap->Interrupt();
775cacde7dfSTodd Fiala #endif
77644d93782SGreg Clayton }
77744d93782SGreg Clayton 
7784446487dSPavel Labath void
7794446487dSPavel Labath IOHandlerEditline::PrintAsync (Stream *stream, const char *s, size_t len)
7804446487dSPavel Labath {
7814446487dSPavel Labath #ifndef LLDB_DISABLE_LIBEDIT
7824446487dSPavel Labath     if (m_editline_ap)
7834446487dSPavel Labath         m_editline_ap->PrintAsync(stream, s, len);
7844446487dSPavel Labath     else
7854446487dSPavel Labath #endif
7864446487dSPavel Labath         IOHandler::PrintAsync(stream, s, len);
7874446487dSPavel Labath }
7884446487dSPavel Labath 
789914b8d98SDeepak Panickal // we may want curses to be disabled for some builds
790914b8d98SDeepak Panickal // for instance, windows
791914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES
792914b8d98SDeepak Panickal 
79344d93782SGreg Clayton #include "lldb/Core/ValueObject.h"
79444d93782SGreg Clayton #include "lldb/Symbol/VariableList.h"
79544d93782SGreg Clayton #include "lldb/Target/Target.h"
79644d93782SGreg Clayton #include "lldb/Target/Process.h"
79744d93782SGreg Clayton #include "lldb/Target/Thread.h"
79844d93782SGreg Clayton #include "lldb/Target/StackFrame.h"
79944d93782SGreg Clayton 
80044d93782SGreg Clayton #define KEY_RETURN   10
80144d93782SGreg Clayton #define KEY_ESCAPE  27
80244d93782SGreg Clayton 
80344d93782SGreg Clayton namespace curses
80444d93782SGreg Clayton {
80544d93782SGreg Clayton     class Menu;
80644d93782SGreg Clayton     class MenuDelegate;
80744d93782SGreg Clayton     class Window;
80844d93782SGreg Clayton     class WindowDelegate;
80944d93782SGreg Clayton     typedef std::shared_ptr<Menu> MenuSP;
81044d93782SGreg Clayton     typedef std::shared_ptr<MenuDelegate> MenuDelegateSP;
81144d93782SGreg Clayton     typedef std::shared_ptr<Window> WindowSP;
81244d93782SGreg Clayton     typedef std::shared_ptr<WindowDelegate> WindowDelegateSP;
81344d93782SGreg Clayton     typedef std::vector<MenuSP> Menus;
81444d93782SGreg Clayton     typedef std::vector<WindowSP> Windows;
81544d93782SGreg Clayton     typedef std::vector<WindowDelegateSP> WindowDelegates;
81644d93782SGreg Clayton 
81744d93782SGreg Clayton #if 0
81844d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point
81944d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size
82044d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect
82144d93782SGreg Clayton #endif
822315b6884SEugene Zelenko 
82344d93782SGreg Clayton     struct Point
82444d93782SGreg Clayton     {
82544d93782SGreg Clayton         int x;
82644d93782SGreg Clayton         int y;
82744d93782SGreg Clayton 
82844d93782SGreg Clayton         Point (int _x = 0, int _y = 0) :
82944d93782SGreg Clayton             x(_x),
83044d93782SGreg Clayton             y(_y)
83144d93782SGreg Clayton         {
83244d93782SGreg Clayton         }
83344d93782SGreg Clayton 
83444d93782SGreg Clayton         void
83544d93782SGreg Clayton         Clear ()
83644d93782SGreg Clayton         {
83744d93782SGreg Clayton             x = 0;
83844d93782SGreg Clayton             y = 0;
83944d93782SGreg Clayton         }
84044d93782SGreg Clayton 
84144d93782SGreg Clayton         Point &
84244d93782SGreg Clayton         operator += (const Point &rhs)
84344d93782SGreg Clayton         {
84444d93782SGreg Clayton             x += rhs.x;
84544d93782SGreg Clayton             y += rhs.y;
84644d93782SGreg Clayton             return *this;
84744d93782SGreg Clayton         }
84844d93782SGreg Clayton 
84944d93782SGreg Clayton         void
85044d93782SGreg Clayton         Dump ()
85144d93782SGreg Clayton         {
85244d93782SGreg Clayton             printf ("(x=%i, y=%i)\n", x, y);
85344d93782SGreg Clayton         }
85444d93782SGreg Clayton     };
85544d93782SGreg Clayton 
85644d93782SGreg Clayton     bool operator == (const Point &lhs, const Point &rhs)
85744d93782SGreg Clayton     {
85844d93782SGreg Clayton         return lhs.x == rhs.x && lhs.y == rhs.y;
85944d93782SGreg Clayton     }
860315b6884SEugene Zelenko 
86144d93782SGreg Clayton     bool operator != (const Point &lhs, const Point &rhs)
86244d93782SGreg Clayton     {
86344d93782SGreg Clayton         return lhs.x != rhs.x || lhs.y != rhs.y;
86444d93782SGreg Clayton     }
86544d93782SGreg Clayton 
86644d93782SGreg Clayton     struct Size
86744d93782SGreg Clayton     {
86844d93782SGreg Clayton         int width;
86944d93782SGreg Clayton         int height;
87044d93782SGreg Clayton         Size (int w = 0, int h = 0) :
87144d93782SGreg Clayton             width (w),
87244d93782SGreg Clayton             height (h)
87344d93782SGreg Clayton         {
87444d93782SGreg Clayton         }
87544d93782SGreg Clayton 
87644d93782SGreg Clayton         void
87744d93782SGreg Clayton         Clear ()
87844d93782SGreg Clayton         {
87944d93782SGreg Clayton             width = 0;
88044d93782SGreg Clayton             height = 0;
88144d93782SGreg Clayton         }
88244d93782SGreg Clayton 
88344d93782SGreg Clayton         void
88444d93782SGreg Clayton         Dump ()
88544d93782SGreg Clayton         {
88644d93782SGreg Clayton             printf ("(w=%i, h=%i)\n", width, height);
88744d93782SGreg Clayton         }
88844d93782SGreg Clayton     };
88944d93782SGreg Clayton 
89044d93782SGreg Clayton     bool operator == (const Size &lhs, const Size &rhs)
89144d93782SGreg Clayton     {
89244d93782SGreg Clayton         return lhs.width == rhs.width && lhs.height == rhs.height;
89344d93782SGreg Clayton     }
894315b6884SEugene Zelenko 
89544d93782SGreg Clayton     bool operator != (const Size &lhs, const Size &rhs)
89644d93782SGreg Clayton     {
89744d93782SGreg Clayton         return lhs.width != rhs.width || lhs.height != rhs.height;
89844d93782SGreg Clayton     }
89944d93782SGreg Clayton 
90044d93782SGreg Clayton     struct Rect
90144d93782SGreg Clayton     {
90244d93782SGreg Clayton         Point origin;
90344d93782SGreg Clayton         Size size;
90444d93782SGreg Clayton 
90544d93782SGreg Clayton         Rect () :
90644d93782SGreg Clayton             origin(),
90744d93782SGreg Clayton             size()
90844d93782SGreg Clayton         {
90944d93782SGreg Clayton         }
91044d93782SGreg Clayton 
91144d93782SGreg Clayton         Rect (const Point &p, const Size &s) :
91244d93782SGreg Clayton             origin (p),
91344d93782SGreg Clayton             size (s)
91444d93782SGreg Clayton         {
91544d93782SGreg Clayton         }
91644d93782SGreg Clayton 
91744d93782SGreg Clayton         void
91844d93782SGreg Clayton         Clear ()
91944d93782SGreg Clayton         {
92044d93782SGreg Clayton             origin.Clear();
92144d93782SGreg Clayton             size.Clear();
92244d93782SGreg Clayton         }
92344d93782SGreg Clayton 
92444d93782SGreg Clayton         void
92544d93782SGreg Clayton         Dump ()
92644d93782SGreg Clayton         {
92744d93782SGreg Clayton             printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height);
92844d93782SGreg Clayton         }
92944d93782SGreg Clayton 
93044d93782SGreg Clayton         void
93144d93782SGreg Clayton         Inset (int w, int h)
93244d93782SGreg Clayton         {
93344d93782SGreg Clayton             if (size.width > w*2)
93444d93782SGreg Clayton                 size.width -= w*2;
93544d93782SGreg Clayton             origin.x += w;
93644d93782SGreg Clayton 
93744d93782SGreg Clayton             if (size.height > h*2)
93844d93782SGreg Clayton                 size.height -= h*2;
93944d93782SGreg Clayton             origin.y += h;
94044d93782SGreg Clayton         }
941315b6884SEugene Zelenko 
94244d93782SGreg Clayton         // Return a status bar rectangle which is the last line of
94344d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
94444d93782SGreg Clayton         // include the status bar area.
94544d93782SGreg Clayton         Rect
94644d93782SGreg Clayton         MakeStatusBar ()
94744d93782SGreg Clayton         {
94844d93782SGreg Clayton             Rect status_bar;
94944d93782SGreg Clayton             if (size.height > 1)
95044d93782SGreg Clayton             {
95144d93782SGreg Clayton                 status_bar.origin.x = origin.x;
95244d93782SGreg Clayton                 status_bar.origin.y = size.height;
95344d93782SGreg Clayton                 status_bar.size.width = size.width;
95444d93782SGreg Clayton                 status_bar.size.height = 1;
95544d93782SGreg Clayton                 --size.height;
95644d93782SGreg Clayton             }
95744d93782SGreg Clayton             return status_bar;
95844d93782SGreg Clayton         }
95944d93782SGreg Clayton 
96044d93782SGreg Clayton         // Return a menubar rectangle which is the first line of
96144d93782SGreg Clayton         // this rectangle. This rectangle will be modified to not
96244d93782SGreg Clayton         // include the menubar area.
96344d93782SGreg Clayton         Rect
96444d93782SGreg Clayton         MakeMenuBar ()
96544d93782SGreg Clayton         {
96644d93782SGreg Clayton             Rect menubar;
96744d93782SGreg Clayton             if (size.height > 1)
96844d93782SGreg Clayton             {
96944d93782SGreg Clayton                 menubar.origin.x = origin.x;
97044d93782SGreg Clayton                 menubar.origin.y = origin.y;
97144d93782SGreg Clayton                 menubar.size.width = size.width;
97244d93782SGreg Clayton                 menubar.size.height = 1;
97344d93782SGreg Clayton                 ++origin.y;
97444d93782SGreg Clayton                 --size.height;
97544d93782SGreg Clayton             }
97644d93782SGreg Clayton             return menubar;
97744d93782SGreg Clayton         }
97844d93782SGreg Clayton 
97944d93782SGreg Clayton         void
98044d93782SGreg Clayton         HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const
98144d93782SGreg Clayton         {
98244d93782SGreg Clayton             float top_height = top_percentage * size.height;
98344d93782SGreg Clayton             HorizontalSplit (top_height, top, bottom);
98444d93782SGreg Clayton         }
98544d93782SGreg Clayton 
98644d93782SGreg Clayton         void
98744d93782SGreg Clayton         HorizontalSplit (int top_height, Rect &top, Rect &bottom) const
98844d93782SGreg Clayton         {
98944d93782SGreg Clayton             top = *this;
99044d93782SGreg Clayton             if (top_height < size.height)
99144d93782SGreg Clayton             {
99244d93782SGreg Clayton                 top.size.height = top_height;
99344d93782SGreg Clayton                 bottom.origin.x = origin.x;
99444d93782SGreg Clayton                 bottom.origin.y = origin.y + top.size.height;
99544d93782SGreg Clayton                 bottom.size.width = size.width;
99644d93782SGreg Clayton                 bottom.size.height = size.height - top.size.height;
99744d93782SGreg Clayton             }
99844d93782SGreg Clayton             else
99944d93782SGreg Clayton             {
100044d93782SGreg Clayton                 bottom.Clear();
100144d93782SGreg Clayton             }
100244d93782SGreg Clayton         }
100344d93782SGreg Clayton 
100444d93782SGreg Clayton         void
100544d93782SGreg Clayton         VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const
100644d93782SGreg Clayton         {
100744d93782SGreg Clayton             float left_width = left_percentage * size.width;
100844d93782SGreg Clayton             VerticalSplit (left_width, left, right);
100944d93782SGreg Clayton         }
101044d93782SGreg Clayton 
101144d93782SGreg Clayton         void
101244d93782SGreg Clayton         VerticalSplit (int left_width, Rect &left, Rect &right) const
101344d93782SGreg Clayton         {
101444d93782SGreg Clayton             left = *this;
101544d93782SGreg Clayton             if (left_width < size.width)
101644d93782SGreg Clayton             {
101744d93782SGreg Clayton                 left.size.width = left_width;
101844d93782SGreg Clayton                 right.origin.x = origin.x + left.size.width;
101944d93782SGreg Clayton                 right.origin.y = origin.y;
102044d93782SGreg Clayton                 right.size.width = size.width - left.size.width;
102144d93782SGreg Clayton                 right.size.height = size.height;
102244d93782SGreg Clayton             }
102344d93782SGreg Clayton             else
102444d93782SGreg Clayton             {
102544d93782SGreg Clayton                 right.Clear();
102644d93782SGreg Clayton             }
102744d93782SGreg Clayton         }
102844d93782SGreg Clayton     };
102944d93782SGreg Clayton 
103044d93782SGreg Clayton     bool operator == (const Rect &lhs, const Rect &rhs)
103144d93782SGreg Clayton     {
103244d93782SGreg Clayton         return lhs.origin == rhs.origin && lhs.size == rhs.size;
103344d93782SGreg Clayton     }
1034315b6884SEugene Zelenko 
103544d93782SGreg Clayton     bool operator != (const Rect &lhs, const Rect &rhs)
103644d93782SGreg Clayton     {
103744d93782SGreg Clayton         return lhs.origin != rhs.origin || lhs.size != rhs.size;
103844d93782SGreg Clayton     }
103944d93782SGreg Clayton 
104044d93782SGreg Clayton     enum HandleCharResult
104144d93782SGreg Clayton     {
104244d93782SGreg Clayton         eKeyNotHandled      = 0,
104344d93782SGreg Clayton         eKeyHandled         = 1,
104444d93782SGreg Clayton         eQuitApplication    = 2
104544d93782SGreg Clayton     };
104644d93782SGreg Clayton 
104744d93782SGreg Clayton     enum class MenuActionResult
104844d93782SGreg Clayton     {
104944d93782SGreg Clayton         Handled,
105044d93782SGreg Clayton         NotHandled,
105144d93782SGreg Clayton         Quit    // Exit all menus and quit
105244d93782SGreg Clayton     };
105344d93782SGreg Clayton 
105444d93782SGreg Clayton     struct KeyHelp
105544d93782SGreg Clayton     {
105644d93782SGreg Clayton         int ch;
105744d93782SGreg Clayton         const char *description;
105844d93782SGreg Clayton     };
105944d93782SGreg Clayton 
106044d93782SGreg Clayton     class WindowDelegate
106144d93782SGreg Clayton     {
106244d93782SGreg Clayton     public:
106344d93782SGreg Clayton         virtual
1064315b6884SEugene Zelenko         ~WindowDelegate() = default;
106544d93782SGreg Clayton 
106644d93782SGreg Clayton         virtual bool
106744d93782SGreg Clayton         WindowDelegateDraw (Window &window, bool force)
106844d93782SGreg Clayton         {
106944d93782SGreg Clayton             return false; // Drawing not handled
107044d93782SGreg Clayton         }
107144d93782SGreg Clayton 
107244d93782SGreg Clayton         virtual HandleCharResult
107344d93782SGreg Clayton         WindowDelegateHandleChar (Window &window, int key)
107444d93782SGreg Clayton         {
107544d93782SGreg Clayton             return eKeyNotHandled;
107644d93782SGreg Clayton         }
107744d93782SGreg Clayton 
107844d93782SGreg Clayton         virtual const char *
107944d93782SGreg Clayton         WindowDelegateGetHelpText ()
108044d93782SGreg Clayton         {
108144d93782SGreg Clayton             return NULL;
108244d93782SGreg Clayton         }
108344d93782SGreg Clayton 
108444d93782SGreg Clayton         virtual KeyHelp *
108544d93782SGreg Clayton         WindowDelegateGetKeyHelp ()
108644d93782SGreg Clayton         {
108744d93782SGreg Clayton             return NULL;
108844d93782SGreg Clayton         }
108944d93782SGreg Clayton     };
109044d93782SGreg Clayton 
109144d93782SGreg Clayton     class HelpDialogDelegate :
109244d93782SGreg Clayton         public WindowDelegate
109344d93782SGreg Clayton     {
109444d93782SGreg Clayton     public:
109544d93782SGreg Clayton         HelpDialogDelegate (const char *text, KeyHelp *key_help_array);
109644d93782SGreg Clayton 
1097bd5ae6b4SGreg Clayton         ~HelpDialogDelegate() override;
109844d93782SGreg Clayton 
1099bd5ae6b4SGreg Clayton         bool
1100bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
110144d93782SGreg Clayton 
1102bd5ae6b4SGreg Clayton         HandleCharResult
1103bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
110444d93782SGreg Clayton 
110544d93782SGreg Clayton         size_t
110644d93782SGreg Clayton         GetNumLines() const
110744d93782SGreg Clayton         {
110844d93782SGreg Clayton             return m_text.GetSize();
110944d93782SGreg Clayton         }
111044d93782SGreg Clayton 
111144d93782SGreg Clayton         size_t
111244d93782SGreg Clayton         GetMaxLineLength () const
111344d93782SGreg Clayton         {
111444d93782SGreg Clayton             return m_text.GetMaxStringLength();
111544d93782SGreg Clayton         }
111644d93782SGreg Clayton 
111744d93782SGreg Clayton     protected:
111844d93782SGreg Clayton         StringList m_text;
111944d93782SGreg Clayton         int m_first_visible_line;
112044d93782SGreg Clayton     };
112144d93782SGreg Clayton 
112244d93782SGreg Clayton     class Window
112344d93782SGreg Clayton     {
112444d93782SGreg Clayton     public:
112544d93782SGreg Clayton         Window (const char *name) :
112644d93782SGreg Clayton             m_name (name),
112744d93782SGreg Clayton             m_window (NULL),
112844d93782SGreg Clayton             m_panel (NULL),
112944d93782SGreg Clayton             m_parent (NULL),
113044d93782SGreg Clayton             m_subwindows (),
113144d93782SGreg Clayton             m_delegate_sp (),
113244d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
113344d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
113444d93782SGreg Clayton             m_delete (false),
113544d93782SGreg Clayton             m_needs_update (true),
113644d93782SGreg Clayton             m_can_activate (true),
113744d93782SGreg Clayton             m_is_subwin (false)
113844d93782SGreg Clayton         {
113944d93782SGreg Clayton         }
114044d93782SGreg Clayton 
114144d93782SGreg Clayton         Window (const char *name, WINDOW *w, bool del = true) :
114244d93782SGreg Clayton             m_name (name),
114344d93782SGreg Clayton             m_window (NULL),
114444d93782SGreg Clayton             m_panel (NULL),
114544d93782SGreg Clayton             m_parent (NULL),
114644d93782SGreg Clayton             m_subwindows (),
114744d93782SGreg Clayton             m_delegate_sp (),
114844d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
114944d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
115044d93782SGreg Clayton             m_delete (del),
115144d93782SGreg Clayton             m_needs_update (true),
115244d93782SGreg Clayton             m_can_activate (true),
115344d93782SGreg Clayton             m_is_subwin (false)
115444d93782SGreg Clayton         {
115544d93782SGreg Clayton             if (w)
115644d93782SGreg Clayton                 Reset(w);
115744d93782SGreg Clayton         }
115844d93782SGreg Clayton 
115944d93782SGreg Clayton         Window (const char *name, const Rect &bounds) :
116044d93782SGreg Clayton             m_name (name),
116144d93782SGreg Clayton             m_window (NULL),
116244d93782SGreg Clayton             m_parent (NULL),
116344d93782SGreg Clayton             m_subwindows (),
116444d93782SGreg Clayton             m_delegate_sp (),
116544d93782SGreg Clayton             m_curr_active_window_idx (UINT32_MAX),
116644d93782SGreg Clayton             m_prev_active_window_idx (UINT32_MAX),
116744d93782SGreg Clayton             m_delete (true),
116844d93782SGreg Clayton             m_needs_update (true),
116944d93782SGreg Clayton             m_can_activate (true),
117044d93782SGreg Clayton             m_is_subwin (false)
117144d93782SGreg Clayton         {
117244d93782SGreg Clayton             Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y));
117344d93782SGreg Clayton         }
117444d93782SGreg Clayton 
117544d93782SGreg Clayton         virtual
117644d93782SGreg Clayton         ~Window ()
117744d93782SGreg Clayton         {
117844d93782SGreg Clayton             RemoveSubWindows ();
117944d93782SGreg Clayton             Reset ();
118044d93782SGreg Clayton         }
118144d93782SGreg Clayton 
118244d93782SGreg Clayton         void
118344d93782SGreg Clayton         Reset (WINDOW *w = NULL, bool del = true)
118444d93782SGreg Clayton         {
118544d93782SGreg Clayton             if (m_window == w)
118644d93782SGreg Clayton                 return;
118744d93782SGreg Clayton 
118844d93782SGreg Clayton             if (m_panel)
118944d93782SGreg Clayton             {
119044d93782SGreg Clayton                 ::del_panel (m_panel);
119144d93782SGreg Clayton                 m_panel = NULL;
119244d93782SGreg Clayton             }
119344d93782SGreg Clayton             if (m_window && m_delete)
119444d93782SGreg Clayton             {
119544d93782SGreg Clayton                 ::delwin (m_window);
119644d93782SGreg Clayton                 m_window = NULL;
119744d93782SGreg Clayton                 m_delete = false;
119844d93782SGreg Clayton             }
119944d93782SGreg Clayton             if (w)
120044d93782SGreg Clayton             {
120144d93782SGreg Clayton                 m_window = w;
120244d93782SGreg Clayton                 m_panel = ::new_panel (m_window);
120344d93782SGreg Clayton                 m_delete = del;
120444d93782SGreg Clayton             }
120544d93782SGreg Clayton         }
120644d93782SGreg Clayton 
120744d93782SGreg Clayton         void    AttributeOn (attr_t attr)   { ::wattron (m_window, attr); }
120844d93782SGreg Clayton         void    AttributeOff (attr_t attr)  { ::wattroff (m_window, attr); }
120944d93782SGreg Clayton         void    Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); }
121044d93782SGreg Clayton         void    Clear ()    { ::wclear (m_window); }
121144d93782SGreg Clayton         void    Erase ()    { ::werase (m_window); }
121244d93782SGreg Clayton         Rect    GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window
121344d93782SGreg Clayton         int     GetChar ()  { return ::wgetch (m_window); }
121444d93782SGreg Clayton         int     GetCursorX ()     { return getcurx (m_window); }
121544d93782SGreg Clayton         int     GetCursorY ()     { return getcury (m_window); }
121644d93782SGreg Clayton         Rect    GetFrame ()    { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system
121744d93782SGreg Clayton         Point   GetParentOrigin() { return Point (GetParentX(), GetParentY()); }
121844d93782SGreg Clayton         Size    GetSize()         { return Size (GetWidth(), GetHeight()); }
121944d93782SGreg Clayton         int     GetParentX ()     { return getparx (m_window); }
122044d93782SGreg Clayton         int     GetParentY ()     { return getpary (m_window); }
122144d93782SGreg Clayton         int     GetMaxX()   { return getmaxx (m_window); }
122244d93782SGreg Clayton         int     GetMaxY()   { return getmaxy (m_window); }
122344d93782SGreg Clayton         int     GetWidth()  { return GetMaxX(); }
122444d93782SGreg Clayton         int     GetHeight() { return GetMaxY(); }
122544d93782SGreg Clayton         void    MoveCursor (int x, int y) {  ::wmove (m_window, y, x); }
122644d93782SGreg Clayton         void    MoveWindow (int x, int y) {  MoveWindow(Point(x,y)); }
122744d93782SGreg Clayton         void    Resize (int w, int h) { ::wresize(m_window, h, w); }
122844d93782SGreg Clayton         void    Resize (const Size &size) { ::wresize(m_window, size.height, size.width); }
122944d93782SGreg Clayton         void    PutChar (int ch)    { ::waddch (m_window, ch); }
123044d93782SGreg Clayton         void    PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); }
123144d93782SGreg Clayton         void    Refresh ()  { ::wrefresh (m_window); }
123244d93782SGreg Clayton         void    DeferredRefresh ()
123344d93782SGreg Clayton         {
123444d93782SGreg Clayton             // We are using panels, so we don't need to call this...
123544d93782SGreg Clayton             //::wnoutrefresh(m_window);
123644d93782SGreg Clayton         }
123744d93782SGreg Clayton         void    SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); }
123844d93782SGreg Clayton         void    UnderlineOn ()  { AttributeOn(A_UNDERLINE); }
123944d93782SGreg Clayton         void    UnderlineOff () { AttributeOff(A_UNDERLINE); }
124044d93782SGreg Clayton 
124144d93782SGreg Clayton         void    PutCStringTruncated (const char *s, int right_pad)
124244d93782SGreg Clayton         {
124344d93782SGreg Clayton             int bytes_left = GetWidth() - GetCursorX();
124444d93782SGreg Clayton             if (bytes_left > right_pad)
124544d93782SGreg Clayton             {
124644d93782SGreg Clayton                 bytes_left -= right_pad;
124744d93782SGreg Clayton                 ::waddnstr (m_window, s, bytes_left);
124844d93782SGreg Clayton             }
124944d93782SGreg Clayton         }
125044d93782SGreg Clayton 
125144d93782SGreg Clayton         void
125244d93782SGreg Clayton         MoveWindow (const Point &origin)
125344d93782SGreg Clayton         {
125444d93782SGreg Clayton             const bool moving_window = origin != GetParentOrigin();
125544d93782SGreg Clayton             if (m_is_subwin && moving_window)
125644d93782SGreg Clayton             {
125744d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
125844d93782SGreg Clayton                 Size size = GetSize();
125944d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
126044d93782SGreg Clayton                                  size.height,
126144d93782SGreg Clayton                                  size.width,
126244d93782SGreg Clayton                                  origin.y,
126344d93782SGreg Clayton                                  origin.x), true);
126444d93782SGreg Clayton             }
126544d93782SGreg Clayton             else
126644d93782SGreg Clayton             {
126744d93782SGreg Clayton                 ::mvwin (m_window, origin.y, origin.x);
126844d93782SGreg Clayton             }
126944d93782SGreg Clayton         }
127044d93782SGreg Clayton 
127144d93782SGreg Clayton         void
127244d93782SGreg Clayton         SetBounds (const Rect &bounds)
127344d93782SGreg Clayton         {
127444d93782SGreg Clayton             const bool moving_window = bounds.origin != GetParentOrigin();
127544d93782SGreg Clayton             if (m_is_subwin && moving_window)
127644d93782SGreg Clayton             {
127744d93782SGreg Clayton                 // Can't move subwindows, must delete and re-create
127844d93782SGreg Clayton                 Reset (::subwin (m_parent->m_window,
127944d93782SGreg Clayton                                  bounds.size.height,
128044d93782SGreg Clayton                                  bounds.size.width,
128144d93782SGreg Clayton                                  bounds.origin.y,
128244d93782SGreg Clayton                                  bounds.origin.x), true);
128344d93782SGreg Clayton             }
128444d93782SGreg Clayton             else
128544d93782SGreg Clayton             {
128644d93782SGreg Clayton                 if (moving_window)
128744d93782SGreg Clayton                     MoveWindow(bounds.origin);
128844d93782SGreg Clayton                 Resize (bounds.size);
128944d93782SGreg Clayton             }
129044d93782SGreg Clayton         }
129144d93782SGreg Clayton 
129244d93782SGreg Clayton         void
129344d93782SGreg Clayton         Printf (const char *format, ...)  __attribute__ ((format (printf, 2, 3)))
129444d93782SGreg Clayton         {
129544d93782SGreg Clayton             va_list args;
129644d93782SGreg Clayton             va_start (args, format);
129744d93782SGreg Clayton             vwprintw(m_window, format, args);
129844d93782SGreg Clayton             va_end (args);
129944d93782SGreg Clayton         }
130044d93782SGreg Clayton 
130144d93782SGreg Clayton         void
130244d93782SGreg Clayton         Touch ()
130344d93782SGreg Clayton         {
130444d93782SGreg Clayton             ::touchwin (m_window);
130544d93782SGreg Clayton             if (m_parent)
130644d93782SGreg Clayton                 m_parent->Touch();
130744d93782SGreg Clayton         }
130844d93782SGreg Clayton 
130944d93782SGreg Clayton         WindowSP
131044d93782SGreg Clayton         CreateSubWindow (const char *name, const Rect &bounds, bool make_active)
131144d93782SGreg Clayton         {
131244d93782SGreg Clayton             WindowSP subwindow_sp;
131344d93782SGreg Clayton             if (m_window)
131444d93782SGreg Clayton             {
131544d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::subwin (m_window,
131644d93782SGreg Clayton                                                               bounds.size.height,
131744d93782SGreg Clayton                                                               bounds.size.width,
131844d93782SGreg Clayton                                                               bounds.origin.y,
131944d93782SGreg Clayton                                                               bounds.origin.x), true));
132044d93782SGreg Clayton                 subwindow_sp->m_is_subwin = true;
132144d93782SGreg Clayton             }
132244d93782SGreg Clayton             else
132344d93782SGreg Clayton             {
132444d93782SGreg Clayton                 subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height,
132544d93782SGreg Clayton                                                               bounds.size.width,
132644d93782SGreg Clayton                                                               bounds.origin.y,
132744d93782SGreg Clayton                                                               bounds.origin.x), true));
132844d93782SGreg Clayton                 subwindow_sp->m_is_subwin = false;
132944d93782SGreg Clayton             }
133044d93782SGreg Clayton             subwindow_sp->m_parent = this;
133144d93782SGreg Clayton             if (make_active)
133244d93782SGreg Clayton             {
133344d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
133444d93782SGreg Clayton                 m_curr_active_window_idx = m_subwindows.size();
133544d93782SGreg Clayton             }
133644d93782SGreg Clayton             m_subwindows.push_back(subwindow_sp);
133744d93782SGreg Clayton             ::top_panel (subwindow_sp->m_panel);
133844d93782SGreg Clayton             m_needs_update = true;
133944d93782SGreg Clayton             return subwindow_sp;
134044d93782SGreg Clayton         }
134144d93782SGreg Clayton 
134244d93782SGreg Clayton         bool
134344d93782SGreg Clayton         RemoveSubWindow (Window *window)
134444d93782SGreg Clayton         {
134544d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
134644d93782SGreg Clayton             size_t i = 0;
134744d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
134844d93782SGreg Clayton             {
134944d93782SGreg Clayton                 if ((*pos).get() == window)
135044d93782SGreg Clayton                 {
135144d93782SGreg Clayton                     if (m_prev_active_window_idx == i)
135244d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
135344d93782SGreg Clayton                     else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i)
135444d93782SGreg Clayton                         --m_prev_active_window_idx;
135544d93782SGreg Clayton 
135644d93782SGreg Clayton                     if (m_curr_active_window_idx == i)
135744d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
135844d93782SGreg Clayton                     else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i)
135944d93782SGreg Clayton                         --m_curr_active_window_idx;
136044d93782SGreg Clayton                     window->Erase();
136144d93782SGreg Clayton                     m_subwindows.erase(pos);
136244d93782SGreg Clayton                     m_needs_update = true;
136344d93782SGreg Clayton                     if (m_parent)
136444d93782SGreg Clayton                         m_parent->Touch();
136544d93782SGreg Clayton                     else
136644d93782SGreg Clayton                         ::touchwin (stdscr);
136744d93782SGreg Clayton                     return true;
136844d93782SGreg Clayton                 }
136944d93782SGreg Clayton             }
137044d93782SGreg Clayton             return false;
137144d93782SGreg Clayton         }
137244d93782SGreg Clayton 
137344d93782SGreg Clayton         WindowSP
137444d93782SGreg Clayton         FindSubWindow (const char *name)
137544d93782SGreg Clayton         {
137644d93782SGreg Clayton             Windows::iterator pos, end = m_subwindows.end();
137744d93782SGreg Clayton             size_t i = 0;
137844d93782SGreg Clayton             for (pos = m_subwindows.begin(); pos != end; ++pos, ++i)
137944d93782SGreg Clayton             {
138044d93782SGreg Clayton                 if ((*pos)->m_name.compare(name) == 0)
138144d93782SGreg Clayton                     return *pos;
138244d93782SGreg Clayton             }
138344d93782SGreg Clayton             return WindowSP();
138444d93782SGreg Clayton         }
138544d93782SGreg Clayton 
138644d93782SGreg Clayton         void
138744d93782SGreg Clayton         RemoveSubWindows ()
138844d93782SGreg Clayton         {
138944d93782SGreg Clayton             m_curr_active_window_idx = UINT32_MAX;
139044d93782SGreg Clayton             m_prev_active_window_idx = UINT32_MAX;
139144d93782SGreg Clayton             for (Windows::iterator pos = m_subwindows.begin();
139244d93782SGreg Clayton                  pos != m_subwindows.end();
139344d93782SGreg Clayton                  pos = m_subwindows.erase(pos))
139444d93782SGreg Clayton             {
139544d93782SGreg Clayton                 (*pos)->Erase();
139644d93782SGreg Clayton             }
139744d93782SGreg Clayton             if (m_parent)
139844d93782SGreg Clayton                 m_parent->Touch();
139944d93782SGreg Clayton             else
140044d93782SGreg Clayton                 ::touchwin (stdscr);
140144d93782SGreg Clayton         }
140244d93782SGreg Clayton 
140344d93782SGreg Clayton         WINDOW *
140444d93782SGreg Clayton         get()
140544d93782SGreg Clayton         {
140644d93782SGreg Clayton             return m_window;
140744d93782SGreg Clayton         }
140844d93782SGreg Clayton 
140944d93782SGreg Clayton         operator WINDOW *()
141044d93782SGreg Clayton         {
141144d93782SGreg Clayton             return m_window;
141244d93782SGreg Clayton         }
141344d93782SGreg Clayton 
141444d93782SGreg Clayton         //----------------------------------------------------------------------
141544d93782SGreg Clayton         // Window drawing utilities
141644d93782SGreg Clayton         //----------------------------------------------------------------------
141744d93782SGreg Clayton         void
141844d93782SGreg Clayton         DrawTitleBox (const char *title, const char *bottom_message = NULL)
141944d93782SGreg Clayton         {
142044d93782SGreg Clayton             attr_t attr = 0;
142144d93782SGreg Clayton             if (IsActive())
142244d93782SGreg Clayton                 attr = A_BOLD | COLOR_PAIR(2);
142344d93782SGreg Clayton             else
142444d93782SGreg Clayton                 attr = 0;
142544d93782SGreg Clayton             if (attr)
142644d93782SGreg Clayton                 AttributeOn(attr);
142744d93782SGreg Clayton 
142844d93782SGreg Clayton             Box();
142944d93782SGreg Clayton             MoveCursor(3, 0);
143044d93782SGreg Clayton 
143144d93782SGreg Clayton             if (title && title[0])
143244d93782SGreg Clayton             {
143344d93782SGreg Clayton                 PutChar ('<');
143444d93782SGreg Clayton                 PutCString (title);
143544d93782SGreg Clayton                 PutChar ('>');
143644d93782SGreg Clayton             }
143744d93782SGreg Clayton 
143844d93782SGreg Clayton             if (bottom_message && bottom_message[0])
143944d93782SGreg Clayton             {
144044d93782SGreg Clayton                 int bottom_message_length = strlen(bottom_message);
144144d93782SGreg Clayton                 int x = GetWidth() - 3 - (bottom_message_length + 2);
144244d93782SGreg Clayton 
144344d93782SGreg Clayton                 if (x > 0)
144444d93782SGreg Clayton                 {
144544d93782SGreg Clayton                     MoveCursor (x, GetHeight() - 1);
144644d93782SGreg Clayton                     PutChar ('[');
144744d93782SGreg Clayton                     PutCString(bottom_message);
144844d93782SGreg Clayton                     PutChar (']');
144944d93782SGreg Clayton                 }
145044d93782SGreg Clayton                 else
145144d93782SGreg Clayton                 {
145244d93782SGreg Clayton                     MoveCursor (1, GetHeight() - 1);
145344d93782SGreg Clayton                     PutChar ('[');
145444d93782SGreg Clayton                     PutCStringTruncated (bottom_message, 1);
145544d93782SGreg Clayton                 }
145644d93782SGreg Clayton             }
145744d93782SGreg Clayton             if (attr)
145844d93782SGreg Clayton                 AttributeOff(attr);
145944d93782SGreg Clayton 
146044d93782SGreg Clayton         }
146144d93782SGreg Clayton 
146244d93782SGreg Clayton         virtual void
146344d93782SGreg Clayton         Draw (bool force)
146444d93782SGreg Clayton         {
146544d93782SGreg Clayton             if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force))
146644d93782SGreg Clayton                 return;
146744d93782SGreg Clayton 
146844d93782SGreg Clayton             for (auto &subwindow_sp : m_subwindows)
146944d93782SGreg Clayton                 subwindow_sp->Draw(force);
147044d93782SGreg Clayton         }
147144d93782SGreg Clayton 
147244d93782SGreg Clayton         bool
147344d93782SGreg Clayton         CreateHelpSubwindow ()
147444d93782SGreg Clayton         {
147544d93782SGreg Clayton             if (m_delegate_sp)
147644d93782SGreg Clayton             {
147744d93782SGreg Clayton                 const char *text = m_delegate_sp->WindowDelegateGetHelpText ();
147844d93782SGreg Clayton                 KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp ();
147944d93782SGreg Clayton                 if ((text && text[0]) || key_help)
148044d93782SGreg Clayton                 {
148144d93782SGreg Clayton                     std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help));
148244d93782SGreg Clayton                     const size_t num_lines = help_delegate_ap->GetNumLines();
148344d93782SGreg Clayton                     const size_t max_length = help_delegate_ap->GetMaxLineLength();
148444d93782SGreg Clayton                     Rect bounds = GetBounds();
148544d93782SGreg Clayton                     bounds.Inset(1, 1);
14863985c8c6SSaleem Abdulrasool                     if (max_length + 4 < static_cast<size_t>(bounds.size.width))
148744d93782SGreg Clayton                     {
148844d93782SGreg Clayton                         bounds.origin.x += (bounds.size.width - max_length + 4)/2;
148944d93782SGreg Clayton                         bounds.size.width = max_length + 4;
149044d93782SGreg Clayton                     }
149144d93782SGreg Clayton                     else
149244d93782SGreg Clayton                     {
149344d93782SGreg Clayton                         if (bounds.size.width > 100)
149444d93782SGreg Clayton                         {
149544d93782SGreg Clayton                             const int inset_w = bounds.size.width / 4;
149644d93782SGreg Clayton                             bounds.origin.x += inset_w;
149744d93782SGreg Clayton                             bounds.size.width -= 2*inset_w;
149844d93782SGreg Clayton                         }
149944d93782SGreg Clayton                     }
150044d93782SGreg Clayton 
15013985c8c6SSaleem Abdulrasool                     if (num_lines + 2 < static_cast<size_t>(bounds.size.height))
150244d93782SGreg Clayton                     {
150344d93782SGreg Clayton                         bounds.origin.y += (bounds.size.height - num_lines + 2)/2;
150444d93782SGreg Clayton                         bounds.size.height = num_lines + 2;
150544d93782SGreg Clayton                     }
150644d93782SGreg Clayton                     else
150744d93782SGreg Clayton                     {
150844d93782SGreg Clayton                         if (bounds.size.height > 100)
150944d93782SGreg Clayton                         {
151044d93782SGreg Clayton                             const int inset_h = bounds.size.height / 4;
151144d93782SGreg Clayton                             bounds.origin.y += inset_h;
151244d93782SGreg Clayton                             bounds.size.height -= 2*inset_h;
151344d93782SGreg Clayton                         }
151444d93782SGreg Clayton                     }
15155fdb09bbSGreg Clayton                     WindowSP help_window_sp;
15165fdb09bbSGreg Clayton                     Window *parent_window = GetParent();
15175fdb09bbSGreg Clayton                     if (parent_window)
15185fdb09bbSGreg Clayton                         help_window_sp = parent_window->CreateSubWindow("Help", bounds, true);
15195fdb09bbSGreg Clayton                     else
15205fdb09bbSGreg Clayton                         help_window_sp = CreateSubWindow("Help", bounds, true);
152144d93782SGreg Clayton                     help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release()));
152244d93782SGreg Clayton                     return true;
152344d93782SGreg Clayton                 }
152444d93782SGreg Clayton             }
152544d93782SGreg Clayton             return false;
152644d93782SGreg Clayton         }
152744d93782SGreg Clayton 
152844d93782SGreg Clayton         virtual HandleCharResult
152944d93782SGreg Clayton         HandleChar (int key)
153044d93782SGreg Clayton         {
153144d93782SGreg Clayton             // Always check the active window first
153244d93782SGreg Clayton             HandleCharResult result = eKeyNotHandled;
153344d93782SGreg Clayton             WindowSP active_window_sp = GetActiveWindow ();
153444d93782SGreg Clayton             if (active_window_sp)
153544d93782SGreg Clayton             {
153644d93782SGreg Clayton                 result = active_window_sp->HandleChar (key);
153744d93782SGreg Clayton                 if (result != eKeyNotHandled)
153844d93782SGreg Clayton                     return result;
153944d93782SGreg Clayton             }
154044d93782SGreg Clayton 
154144d93782SGreg Clayton             if (m_delegate_sp)
154244d93782SGreg Clayton             {
154344d93782SGreg Clayton                 result = m_delegate_sp->WindowDelegateHandleChar (*this, key);
154444d93782SGreg Clayton                 if (result != eKeyNotHandled)
154544d93782SGreg Clayton                     return result;
154644d93782SGreg Clayton             }
154744d93782SGreg Clayton 
154844d93782SGreg Clayton             // Then check for any windows that want any keys
154944d93782SGreg Clayton             // that weren't handled. This is typically only
155044d93782SGreg Clayton             // for a menubar.
155144d93782SGreg Clayton             // Make a copy of the subwindows in case any HandleChar()
155244d93782SGreg Clayton             // functions muck with the subwindows. If we don't do this,
155344d93782SGreg Clayton             // we can crash when iterating over the subwindows.
155444d93782SGreg Clayton             Windows subwindows (m_subwindows);
155544d93782SGreg Clayton             for (auto subwindow_sp : subwindows)
155644d93782SGreg Clayton             {
155744d93782SGreg Clayton                 if (subwindow_sp->m_can_activate == false)
155844d93782SGreg Clayton                 {
155944d93782SGreg Clayton                     HandleCharResult result = subwindow_sp->HandleChar(key);
156044d93782SGreg Clayton                     if (result != eKeyNotHandled)
156144d93782SGreg Clayton                         return result;
156244d93782SGreg Clayton                 }
156344d93782SGreg Clayton             }
156444d93782SGreg Clayton 
156544d93782SGreg Clayton             return eKeyNotHandled;
156644d93782SGreg Clayton         }
156744d93782SGreg Clayton 
156844d93782SGreg Clayton         bool
156944d93782SGreg Clayton         SetActiveWindow (Window *window)
157044d93782SGreg Clayton         {
157144d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
157244d93782SGreg Clayton             for (size_t i=0; i<num_subwindows; ++i)
157344d93782SGreg Clayton             {
157444d93782SGreg Clayton                 if (m_subwindows[i].get() == window)
157544d93782SGreg Clayton                 {
157644d93782SGreg Clayton                     m_prev_active_window_idx = m_curr_active_window_idx;
157744d93782SGreg Clayton                     ::top_panel (window->m_panel);
157844d93782SGreg Clayton                     m_curr_active_window_idx = i;
157944d93782SGreg Clayton                     return true;
158044d93782SGreg Clayton                 }
158144d93782SGreg Clayton             }
158244d93782SGreg Clayton             return false;
158344d93782SGreg Clayton         }
158444d93782SGreg Clayton 
158544d93782SGreg Clayton         WindowSP
158644d93782SGreg Clayton         GetActiveWindow ()
158744d93782SGreg Clayton         {
158844d93782SGreg Clayton             if (!m_subwindows.empty())
158944d93782SGreg Clayton             {
159044d93782SGreg Clayton                 if (m_curr_active_window_idx >= m_subwindows.size())
159144d93782SGreg Clayton                 {
159244d93782SGreg Clayton                     if (m_prev_active_window_idx < m_subwindows.size())
159344d93782SGreg Clayton                     {
159444d93782SGreg Clayton                         m_curr_active_window_idx = m_prev_active_window_idx;
159544d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
159644d93782SGreg Clayton                     }
159744d93782SGreg Clayton                     else if (IsActive())
159844d93782SGreg Clayton                     {
159944d93782SGreg Clayton                         m_prev_active_window_idx = UINT32_MAX;
160044d93782SGreg Clayton                         m_curr_active_window_idx = UINT32_MAX;
160144d93782SGreg Clayton 
160244d93782SGreg Clayton                         // Find first window that wants to be active if this window is active
160344d93782SGreg Clayton                         const size_t num_subwindows = m_subwindows.size();
160444d93782SGreg Clayton                         for (size_t i=0; i<num_subwindows; ++i)
160544d93782SGreg Clayton                         {
160644d93782SGreg Clayton                             if (m_subwindows[i]->GetCanBeActive())
160744d93782SGreg Clayton                             {
160844d93782SGreg Clayton                                 m_curr_active_window_idx = i;
160944d93782SGreg Clayton                                 break;
161044d93782SGreg Clayton                             }
161144d93782SGreg Clayton                         }
161244d93782SGreg Clayton                     }
161344d93782SGreg Clayton                 }
161444d93782SGreg Clayton 
161544d93782SGreg Clayton                 if (m_curr_active_window_idx < m_subwindows.size())
161644d93782SGreg Clayton                     return m_subwindows[m_curr_active_window_idx];
161744d93782SGreg Clayton             }
161844d93782SGreg Clayton             return WindowSP();
161944d93782SGreg Clayton         }
162044d93782SGreg Clayton 
162144d93782SGreg Clayton         bool
162244d93782SGreg Clayton         GetCanBeActive () const
162344d93782SGreg Clayton         {
162444d93782SGreg Clayton             return m_can_activate;
162544d93782SGreg Clayton         }
162644d93782SGreg Clayton 
162744d93782SGreg Clayton         void
162844d93782SGreg Clayton         SetCanBeActive (bool b)
162944d93782SGreg Clayton         {
163044d93782SGreg Clayton             m_can_activate = b;
163144d93782SGreg Clayton         }
163244d93782SGreg Clayton 
163344d93782SGreg Clayton         const WindowDelegateSP &
163444d93782SGreg Clayton         GetDelegate () const
163544d93782SGreg Clayton         {
163644d93782SGreg Clayton             return m_delegate_sp;
163744d93782SGreg Clayton         }
163844d93782SGreg Clayton 
163944d93782SGreg Clayton         void
164044d93782SGreg Clayton         SetDelegate (const WindowDelegateSP &delegate_sp)
164144d93782SGreg Clayton         {
164244d93782SGreg Clayton             m_delegate_sp = delegate_sp;
164344d93782SGreg Clayton         }
164444d93782SGreg Clayton 
164544d93782SGreg Clayton         Window *
164644d93782SGreg Clayton         GetParent () const
164744d93782SGreg Clayton         {
164844d93782SGreg Clayton             return m_parent;
164944d93782SGreg Clayton         }
165044d93782SGreg Clayton 
165144d93782SGreg Clayton         bool
165244d93782SGreg Clayton         IsActive () const
165344d93782SGreg Clayton         {
165444d93782SGreg Clayton             if (m_parent)
165544d93782SGreg Clayton                 return m_parent->GetActiveWindow().get() == this;
165644d93782SGreg Clayton             else
165744d93782SGreg Clayton                 return true; // Top level window is always active
165844d93782SGreg Clayton         }
165944d93782SGreg Clayton 
166044d93782SGreg Clayton         void
166144d93782SGreg Clayton         SelectNextWindowAsActive ()
166244d93782SGreg Clayton         {
166344d93782SGreg Clayton             // Move active focus to next window
166444d93782SGreg Clayton             const size_t num_subwindows = m_subwindows.size();
166544d93782SGreg Clayton             if (m_curr_active_window_idx == UINT32_MAX)
166644d93782SGreg Clayton             {
166744d93782SGreg Clayton                 uint32_t idx = 0;
166844d93782SGreg Clayton                 for (auto subwindow_sp : m_subwindows)
166944d93782SGreg Clayton                 {
167044d93782SGreg Clayton                     if (subwindow_sp->GetCanBeActive())
167144d93782SGreg Clayton                     {
167244d93782SGreg Clayton                         m_curr_active_window_idx = idx;
167344d93782SGreg Clayton                         break;
167444d93782SGreg Clayton                     }
167544d93782SGreg Clayton                     ++idx;
167644d93782SGreg Clayton                 }
167744d93782SGreg Clayton             }
167844d93782SGreg Clayton             else if (m_curr_active_window_idx + 1 < num_subwindows)
167944d93782SGreg Clayton             {
168044d93782SGreg Clayton                 bool handled = false;
168144d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
168244d93782SGreg Clayton                 for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx)
168344d93782SGreg Clayton                 {
168444d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
168544d93782SGreg Clayton                     {
168644d93782SGreg Clayton                         m_curr_active_window_idx = idx;
168744d93782SGreg Clayton                         handled = true;
168844d93782SGreg Clayton                         break;
168944d93782SGreg Clayton                     }
169044d93782SGreg Clayton                 }
169144d93782SGreg Clayton                 if (!handled)
169244d93782SGreg Clayton                 {
169344d93782SGreg Clayton                     for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx)
169444d93782SGreg Clayton                     {
169544d93782SGreg Clayton                         if (m_subwindows[idx]->GetCanBeActive())
169644d93782SGreg Clayton                         {
169744d93782SGreg Clayton                             m_curr_active_window_idx = idx;
169844d93782SGreg Clayton                             break;
169944d93782SGreg Clayton                         }
170044d93782SGreg Clayton                     }
170144d93782SGreg Clayton                 }
170244d93782SGreg Clayton             }
170344d93782SGreg Clayton             else
170444d93782SGreg Clayton             {
170544d93782SGreg Clayton                 m_prev_active_window_idx = m_curr_active_window_idx;
170644d93782SGreg Clayton                 for (size_t idx=0; idx<num_subwindows; ++idx)
170744d93782SGreg Clayton                 {
170844d93782SGreg Clayton                     if (m_subwindows[idx]->GetCanBeActive())
170944d93782SGreg Clayton                     {
171044d93782SGreg Clayton                         m_curr_active_window_idx = idx;
171144d93782SGreg Clayton                         break;
171244d93782SGreg Clayton                     }
171344d93782SGreg Clayton                 }
171444d93782SGreg Clayton             }
171544d93782SGreg Clayton         }
171644d93782SGreg Clayton 
171744d93782SGreg Clayton         const char *
171844d93782SGreg Clayton         GetName () const
171944d93782SGreg Clayton         {
172044d93782SGreg Clayton             return m_name.c_str();
172144d93782SGreg Clayton         }
1722315b6884SEugene Zelenko 
172344d93782SGreg Clayton     protected:
172444d93782SGreg Clayton         std::string m_name;
172544d93782SGreg Clayton         WINDOW *m_window;
172644d93782SGreg Clayton         PANEL *m_panel;
172744d93782SGreg Clayton         Window *m_parent;
172844d93782SGreg Clayton         Windows m_subwindows;
172944d93782SGreg Clayton         WindowDelegateSP m_delegate_sp;
173044d93782SGreg Clayton         uint32_t m_curr_active_window_idx;
173144d93782SGreg Clayton         uint32_t m_prev_active_window_idx;
173244d93782SGreg Clayton         bool m_delete;
173344d93782SGreg Clayton         bool m_needs_update;
173444d93782SGreg Clayton         bool m_can_activate;
173544d93782SGreg Clayton         bool m_is_subwin;
173644d93782SGreg Clayton 
173744d93782SGreg Clayton     private:
173844d93782SGreg Clayton         DISALLOW_COPY_AND_ASSIGN(Window);
173944d93782SGreg Clayton     };
174044d93782SGreg Clayton 
174144d93782SGreg Clayton     class MenuDelegate
174244d93782SGreg Clayton     {
174344d93782SGreg Clayton     public:
1744315b6884SEugene Zelenko         virtual ~MenuDelegate() = default;
174544d93782SGreg Clayton 
174644d93782SGreg Clayton         virtual MenuActionResult
174744d93782SGreg Clayton         MenuDelegateAction (Menu &menu) = 0;
174844d93782SGreg Clayton     };
174944d93782SGreg Clayton 
175044d93782SGreg Clayton     class Menu : public WindowDelegate
175144d93782SGreg Clayton     {
175244d93782SGreg Clayton     public:
175344d93782SGreg Clayton         enum class Type
175444d93782SGreg Clayton         {
175544d93782SGreg Clayton             Invalid,
175644d93782SGreg Clayton             Bar,
175744d93782SGreg Clayton             Item,
175844d93782SGreg Clayton             Separator
175944d93782SGreg Clayton         };
176044d93782SGreg Clayton 
176144d93782SGreg Clayton         // Menubar or separator constructor
176244d93782SGreg Clayton         Menu (Type type);
176344d93782SGreg Clayton 
176444d93782SGreg Clayton         // Menuitem constructor
176544d93782SGreg Clayton         Menu (const char *name,
176644d93782SGreg Clayton               const char *key_name,
176744d93782SGreg Clayton               int key_value,
176844d93782SGreg Clayton               uint64_t identifier);
176944d93782SGreg Clayton 
1770315b6884SEugene Zelenko         ~Menu() override = default;
177144d93782SGreg Clayton 
177244d93782SGreg Clayton         const MenuDelegateSP &
177344d93782SGreg Clayton         GetDelegate () const
177444d93782SGreg Clayton         {
177544d93782SGreg Clayton             return m_delegate_sp;
177644d93782SGreg Clayton         }
177744d93782SGreg Clayton 
177844d93782SGreg Clayton         void
177944d93782SGreg Clayton         SetDelegate (const MenuDelegateSP &delegate_sp)
178044d93782SGreg Clayton         {
178144d93782SGreg Clayton             m_delegate_sp = delegate_sp;
178244d93782SGreg Clayton         }
178344d93782SGreg Clayton 
178444d93782SGreg Clayton         void
178544d93782SGreg Clayton         RecalculateNameLengths();
178644d93782SGreg Clayton 
178744d93782SGreg Clayton         void
178844d93782SGreg Clayton         AddSubmenu (const MenuSP &menu_sp);
178944d93782SGreg Clayton 
179044d93782SGreg Clayton         int
179144d93782SGreg Clayton         DrawAndRunMenu (Window &window);
179244d93782SGreg Clayton 
179344d93782SGreg Clayton         void
179444d93782SGreg Clayton         DrawMenuTitle (Window &window, bool highlight);
179544d93782SGreg Clayton 
1796bd5ae6b4SGreg Clayton         bool
1797bd5ae6b4SGreg Clayton         WindowDelegateDraw (Window &window, bool force) override;
179844d93782SGreg Clayton 
1799bd5ae6b4SGreg Clayton         HandleCharResult
1800bd5ae6b4SGreg Clayton         WindowDelegateHandleChar (Window &window, int key) override;
180144d93782SGreg Clayton 
180244d93782SGreg Clayton         MenuActionResult
180344d93782SGreg Clayton         ActionPrivate (Menu &menu)
180444d93782SGreg Clayton         {
180544d93782SGreg Clayton             MenuActionResult result = MenuActionResult::NotHandled;
180644d93782SGreg Clayton             if (m_delegate_sp)
180744d93782SGreg Clayton             {
180844d93782SGreg Clayton                 result = m_delegate_sp->MenuDelegateAction (menu);
180944d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
181044d93782SGreg Clayton                     return result;
181144d93782SGreg Clayton             }
181244d93782SGreg Clayton             else if (m_parent)
181344d93782SGreg Clayton             {
181444d93782SGreg Clayton                 result = m_parent->ActionPrivate(menu);
181544d93782SGreg Clayton                 if (result != MenuActionResult::NotHandled)
181644d93782SGreg Clayton                     return result;
181744d93782SGreg Clayton             }
181844d93782SGreg Clayton             return m_canned_result;
181944d93782SGreg Clayton         }
182044d93782SGreg Clayton 
182144d93782SGreg Clayton         MenuActionResult
182244d93782SGreg Clayton         Action ()
182344d93782SGreg Clayton         {
182444d93782SGreg Clayton             // Call the recursive action so it can try to handle it
182544d93782SGreg Clayton             // with the menu delegate, and if not, try our parent menu
182644d93782SGreg Clayton             return ActionPrivate (*this);
182744d93782SGreg Clayton         }
182844d93782SGreg Clayton 
182944d93782SGreg Clayton         void
183044d93782SGreg Clayton         SetCannedResult (MenuActionResult result)
183144d93782SGreg Clayton         {
183244d93782SGreg Clayton             m_canned_result = result;
183344d93782SGreg Clayton         }
183444d93782SGreg Clayton 
183544d93782SGreg Clayton         Menus &
183644d93782SGreg Clayton         GetSubmenus()
183744d93782SGreg Clayton         {
183844d93782SGreg Clayton             return m_submenus;
183944d93782SGreg Clayton         }
184044d93782SGreg Clayton 
184144d93782SGreg Clayton         const Menus &
184244d93782SGreg Clayton         GetSubmenus() const
184344d93782SGreg Clayton         {
184444d93782SGreg Clayton             return m_submenus;
184544d93782SGreg Clayton         }
184644d93782SGreg Clayton 
184744d93782SGreg Clayton         int
184844d93782SGreg Clayton         GetSelectedSubmenuIndex () const
184944d93782SGreg Clayton         {
185044d93782SGreg Clayton             return m_selected;
185144d93782SGreg Clayton         }
185244d93782SGreg Clayton 
185344d93782SGreg Clayton         void
185444d93782SGreg Clayton         SetSelectedSubmenuIndex (int idx)
185544d93782SGreg Clayton         {
185644d93782SGreg Clayton             m_selected = idx;
185744d93782SGreg Clayton         }
185844d93782SGreg Clayton 
185944d93782SGreg Clayton         Type
186044d93782SGreg Clayton         GetType () const
186144d93782SGreg Clayton         {
186244d93782SGreg Clayton             return m_type;
186344d93782SGreg Clayton         }
186444d93782SGreg Clayton 
186544d93782SGreg Clayton         int
186644d93782SGreg Clayton         GetStartingColumn() const
186744d93782SGreg Clayton         {
186844d93782SGreg Clayton             return m_start_col;
186944d93782SGreg Clayton         }
187044d93782SGreg Clayton 
187144d93782SGreg Clayton         void
187244d93782SGreg Clayton         SetStartingColumn(int col)
187344d93782SGreg Clayton         {
187444d93782SGreg Clayton             m_start_col = col;
187544d93782SGreg Clayton         }
187644d93782SGreg Clayton 
187744d93782SGreg Clayton         int
187844d93782SGreg Clayton         GetKeyValue() const
187944d93782SGreg Clayton         {
188044d93782SGreg Clayton             return m_key_value;
188144d93782SGreg Clayton         }
188244d93782SGreg Clayton 
188344d93782SGreg Clayton         void
188444d93782SGreg Clayton         SetKeyValue(int key_value)
188544d93782SGreg Clayton         {
188644d93782SGreg Clayton             m_key_value = key_value;
188744d93782SGreg Clayton         }
188844d93782SGreg Clayton 
188944d93782SGreg Clayton         std::string &
189044d93782SGreg Clayton         GetName()
189144d93782SGreg Clayton         {
189244d93782SGreg Clayton             return m_name;
189344d93782SGreg Clayton         }
189444d93782SGreg Clayton 
189544d93782SGreg Clayton         std::string &
189644d93782SGreg Clayton         GetKeyName()
189744d93782SGreg Clayton         {
189844d93782SGreg Clayton             return m_key_name;
189944d93782SGreg Clayton         }
190044d93782SGreg Clayton 
190144d93782SGreg Clayton         int
190244d93782SGreg Clayton         GetDrawWidth () const
190344d93782SGreg Clayton         {
190444d93782SGreg Clayton             return m_max_submenu_name_length + m_max_submenu_key_name_length + 8;
190544d93782SGreg Clayton         }
190644d93782SGreg Clayton 
190744d93782SGreg Clayton         uint64_t
190844d93782SGreg Clayton         GetIdentifier() const
190944d93782SGreg Clayton         {
191044d93782SGreg Clayton             return m_identifier;
191144d93782SGreg Clayton         }
191244d93782SGreg Clayton 
191344d93782SGreg Clayton         void
191444d93782SGreg Clayton         SetIdentifier (uint64_t identifier)
191544d93782SGreg Clayton         {
191644d93782SGreg Clayton             m_identifier = identifier;
191744d93782SGreg Clayton         }
191844d93782SGreg Clayton 
191944d93782SGreg Clayton     protected:
192044d93782SGreg Clayton         std::string m_name;
192144d93782SGreg Clayton         std::string m_key_name;
192244d93782SGreg Clayton         uint64_t m_identifier;
192344d93782SGreg Clayton         Type m_type;
192444d93782SGreg Clayton         int m_key_value;
192544d93782SGreg Clayton         int m_start_col;
192644d93782SGreg Clayton         int m_max_submenu_name_length;
192744d93782SGreg Clayton         int m_max_submenu_key_name_length;
192844d93782SGreg Clayton         int m_selected;
192944d93782SGreg Clayton         Menu *m_parent;
193044d93782SGreg Clayton         Menus m_submenus;
193144d93782SGreg Clayton         WindowSP m_menu_window_sp;
193244d93782SGreg Clayton         MenuActionResult m_canned_result;
193344d93782SGreg Clayton         MenuDelegateSP m_delegate_sp;
193444d93782SGreg Clayton     };
193544d93782SGreg Clayton 
193644d93782SGreg Clayton     // Menubar or separator constructor
193744d93782SGreg Clayton     Menu::Menu (Type type) :
193844d93782SGreg Clayton         m_name (),
193944d93782SGreg Clayton         m_key_name (),
194044d93782SGreg Clayton         m_identifier (0),
194144d93782SGreg Clayton         m_type (type),
194244d93782SGreg Clayton         m_key_value (0),
194344d93782SGreg Clayton         m_start_col (0),
194444d93782SGreg Clayton         m_max_submenu_name_length (0),
194544d93782SGreg Clayton         m_max_submenu_key_name_length (0),
194644d93782SGreg Clayton         m_selected (0),
194744d93782SGreg Clayton         m_parent (NULL),
194844d93782SGreg Clayton         m_submenus (),
194944d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
195044d93782SGreg Clayton         m_delegate_sp()
195144d93782SGreg Clayton     {
195244d93782SGreg Clayton     }
195344d93782SGreg Clayton 
195444d93782SGreg Clayton     // Menuitem constructor
195544d93782SGreg Clayton     Menu::Menu (const char *name,
195644d93782SGreg Clayton                 const char *key_name,
195744d93782SGreg Clayton                 int key_value,
195844d93782SGreg Clayton                 uint64_t identifier) :
195944d93782SGreg Clayton         m_name (),
196044d93782SGreg Clayton         m_key_name (),
196144d93782SGreg Clayton         m_identifier (identifier),
196244d93782SGreg Clayton         m_type (Type::Invalid),
196344d93782SGreg Clayton         m_key_value (key_value),
196444d93782SGreg Clayton         m_start_col (0),
196544d93782SGreg Clayton         m_max_submenu_name_length (0),
196644d93782SGreg Clayton         m_max_submenu_key_name_length (0),
196744d93782SGreg Clayton         m_selected (0),
196844d93782SGreg Clayton         m_parent (NULL),
196944d93782SGreg Clayton         m_submenus (),
197044d93782SGreg Clayton         m_canned_result (MenuActionResult::NotHandled),
197144d93782SGreg Clayton         m_delegate_sp()
197244d93782SGreg Clayton     {
197344d93782SGreg Clayton         if (name && name[0])
197444d93782SGreg Clayton         {
197544d93782SGreg Clayton             m_name = name;
197644d93782SGreg Clayton             m_type = Type::Item;
197744d93782SGreg Clayton             if (key_name && key_name[0])
197844d93782SGreg Clayton                 m_key_name = key_name;
197944d93782SGreg Clayton         }
198044d93782SGreg Clayton         else
198144d93782SGreg Clayton         {
198244d93782SGreg Clayton             m_type = Type::Separator;
198344d93782SGreg Clayton         }
198444d93782SGreg Clayton     }
198544d93782SGreg Clayton 
198644d93782SGreg Clayton     void
198744d93782SGreg Clayton     Menu::RecalculateNameLengths()
198844d93782SGreg Clayton     {
198944d93782SGreg Clayton         m_max_submenu_name_length = 0;
199044d93782SGreg Clayton         m_max_submenu_key_name_length = 0;
199144d93782SGreg Clayton         Menus &submenus = GetSubmenus();
199244d93782SGreg Clayton         const size_t num_submenus = submenus.size();
199344d93782SGreg Clayton         for (size_t i=0; i<num_submenus; ++i)
199444d93782SGreg Clayton         {
199544d93782SGreg Clayton             Menu *submenu = submenus[i].get();
19963985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size())
199744d93782SGreg Clayton                 m_max_submenu_name_length = submenu->m_name.size();
19983985c8c6SSaleem Abdulrasool             if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size())
199944d93782SGreg Clayton                 m_max_submenu_key_name_length = submenu->m_key_name.size();
200044d93782SGreg Clayton         }
200144d93782SGreg Clayton     }
200244d93782SGreg Clayton 
200344d93782SGreg Clayton     void
200444d93782SGreg Clayton     Menu::AddSubmenu (const MenuSP &menu_sp)
200544d93782SGreg Clayton     {
200644d93782SGreg Clayton         menu_sp->m_parent = this;
20073985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size())
200844d93782SGreg Clayton             m_max_submenu_name_length = menu_sp->m_name.size();
20093985c8c6SSaleem Abdulrasool         if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size())
201044d93782SGreg Clayton             m_max_submenu_key_name_length = menu_sp->m_key_name.size();
201144d93782SGreg Clayton         m_submenus.push_back(menu_sp);
201244d93782SGreg Clayton     }
201344d93782SGreg Clayton 
201444d93782SGreg Clayton     void
201544d93782SGreg Clayton     Menu::DrawMenuTitle (Window &window, bool highlight)
201644d93782SGreg Clayton     {
201744d93782SGreg Clayton         if (m_type == Type::Separator)
201844d93782SGreg Clayton         {
201944d93782SGreg Clayton             window.MoveCursor(0, window.GetCursorY());
202044d93782SGreg Clayton             window.PutChar(ACS_LTEE);
202144d93782SGreg Clayton             int width = window.GetWidth();
202244d93782SGreg Clayton             if (width > 2)
202344d93782SGreg Clayton             {
202444d93782SGreg Clayton                 width -= 2;
20253985c8c6SSaleem Abdulrasool                 for (int i=0; i< width; ++i)
202644d93782SGreg Clayton                     window.PutChar(ACS_HLINE);
202744d93782SGreg Clayton             }
202844d93782SGreg Clayton             window.PutChar(ACS_RTEE);
202944d93782SGreg Clayton         }
203044d93782SGreg Clayton         else
203144d93782SGreg Clayton         {
203244d93782SGreg Clayton             const int shortcut_key = m_key_value;
203344d93782SGreg Clayton             bool underlined_shortcut = false;
203444d93782SGreg Clayton             const attr_t hilgight_attr = A_REVERSE;
203544d93782SGreg Clayton             if (highlight)
203644d93782SGreg Clayton                 window.AttributeOn(hilgight_attr);
203744d93782SGreg Clayton             if (isprint(shortcut_key))
203844d93782SGreg Clayton             {
203944d93782SGreg Clayton                 size_t lower_pos = m_name.find(tolower(shortcut_key));
204044d93782SGreg Clayton                 size_t upper_pos = m_name.find(toupper(shortcut_key));
204144d93782SGreg Clayton                 const char *name = m_name.c_str();
204244d93782SGreg Clayton                 size_t pos = std::min<size_t>(lower_pos, upper_pos);
204344d93782SGreg Clayton                 if (pos != std::string::npos)
204444d93782SGreg Clayton                 {
204544d93782SGreg Clayton                     underlined_shortcut = true;
204644d93782SGreg Clayton                     if (pos > 0)
204744d93782SGreg Clayton                     {
204844d93782SGreg Clayton                         window.PutCString(name, pos);
204944d93782SGreg Clayton                         name += pos;
205044d93782SGreg Clayton                     }
205144d93782SGreg Clayton                     const attr_t shortcut_attr = A_UNDERLINE|A_BOLD;
205244d93782SGreg Clayton                     window.AttributeOn (shortcut_attr);
205344d93782SGreg Clayton                     window.PutChar(name[0]);
205444d93782SGreg Clayton                     window.AttributeOff(shortcut_attr);
205544d93782SGreg Clayton                     name++;
205644d93782SGreg Clayton                     if (name[0])
205744d93782SGreg Clayton                         window.PutCString(name);
205844d93782SGreg Clayton                 }
205944d93782SGreg Clayton             }
206044d93782SGreg Clayton 
206144d93782SGreg Clayton             if (!underlined_shortcut)
206244d93782SGreg Clayton             {
206344d93782SGreg Clayton                 window.PutCString(m_name.c_str());
206444d93782SGreg Clayton             }
206544d93782SGreg Clayton 
206644d93782SGreg Clayton             if (highlight)
206744d93782SGreg Clayton                 window.AttributeOff(hilgight_attr);
206844d93782SGreg Clayton 
206944d93782SGreg Clayton             if (m_key_name.empty())
207044d93782SGreg Clayton             {
207144d93782SGreg Clayton                 if (!underlined_shortcut && isprint(m_key_value))
207244d93782SGreg Clayton                 {
207344d93782SGreg Clayton                     window.AttributeOn (COLOR_PAIR(3));
207444d93782SGreg Clayton                     window.Printf (" (%c)", m_key_value);
207544d93782SGreg Clayton                     window.AttributeOff (COLOR_PAIR(3));
207644d93782SGreg Clayton                 }
207744d93782SGreg Clayton             }
207844d93782SGreg Clayton             else
207944d93782SGreg Clayton             {
208044d93782SGreg Clayton                 window.AttributeOn (COLOR_PAIR(3));
208144d93782SGreg Clayton                 window.Printf (" (%s)", m_key_name.c_str());
208244d93782SGreg Clayton                 window.AttributeOff (COLOR_PAIR(3));
208344d93782SGreg Clayton             }
208444d93782SGreg Clayton         }
208544d93782SGreg Clayton     }
208644d93782SGreg Clayton 
208744d93782SGreg Clayton     bool
208844d93782SGreg Clayton     Menu::WindowDelegateDraw (Window &window, bool force)
208944d93782SGreg Clayton     {
209044d93782SGreg Clayton         Menus &submenus = GetSubmenus();
209144d93782SGreg Clayton         const size_t num_submenus = submenus.size();
209244d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
209344d93782SGreg Clayton         Menu::Type menu_type = GetType ();
209444d93782SGreg Clayton         switch (menu_type)
209544d93782SGreg Clayton         {
209644d93782SGreg Clayton         case  Menu::Type::Bar:
209744d93782SGreg Clayton             {
209844d93782SGreg Clayton                 window.SetBackground(2);
209944d93782SGreg Clayton                 window.MoveCursor(0, 0);
210044d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
210144d93782SGreg Clayton                 {
210244d93782SGreg Clayton                     Menu *menu = submenus[i].get();
210344d93782SGreg Clayton                     if (i > 0)
210444d93782SGreg Clayton                         window.PutChar(' ');
210544d93782SGreg Clayton                     menu->SetStartingColumn (window.GetCursorX());
210644d93782SGreg Clayton                     window.PutCString("| ");
210744d93782SGreg Clayton                     menu->DrawMenuTitle (window, false);
210844d93782SGreg Clayton                 }
210944d93782SGreg Clayton                 window.PutCString(" |");
211044d93782SGreg Clayton                 window.DeferredRefresh();
211144d93782SGreg Clayton             }
211244d93782SGreg Clayton             break;
211344d93782SGreg Clayton 
211444d93782SGreg Clayton         case Menu::Type::Item:
211544d93782SGreg Clayton             {
211644d93782SGreg Clayton                 int y = 1;
211744d93782SGreg Clayton                 int x = 3;
211844d93782SGreg Clayton                 // Draw the menu
211944d93782SGreg Clayton                 int cursor_x = 0;
212044d93782SGreg Clayton                 int cursor_y = 0;
212144d93782SGreg Clayton                 window.Erase();
212244d93782SGreg Clayton                 window.SetBackground(2);
212344d93782SGreg Clayton                 window.Box();
212444d93782SGreg Clayton                 for (size_t i=0; i<num_submenus; ++i)
212544d93782SGreg Clayton                 {
21263985c8c6SSaleem Abdulrasool                     const bool is_selected =
21273985c8c6SSaleem Abdulrasool                       (i == static_cast<size_t>(selected_idx));
212844d93782SGreg Clayton                     window.MoveCursor(x, y + i);
212944d93782SGreg Clayton                     if (is_selected)
213044d93782SGreg Clayton                     {
213144d93782SGreg Clayton                         // Remember where we want the cursor to be
213244d93782SGreg Clayton                         cursor_x = x-1;
213344d93782SGreg Clayton                         cursor_y = y+i;
213444d93782SGreg Clayton                     }
213544d93782SGreg Clayton                     submenus[i]->DrawMenuTitle (window, is_selected);
213644d93782SGreg Clayton                 }
213744d93782SGreg Clayton                 window.MoveCursor(cursor_x, cursor_y);
213844d93782SGreg Clayton                 window.DeferredRefresh();
213944d93782SGreg Clayton             }
214044d93782SGreg Clayton             break;
214144d93782SGreg Clayton 
214244d93782SGreg Clayton         default:
214344d93782SGreg Clayton         case Menu::Type::Separator:
214444d93782SGreg Clayton             break;
214544d93782SGreg Clayton         }
214644d93782SGreg Clayton         return true; // Drawing handled...
214744d93782SGreg Clayton     }
214844d93782SGreg Clayton 
214944d93782SGreg Clayton     HandleCharResult
215044d93782SGreg Clayton     Menu::WindowDelegateHandleChar (Window &window, int key)
215144d93782SGreg Clayton     {
215244d93782SGreg Clayton         HandleCharResult result = eKeyNotHandled;
215344d93782SGreg Clayton 
215444d93782SGreg Clayton         Menus &submenus = GetSubmenus();
215544d93782SGreg Clayton         const size_t num_submenus = submenus.size();
215644d93782SGreg Clayton         const int selected_idx = GetSelectedSubmenuIndex();
215744d93782SGreg Clayton         Menu::Type menu_type = GetType ();
215844d93782SGreg Clayton         if (menu_type == Menu::Type::Bar)
215944d93782SGreg Clayton         {
216044d93782SGreg Clayton             MenuSP run_menu_sp;
216144d93782SGreg Clayton             switch (key)
216244d93782SGreg Clayton             {
216344d93782SGreg Clayton                 case KEY_DOWN:
216444d93782SGreg Clayton                 case KEY_UP:
216544d93782SGreg Clayton                     // Show last menu or first menu
21663985c8c6SSaleem Abdulrasool                     if (selected_idx < static_cast<int>(num_submenus))
216744d93782SGreg Clayton                         run_menu_sp = submenus[selected_idx];
216844d93782SGreg Clayton                     else if (!submenus.empty())
216944d93782SGreg Clayton                         run_menu_sp = submenus.front();
217044d93782SGreg Clayton                     result = eKeyHandled;
217144d93782SGreg Clayton                     break;
217244d93782SGreg Clayton 
217344d93782SGreg Clayton                 case KEY_RIGHT:
217444d93782SGreg Clayton                 {
217544d93782SGreg Clayton                     ++m_selected;
21763985c8c6SSaleem Abdulrasool                     if (m_selected >= static_cast<int>(num_submenus))
217744d93782SGreg Clayton                         m_selected = 0;
21783985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
217944d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
218044d93782SGreg Clayton                     else if (!submenus.empty())
218144d93782SGreg Clayton                         run_menu_sp = submenus.front();
218244d93782SGreg Clayton                     result = eKeyHandled;
218344d93782SGreg Clayton                 }
218444d93782SGreg Clayton                     break;
218544d93782SGreg Clayton 
218644d93782SGreg Clayton                 case KEY_LEFT:
218744d93782SGreg Clayton                 {
218844d93782SGreg Clayton                     --m_selected;
218944d93782SGreg Clayton                     if (m_selected < 0)
219044d93782SGreg Clayton                         m_selected = num_submenus - 1;
21913985c8c6SSaleem Abdulrasool                     if (m_selected < static_cast<int>(num_submenus))
219244d93782SGreg Clayton                         run_menu_sp = submenus[m_selected];
219344d93782SGreg Clayton                     else if (!submenus.empty())
219444d93782SGreg Clayton                         run_menu_sp = submenus.front();
219544d93782SGreg Clayton                     result = eKeyHandled;
219644d93782SGreg Clayton                 }
219744d93782SGreg Clayton                     break;
219844d93782SGreg Clayton 
219944d93782SGreg Clayton                 default:
220044d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
220144d93782SGreg Clayton                     {
220244d93782SGreg Clayton                         if (submenus[i]->GetKeyValue() == key)
220344d93782SGreg Clayton                         {
220444d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
220544d93782SGreg Clayton                             run_menu_sp = submenus[i];
220644d93782SGreg Clayton                             result = eKeyHandled;
220744d93782SGreg Clayton                             break;
220844d93782SGreg Clayton                         }
220944d93782SGreg Clayton                     }
221044d93782SGreg Clayton                     break;
221144d93782SGreg Clayton             }
221244d93782SGreg Clayton 
221344d93782SGreg Clayton             if (run_menu_sp)
221444d93782SGreg Clayton             {
221544d93782SGreg Clayton                 // Run the action on this menu in case we need to populate the
221644d93782SGreg Clayton                 // menu with dynamic content and also in case check marks, and
221785025451SKamil Rytarowski                 // any other menu decorations need to be calculated
221844d93782SGreg Clayton                 if (run_menu_sp->Action() == MenuActionResult::Quit)
221944d93782SGreg Clayton                     return eQuitApplication;
222044d93782SGreg Clayton 
222144d93782SGreg Clayton                 Rect menu_bounds;
222244d93782SGreg Clayton                 menu_bounds.origin.x = run_menu_sp->GetStartingColumn();
222344d93782SGreg Clayton                 menu_bounds.origin.y = 1;
222444d93782SGreg Clayton                 menu_bounds.size.width = run_menu_sp->GetDrawWidth();
222544d93782SGreg Clayton                 menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2;
222644d93782SGreg Clayton                 if (m_menu_window_sp)
222744d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(m_menu_window_sp.get());
222844d93782SGreg Clayton 
222944d93782SGreg Clayton                 m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(),
223044d93782SGreg Clayton                                                                         menu_bounds,
223144d93782SGreg Clayton                                                                         true);
223244d93782SGreg Clayton                 m_menu_window_sp->SetDelegate (run_menu_sp);
223344d93782SGreg Clayton             }
223444d93782SGreg Clayton         }
223544d93782SGreg Clayton         else if (menu_type == Menu::Type::Item)
223644d93782SGreg Clayton         {
223744d93782SGreg Clayton             switch (key)
223844d93782SGreg Clayton             {
223944d93782SGreg Clayton                 case KEY_DOWN:
224044d93782SGreg Clayton                     if (m_submenus.size() > 1)
224144d93782SGreg Clayton                     {
224244d93782SGreg Clayton                         const int start_select = m_selected;
224344d93782SGreg Clayton                         while (++m_selected != start_select)
224444d93782SGreg Clayton                         {
22453985c8c6SSaleem Abdulrasool                             if (static_cast<size_t>(m_selected) >= num_submenus)
224644d93782SGreg Clayton                                 m_selected = 0;
224744d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
224844d93782SGreg Clayton                                 continue;
224944d93782SGreg Clayton                             else
225044d93782SGreg Clayton                                 break;
225144d93782SGreg Clayton                         }
225244d93782SGreg Clayton                         return eKeyHandled;
225344d93782SGreg Clayton                     }
225444d93782SGreg Clayton                     break;
225544d93782SGreg Clayton 
225644d93782SGreg Clayton                 case KEY_UP:
225744d93782SGreg Clayton                     if (m_submenus.size() > 1)
225844d93782SGreg Clayton                     {
225944d93782SGreg Clayton                         const int start_select = m_selected;
226044d93782SGreg Clayton                         while (--m_selected != start_select)
226144d93782SGreg Clayton                         {
22623985c8c6SSaleem Abdulrasool                             if (m_selected < static_cast<int>(0))
226344d93782SGreg Clayton                                 m_selected = num_submenus - 1;
226444d93782SGreg Clayton                             if (m_submenus[m_selected]->GetType() == Type::Separator)
226544d93782SGreg Clayton                                 continue;
226644d93782SGreg Clayton                             else
226744d93782SGreg Clayton                                 break;
226844d93782SGreg Clayton                         }
226944d93782SGreg Clayton                         return eKeyHandled;
227044d93782SGreg Clayton                     }
227144d93782SGreg Clayton                     break;
227244d93782SGreg Clayton 
227344d93782SGreg Clayton                 case KEY_RETURN:
22743985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(selected_idx) < num_submenus)
227544d93782SGreg Clayton                     {
227644d93782SGreg Clayton                         if (submenus[selected_idx]->Action() == MenuActionResult::Quit)
227744d93782SGreg Clayton                             return eQuitApplication;
227844d93782SGreg Clayton                         window.GetParent()->RemoveSubWindow(&window);
227944d93782SGreg Clayton                         return eKeyHandled;
228044d93782SGreg Clayton                     }
228144d93782SGreg Clayton                     break;
228244d93782SGreg Clayton 
228344d93782SGreg Clayton                 case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences
228444d93782SGreg Clayton                     window.GetParent()->RemoveSubWindow(&window);
228544d93782SGreg Clayton                     return eKeyHandled;
228644d93782SGreg Clayton 
228744d93782SGreg Clayton                 default:
228844d93782SGreg Clayton                 {
228944d93782SGreg Clayton                     for (size_t i=0; i<num_submenus; ++i)
229044d93782SGreg Clayton                     {
229144d93782SGreg Clayton                         Menu *menu = submenus[i].get();
229244d93782SGreg Clayton                         if (menu->GetKeyValue() == key)
229344d93782SGreg Clayton                         {
229444d93782SGreg Clayton                             SetSelectedSubmenuIndex(i);
229544d93782SGreg Clayton                             window.GetParent()->RemoveSubWindow(&window);
229644d93782SGreg Clayton                             if (menu->Action() == MenuActionResult::Quit)
229744d93782SGreg Clayton                                 return eQuitApplication;
229844d93782SGreg Clayton                             return eKeyHandled;
229944d93782SGreg Clayton                         }
230044d93782SGreg Clayton                     }
230144d93782SGreg Clayton                 }
230244d93782SGreg Clayton                     break;
230344d93782SGreg Clayton 
230444d93782SGreg Clayton             }
230544d93782SGreg Clayton         }
230644d93782SGreg Clayton         else if (menu_type == Menu::Type::Separator)
230744d93782SGreg Clayton         {
230844d93782SGreg Clayton         }
230944d93782SGreg Clayton         return result;
231044d93782SGreg Clayton     }
231144d93782SGreg Clayton 
231244d93782SGreg Clayton     class Application
231344d93782SGreg Clayton     {
231444d93782SGreg Clayton     public:
231544d93782SGreg Clayton         Application (FILE *in, FILE *out) :
231644d93782SGreg Clayton             m_window_sp(),
231744d93782SGreg Clayton             m_screen (NULL),
231844d93782SGreg Clayton             m_in (in),
231944d93782SGreg Clayton             m_out (out)
232044d93782SGreg Clayton         {
232144d93782SGreg Clayton 
232244d93782SGreg Clayton         }
232344d93782SGreg Clayton 
232444d93782SGreg Clayton         ~Application ()
232544d93782SGreg Clayton         {
232644d93782SGreg Clayton             m_window_delegates.clear();
232744d93782SGreg Clayton             m_window_sp.reset();
232844d93782SGreg Clayton             if (m_screen)
232944d93782SGreg Clayton             {
233044d93782SGreg Clayton                 ::delscreen(m_screen);
233144d93782SGreg Clayton                 m_screen = NULL;
233244d93782SGreg Clayton             }
233344d93782SGreg Clayton         }
233444d93782SGreg Clayton 
233544d93782SGreg Clayton         void
233644d93782SGreg Clayton         Initialize ()
233744d93782SGreg Clayton         {
233844d93782SGreg Clayton             ::setlocale(LC_ALL, "");
233944d93782SGreg Clayton             ::setlocale(LC_CTYPE, "");
234044d93782SGreg Clayton #if 0
234144d93782SGreg Clayton             ::initscr();
234244d93782SGreg Clayton #else
234344d93782SGreg Clayton             m_screen = ::newterm(NULL, m_out, m_in);
234444d93782SGreg Clayton #endif
234544d93782SGreg Clayton             ::start_color();
234644d93782SGreg Clayton             ::curs_set(0);
234744d93782SGreg Clayton             ::noecho();
234844d93782SGreg Clayton             ::keypad(stdscr,TRUE);
234944d93782SGreg Clayton         }
235044d93782SGreg Clayton 
235144d93782SGreg Clayton         void
235244d93782SGreg Clayton         Terminate ()
235344d93782SGreg Clayton         {
235444d93782SGreg Clayton             ::endwin();
235544d93782SGreg Clayton         }
235644d93782SGreg Clayton 
235744d93782SGreg Clayton         void
235844d93782SGreg Clayton         Run (Debugger &debugger)
235944d93782SGreg Clayton         {
236044d93782SGreg Clayton             bool done = false;
236144d93782SGreg Clayton             int delay_in_tenths_of_a_second = 1;
236244d93782SGreg Clayton 
236344d93782SGreg Clayton             // Alas the threading model in curses is a bit lame so we need to
236444d93782SGreg Clayton             // resort to polling every 0.5 seconds. We could poll for stdin
236544d93782SGreg Clayton             // ourselves and then pass the keys down but then we need to
236644d93782SGreg Clayton             // translate all of the escape sequences ourselves. So we resort to
236744d93782SGreg Clayton             // polling for input because we need to receive async process events
236844d93782SGreg Clayton             // while in this loop.
236944d93782SGreg Clayton 
237044d93782SGreg Clayton             halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar()
237144d93782SGreg Clayton 
2372*583bbb1dSJim Ingham             ListenerSP listener_sp (Listener::MakeListener("lldb.IOHandler.curses.Application"));
237344d93782SGreg Clayton             ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass());
237444d93782SGreg Clayton             ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass());
237544d93782SGreg Clayton             ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass());
237644d93782SGreg Clayton             debugger.EnableForwardEvents (listener_sp);
237744d93782SGreg Clayton 
237844d93782SGreg Clayton             bool update = true;
237944d93782SGreg Clayton #if defined(__APPLE__)
238044d93782SGreg Clayton             std::deque<int> escape_chars;
238144d93782SGreg Clayton #endif
238244d93782SGreg Clayton 
238344d93782SGreg Clayton             while (!done)
238444d93782SGreg Clayton             {
238544d93782SGreg Clayton                 if (update)
238644d93782SGreg Clayton                 {
238744d93782SGreg Clayton                     m_window_sp->Draw(false);
238844d93782SGreg Clayton                     // All windows should be calling Window::DeferredRefresh() instead
238944d93782SGreg Clayton                     // of Window::Refresh() so we can do a single update and avoid
239044d93782SGreg Clayton                     // any screen blinking
239144d93782SGreg Clayton                     update_panels();
239244d93782SGreg Clayton 
239344d93782SGreg Clayton                     // Cursor hiding isn't working on MacOSX, so hide it in the top left corner
239444d93782SGreg Clayton                     m_window_sp->MoveCursor(0, 0);
239544d93782SGreg Clayton 
239644d93782SGreg Clayton                     doupdate();
239744d93782SGreg Clayton                     update = false;
239844d93782SGreg Clayton                 }
239944d93782SGreg Clayton 
240044d93782SGreg Clayton #if defined(__APPLE__)
240144d93782SGreg Clayton                 // Terminal.app doesn't map its function keys correctly, F1-F4 default to:
240244d93782SGreg Clayton                 // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible
240344d93782SGreg Clayton                 int ch;
240444d93782SGreg Clayton                 if (escape_chars.empty())
240544d93782SGreg Clayton                     ch = m_window_sp->GetChar();
240644d93782SGreg Clayton                 else
240744d93782SGreg Clayton                 {
240844d93782SGreg Clayton                     ch = escape_chars.front();
240944d93782SGreg Clayton                     escape_chars.pop_front();
241044d93782SGreg Clayton                 }
241144d93782SGreg Clayton                 if (ch == KEY_ESCAPE)
241244d93782SGreg Clayton                 {
241344d93782SGreg Clayton                     int ch2 = m_window_sp->GetChar();
241444d93782SGreg Clayton                     if (ch2 == 'O')
241544d93782SGreg Clayton                     {
241644d93782SGreg Clayton                         int ch3 = m_window_sp->GetChar();
241744d93782SGreg Clayton                         switch (ch3)
241844d93782SGreg Clayton                         {
241944d93782SGreg Clayton                             case 'P': ch = KEY_F(1); break;
242044d93782SGreg Clayton                             case 'Q': ch = KEY_F(2); break;
242144d93782SGreg Clayton                             case 'R': ch = KEY_F(3); break;
242244d93782SGreg Clayton                             case 'S': ch = KEY_F(4); break;
242344d93782SGreg Clayton                             default:
242444d93782SGreg Clayton                                 escape_chars.push_back(ch2);
242544d93782SGreg Clayton                                 if (ch3 != -1)
242644d93782SGreg Clayton                                     escape_chars.push_back(ch3);
242744d93782SGreg Clayton                                 break;
242844d93782SGreg Clayton                         }
242944d93782SGreg Clayton                     }
243044d93782SGreg Clayton                     else if (ch2 != -1)
243144d93782SGreg Clayton                         escape_chars.push_back(ch2);
243244d93782SGreg Clayton                 }
243344d93782SGreg Clayton #else
243444d93782SGreg Clayton                 int ch = m_window_sp->GetChar();
243544d93782SGreg Clayton 
243644d93782SGreg Clayton #endif
243744d93782SGreg Clayton                 if (ch == -1)
243844d93782SGreg Clayton                 {
243944d93782SGreg Clayton                     if (feof(m_in) || ferror(m_in))
244044d93782SGreg Clayton                     {
244144d93782SGreg Clayton                         done = true;
244244d93782SGreg Clayton                     }
244344d93782SGreg Clayton                     else
244444d93782SGreg Clayton                     {
244544d93782SGreg Clayton                         // Just a timeout from using halfdelay(), check for events
244644d93782SGreg Clayton                         EventSP event_sp;
244744d93782SGreg Clayton                         while (listener_sp->PeekAtNextEvent())
244844d93782SGreg Clayton                         {
244944d93782SGreg Clayton                             listener_sp->GetNextEvent(event_sp);
245044d93782SGreg Clayton 
245144d93782SGreg Clayton                             if (event_sp)
245244d93782SGreg Clayton                             {
245344d93782SGreg Clayton                                 Broadcaster *broadcaster = event_sp->GetBroadcaster();
245444d93782SGreg Clayton                                 if (broadcaster)
245544d93782SGreg Clayton                                 {
245644d93782SGreg Clayton                                     //uint32_t event_type = event_sp->GetType();
245744d93782SGreg Clayton                                     ConstString broadcaster_class (broadcaster->GetBroadcasterClass());
245844d93782SGreg Clayton                                     if (broadcaster_class == broadcaster_class_process)
245944d93782SGreg Clayton                                     {
2460ec990867SGreg Clayton                                         debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
246144d93782SGreg Clayton                                         update = true;
246244d93782SGreg Clayton                                         continue; // Don't get any key, just update our view
246344d93782SGreg Clayton                                     }
246444d93782SGreg Clayton                                 }
246544d93782SGreg Clayton                             }
246644d93782SGreg Clayton                         }
246744d93782SGreg Clayton                     }
246844d93782SGreg Clayton                 }
246944d93782SGreg Clayton                 else
247044d93782SGreg Clayton                 {
247144d93782SGreg Clayton                     HandleCharResult key_result = m_window_sp->HandleChar(ch);
247244d93782SGreg Clayton                     switch (key_result)
247344d93782SGreg Clayton                     {
247444d93782SGreg Clayton                         case eKeyHandled:
2475ec990867SGreg Clayton                             debugger.GetCommandInterpreter().UpdateExecutionContext(NULL);
247644d93782SGreg Clayton                             update = true;
247744d93782SGreg Clayton                             break;
247844d93782SGreg Clayton                         case eKeyNotHandled:
247944d93782SGreg Clayton                             break;
248044d93782SGreg Clayton                         case eQuitApplication:
248144d93782SGreg Clayton                             done = true;
248244d93782SGreg Clayton                             break;
248344d93782SGreg Clayton                     }
248444d93782SGreg Clayton                 }
248544d93782SGreg Clayton             }
248644d93782SGreg Clayton 
248744d93782SGreg Clayton             debugger.CancelForwardEvents (listener_sp);
248844d93782SGreg Clayton         }
248944d93782SGreg Clayton 
249044d93782SGreg Clayton         WindowSP &
249144d93782SGreg Clayton         GetMainWindow ()
249244d93782SGreg Clayton         {
249344d93782SGreg Clayton             if (!m_window_sp)
249444d93782SGreg Clayton                 m_window_sp.reset (new Window ("main", stdscr, false));
249544d93782SGreg Clayton             return m_window_sp;
249644d93782SGreg Clayton         }
249744d93782SGreg Clayton 
249844d93782SGreg Clayton         WindowDelegates &
249944d93782SGreg Clayton         GetWindowDelegates ()
250044d93782SGreg Clayton         {
250144d93782SGreg Clayton             return m_window_delegates;
250244d93782SGreg Clayton         }
250344d93782SGreg Clayton 
250444d93782SGreg Clayton     protected:
250544d93782SGreg Clayton         WindowSP m_window_sp;
250644d93782SGreg Clayton         WindowDelegates m_window_delegates;
250744d93782SGreg Clayton         SCREEN *m_screen;
250844d93782SGreg Clayton         FILE *m_in;
250944d93782SGreg Clayton         FILE *m_out;
251044d93782SGreg Clayton     };
251144d93782SGreg Clayton 
251244d93782SGreg Clayton } // namespace curses
251344d93782SGreg Clayton 
251444d93782SGreg Clayton using namespace curses;
251544d93782SGreg Clayton 
251644d93782SGreg Clayton struct Row
251744d93782SGreg Clayton {
251844d93782SGreg Clayton     ValueObjectSP valobj;
251944d93782SGreg Clayton     Row *parent;
252044d93782SGreg Clayton     int row_idx;
252144d93782SGreg Clayton     int x;
252244d93782SGreg Clayton     int y;
252344d93782SGreg Clayton     bool might_have_children;
252444d93782SGreg Clayton     bool expanded;
252544d93782SGreg Clayton     bool calculated_children;
252644d93782SGreg Clayton     std::vector<Row> children;
252744d93782SGreg Clayton 
252844d93782SGreg Clayton     Row (const ValueObjectSP &v, Row *p) :
252944d93782SGreg Clayton     valobj (v),
253044d93782SGreg Clayton     parent (p),
253144d93782SGreg Clayton     row_idx(0),
253244d93782SGreg Clayton     x(1),
253344d93782SGreg Clayton     y(1),
253444d93782SGreg Clayton     might_have_children (v ? v->MightHaveChildren() : false),
253544d93782SGreg Clayton     expanded (false),
253644d93782SGreg Clayton     calculated_children (false),
253744d93782SGreg Clayton     children()
253844d93782SGreg Clayton     {
253944d93782SGreg Clayton     }
254044d93782SGreg Clayton 
254144d93782SGreg Clayton     size_t
254244d93782SGreg Clayton     GetDepth () const
254344d93782SGreg Clayton     {
254444d93782SGreg Clayton         if (parent)
254544d93782SGreg Clayton             return 1 + parent->GetDepth();
254644d93782SGreg Clayton         return 0;
254744d93782SGreg Clayton     }
254844d93782SGreg Clayton 
254944d93782SGreg Clayton     void
255044d93782SGreg Clayton     Expand()
255144d93782SGreg Clayton     {
255244d93782SGreg Clayton         expanded = true;
255344d93782SGreg Clayton         if (!calculated_children)
255444d93782SGreg Clayton         {
255544d93782SGreg Clayton             calculated_children = true;
255644d93782SGreg Clayton             if (valobj)
255744d93782SGreg Clayton             {
255844d93782SGreg Clayton                 const size_t num_children = valobj->GetNumChildren();
255944d93782SGreg Clayton                 for (size_t i=0; i<num_children; ++i)
256044d93782SGreg Clayton                 {
256144d93782SGreg Clayton                     children.push_back(Row (valobj->GetChildAtIndex(i, true), this));
256244d93782SGreg Clayton                 }
256344d93782SGreg Clayton             }
256444d93782SGreg Clayton         }
256544d93782SGreg Clayton     }
256644d93782SGreg Clayton 
256744d93782SGreg Clayton     void
256844d93782SGreg Clayton     Unexpand ()
256944d93782SGreg Clayton     {
257044d93782SGreg Clayton         expanded = false;
257144d93782SGreg Clayton     }
257244d93782SGreg Clayton 
257344d93782SGreg Clayton     void
257444d93782SGreg Clayton     DrawTree (Window &window)
257544d93782SGreg Clayton     {
257644d93782SGreg Clayton         if (parent)
257744d93782SGreg Clayton             parent->DrawTreeForChild (window, this, 0);
257844d93782SGreg Clayton 
257944d93782SGreg Clayton         if (might_have_children)
258044d93782SGreg Clayton         {
258144d93782SGreg Clayton             // It we can get UTF8 characters to work we should try to use the "symbol"
258244d93782SGreg Clayton             // UTF8 string below
258344d93782SGreg Clayton //            const char *symbol = "";
258444d93782SGreg Clayton //            if (row.expanded)
258544d93782SGreg Clayton //                symbol = "\xe2\x96\xbd ";
258644d93782SGreg Clayton //            else
258744d93782SGreg Clayton //                symbol = "\xe2\x96\xb7 ";
258844d93782SGreg Clayton //            window.PutCString (symbol);
258944d93782SGreg Clayton 
259044d93782SGreg Clayton             // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
259144d93782SGreg Clayton             // 'v' or '>' character...
259244d93782SGreg Clayton //            if (expanded)
259344d93782SGreg Clayton //                window.PutChar (ACS_DARROW);
259444d93782SGreg Clayton //            else
259544d93782SGreg Clayton //                window.PutChar (ACS_RARROW);
259644d93782SGreg Clayton             // Since we can't find any good looking right arrow/down arrow
259744d93782SGreg Clayton             // symbols, just use a diamond...
259844d93782SGreg Clayton             window.PutChar (ACS_DIAMOND);
259944d93782SGreg Clayton             window.PutChar (ACS_HLINE);
260044d93782SGreg Clayton         }
260144d93782SGreg Clayton     }
260244d93782SGreg Clayton 
260344d93782SGreg Clayton     void
260444d93782SGreg Clayton     DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth)
260544d93782SGreg Clayton     {
260644d93782SGreg Clayton         if (parent)
260744d93782SGreg Clayton             parent->DrawTreeForChild (window, this, reverse_depth + 1);
260844d93782SGreg Clayton 
260944d93782SGreg Clayton         if (&children.back() == child)
261044d93782SGreg Clayton         {
261144d93782SGreg Clayton             // Last child
261244d93782SGreg Clayton             if (reverse_depth == 0)
261344d93782SGreg Clayton             {
261444d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
261544d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
261644d93782SGreg Clayton             }
261744d93782SGreg Clayton             else
261844d93782SGreg Clayton             {
261944d93782SGreg Clayton                 window.PutChar (' ');
262044d93782SGreg Clayton                 window.PutChar (' ');
262144d93782SGreg Clayton             }
262244d93782SGreg Clayton         }
262344d93782SGreg Clayton         else
262444d93782SGreg Clayton         {
262544d93782SGreg Clayton             if (reverse_depth == 0)
262644d93782SGreg Clayton             {
262744d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
262844d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
262944d93782SGreg Clayton             }
263044d93782SGreg Clayton             else
263144d93782SGreg Clayton             {
263244d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
263344d93782SGreg Clayton                 window.PutChar (' ');
263444d93782SGreg Clayton             }
263544d93782SGreg Clayton         }
263644d93782SGreg Clayton     }
263744d93782SGreg Clayton };
263844d93782SGreg Clayton 
263944d93782SGreg Clayton struct DisplayOptions
264044d93782SGreg Clayton {
264144d93782SGreg Clayton     bool show_types;
264244d93782SGreg Clayton };
264344d93782SGreg Clayton 
264444d93782SGreg Clayton class TreeItem;
264544d93782SGreg Clayton 
264644d93782SGreg Clayton class TreeDelegate
264744d93782SGreg Clayton {
264844d93782SGreg Clayton public:
264944d93782SGreg Clayton     TreeDelegate() {}
2650315b6884SEugene Zelenko     virtual ~TreeDelegate() = default;
2651315b6884SEugene Zelenko 
265244d93782SGreg Clayton     virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0;
265344d93782SGreg Clayton     virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0;
265444d93782SGreg Clayton     virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views
265544d93782SGreg Clayton };
2656315b6884SEugene Zelenko 
265744d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP;
265844d93782SGreg Clayton 
265944d93782SGreg Clayton class TreeItem
266044d93782SGreg Clayton {
266144d93782SGreg Clayton public:
266244d93782SGreg Clayton 
266344d93782SGreg Clayton     TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) :
266444d93782SGreg Clayton         m_parent (parent),
266544d93782SGreg Clayton         m_delegate (delegate),
2666ec990867SGreg Clayton         m_user_data (NULL),
266744d93782SGreg Clayton         m_identifier (0),
266844d93782SGreg Clayton         m_row_idx (-1),
266944d93782SGreg Clayton         m_children (),
267044d93782SGreg Clayton         m_might_have_children (might_have_children),
267144d93782SGreg Clayton         m_is_expanded (false)
267244d93782SGreg Clayton     {
267344d93782SGreg Clayton     }
267444d93782SGreg Clayton 
267544d93782SGreg Clayton     TreeItem &
267644d93782SGreg Clayton     operator=(const TreeItem &rhs)
267744d93782SGreg Clayton     {
267844d93782SGreg Clayton         if (this != &rhs)
267944d93782SGreg Clayton         {
268044d93782SGreg Clayton             m_parent = rhs.m_parent;
268144d93782SGreg Clayton             m_delegate = rhs.m_delegate;
2682ec990867SGreg Clayton             m_user_data = rhs.m_user_data;
268344d93782SGreg Clayton             m_identifier = rhs.m_identifier;
268444d93782SGreg Clayton             m_row_idx = rhs.m_row_idx;
268544d93782SGreg Clayton             m_children = rhs.m_children;
268644d93782SGreg Clayton             m_might_have_children = rhs.m_might_have_children;
268744d93782SGreg Clayton             m_is_expanded = rhs.m_is_expanded;
268844d93782SGreg Clayton         }
268944d93782SGreg Clayton         return *this;
269044d93782SGreg Clayton     }
269144d93782SGreg Clayton 
269244d93782SGreg Clayton     size_t
269344d93782SGreg Clayton     GetDepth () const
269444d93782SGreg Clayton     {
269544d93782SGreg Clayton         if (m_parent)
269644d93782SGreg Clayton             return 1 + m_parent->GetDepth();
269744d93782SGreg Clayton         return 0;
269844d93782SGreg Clayton     }
269944d93782SGreg Clayton 
270044d93782SGreg Clayton     int
270144d93782SGreg Clayton     GetRowIndex () const
270244d93782SGreg Clayton     {
270344d93782SGreg Clayton         return m_row_idx;
270444d93782SGreg Clayton     }
270544d93782SGreg Clayton 
270644d93782SGreg Clayton     void
270744d93782SGreg Clayton     ClearChildren ()
270844d93782SGreg Clayton     {
270944d93782SGreg Clayton         m_children.clear();
271044d93782SGreg Clayton     }
271144d93782SGreg Clayton 
271244d93782SGreg Clayton     void
271344d93782SGreg Clayton     Resize (size_t n, const TreeItem &t)
271444d93782SGreg Clayton     {
271544d93782SGreg Clayton         m_children.resize(n, t);
271644d93782SGreg Clayton     }
271744d93782SGreg Clayton 
271844d93782SGreg Clayton     TreeItem &
271944d93782SGreg Clayton     operator [](size_t i)
272044d93782SGreg Clayton     {
272144d93782SGreg Clayton         return m_children[i];
272244d93782SGreg Clayton     }
272344d93782SGreg Clayton 
272444d93782SGreg Clayton     void
272544d93782SGreg Clayton     SetRowIndex (int row_idx)
272644d93782SGreg Clayton     {
272744d93782SGreg Clayton         m_row_idx = row_idx;
272844d93782SGreg Clayton     }
272944d93782SGreg Clayton 
273044d93782SGreg Clayton     size_t
273144d93782SGreg Clayton     GetNumChildren ()
273244d93782SGreg Clayton     {
273344d93782SGreg Clayton         m_delegate.TreeDelegateGenerateChildren (*this);
273444d93782SGreg Clayton         return m_children.size();
273544d93782SGreg Clayton     }
273644d93782SGreg Clayton 
273744d93782SGreg Clayton     void
273844d93782SGreg Clayton     ItemWasSelected ()
273944d93782SGreg Clayton     {
274044d93782SGreg Clayton         m_delegate.TreeDelegateItemSelected(*this);
274144d93782SGreg Clayton     }
2742315b6884SEugene Zelenko 
274344d93782SGreg Clayton     void
274444d93782SGreg Clayton     CalculateRowIndexes (int &row_idx)
274544d93782SGreg Clayton     {
274644d93782SGreg Clayton         SetRowIndex(row_idx);
274744d93782SGreg Clayton         ++row_idx;
274844d93782SGreg Clayton 
2749ec990867SGreg Clayton         const bool expanded = IsExpanded();
2750ec990867SGreg Clayton 
2751ec990867SGreg Clayton         // The root item must calculate its children,
2752ec990867SGreg Clayton         // or we must calculate the number of children
2753ec990867SGreg Clayton         // if the item is expanded
2754ec990867SGreg Clayton         if (m_parent == NULL || expanded)
275544d93782SGreg Clayton             GetNumChildren();
275644d93782SGreg Clayton 
275744d93782SGreg Clayton         for (auto &item : m_children)
275844d93782SGreg Clayton         {
275944d93782SGreg Clayton             if (expanded)
276044d93782SGreg Clayton                 item.CalculateRowIndexes(row_idx);
276144d93782SGreg Clayton             else
276244d93782SGreg Clayton                 item.SetRowIndex(-1);
276344d93782SGreg Clayton         }
276444d93782SGreg Clayton     }
276544d93782SGreg Clayton 
276644d93782SGreg Clayton     TreeItem *
276744d93782SGreg Clayton     GetParent ()
276844d93782SGreg Clayton     {
276944d93782SGreg Clayton         return m_parent;
277044d93782SGreg Clayton     }
277144d93782SGreg Clayton 
277244d93782SGreg Clayton     bool
277344d93782SGreg Clayton     IsExpanded () const
277444d93782SGreg Clayton     {
277544d93782SGreg Clayton         return m_is_expanded;
277644d93782SGreg Clayton     }
277744d93782SGreg Clayton 
277844d93782SGreg Clayton     void
277944d93782SGreg Clayton     Expand()
278044d93782SGreg Clayton     {
278144d93782SGreg Clayton         m_is_expanded = true;
278244d93782SGreg Clayton     }
278344d93782SGreg Clayton 
278444d93782SGreg Clayton     void
278544d93782SGreg Clayton     Unexpand ()
278644d93782SGreg Clayton     {
278744d93782SGreg Clayton         m_is_expanded = false;
278844d93782SGreg Clayton     }
278944d93782SGreg Clayton 
279044d93782SGreg Clayton     bool
279144d93782SGreg Clayton     Draw (Window &window,
279244d93782SGreg Clayton           const int first_visible_row,
279344d93782SGreg Clayton           const uint32_t selected_row_idx,
279444d93782SGreg Clayton           int &row_idx,
279544d93782SGreg Clayton           int &num_rows_left)
279644d93782SGreg Clayton     {
279744d93782SGreg Clayton         if (num_rows_left <= 0)
279844d93782SGreg Clayton             return false;
279944d93782SGreg Clayton 
280044d93782SGreg Clayton         if (m_row_idx >= first_visible_row)
280144d93782SGreg Clayton         {
280244d93782SGreg Clayton             window.MoveCursor(2, row_idx + 1);
280344d93782SGreg Clayton 
280444d93782SGreg Clayton             if (m_parent)
280544d93782SGreg Clayton                 m_parent->DrawTreeForChild (window, this, 0);
280644d93782SGreg Clayton 
280744d93782SGreg Clayton             if (m_might_have_children)
280844d93782SGreg Clayton             {
280944d93782SGreg Clayton                 // It we can get UTF8 characters to work we should try to use the "symbol"
281044d93782SGreg Clayton                 // UTF8 string below
281144d93782SGreg Clayton                 //            const char *symbol = "";
281244d93782SGreg Clayton                 //            if (row.expanded)
281344d93782SGreg Clayton                 //                symbol = "\xe2\x96\xbd ";
281444d93782SGreg Clayton                 //            else
281544d93782SGreg Clayton                 //                symbol = "\xe2\x96\xb7 ";
281644d93782SGreg Clayton                 //            window.PutCString (symbol);
281744d93782SGreg Clayton 
281844d93782SGreg Clayton                 // The ACS_DARROW and ACS_RARROW don't look very nice they are just a
281944d93782SGreg Clayton                 // 'v' or '>' character...
282044d93782SGreg Clayton                 //            if (expanded)
282144d93782SGreg Clayton                 //                window.PutChar (ACS_DARROW);
282244d93782SGreg Clayton                 //            else
282344d93782SGreg Clayton                 //                window.PutChar (ACS_RARROW);
282444d93782SGreg Clayton                 // Since we can't find any good looking right arrow/down arrow
282544d93782SGreg Clayton                 // symbols, just use a diamond...
282644d93782SGreg Clayton                 window.PutChar (ACS_DIAMOND);
282744d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
282844d93782SGreg Clayton             }
28293985c8c6SSaleem Abdulrasool             bool highlight =
28303985c8c6SSaleem Abdulrasool               (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive();
283144d93782SGreg Clayton 
283244d93782SGreg Clayton             if (highlight)
283344d93782SGreg Clayton                 window.AttributeOn(A_REVERSE);
283444d93782SGreg Clayton 
283544d93782SGreg Clayton             m_delegate.TreeDelegateDrawTreeItem(*this, window);
283644d93782SGreg Clayton 
283744d93782SGreg Clayton             if (highlight)
283844d93782SGreg Clayton                 window.AttributeOff(A_REVERSE);
283944d93782SGreg Clayton             ++row_idx;
284044d93782SGreg Clayton             --num_rows_left;
284144d93782SGreg Clayton         }
284244d93782SGreg Clayton 
284344d93782SGreg Clayton         if (num_rows_left <= 0)
284444d93782SGreg Clayton             return false; // We are done drawing...
284544d93782SGreg Clayton 
284644d93782SGreg Clayton         if (IsExpanded())
284744d93782SGreg Clayton         {
284844d93782SGreg Clayton             for (auto &item : m_children)
284944d93782SGreg Clayton             {
285044d93782SGreg Clayton                 // If we displayed all the rows and item.Draw() returns
285144d93782SGreg Clayton                 // false we are done drawing and can exit this for loop
285244d93782SGreg Clayton                 if (item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left) == false)
285344d93782SGreg Clayton                     break;
285444d93782SGreg Clayton             }
285544d93782SGreg Clayton         }
285644d93782SGreg Clayton         return num_rows_left >= 0; // Return true if not done drawing yet
285744d93782SGreg Clayton     }
285844d93782SGreg Clayton 
285944d93782SGreg Clayton     void
286044d93782SGreg Clayton     DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth)
286144d93782SGreg Clayton     {
286244d93782SGreg Clayton         if (m_parent)
286344d93782SGreg Clayton             m_parent->DrawTreeForChild (window, this, reverse_depth + 1);
286444d93782SGreg Clayton 
286544d93782SGreg Clayton         if (&m_children.back() == child)
286644d93782SGreg Clayton         {
286744d93782SGreg Clayton             // Last child
286844d93782SGreg Clayton             if (reverse_depth == 0)
286944d93782SGreg Clayton             {
287044d93782SGreg Clayton                 window.PutChar (ACS_LLCORNER);
287144d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
287244d93782SGreg Clayton             }
287344d93782SGreg Clayton             else
287444d93782SGreg Clayton             {
287544d93782SGreg Clayton                 window.PutChar (' ');
287644d93782SGreg Clayton                 window.PutChar (' ');
287744d93782SGreg Clayton             }
287844d93782SGreg Clayton         }
287944d93782SGreg Clayton         else
288044d93782SGreg Clayton         {
288144d93782SGreg Clayton             if (reverse_depth == 0)
288244d93782SGreg Clayton             {
288344d93782SGreg Clayton                 window.PutChar (ACS_LTEE);
288444d93782SGreg Clayton                 window.PutChar (ACS_HLINE);
288544d93782SGreg Clayton             }
288644d93782SGreg Clayton             else
288744d93782SGreg Clayton             {
288844d93782SGreg Clayton                 window.PutChar (ACS_VLINE);
288944d93782SGreg Clayton                 window.PutChar (' ');
289044d93782SGreg Clayton             }
289144d93782SGreg Clayton         }
289244d93782SGreg Clayton     }
289344d93782SGreg Clayton 
289444d93782SGreg Clayton     TreeItem *
289544d93782SGreg Clayton     GetItemForRowIndex (uint32_t row_idx)
289644d93782SGreg Clayton     {
28973985c8c6SSaleem Abdulrasool         if (static_cast<uint32_t>(m_row_idx) == row_idx)
289844d93782SGreg Clayton             return this;
289944d93782SGreg Clayton         if (m_children.empty())
290044d93782SGreg Clayton             return NULL;
290144d93782SGreg Clayton         if (IsExpanded())
290244d93782SGreg Clayton         {
290344d93782SGreg Clayton             for (auto &item : m_children)
290444d93782SGreg Clayton             {
290544d93782SGreg Clayton                 TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx);
290644d93782SGreg Clayton                 if (selected_item_ptr)
290744d93782SGreg Clayton                     return selected_item_ptr;
290844d93782SGreg Clayton             }
290944d93782SGreg Clayton         }
291044d93782SGreg Clayton         return NULL;
291144d93782SGreg Clayton     }
291244d93782SGreg Clayton 
2913ec990867SGreg Clayton     void *
2914ec990867SGreg Clayton     GetUserData() const
2915ec990867SGreg Clayton     {
2916ec990867SGreg Clayton         return m_user_data;
2917ec990867SGreg Clayton     }
2918ec990867SGreg Clayton 
2919ec990867SGreg Clayton     void
2920ec990867SGreg Clayton     SetUserData (void *user_data)
2921ec990867SGreg Clayton     {
2922ec990867SGreg Clayton         m_user_data = user_data;
2923ec990867SGreg Clayton     }
2924ec990867SGreg Clayton 
292544d93782SGreg Clayton     uint64_t
292644d93782SGreg Clayton     GetIdentifier() const
292744d93782SGreg Clayton     {
292844d93782SGreg Clayton         return m_identifier;
292944d93782SGreg Clayton     }
293044d93782SGreg Clayton 
293144d93782SGreg Clayton     void
293244d93782SGreg Clayton     SetIdentifier (uint64_t identifier)
293344d93782SGreg Clayton     {
293444d93782SGreg Clayton         m_identifier = identifier;
293544d93782SGreg Clayton     }
293644d93782SGreg Clayton 
2937ec990867SGreg Clayton     void
2938ec990867SGreg Clayton     SetMightHaveChildren (bool b)
2939ec990867SGreg Clayton     {
2940ec990867SGreg Clayton         m_might_have_children = b;
2941ec990867SGreg Clayton     }
2942ec990867SGreg Clayton 
294344d93782SGreg Clayton protected:
294444d93782SGreg Clayton     TreeItem *m_parent;
294544d93782SGreg Clayton     TreeDelegate &m_delegate;
2946ec990867SGreg Clayton     void *m_user_data;
294744d93782SGreg Clayton     uint64_t m_identifier;
294844d93782SGreg Clayton     int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item
294944d93782SGreg Clayton     std::vector<TreeItem> m_children;
295044d93782SGreg Clayton     bool m_might_have_children;
295144d93782SGreg Clayton     bool m_is_expanded;
295244d93782SGreg Clayton };
295344d93782SGreg Clayton 
295444d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate
295544d93782SGreg Clayton {
295644d93782SGreg Clayton public:
295744d93782SGreg Clayton     TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) :
295844d93782SGreg Clayton         m_debugger (debugger),
295944d93782SGreg Clayton         m_delegate_sp (delegate_sp),
296044d93782SGreg Clayton         m_root (NULL, *delegate_sp, true),
296144d93782SGreg Clayton         m_selected_item (NULL),
296244d93782SGreg Clayton         m_num_rows (0),
296344d93782SGreg Clayton         m_selected_row_idx (0),
296444d93782SGreg Clayton         m_first_visible_row (0),
296544d93782SGreg Clayton         m_min_x (0),
296644d93782SGreg Clayton         m_min_y (0),
296744d93782SGreg Clayton         m_max_x (0),
296844d93782SGreg Clayton         m_max_y (0)
296944d93782SGreg Clayton     {
297044d93782SGreg Clayton     }
297144d93782SGreg Clayton 
297244d93782SGreg Clayton     int
297344d93782SGreg Clayton     NumVisibleRows () const
297444d93782SGreg Clayton     {
297544d93782SGreg Clayton         return m_max_y - m_min_y;
297644d93782SGreg Clayton     }
297744d93782SGreg Clayton 
2978bd5ae6b4SGreg Clayton     bool
2979bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
298044d93782SGreg Clayton     {
298144d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
298244d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
298344d93782SGreg Clayton 
298444d93782SGreg Clayton         bool display_content = false;
298544d93782SGreg Clayton         if (process)
298644d93782SGreg Clayton         {
298744d93782SGreg Clayton             StateType state = process->GetState();
298844d93782SGreg Clayton             if (StateIsStoppedState(state, true))
298944d93782SGreg Clayton             {
299044d93782SGreg Clayton                 // We are stopped, so it is ok to
299144d93782SGreg Clayton                 display_content = true;
299244d93782SGreg Clayton             }
299344d93782SGreg Clayton             else if (StateIsRunningState(state))
299444d93782SGreg Clayton             {
299544d93782SGreg Clayton                 return true; // Don't do any updating when we are running
299644d93782SGreg Clayton             }
299744d93782SGreg Clayton         }
299844d93782SGreg Clayton 
299944d93782SGreg Clayton         m_min_x = 2;
300044d93782SGreg Clayton         m_min_y = 1;
300144d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
300244d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
300344d93782SGreg Clayton 
300444d93782SGreg Clayton         window.Erase();
300544d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
300644d93782SGreg Clayton 
300744d93782SGreg Clayton         if (display_content)
300844d93782SGreg Clayton         {
300944d93782SGreg Clayton             const int num_visible_rows = NumVisibleRows();
301044d93782SGreg Clayton             m_num_rows = 0;
301144d93782SGreg Clayton             m_root.CalculateRowIndexes(m_num_rows);
301244d93782SGreg Clayton 
301344d93782SGreg Clayton             // If we unexpanded while having something selected our
301444d93782SGreg Clayton             // total number of rows is less than the num visible rows,
301544d93782SGreg Clayton             // then make sure we show all the rows by setting the first
301644d93782SGreg Clayton             // visible row accordingly.
301744d93782SGreg Clayton             if (m_first_visible_row > 0 && m_num_rows < num_visible_rows)
301844d93782SGreg Clayton                 m_first_visible_row = 0;
301944d93782SGreg Clayton 
302044d93782SGreg Clayton             // Make sure the selected row is always visible
302144d93782SGreg Clayton             if (m_selected_row_idx < m_first_visible_row)
302244d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx;
302344d93782SGreg Clayton             else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
302444d93782SGreg Clayton                 m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
302544d93782SGreg Clayton 
302644d93782SGreg Clayton             int row_idx = 0;
302744d93782SGreg Clayton             int num_rows_left = num_visible_rows;
302844d93782SGreg Clayton             m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left);
302944d93782SGreg Clayton             // Get the selected row
303044d93782SGreg Clayton             m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx);
303144d93782SGreg Clayton         }
303244d93782SGreg Clayton         else
303344d93782SGreg Clayton         {
303444d93782SGreg Clayton             m_selected_item = NULL;
303544d93782SGreg Clayton         }
303644d93782SGreg Clayton 
303744d93782SGreg Clayton         window.DeferredRefresh();
303844d93782SGreg Clayton 
303944d93782SGreg Clayton 
304044d93782SGreg Clayton         return true; // Drawing handled
304144d93782SGreg Clayton     }
304244d93782SGreg Clayton 
3043bd5ae6b4SGreg Clayton     const char *
3044bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
304544d93782SGreg Clayton     {
304644d93782SGreg Clayton         return "Thread window keyboard shortcuts:";
304744d93782SGreg Clayton     }
304844d93782SGreg Clayton 
3049bd5ae6b4SGreg Clayton     KeyHelp *
3050bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
305144d93782SGreg Clayton     {
305244d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
305344d93782SGreg Clayton             { KEY_UP, "Select previous item" },
305444d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
305544d93782SGreg Clayton             { KEY_RIGHT, "Expand the selected item" },
305644d93782SGreg Clayton             { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" },
305744d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
305844d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
305944d93782SGreg Clayton             { 'h', "Show help dialog" },
306044d93782SGreg Clayton             { ' ', "Toggle item expansion" },
306144d93782SGreg Clayton             { ',', "Page up" },
306244d93782SGreg Clayton             { '.', "Page down" },
306344d93782SGreg Clayton             { '\0', NULL }
306444d93782SGreg Clayton         };
306544d93782SGreg Clayton         return g_source_view_key_help;
306644d93782SGreg Clayton     }
306744d93782SGreg Clayton 
3068bd5ae6b4SGreg Clayton     HandleCharResult
3069bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
307044d93782SGreg Clayton     {
307144d93782SGreg Clayton         switch(c)
307244d93782SGreg Clayton         {
307344d93782SGreg Clayton             case ',':
307444d93782SGreg Clayton             case KEY_PPAGE:
307544d93782SGreg Clayton                 // Page up key
307644d93782SGreg Clayton                 if (m_first_visible_row > 0)
307744d93782SGreg Clayton                 {
307844d93782SGreg Clayton                     if (m_first_visible_row > m_max_y)
307944d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
308044d93782SGreg Clayton                     else
308144d93782SGreg Clayton                         m_first_visible_row = 0;
308244d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
308344d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
308444d93782SGreg Clayton                     if (m_selected_item)
308544d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
308644d93782SGreg Clayton                 }
308744d93782SGreg Clayton                 return eKeyHandled;
308844d93782SGreg Clayton 
308944d93782SGreg Clayton             case '.':
309044d93782SGreg Clayton             case KEY_NPAGE:
309144d93782SGreg Clayton                 // Page down key
309244d93782SGreg Clayton                 if (m_num_rows > m_max_y)
309344d93782SGreg Clayton                 {
309444d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
309544d93782SGreg Clayton                     {
309644d93782SGreg Clayton                         m_first_visible_row += m_max_y;
309744d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
309844d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
309944d93782SGreg Clayton                         if (m_selected_item)
310044d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
310144d93782SGreg Clayton                     }
310244d93782SGreg Clayton                 }
310344d93782SGreg Clayton                 return eKeyHandled;
310444d93782SGreg Clayton 
310544d93782SGreg Clayton             case KEY_UP:
310644d93782SGreg Clayton                 if (m_selected_row_idx > 0)
310744d93782SGreg Clayton                 {
310844d93782SGreg Clayton                     --m_selected_row_idx;
310944d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
311044d93782SGreg Clayton                     if (m_selected_item)
311144d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
311244d93782SGreg Clayton                 }
311344d93782SGreg Clayton                 return eKeyHandled;
3114315b6884SEugene Zelenko 
311544d93782SGreg Clayton             case KEY_DOWN:
311644d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
311744d93782SGreg Clayton                 {
311844d93782SGreg Clayton                     ++m_selected_row_idx;
311944d93782SGreg Clayton                     m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
312044d93782SGreg Clayton                     if (m_selected_item)
312144d93782SGreg Clayton                         m_selected_item->ItemWasSelected ();
312244d93782SGreg Clayton                 }
312344d93782SGreg Clayton                 return eKeyHandled;
312444d93782SGreg Clayton 
312544d93782SGreg Clayton             case KEY_RIGHT:
312644d93782SGreg Clayton                 if (m_selected_item)
312744d93782SGreg Clayton                 {
312844d93782SGreg Clayton                     if (!m_selected_item->IsExpanded())
312944d93782SGreg Clayton                         m_selected_item->Expand();
313044d93782SGreg Clayton                 }
313144d93782SGreg Clayton                 return eKeyHandled;
313244d93782SGreg Clayton 
313344d93782SGreg Clayton             case KEY_LEFT:
313444d93782SGreg Clayton                 if (m_selected_item)
313544d93782SGreg Clayton                 {
313644d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
313744d93782SGreg Clayton                         m_selected_item->Unexpand();
313844d93782SGreg Clayton                     else if (m_selected_item->GetParent())
313944d93782SGreg Clayton                     {
314044d93782SGreg Clayton                         m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex();
314144d93782SGreg Clayton                         m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx);
314244d93782SGreg Clayton                         if (m_selected_item)
314344d93782SGreg Clayton                             m_selected_item->ItemWasSelected ();
314444d93782SGreg Clayton                     }
314544d93782SGreg Clayton                 }
314644d93782SGreg Clayton                 return eKeyHandled;
314744d93782SGreg Clayton 
314844d93782SGreg Clayton             case ' ':
314944d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
315044d93782SGreg Clayton                 if (m_selected_item)
315144d93782SGreg Clayton                 {
315244d93782SGreg Clayton                     if (m_selected_item->IsExpanded())
315344d93782SGreg Clayton                         m_selected_item->Unexpand();
315444d93782SGreg Clayton                     else
315544d93782SGreg Clayton                         m_selected_item->Expand();
315644d93782SGreg Clayton                 }
315744d93782SGreg Clayton                 return eKeyHandled;
315844d93782SGreg Clayton 
315944d93782SGreg Clayton             case 'h':
316044d93782SGreg Clayton                 window.CreateHelpSubwindow ();
316144d93782SGreg Clayton                 return eKeyHandled;
316244d93782SGreg Clayton 
316344d93782SGreg Clayton             default:
316444d93782SGreg Clayton                 break;
316544d93782SGreg Clayton         }
316644d93782SGreg Clayton         return eKeyNotHandled;
316744d93782SGreg Clayton     }
316844d93782SGreg Clayton 
316944d93782SGreg Clayton protected:
317044d93782SGreg Clayton     Debugger &m_debugger;
317144d93782SGreg Clayton     TreeDelegateSP m_delegate_sp;
317244d93782SGreg Clayton     TreeItem m_root;
317344d93782SGreg Clayton     TreeItem *m_selected_item;
317444d93782SGreg Clayton     int m_num_rows;
317544d93782SGreg Clayton     int m_selected_row_idx;
317644d93782SGreg Clayton     int m_first_visible_row;
317744d93782SGreg Clayton     int m_min_x;
317844d93782SGreg Clayton     int m_min_y;
317944d93782SGreg Clayton     int m_max_x;
318044d93782SGreg Clayton     int m_max_y;
318144d93782SGreg Clayton };
318244d93782SGreg Clayton 
318344d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate
318444d93782SGreg Clayton {
318544d93782SGreg Clayton public:
3186ec990867SGreg Clayton     FrameTreeDelegate () :
3187ec990867SGreg Clayton         TreeDelegate()
318844d93782SGreg Clayton     {
3189554f68d3SGreg Clayton         FormatEntity::Parse ("frame #${frame.index}: {${function.name}${function.pc-offset}}}",
3190554f68d3SGreg Clayton                              m_format);
319144d93782SGreg Clayton     }
319244d93782SGreg Clayton 
3193315b6884SEugene Zelenko     ~FrameTreeDelegate() override = default;
319444d93782SGreg Clayton 
3195bd5ae6b4SGreg Clayton     void
3196bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
319744d93782SGreg Clayton     {
3198ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3199ec990867SGreg Clayton         if (thread)
320044d93782SGreg Clayton         {
320144d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3202ec990867SGreg Clayton             StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx);
320344d93782SGreg Clayton             if (frame_sp)
320444d93782SGreg Clayton             {
320544d93782SGreg Clayton                 StreamString strm;
320644d93782SGreg Clayton                 const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
320744d93782SGreg Clayton                 ExecutionContext exe_ctx (frame_sp);
3208554f68d3SGreg Clayton                 if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, NULL, NULL, false, false))
320944d93782SGreg Clayton                 {
321044d93782SGreg Clayton                     int right_pad = 1;
321144d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
321244d93782SGreg Clayton                 }
321344d93782SGreg Clayton             }
321444d93782SGreg Clayton         }
321544d93782SGreg Clayton     }
3216315b6884SEugene Zelenko 
3217bd5ae6b4SGreg Clayton     void
3218bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item)  override
321944d93782SGreg Clayton     {
322044d93782SGreg Clayton         // No children for frames yet...
322144d93782SGreg Clayton     }
322244d93782SGreg Clayton 
3223bd5ae6b4SGreg Clayton     bool
3224bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
322544d93782SGreg Clayton     {
3226ec990867SGreg Clayton         Thread* thread = (Thread*)item.GetUserData();
3227ec990867SGreg Clayton         if (thread)
322844d93782SGreg Clayton         {
3229ec990867SGreg Clayton             thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID());
323044d93782SGreg Clayton             const uint64_t frame_idx = item.GetIdentifier();
3231ec990867SGreg Clayton             thread->SetSelectedFrameByIndex(frame_idx);
323244d93782SGreg Clayton             return true;
323344d93782SGreg Clayton         }
323444d93782SGreg Clayton         return false;
323544d93782SGreg Clayton     }
3236315b6884SEugene Zelenko 
3237554f68d3SGreg Clayton protected:
3238554f68d3SGreg Clayton     FormatEntity::Entry m_format;
323944d93782SGreg Clayton };
324044d93782SGreg Clayton 
324144d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate
324244d93782SGreg Clayton {
324344d93782SGreg Clayton public:
324444d93782SGreg Clayton     ThreadTreeDelegate (Debugger &debugger) :
324544d93782SGreg Clayton         TreeDelegate(),
324644d93782SGreg Clayton         m_debugger (debugger),
324744d93782SGreg Clayton         m_tid (LLDB_INVALID_THREAD_ID),
324844d93782SGreg Clayton         m_stop_id (UINT32_MAX)
324944d93782SGreg Clayton     {
3250554f68d3SGreg Clayton         FormatEntity::Parse ("thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}",
3251554f68d3SGreg Clayton                              m_format);
325244d93782SGreg Clayton     }
325344d93782SGreg Clayton 
3254315b6884SEugene Zelenko     ~ThreadTreeDelegate() override = default;
325544d93782SGreg Clayton 
3256ec990867SGreg Clayton     ProcessSP
3257ec990867SGreg Clayton     GetProcess ()
3258ec990867SGreg Clayton     {
3259ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3260ec990867SGreg Clayton     }
3261ec990867SGreg Clayton 
3262ec990867SGreg Clayton     ThreadSP
3263ec990867SGreg Clayton     GetThread (const TreeItem &item)
3264ec990867SGreg Clayton     {
3265ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3266ec990867SGreg Clayton         if (process_sp)
3267ec990867SGreg Clayton             return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier());
3268ec990867SGreg Clayton         return ThreadSP();
3269ec990867SGreg Clayton     }
3270ec990867SGreg Clayton 
3271bd5ae6b4SGreg Clayton     void
3272bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
327344d93782SGreg Clayton     {
3274ec990867SGreg Clayton         ThreadSP thread_sp = GetThread (item);
327544d93782SGreg Clayton         if (thread_sp)
327644d93782SGreg Clayton         {
327744d93782SGreg Clayton             StreamString strm;
327844d93782SGreg Clayton             ExecutionContext exe_ctx (thread_sp);
3279554f68d3SGreg Clayton             if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
328044d93782SGreg Clayton             {
328144d93782SGreg Clayton                 int right_pad = 1;
328244d93782SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
328344d93782SGreg Clayton             }
328444d93782SGreg Clayton         }
328544d93782SGreg Clayton     }
3286315b6884SEugene Zelenko 
3287bd5ae6b4SGreg Clayton     void
3288bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
328944d93782SGreg Clayton     {
3290ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
329144d93782SGreg Clayton         if (process_sp && process_sp->IsAlive())
329244d93782SGreg Clayton         {
329344d93782SGreg Clayton             StateType state = process_sp->GetState();
329444d93782SGreg Clayton             if (StateIsStoppedState(state, true))
329544d93782SGreg Clayton             {
3296ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
329744d93782SGreg Clayton                 if (thread_sp)
329844d93782SGreg Clayton                 {
329944d93782SGreg Clayton                     if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid)
330044d93782SGreg Clayton                         return; // Children are already up to date
3301ec990867SGreg Clayton                     if (!m_frame_delegate_sp)
330244d93782SGreg Clayton                     {
330344d93782SGreg Clayton                         // Always expand the thread item the first time we show it
3304ec990867SGreg Clayton                         m_frame_delegate_sp.reset (new FrameTreeDelegate());
330544d93782SGreg Clayton                     }
330644d93782SGreg Clayton 
330744d93782SGreg Clayton                     m_stop_id = process_sp->GetStopID();
330844d93782SGreg Clayton                     m_tid = thread_sp->GetID();
330944d93782SGreg Clayton 
331044d93782SGreg Clayton                     TreeItem t (&item, *m_frame_delegate_sp, false);
331144d93782SGreg Clayton                     size_t num_frames = thread_sp->GetStackFrameCount();
331244d93782SGreg Clayton                     item.Resize (num_frames, t);
331344d93782SGreg Clayton                     for (size_t i=0; i<num_frames; ++i)
331444d93782SGreg Clayton                     {
3315ec990867SGreg Clayton                         item[i].SetUserData(thread_sp.get());
331644d93782SGreg Clayton                         item[i].SetIdentifier(i);
331744d93782SGreg Clayton                     }
331844d93782SGreg Clayton                 }
331944d93782SGreg Clayton                 return;
332044d93782SGreg Clayton             }
332144d93782SGreg Clayton         }
332244d93782SGreg Clayton         item.ClearChildren();
332344d93782SGreg Clayton     }
332444d93782SGreg Clayton 
3325bd5ae6b4SGreg Clayton     bool
3326bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
332744d93782SGreg Clayton     {
3328ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3329ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3330ec990867SGreg Clayton         {
3331ec990867SGreg Clayton             StateType state = process_sp->GetState();
3332ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3333ec990867SGreg Clayton             {
3334ec990867SGreg Clayton                 ThreadSP thread_sp = GetThread (item);
333544d93782SGreg Clayton                 if (thread_sp)
333644d93782SGreg Clayton                 {
333744d93782SGreg Clayton                     ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList();
333844d93782SGreg Clayton                     Mutex::Locker locker (thread_list.GetMutex());
333944d93782SGreg Clayton                     ThreadSP selected_thread_sp = thread_list.GetSelectedThread();
334044d93782SGreg Clayton                     if (selected_thread_sp->GetID() != thread_sp->GetID())
334144d93782SGreg Clayton                     {
334244d93782SGreg Clayton                         thread_list.SetSelectedThreadByID(thread_sp->GetID());
334344d93782SGreg Clayton                         return true;
334444d93782SGreg Clayton                     }
334544d93782SGreg Clayton                 }
3346ec990867SGreg Clayton             }
3347ec990867SGreg Clayton         }
334844d93782SGreg Clayton         return false;
334944d93782SGreg Clayton     }
335044d93782SGreg Clayton 
335144d93782SGreg Clayton protected:
335244d93782SGreg Clayton     Debugger &m_debugger;
335344d93782SGreg Clayton     std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp;
335444d93782SGreg Clayton     lldb::user_id_t m_tid;
335544d93782SGreg Clayton     uint32_t m_stop_id;
3356554f68d3SGreg Clayton     FormatEntity::Entry m_format;
335744d93782SGreg Clayton };
335844d93782SGreg Clayton 
3359ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate
3360ec990867SGreg Clayton {
3361ec990867SGreg Clayton public:
3362ec990867SGreg Clayton     ThreadsTreeDelegate (Debugger &debugger) :
3363ec990867SGreg Clayton         TreeDelegate(),
3364ec990867SGreg Clayton         m_thread_delegate_sp (),
3365ec990867SGreg Clayton         m_debugger (debugger),
3366ec990867SGreg Clayton         m_stop_id (UINT32_MAX)
3367ec990867SGreg Clayton     {
3368554f68d3SGreg Clayton         FormatEntity::Parse("process ${process.id}{, name = ${process.name}}",
3369554f68d3SGreg Clayton                             m_format);
3370ec990867SGreg Clayton     }
3371ec990867SGreg Clayton 
3372315b6884SEugene Zelenko     ~ThreadsTreeDelegate() override = default;
3373ec990867SGreg Clayton 
3374ec990867SGreg Clayton     ProcessSP
3375ec990867SGreg Clayton     GetProcess ()
3376ec990867SGreg Clayton     {
3377ec990867SGreg Clayton         return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP();
3378ec990867SGreg Clayton     }
3379ec990867SGreg Clayton 
3380bd5ae6b4SGreg Clayton     void
3381bd5ae6b4SGreg Clayton     TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override
3382ec990867SGreg Clayton     {
3383ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3384ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3385ec990867SGreg Clayton         {
3386ec990867SGreg Clayton             StreamString strm;
3387ec990867SGreg Clayton             ExecutionContext exe_ctx (process_sp);
3388554f68d3SGreg Clayton             if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
3389ec990867SGreg Clayton             {
3390ec990867SGreg Clayton                 int right_pad = 1;
3391ec990867SGreg Clayton                 window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
3392ec990867SGreg Clayton             }
3393ec990867SGreg Clayton         }
3394ec990867SGreg Clayton     }
3395ec990867SGreg Clayton 
3396bd5ae6b4SGreg Clayton     void
3397bd5ae6b4SGreg Clayton     TreeDelegateGenerateChildren (TreeItem &item) override
3398ec990867SGreg Clayton     {
3399ec990867SGreg Clayton         ProcessSP process_sp = GetProcess ();
3400ec990867SGreg Clayton         if (process_sp && process_sp->IsAlive())
3401ec990867SGreg Clayton         {
3402ec990867SGreg Clayton             StateType state = process_sp->GetState();
3403ec990867SGreg Clayton             if (StateIsStoppedState(state, true))
3404ec990867SGreg Clayton             {
3405ec990867SGreg Clayton                 const uint32_t stop_id = process_sp->GetStopID();
3406ec990867SGreg Clayton                 if (m_stop_id == stop_id)
3407ec990867SGreg Clayton                     return; // Children are already up to date
3408ec990867SGreg Clayton 
3409ec990867SGreg Clayton                 m_stop_id = stop_id;
3410ec990867SGreg Clayton 
3411ec990867SGreg Clayton                 if (!m_thread_delegate_sp)
3412ec990867SGreg Clayton                 {
3413ec990867SGreg Clayton                     // Always expand the thread item the first time we show it
3414ec990867SGreg Clayton                     //item.Expand();
3415ec990867SGreg Clayton                     m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger));
3416ec990867SGreg Clayton                 }
3417ec990867SGreg Clayton 
3418ec990867SGreg Clayton                 TreeItem t (&item, *m_thread_delegate_sp, false);
3419ec990867SGreg Clayton                 ThreadList &threads = process_sp->GetThreadList();
3420ec990867SGreg Clayton                 Mutex::Locker locker (threads.GetMutex());
3421ec990867SGreg Clayton                 size_t num_threads = threads.GetSize();
3422ec990867SGreg Clayton                 item.Resize (num_threads, t);
3423ec990867SGreg Clayton                 for (size_t i=0; i<num_threads; ++i)
3424ec990867SGreg Clayton                 {
3425ec990867SGreg Clayton                     item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID());
3426ec990867SGreg Clayton                     item[i].SetMightHaveChildren(true);
3427ec990867SGreg Clayton                 }
3428ec990867SGreg Clayton                 return;
3429ec990867SGreg Clayton             }
3430ec990867SGreg Clayton         }
3431ec990867SGreg Clayton         item.ClearChildren();
3432ec990867SGreg Clayton     }
3433ec990867SGreg Clayton 
3434bd5ae6b4SGreg Clayton     bool
3435bd5ae6b4SGreg Clayton     TreeDelegateItemSelected (TreeItem &item) override
3436ec990867SGreg Clayton     {
3437ec990867SGreg Clayton         return false;
3438ec990867SGreg Clayton     }
3439ec990867SGreg Clayton 
3440ec990867SGreg Clayton protected:
3441ec990867SGreg Clayton     std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp;
3442ec990867SGreg Clayton     Debugger &m_debugger;
3443ec990867SGreg Clayton     uint32_t m_stop_id;
3444554f68d3SGreg Clayton     FormatEntity::Entry m_format;
3445ec990867SGreg Clayton };
3446ec990867SGreg Clayton 
344744d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate
344844d93782SGreg Clayton {
344944d93782SGreg Clayton public:
345044d93782SGreg Clayton     ValueObjectListDelegate () :
345144d93782SGreg Clayton         m_valobj_list (),
345244d93782SGreg Clayton         m_rows (),
345344d93782SGreg Clayton         m_selected_row (NULL),
345444d93782SGreg Clayton         m_selected_row_idx (0),
345544d93782SGreg Clayton         m_first_visible_row (0),
345644d93782SGreg Clayton         m_num_rows (0),
345744d93782SGreg Clayton         m_max_x (0),
345844d93782SGreg Clayton         m_max_y (0)
345944d93782SGreg Clayton     {
346044d93782SGreg Clayton     }
346144d93782SGreg Clayton 
346244d93782SGreg Clayton     ValueObjectListDelegate (ValueObjectList &valobj_list) :
346344d93782SGreg Clayton         m_valobj_list (valobj_list),
346444d93782SGreg Clayton         m_rows (),
346544d93782SGreg Clayton         m_selected_row (NULL),
346644d93782SGreg Clayton         m_selected_row_idx (0),
346744d93782SGreg Clayton         m_first_visible_row (0),
346844d93782SGreg Clayton         m_num_rows (0),
346944d93782SGreg Clayton         m_max_x (0),
347044d93782SGreg Clayton         m_max_y (0)
347144d93782SGreg Clayton     {
347244d93782SGreg Clayton         SetValues (valobj_list);
347344d93782SGreg Clayton     }
347444d93782SGreg Clayton 
3475315b6884SEugene Zelenko     ~ValueObjectListDelegate() override = default;
347644d93782SGreg Clayton 
347744d93782SGreg Clayton     void
347844d93782SGreg Clayton     SetValues (ValueObjectList &valobj_list)
347944d93782SGreg Clayton     {
348044d93782SGreg Clayton         m_selected_row = NULL;
348144d93782SGreg Clayton         m_selected_row_idx = 0;
348244d93782SGreg Clayton         m_first_visible_row = 0;
348344d93782SGreg Clayton         m_num_rows = 0;
348444d93782SGreg Clayton         m_rows.clear();
348544d93782SGreg Clayton         m_valobj_list = valobj_list;
348644d93782SGreg Clayton         const size_t num_values = m_valobj_list.GetSize();
348744d93782SGreg Clayton         for (size_t i=0; i<num_values; ++i)
348844d93782SGreg Clayton             m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), NULL));
348944d93782SGreg Clayton     }
349044d93782SGreg Clayton 
3491bd5ae6b4SGreg Clayton     bool
3492bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
349344d93782SGreg Clayton     {
349444d93782SGreg Clayton         m_num_rows = 0;
349544d93782SGreg Clayton         m_min_x = 2;
349644d93782SGreg Clayton         m_min_y = 1;
349744d93782SGreg Clayton         m_max_x = window.GetWidth() - 1;
349844d93782SGreg Clayton         m_max_y = window.GetHeight() - 1;
349944d93782SGreg Clayton 
350044d93782SGreg Clayton         window.Erase();
350144d93782SGreg Clayton         window.DrawTitleBox (window.GetName());
350244d93782SGreg Clayton 
350344d93782SGreg Clayton         const int num_visible_rows = NumVisibleRows();
350444d93782SGreg Clayton         const int num_rows = CalculateTotalNumberRows (m_rows);
350544d93782SGreg Clayton 
350644d93782SGreg Clayton         // If we unexpanded while having something selected our
350744d93782SGreg Clayton         // total number of rows is less than the num visible rows,
350844d93782SGreg Clayton         // then make sure we show all the rows by setting the first
350944d93782SGreg Clayton         // visible row accordingly.
351044d93782SGreg Clayton         if (m_first_visible_row > 0 && num_rows < num_visible_rows)
351144d93782SGreg Clayton             m_first_visible_row = 0;
351244d93782SGreg Clayton 
351344d93782SGreg Clayton         // Make sure the selected row is always visible
351444d93782SGreg Clayton         if (m_selected_row_idx < m_first_visible_row)
351544d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx;
351644d93782SGreg Clayton         else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx)
351744d93782SGreg Clayton             m_first_visible_row = m_selected_row_idx - num_visible_rows + 1;
351844d93782SGreg Clayton 
351944d93782SGreg Clayton         DisplayRows (window, m_rows, g_options);
352044d93782SGreg Clayton 
352144d93782SGreg Clayton         window.DeferredRefresh();
352244d93782SGreg Clayton 
352344d93782SGreg Clayton         // Get the selected row
352444d93782SGreg Clayton         m_selected_row = GetRowForRowIndex (m_selected_row_idx);
352544d93782SGreg Clayton         // Keep the cursor on the selected row so the highlight and the cursor
352644d93782SGreg Clayton         // are always on the same line
352744d93782SGreg Clayton         if (m_selected_row)
352844d93782SGreg Clayton             window.MoveCursor (m_selected_row->x,
352944d93782SGreg Clayton                                m_selected_row->y);
353044d93782SGreg Clayton 
353144d93782SGreg Clayton         return true; // Drawing handled
353244d93782SGreg Clayton     }
353344d93782SGreg Clayton 
3534bd5ae6b4SGreg Clayton     KeyHelp *
3535bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
353644d93782SGreg Clayton     {
353744d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
353844d93782SGreg Clayton             { KEY_UP, "Select previous item" },
353944d93782SGreg Clayton             { KEY_DOWN, "Select next item" },
354044d93782SGreg Clayton             { KEY_RIGHT, "Expand selected item" },
354144d93782SGreg Clayton             { KEY_LEFT, "Unexpand selected item or select parent if not expanded" },
354244d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
354344d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
354444d93782SGreg Clayton             { 'A', "Format as annotated address" },
354544d93782SGreg Clayton             { 'b', "Format as binary" },
354644d93782SGreg Clayton             { 'B', "Format as hex bytes with ASCII" },
354744d93782SGreg Clayton             { 'c', "Format as character" },
354844d93782SGreg Clayton             { 'd', "Format as a signed integer" },
354944d93782SGreg Clayton             { 'D', "Format selected value using the default format for the type" },
355044d93782SGreg Clayton             { 'f', "Format as float" },
355144d93782SGreg Clayton             { 'h', "Show help dialog" },
355244d93782SGreg Clayton             { 'i', "Format as instructions" },
355344d93782SGreg Clayton             { 'o', "Format as octal" },
355444d93782SGreg Clayton             { 'p', "Format as pointer" },
355544d93782SGreg Clayton             { 's', "Format as C string" },
355644d93782SGreg Clayton             { 't', "Toggle showing/hiding type names" },
355744d93782SGreg Clayton             { 'u', "Format as an unsigned integer" },
355844d93782SGreg Clayton             { 'x', "Format as hex" },
355944d93782SGreg Clayton             { 'X', "Format as uppercase hex" },
356044d93782SGreg Clayton             { ' ', "Toggle item expansion" },
356144d93782SGreg Clayton             { ',', "Page up" },
356244d93782SGreg Clayton             { '.', "Page down" },
356344d93782SGreg Clayton             { '\0', NULL }
356444d93782SGreg Clayton         };
356544d93782SGreg Clayton         return g_source_view_key_help;
356644d93782SGreg Clayton     }
356744d93782SGreg Clayton 
3568bd5ae6b4SGreg Clayton     HandleCharResult
3569bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
357044d93782SGreg Clayton     {
357144d93782SGreg Clayton         switch(c)
357244d93782SGreg Clayton         {
357344d93782SGreg Clayton             case 'x':
357444d93782SGreg Clayton             case 'X':
357544d93782SGreg Clayton             case 'o':
357644d93782SGreg Clayton             case 's':
357744d93782SGreg Clayton             case 'u':
357844d93782SGreg Clayton             case 'd':
357944d93782SGreg Clayton             case 'D':
358044d93782SGreg Clayton             case 'i':
358144d93782SGreg Clayton             case 'A':
358244d93782SGreg Clayton             case 'p':
358344d93782SGreg Clayton             case 'c':
358444d93782SGreg Clayton             case 'b':
358544d93782SGreg Clayton             case 'B':
358644d93782SGreg Clayton             case 'f':
358744d93782SGreg Clayton                 // Change the format for the currently selected item
358844d93782SGreg Clayton                 if (m_selected_row)
358944d93782SGreg Clayton                     m_selected_row->valobj->SetFormat (FormatForChar (c));
359044d93782SGreg Clayton                 return eKeyHandled;
359144d93782SGreg Clayton 
359244d93782SGreg Clayton             case 't':
359344d93782SGreg Clayton                 // Toggle showing type names
359444d93782SGreg Clayton                 g_options.show_types = !g_options.show_types;
359544d93782SGreg Clayton                 return eKeyHandled;
359644d93782SGreg Clayton 
359744d93782SGreg Clayton             case ',':
359844d93782SGreg Clayton             case KEY_PPAGE:
359944d93782SGreg Clayton                 // Page up key
360044d93782SGreg Clayton                 if (m_first_visible_row > 0)
360144d93782SGreg Clayton                 {
36023985c8c6SSaleem Abdulrasool                     if (static_cast<int>(m_first_visible_row) > m_max_y)
360344d93782SGreg Clayton                         m_first_visible_row -= m_max_y;
360444d93782SGreg Clayton                     else
360544d93782SGreg Clayton                         m_first_visible_row = 0;
360644d93782SGreg Clayton                     m_selected_row_idx = m_first_visible_row;
360744d93782SGreg Clayton                 }
360844d93782SGreg Clayton                 return eKeyHandled;
360944d93782SGreg Clayton 
361044d93782SGreg Clayton             case '.':
361144d93782SGreg Clayton             case KEY_NPAGE:
361244d93782SGreg Clayton                 // Page down key
36133985c8c6SSaleem Abdulrasool                 if (m_num_rows > static_cast<size_t>(m_max_y))
361444d93782SGreg Clayton                 {
361544d93782SGreg Clayton                     if (m_first_visible_row + m_max_y < m_num_rows)
361644d93782SGreg Clayton                     {
361744d93782SGreg Clayton                         m_first_visible_row += m_max_y;
361844d93782SGreg Clayton                         m_selected_row_idx = m_first_visible_row;
361944d93782SGreg Clayton                     }
362044d93782SGreg Clayton                 }
362144d93782SGreg Clayton                 return eKeyHandled;
362244d93782SGreg Clayton 
362344d93782SGreg Clayton             case KEY_UP:
362444d93782SGreg Clayton                 if (m_selected_row_idx > 0)
362544d93782SGreg Clayton                     --m_selected_row_idx;
362644d93782SGreg Clayton                 return eKeyHandled;
3627315b6884SEugene Zelenko 
362844d93782SGreg Clayton             case KEY_DOWN:
362944d93782SGreg Clayton                 if (m_selected_row_idx + 1 < m_num_rows)
363044d93782SGreg Clayton                     ++m_selected_row_idx;
363144d93782SGreg Clayton                 return eKeyHandled;
363244d93782SGreg Clayton 
363344d93782SGreg Clayton             case KEY_RIGHT:
363444d93782SGreg Clayton                 if (m_selected_row)
363544d93782SGreg Clayton                 {
363644d93782SGreg Clayton                     if (!m_selected_row->expanded)
363744d93782SGreg Clayton                         m_selected_row->Expand();
363844d93782SGreg Clayton                 }
363944d93782SGreg Clayton                 return eKeyHandled;
364044d93782SGreg Clayton 
364144d93782SGreg Clayton             case KEY_LEFT:
364244d93782SGreg Clayton                 if (m_selected_row)
364344d93782SGreg Clayton                 {
364444d93782SGreg Clayton                     if (m_selected_row->expanded)
364544d93782SGreg Clayton                         m_selected_row->Unexpand();
364644d93782SGreg Clayton                     else if (m_selected_row->parent)
364744d93782SGreg Clayton                         m_selected_row_idx = m_selected_row->parent->row_idx;
364844d93782SGreg Clayton                 }
364944d93782SGreg Clayton                 return eKeyHandled;
365044d93782SGreg Clayton 
365144d93782SGreg Clayton             case ' ':
365244d93782SGreg Clayton                 // Toggle expansion state when SPACE is pressed
365344d93782SGreg Clayton                 if (m_selected_row)
365444d93782SGreg Clayton                 {
365544d93782SGreg Clayton                     if (m_selected_row->expanded)
365644d93782SGreg Clayton                         m_selected_row->Unexpand();
365744d93782SGreg Clayton                     else
365844d93782SGreg Clayton                         m_selected_row->Expand();
365944d93782SGreg Clayton                 }
366044d93782SGreg Clayton                 return eKeyHandled;
366144d93782SGreg Clayton 
366244d93782SGreg Clayton             case 'h':
366344d93782SGreg Clayton                 window.CreateHelpSubwindow ();
366444d93782SGreg Clayton                 return eKeyHandled;
366544d93782SGreg Clayton 
366644d93782SGreg Clayton             default:
366744d93782SGreg Clayton                 break;
366844d93782SGreg Clayton         }
366944d93782SGreg Clayton         return eKeyNotHandled;
367044d93782SGreg Clayton     }
367144d93782SGreg Clayton 
367244d93782SGreg Clayton protected:
367344d93782SGreg Clayton     ValueObjectList m_valobj_list;
367444d93782SGreg Clayton     std::vector<Row> m_rows;
367544d93782SGreg Clayton     Row *m_selected_row;
367644d93782SGreg Clayton     uint32_t m_selected_row_idx;
367744d93782SGreg Clayton     uint32_t m_first_visible_row;
367844d93782SGreg Clayton     uint32_t m_num_rows;
367944d93782SGreg Clayton     int m_min_x;
368044d93782SGreg Clayton     int m_min_y;
368144d93782SGreg Clayton     int m_max_x;
368244d93782SGreg Clayton     int m_max_y;
368344d93782SGreg Clayton 
368444d93782SGreg Clayton     static Format
368544d93782SGreg Clayton     FormatForChar (int c)
368644d93782SGreg Clayton     {
368744d93782SGreg Clayton         switch (c)
368844d93782SGreg Clayton         {
368944d93782SGreg Clayton             case 'x': return eFormatHex;
369044d93782SGreg Clayton             case 'X': return eFormatHexUppercase;
369144d93782SGreg Clayton             case 'o': return eFormatOctal;
369244d93782SGreg Clayton             case 's': return eFormatCString;
369344d93782SGreg Clayton             case 'u': return eFormatUnsigned;
369444d93782SGreg Clayton             case 'd': return eFormatDecimal;
369544d93782SGreg Clayton             case 'D': return eFormatDefault;
369644d93782SGreg Clayton             case 'i': return eFormatInstruction;
369744d93782SGreg Clayton             case 'A': return eFormatAddressInfo;
369844d93782SGreg Clayton             case 'p': return eFormatPointer;
369944d93782SGreg Clayton             case 'c': return eFormatChar;
370044d93782SGreg Clayton             case 'b': return eFormatBinary;
370144d93782SGreg Clayton             case 'B': return eFormatBytesWithASCII;
370244d93782SGreg Clayton             case 'f': return eFormatFloat;
370344d93782SGreg Clayton         }
370444d93782SGreg Clayton         return eFormatDefault;
370544d93782SGreg Clayton     }
370644d93782SGreg Clayton 
370744d93782SGreg Clayton     bool
370844d93782SGreg Clayton     DisplayRowObject (Window &window,
370944d93782SGreg Clayton                       Row &row,
371044d93782SGreg Clayton                       DisplayOptions &options,
371144d93782SGreg Clayton                       bool highlight,
371244d93782SGreg Clayton                       bool last_child)
371344d93782SGreg Clayton     {
371444d93782SGreg Clayton         ValueObject *valobj = row.valobj.get();
371544d93782SGreg Clayton 
371644d93782SGreg Clayton         if (valobj == NULL)
371744d93782SGreg Clayton             return false;
371844d93782SGreg Clayton 
371944d93782SGreg Clayton         const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : NULL;
372044d93782SGreg Clayton         const char *name = valobj->GetName().GetCString();
372144d93782SGreg Clayton         const char *value = valobj->GetValueAsCString ();
372244d93782SGreg Clayton         const char *summary = valobj->GetSummaryAsCString ();
372344d93782SGreg Clayton 
372444d93782SGreg Clayton         window.MoveCursor (row.x, row.y);
372544d93782SGreg Clayton 
372644d93782SGreg Clayton         row.DrawTree (window);
372744d93782SGreg Clayton 
372844d93782SGreg Clayton         if (highlight)
372944d93782SGreg Clayton             window.AttributeOn(A_REVERSE);
373044d93782SGreg Clayton 
373144d93782SGreg Clayton         if (type_name && type_name[0])
373244d93782SGreg Clayton             window.Printf ("(%s) ", type_name);
373344d93782SGreg Clayton 
373444d93782SGreg Clayton         if (name && name[0])
373544d93782SGreg Clayton             window.PutCString(name);
373644d93782SGreg Clayton 
373744d93782SGreg Clayton         attr_t changd_attr = 0;
373844d93782SGreg Clayton         if (valobj->GetValueDidChange())
373944d93782SGreg Clayton             changd_attr = COLOR_PAIR(5) | A_BOLD;
374044d93782SGreg Clayton 
374144d93782SGreg Clayton         if (value && value[0])
374244d93782SGreg Clayton         {
374344d93782SGreg Clayton             window.PutCString(" = ");
374444d93782SGreg Clayton             if (changd_attr)
374544d93782SGreg Clayton                 window.AttributeOn(changd_attr);
374644d93782SGreg Clayton             window.PutCString (value);
374744d93782SGreg Clayton             if (changd_attr)
374844d93782SGreg Clayton                 window.AttributeOff(changd_attr);
374944d93782SGreg Clayton         }
375044d93782SGreg Clayton 
375144d93782SGreg Clayton         if (summary && summary[0])
375244d93782SGreg Clayton         {
375344d93782SGreg Clayton             window.PutChar(' ');
375444d93782SGreg Clayton             if (changd_attr)
375544d93782SGreg Clayton                 window.AttributeOn(changd_attr);
375644d93782SGreg Clayton             window.PutCString(summary);
375744d93782SGreg Clayton             if (changd_attr)
375844d93782SGreg Clayton                 window.AttributeOff(changd_attr);
375944d93782SGreg Clayton         }
376044d93782SGreg Clayton 
376144d93782SGreg Clayton         if (highlight)
376244d93782SGreg Clayton             window.AttributeOff (A_REVERSE);
376344d93782SGreg Clayton 
376444d93782SGreg Clayton         return true;
376544d93782SGreg Clayton     }
3766315b6884SEugene Zelenko 
376744d93782SGreg Clayton     void
376844d93782SGreg Clayton     DisplayRows (Window &window,
376944d93782SGreg Clayton                  std::vector<Row> &rows,
377044d93782SGreg Clayton                  DisplayOptions &options)
377144d93782SGreg Clayton     {
377244d93782SGreg Clayton         // >   0x25B7
377344d93782SGreg Clayton         // \/  0x25BD
377444d93782SGreg Clayton 
377544d93782SGreg Clayton         bool window_is_active = window.IsActive();
377644d93782SGreg Clayton         for (auto &row : rows)
377744d93782SGreg Clayton         {
377844d93782SGreg Clayton             const bool last_child = row.parent && &rows[rows.size()-1] == &row;
377944d93782SGreg Clayton             // Save the row index in each Row structure
378044d93782SGreg Clayton             row.row_idx = m_num_rows;
378144d93782SGreg Clayton             if ((m_num_rows >= m_first_visible_row) &&
37823985c8c6SSaleem Abdulrasool                 ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows())))
378344d93782SGreg Clayton             {
378444d93782SGreg Clayton                 row.x = m_min_x;
378544d93782SGreg Clayton                 row.y = m_num_rows - m_first_visible_row + 1;
378644d93782SGreg Clayton                 if (DisplayRowObject (window,
378744d93782SGreg Clayton                                       row,
378844d93782SGreg Clayton                                       options,
378944d93782SGreg Clayton                                       window_is_active && m_num_rows == m_selected_row_idx,
379044d93782SGreg Clayton                                       last_child))
379144d93782SGreg Clayton                 {
379244d93782SGreg Clayton                     ++m_num_rows;
379344d93782SGreg Clayton                 }
379444d93782SGreg Clayton                 else
379544d93782SGreg Clayton                 {
379644d93782SGreg Clayton                     row.x = 0;
379744d93782SGreg Clayton                     row.y = 0;
379844d93782SGreg Clayton                 }
379944d93782SGreg Clayton             }
380044d93782SGreg Clayton             else
380144d93782SGreg Clayton             {
380244d93782SGreg Clayton                 row.x = 0;
380344d93782SGreg Clayton                 row.y = 0;
380444d93782SGreg Clayton                 ++m_num_rows;
380544d93782SGreg Clayton             }
380644d93782SGreg Clayton 
380744d93782SGreg Clayton             if (row.expanded && !row.children.empty())
380844d93782SGreg Clayton             {
380944d93782SGreg Clayton                 DisplayRows (window,
381044d93782SGreg Clayton                              row.children,
381144d93782SGreg Clayton                              options);
381244d93782SGreg Clayton             }
381344d93782SGreg Clayton         }
381444d93782SGreg Clayton     }
381544d93782SGreg Clayton 
381644d93782SGreg Clayton     int
381744d93782SGreg Clayton     CalculateTotalNumberRows (const std::vector<Row> &rows)
381844d93782SGreg Clayton     {
381944d93782SGreg Clayton         int row_count = 0;
382044d93782SGreg Clayton         for (const auto &row : rows)
382144d93782SGreg Clayton         {
382244d93782SGreg Clayton             ++row_count;
382344d93782SGreg Clayton             if (row.expanded)
382444d93782SGreg Clayton                 row_count += CalculateTotalNumberRows(row.children);
382544d93782SGreg Clayton         }
382644d93782SGreg Clayton         return row_count;
382744d93782SGreg Clayton     }
3828315b6884SEugene Zelenko 
382944d93782SGreg Clayton     static Row *
383044d93782SGreg Clayton     GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index)
383144d93782SGreg Clayton     {
383244d93782SGreg Clayton         for (auto &row : rows)
383344d93782SGreg Clayton         {
383444d93782SGreg Clayton             if (row_index == 0)
383544d93782SGreg Clayton                 return &row;
383644d93782SGreg Clayton             else
383744d93782SGreg Clayton             {
383844d93782SGreg Clayton                 --row_index;
383944d93782SGreg Clayton                 if (row.expanded && !row.children.empty())
384044d93782SGreg Clayton                 {
384144d93782SGreg Clayton                     Row *result = GetRowForRowIndexImpl (row.children, row_index);
384244d93782SGreg Clayton                     if (result)
384344d93782SGreg Clayton                         return result;
384444d93782SGreg Clayton                 }
384544d93782SGreg Clayton             }
384644d93782SGreg Clayton         }
384744d93782SGreg Clayton         return NULL;
384844d93782SGreg Clayton     }
384944d93782SGreg Clayton 
385044d93782SGreg Clayton     Row *
385144d93782SGreg Clayton     GetRowForRowIndex (size_t row_index)
385244d93782SGreg Clayton     {
385344d93782SGreg Clayton         return GetRowForRowIndexImpl (m_rows, row_index);
385444d93782SGreg Clayton     }
385544d93782SGreg Clayton 
385644d93782SGreg Clayton     int
385744d93782SGreg Clayton     NumVisibleRows () const
385844d93782SGreg Clayton     {
385944d93782SGreg Clayton         return m_max_y - m_min_y;
386044d93782SGreg Clayton     }
386144d93782SGreg Clayton 
386244d93782SGreg Clayton     static DisplayOptions g_options;
386344d93782SGreg Clayton };
386444d93782SGreg Clayton 
386544d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate
386644d93782SGreg Clayton {
386744d93782SGreg Clayton public:
386844d93782SGreg Clayton     FrameVariablesWindowDelegate (Debugger &debugger) :
386944d93782SGreg Clayton         ValueObjectListDelegate (),
387044d93782SGreg Clayton         m_debugger (debugger),
387144d93782SGreg Clayton         m_frame_block (NULL)
387244d93782SGreg Clayton     {
387344d93782SGreg Clayton     }
387444d93782SGreg Clayton 
3875315b6884SEugene Zelenko     ~FrameVariablesWindowDelegate() override = default;
387644d93782SGreg Clayton 
3877bd5ae6b4SGreg Clayton     const char *
3878bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
387944d93782SGreg Clayton     {
388044d93782SGreg Clayton         return "Frame variable window keyboard shortcuts:";
388144d93782SGreg Clayton     }
388244d93782SGreg Clayton 
3883bd5ae6b4SGreg Clayton     bool
3884bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
388544d93782SGreg Clayton     {
388644d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
388744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
388844d93782SGreg Clayton         Block *frame_block = NULL;
388944d93782SGreg Clayton         StackFrame *frame = NULL;
389044d93782SGreg Clayton 
389144d93782SGreg Clayton         if (process)
389244d93782SGreg Clayton         {
389344d93782SGreg Clayton             StateType state = process->GetState();
389444d93782SGreg Clayton             if (StateIsStoppedState(state, true))
389544d93782SGreg Clayton             {
389644d93782SGreg Clayton                 frame = exe_ctx.GetFramePtr();
389744d93782SGreg Clayton                 if (frame)
389844d93782SGreg Clayton                     frame_block = frame->GetFrameBlock ();
389944d93782SGreg Clayton             }
390044d93782SGreg Clayton             else if (StateIsRunningState(state))
390144d93782SGreg Clayton             {
390244d93782SGreg Clayton                 return true; // Don't do any updating when we are running
390344d93782SGreg Clayton             }
390444d93782SGreg Clayton         }
390544d93782SGreg Clayton 
390644d93782SGreg Clayton         ValueObjectList local_values;
390744d93782SGreg Clayton         if (frame_block)
390844d93782SGreg Clayton         {
390944d93782SGreg Clayton             // Only update the variables if they have changed
391044d93782SGreg Clayton             if (m_frame_block != frame_block)
391144d93782SGreg Clayton             {
391244d93782SGreg Clayton                 m_frame_block = frame_block;
391344d93782SGreg Clayton 
391444d93782SGreg Clayton                 VariableList *locals = frame->GetVariableList(true);
391544d93782SGreg Clayton                 if (locals)
391644d93782SGreg Clayton                 {
391744d93782SGreg Clayton                     const DynamicValueType use_dynamic = eDynamicDontRunTarget;
391844d93782SGreg Clayton                     const size_t num_locals = locals->GetSize();
391944d93782SGreg Clayton                     for (size_t i=0; i<num_locals; ++i)
3920eb72dc7dSGreg Clayton                     {
3921eb72dc7dSGreg Clayton                         ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic);
3922eb72dc7dSGreg Clayton                         if (value_sp)
3923eb72dc7dSGreg Clayton                         {
3924eb72dc7dSGreg Clayton                             ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue();
3925eb72dc7dSGreg Clayton                             if (synthetic_value_sp)
3926eb72dc7dSGreg Clayton                                 local_values.Append(synthetic_value_sp);
3927eb72dc7dSGreg Clayton                             else
3928eb72dc7dSGreg Clayton                                 local_values.Append(value_sp);
3929eb72dc7dSGreg Clayton 
3930eb72dc7dSGreg Clayton                         }
3931eb72dc7dSGreg Clayton                     }
393244d93782SGreg Clayton                     // Update the values
393344d93782SGreg Clayton                     SetValues(local_values);
393444d93782SGreg Clayton                 }
393544d93782SGreg Clayton             }
393644d93782SGreg Clayton         }
393744d93782SGreg Clayton         else
393844d93782SGreg Clayton         {
393944d93782SGreg Clayton             m_frame_block = NULL;
394044d93782SGreg Clayton             // Update the values with an empty list if there is no frame
394144d93782SGreg Clayton             SetValues(local_values);
394244d93782SGreg Clayton         }
394344d93782SGreg Clayton 
394444d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
394544d93782SGreg Clayton     }
394644d93782SGreg Clayton 
394744d93782SGreg Clayton protected:
394844d93782SGreg Clayton     Debugger &m_debugger;
394944d93782SGreg Clayton     Block *m_frame_block;
395044d93782SGreg Clayton };
395144d93782SGreg Clayton 
395244d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate
395344d93782SGreg Clayton {
395444d93782SGreg Clayton public:
395544d93782SGreg Clayton     RegistersWindowDelegate (Debugger &debugger) :
395644d93782SGreg Clayton         ValueObjectListDelegate (),
395744d93782SGreg Clayton         m_debugger (debugger)
395844d93782SGreg Clayton     {
395944d93782SGreg Clayton     }
396044d93782SGreg Clayton 
3961315b6884SEugene Zelenko     ~RegistersWindowDelegate() override = default;
396244d93782SGreg Clayton 
3963bd5ae6b4SGreg Clayton     const char *
3964bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
396544d93782SGreg Clayton     {
396644d93782SGreg Clayton         return "Register window keyboard shortcuts:";
396744d93782SGreg Clayton     }
396844d93782SGreg Clayton 
3969bd5ae6b4SGreg Clayton     bool
3970bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
397144d93782SGreg Clayton     {
397244d93782SGreg Clayton         ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext());
397344d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
397444d93782SGreg Clayton 
397544d93782SGreg Clayton         ValueObjectList value_list;
397644d93782SGreg Clayton         if (frame)
397744d93782SGreg Clayton         {
397844d93782SGreg Clayton             if (frame->GetStackID() != m_stack_id)
397944d93782SGreg Clayton             {
398044d93782SGreg Clayton                 m_stack_id = frame->GetStackID();
398144d93782SGreg Clayton                 RegisterContextSP reg_ctx (frame->GetRegisterContext());
398244d93782SGreg Clayton                 if (reg_ctx)
398344d93782SGreg Clayton                 {
398444d93782SGreg Clayton                     const uint32_t num_sets = reg_ctx->GetRegisterSetCount();
398544d93782SGreg Clayton                     for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx)
398644d93782SGreg Clayton                     {
398744d93782SGreg Clayton                         value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx));
398844d93782SGreg Clayton                     }
398944d93782SGreg Clayton                 }
399044d93782SGreg Clayton                 SetValues(value_list);
399144d93782SGreg Clayton             }
399244d93782SGreg Clayton         }
399344d93782SGreg Clayton         else
399444d93782SGreg Clayton         {
399544d93782SGreg Clayton             Process *process = exe_ctx.GetProcessPtr();
399644d93782SGreg Clayton             if (process && process->IsAlive())
399744d93782SGreg Clayton                 return true; // Don't do any updating if we are running
399844d93782SGreg Clayton             else
399944d93782SGreg Clayton             {
400044d93782SGreg Clayton                 // Update the values with an empty list if there
400144d93782SGreg Clayton                 // is no process or the process isn't alive anymore
400244d93782SGreg Clayton                 SetValues(value_list);
400344d93782SGreg Clayton             }
400444d93782SGreg Clayton         }
400544d93782SGreg Clayton         return ValueObjectListDelegate::WindowDelegateDraw (window, force);
400644d93782SGreg Clayton     }
400744d93782SGreg Clayton 
400844d93782SGreg Clayton protected:
400944d93782SGreg Clayton     Debugger &m_debugger;
401044d93782SGreg Clayton     StackID m_stack_id;
401144d93782SGreg Clayton };
401244d93782SGreg Clayton 
401344d93782SGreg Clayton static const char *
401444d93782SGreg Clayton CursesKeyToCString (int ch)
401544d93782SGreg Clayton {
401644d93782SGreg Clayton     static char g_desc[32];
401744d93782SGreg Clayton     if (ch >= KEY_F0 && ch < KEY_F0 + 64)
401844d93782SGreg Clayton     {
401944d93782SGreg Clayton         snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0);
402044d93782SGreg Clayton         return g_desc;
402144d93782SGreg Clayton     }
402244d93782SGreg Clayton     switch (ch)
402344d93782SGreg Clayton     {
402444d93782SGreg Clayton         case KEY_DOWN:  return "down";
402544d93782SGreg Clayton         case KEY_UP:    return "up";
402644d93782SGreg Clayton         case KEY_LEFT:  return "left";
402744d93782SGreg Clayton         case KEY_RIGHT: return "right";
402844d93782SGreg Clayton         case KEY_HOME:  return "home";
402944d93782SGreg Clayton         case KEY_BACKSPACE: return "backspace";
403044d93782SGreg Clayton         case KEY_DL:        return "delete-line";
403144d93782SGreg Clayton         case KEY_IL:        return "insert-line";
403244d93782SGreg Clayton         case KEY_DC:        return "delete-char";
403344d93782SGreg Clayton         case KEY_IC:        return "insert-char";
403444d93782SGreg Clayton         case KEY_CLEAR:     return "clear";
403544d93782SGreg Clayton         case KEY_EOS:       return "clear-to-eos";
403644d93782SGreg Clayton         case KEY_EOL:       return "clear-to-eol";
403744d93782SGreg Clayton         case KEY_SF:        return "scroll-forward";
403844d93782SGreg Clayton         case KEY_SR:        return "scroll-backward";
403944d93782SGreg Clayton         case KEY_NPAGE:     return "page-down";
404044d93782SGreg Clayton         case KEY_PPAGE:     return "page-up";
404144d93782SGreg Clayton         case KEY_STAB:      return "set-tab";
404244d93782SGreg Clayton         case KEY_CTAB:      return "clear-tab";
404344d93782SGreg Clayton         case KEY_CATAB:     return "clear-all-tabs";
404444d93782SGreg Clayton         case KEY_ENTER:     return "enter";
404544d93782SGreg Clayton         case KEY_PRINT:     return "print";
404644d93782SGreg Clayton         case KEY_LL:        return "lower-left key";
404744d93782SGreg Clayton         case KEY_A1:        return "upper left of keypad";
404844d93782SGreg Clayton         case KEY_A3:        return "upper right of keypad";
404944d93782SGreg Clayton         case KEY_B2:        return "center of keypad";
405044d93782SGreg Clayton         case KEY_C1:        return "lower left of keypad";
405144d93782SGreg Clayton         case KEY_C3:        return "lower right of keypad";
405244d93782SGreg Clayton         case KEY_BTAB:      return "back-tab key";
405344d93782SGreg Clayton         case KEY_BEG:       return "begin key";
405444d93782SGreg Clayton         case KEY_CANCEL:    return "cancel key";
405544d93782SGreg Clayton         case KEY_CLOSE:     return "close key";
405644d93782SGreg Clayton         case KEY_COMMAND:   return "command key";
405744d93782SGreg Clayton         case KEY_COPY:      return "copy key";
405844d93782SGreg Clayton         case KEY_CREATE:    return "create key";
405944d93782SGreg Clayton         case KEY_END:       return "end key";
406044d93782SGreg Clayton         case KEY_EXIT:      return "exit key";
406144d93782SGreg Clayton         case KEY_FIND:      return "find key";
406244d93782SGreg Clayton         case KEY_HELP:      return "help key";
406344d93782SGreg Clayton         case KEY_MARK:      return "mark key";
406444d93782SGreg Clayton         case KEY_MESSAGE:   return "message key";
406544d93782SGreg Clayton         case KEY_MOVE:      return "move key";
406644d93782SGreg Clayton         case KEY_NEXT:      return "next key";
406744d93782SGreg Clayton         case KEY_OPEN:      return "open key";
406844d93782SGreg Clayton         case KEY_OPTIONS:   return "options key";
406944d93782SGreg Clayton         case KEY_PREVIOUS:  return "previous key";
407044d93782SGreg Clayton         case KEY_REDO:      return "redo key";
407144d93782SGreg Clayton         case KEY_REFERENCE: return "reference key";
407244d93782SGreg Clayton         case KEY_REFRESH:   return "refresh key";
407344d93782SGreg Clayton         case KEY_REPLACE:   return "replace key";
407444d93782SGreg Clayton         case KEY_RESTART:   return "restart key";
407544d93782SGreg Clayton         case KEY_RESUME:    return "resume key";
407644d93782SGreg Clayton         case KEY_SAVE:      return "save key";
407744d93782SGreg Clayton         case KEY_SBEG:      return "shifted begin key";
407844d93782SGreg Clayton         case KEY_SCANCEL:   return "shifted cancel key";
407944d93782SGreg Clayton         case KEY_SCOMMAND:  return "shifted command key";
408044d93782SGreg Clayton         case KEY_SCOPY:     return "shifted copy key";
408144d93782SGreg Clayton         case KEY_SCREATE:   return "shifted create key";
408244d93782SGreg Clayton         case KEY_SDC:       return "shifted delete-character key";
408344d93782SGreg Clayton         case KEY_SDL:       return "shifted delete-line key";
408444d93782SGreg Clayton         case KEY_SELECT:    return "select key";
408544d93782SGreg Clayton         case KEY_SEND:      return "shifted end key";
408644d93782SGreg Clayton         case KEY_SEOL:      return "shifted clear-to-end-of-line key";
408744d93782SGreg Clayton         case KEY_SEXIT:     return "shifted exit key";
408844d93782SGreg Clayton         case KEY_SFIND:     return "shifted find key";
408944d93782SGreg Clayton         case KEY_SHELP:     return "shifted help key";
409044d93782SGreg Clayton         case KEY_SHOME:     return "shifted home key";
409144d93782SGreg Clayton         case KEY_SIC:       return "shifted insert-character key";
409244d93782SGreg Clayton         case KEY_SLEFT:     return "shifted left-arrow key";
409344d93782SGreg Clayton         case KEY_SMESSAGE:  return "shifted message key";
409444d93782SGreg Clayton         case KEY_SMOVE:     return "shifted move key";
409544d93782SGreg Clayton         case KEY_SNEXT:     return "shifted next key";
409644d93782SGreg Clayton         case KEY_SOPTIONS:  return "shifted options key";
409744d93782SGreg Clayton         case KEY_SPREVIOUS: return "shifted previous key";
409844d93782SGreg Clayton         case KEY_SPRINT:    return "shifted print key";
409944d93782SGreg Clayton         case KEY_SREDO:     return "shifted redo key";
410044d93782SGreg Clayton         case KEY_SREPLACE:  return "shifted replace key";
410144d93782SGreg Clayton         case KEY_SRIGHT:    return "shifted right-arrow key";
410244d93782SGreg Clayton         case KEY_SRSUME:    return "shifted resume key";
410344d93782SGreg Clayton         case KEY_SSAVE:     return "shifted save key";
410444d93782SGreg Clayton         case KEY_SSUSPEND:  return "shifted suspend key";
410544d93782SGreg Clayton         case KEY_SUNDO:     return "shifted undo key";
410644d93782SGreg Clayton         case KEY_SUSPEND:   return "suspend key";
410744d93782SGreg Clayton         case KEY_UNDO:      return "undo key";
410844d93782SGreg Clayton         case KEY_MOUSE:     return "Mouse event has occurred";
410944d93782SGreg Clayton         case KEY_RESIZE:    return "Terminal resize event";
411027801f4fSBruce Mitchener #ifdef KEY_EVENT
411144d93782SGreg Clayton         case KEY_EVENT:     return "We were interrupted by an event";
411227801f4fSBruce Mitchener #endif
411344d93782SGreg Clayton         case KEY_RETURN:    return "return";
411444d93782SGreg Clayton         case ' ':           return "space";
41155fdb09bbSGreg Clayton         case '\t':          return "tab";
411644d93782SGreg Clayton         case KEY_ESCAPE:    return "escape";
411744d93782SGreg Clayton         default:
411844d93782SGreg Clayton             if (isprint(ch))
411944d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "%c", ch);
412044d93782SGreg Clayton             else
412144d93782SGreg Clayton                 snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch);
412244d93782SGreg Clayton             return g_desc;
412344d93782SGreg Clayton     }
412444d93782SGreg Clayton     return NULL;
412544d93782SGreg Clayton }
412644d93782SGreg Clayton 
412744d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) :
412844d93782SGreg Clayton     m_text (),
412944d93782SGreg Clayton     m_first_visible_line (0)
413044d93782SGreg Clayton {
413144d93782SGreg Clayton     if (text && text[0])
413244d93782SGreg Clayton     {
413344d93782SGreg Clayton         m_text.SplitIntoLines(text);
413444d93782SGreg Clayton         m_text.AppendString("");
413544d93782SGreg Clayton     }
413644d93782SGreg Clayton     if (key_help_array)
413744d93782SGreg Clayton     {
413844d93782SGreg Clayton         for (KeyHelp *key = key_help_array; key->ch; ++key)
413944d93782SGreg Clayton         {
414044d93782SGreg Clayton             StreamString key_description;
414144d93782SGreg Clayton             key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description);
414244d93782SGreg Clayton             m_text.AppendString(std::move(key_description.GetString()));
414344d93782SGreg Clayton         }
414444d93782SGreg Clayton     }
414544d93782SGreg Clayton }
414644d93782SGreg Clayton 
4147315b6884SEugene Zelenko HelpDialogDelegate::~HelpDialogDelegate() = default;
414844d93782SGreg Clayton 
414944d93782SGreg Clayton bool
415044d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force)
415144d93782SGreg Clayton {
415244d93782SGreg Clayton     window.Erase();
415344d93782SGreg Clayton     const int window_height = window.GetHeight();
415444d93782SGreg Clayton     int x = 2;
415544d93782SGreg Clayton     int y = 1;
415644d93782SGreg Clayton     const int min_y = y;
415744d93782SGreg Clayton     const int max_y = window_height - 1 - y;
41583985c8c6SSaleem Abdulrasool     const size_t num_visible_lines = max_y - min_y + 1;
415944d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
416044d93782SGreg Clayton     const char *bottom_message;
416144d93782SGreg Clayton     if (num_lines <= num_visible_lines)
416244d93782SGreg Clayton         bottom_message = "Press any key to exit";
416344d93782SGreg Clayton     else
416444d93782SGreg Clayton         bottom_message = "Use arrows to scroll, any other key to exit";
416544d93782SGreg Clayton     window.DrawTitleBox(window.GetName(), bottom_message);
416644d93782SGreg Clayton     while (y <= max_y)
416744d93782SGreg Clayton     {
416844d93782SGreg Clayton         window.MoveCursor(x, y);
416944d93782SGreg Clayton         window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1);
417044d93782SGreg Clayton         ++y;
417144d93782SGreg Clayton     }
417244d93782SGreg Clayton     return true;
417344d93782SGreg Clayton }
417444d93782SGreg Clayton 
417544d93782SGreg Clayton HandleCharResult
417644d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key)
417744d93782SGreg Clayton {
417844d93782SGreg Clayton     bool done = false;
417944d93782SGreg Clayton     const size_t num_lines = m_text.GetSize();
418044d93782SGreg Clayton     const size_t num_visible_lines = window.GetHeight() - 2;
418144d93782SGreg Clayton 
418244d93782SGreg Clayton     if (num_lines <= num_visible_lines)
418344d93782SGreg Clayton     {
418444d93782SGreg Clayton         done = true;
418544d93782SGreg Clayton         // If we have all lines visible and don't need scrolling, then any
418644d93782SGreg Clayton         // key press will cause us to exit
418744d93782SGreg Clayton     }
418844d93782SGreg Clayton     else
418944d93782SGreg Clayton     {
419044d93782SGreg Clayton         switch (key)
419144d93782SGreg Clayton         {
419244d93782SGreg Clayton             case KEY_UP:
419344d93782SGreg Clayton                 if (m_first_visible_line > 0)
419444d93782SGreg Clayton                     --m_first_visible_line;
419544d93782SGreg Clayton                 break;
419644d93782SGreg Clayton 
419744d93782SGreg Clayton             case KEY_DOWN:
419844d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
419944d93782SGreg Clayton                     ++m_first_visible_line;
420044d93782SGreg Clayton                 break;
420144d93782SGreg Clayton 
420244d93782SGreg Clayton             case KEY_PPAGE:
420344d93782SGreg Clayton             case ',':
420444d93782SGreg Clayton                 if (m_first_visible_line > 0)
420544d93782SGreg Clayton                 {
42063985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines)
420744d93782SGreg Clayton                         m_first_visible_line -= num_visible_lines;
420844d93782SGreg Clayton                     else
420944d93782SGreg Clayton                         m_first_visible_line = 0;
421044d93782SGreg Clayton                 }
421144d93782SGreg Clayton                 break;
4212315b6884SEugene Zelenko 
421344d93782SGreg Clayton             case KEY_NPAGE:
421444d93782SGreg Clayton             case '.':
421544d93782SGreg Clayton                 if (m_first_visible_line + num_visible_lines < num_lines)
421644d93782SGreg Clayton                 {
421744d93782SGreg Clayton                     m_first_visible_line += num_visible_lines;
42183985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > num_lines)
421944d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
422044d93782SGreg Clayton                 }
422144d93782SGreg Clayton                 break;
4222315b6884SEugene Zelenko 
422344d93782SGreg Clayton             default:
422444d93782SGreg Clayton                 done = true;
422544d93782SGreg Clayton                 break;
422644d93782SGreg Clayton         }
422744d93782SGreg Clayton     }
422844d93782SGreg Clayton     if (done)
422944d93782SGreg Clayton         window.GetParent()->RemoveSubWindow(&window);
423044d93782SGreg Clayton     return eKeyHandled;
423144d93782SGreg Clayton }
423244d93782SGreg Clayton 
423344d93782SGreg Clayton class ApplicationDelegate :
423444d93782SGreg Clayton     public WindowDelegate,
423544d93782SGreg Clayton     public MenuDelegate
423644d93782SGreg Clayton {
423744d93782SGreg Clayton public:
423844d93782SGreg Clayton     enum {
423944d93782SGreg Clayton         eMenuID_LLDB = 1,
424044d93782SGreg Clayton         eMenuID_LLDBAbout,
424144d93782SGreg Clayton         eMenuID_LLDBExit,
424244d93782SGreg Clayton 
424344d93782SGreg Clayton         eMenuID_Target,
424444d93782SGreg Clayton         eMenuID_TargetCreate,
424544d93782SGreg Clayton         eMenuID_TargetDelete,
424644d93782SGreg Clayton 
424744d93782SGreg Clayton         eMenuID_Process,
424844d93782SGreg Clayton         eMenuID_ProcessAttach,
424944d93782SGreg Clayton         eMenuID_ProcessDetach,
425044d93782SGreg Clayton         eMenuID_ProcessLaunch,
425144d93782SGreg Clayton         eMenuID_ProcessContinue,
425244d93782SGreg Clayton         eMenuID_ProcessHalt,
425344d93782SGreg Clayton         eMenuID_ProcessKill,
425444d93782SGreg Clayton 
425544d93782SGreg Clayton         eMenuID_Thread,
425644d93782SGreg Clayton         eMenuID_ThreadStepIn,
425744d93782SGreg Clayton         eMenuID_ThreadStepOver,
425844d93782SGreg Clayton         eMenuID_ThreadStepOut,
425944d93782SGreg Clayton 
426044d93782SGreg Clayton         eMenuID_View,
426144d93782SGreg Clayton         eMenuID_ViewBacktrace,
426244d93782SGreg Clayton         eMenuID_ViewRegisters,
426344d93782SGreg Clayton         eMenuID_ViewSource,
426444d93782SGreg Clayton         eMenuID_ViewVariables,
426544d93782SGreg Clayton 
426644d93782SGreg Clayton         eMenuID_Help,
426744d93782SGreg Clayton         eMenuID_HelpGUIHelp
426844d93782SGreg Clayton     };
426944d93782SGreg Clayton 
427044d93782SGreg Clayton     ApplicationDelegate (Application &app, Debugger &debugger) :
427144d93782SGreg Clayton         WindowDelegate (),
427244d93782SGreg Clayton         MenuDelegate (),
427344d93782SGreg Clayton         m_app (app),
427444d93782SGreg Clayton         m_debugger (debugger)
427544d93782SGreg Clayton     {
427644d93782SGreg Clayton     }
427744d93782SGreg Clayton 
4278315b6884SEugene Zelenko     ~ApplicationDelegate() override = default;
4279bd5ae6b4SGreg Clayton 
4280bd5ae6b4SGreg Clayton     bool
4281bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
428244d93782SGreg Clayton     {
428344d93782SGreg Clayton         return false; // Drawing not handled, let standard window drawing happen
428444d93782SGreg Clayton     }
428544d93782SGreg Clayton 
4286bd5ae6b4SGreg Clayton     HandleCharResult
4287bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int key) override
428844d93782SGreg Clayton     {
42895fdb09bbSGreg Clayton         switch (key)
429044d93782SGreg Clayton         {
42915fdb09bbSGreg Clayton             case '\t':
429244d93782SGreg Clayton                 window.SelectNextWindowAsActive();
429344d93782SGreg Clayton                 return eKeyHandled;
42945fdb09bbSGreg Clayton 
42955fdb09bbSGreg Clayton             case 'h':
42965fdb09bbSGreg Clayton                 window.CreateHelpSubwindow();
42975fdb09bbSGreg Clayton                 return eKeyHandled;
42985fdb09bbSGreg Clayton 
42995fdb09bbSGreg Clayton             case KEY_ESCAPE:
43005fdb09bbSGreg Clayton                 return eQuitApplication;
43015fdb09bbSGreg Clayton 
43025fdb09bbSGreg Clayton             default:
43035fdb09bbSGreg Clayton                 break;
430444d93782SGreg Clayton         }
430544d93782SGreg Clayton         return eKeyNotHandled;
430644d93782SGreg Clayton     }
430744d93782SGreg Clayton 
4308bd5ae6b4SGreg Clayton     const char *
4309bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
43105fdb09bbSGreg Clayton     {
43115fdb09bbSGreg Clayton         return "Welcome to the LLDB curses GUI.\n\n"
43125fdb09bbSGreg Clayton         "Press the TAB key to change the selected view.\n"
43135fdb09bbSGreg Clayton         "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n"
43145fdb09bbSGreg Clayton         "Common key bindings for all views:";
43155fdb09bbSGreg Clayton     }
43165fdb09bbSGreg Clayton 
4317bd5ae6b4SGreg Clayton     KeyHelp *
4318bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
43195fdb09bbSGreg Clayton     {
43205fdb09bbSGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
43215fdb09bbSGreg Clayton             { '\t', "Select next view" },
43225fdb09bbSGreg Clayton             { 'h', "Show help dialog with view specific key bindings" },
43235fdb09bbSGreg Clayton             { ',', "Page up" },
43245fdb09bbSGreg Clayton             { '.', "Page down" },
43255fdb09bbSGreg Clayton             { KEY_UP, "Select previous" },
43265fdb09bbSGreg Clayton             { KEY_DOWN, "Select next" },
43275fdb09bbSGreg Clayton             { KEY_LEFT, "Unexpand or select parent" },
43285fdb09bbSGreg Clayton             { KEY_RIGHT, "Expand" },
43295fdb09bbSGreg Clayton             { KEY_PPAGE, "Page up" },
43305fdb09bbSGreg Clayton             { KEY_NPAGE, "Page down" },
43315fdb09bbSGreg Clayton             { '\0', NULL }
43325fdb09bbSGreg Clayton         };
43335fdb09bbSGreg Clayton         return g_source_view_key_help;
43345fdb09bbSGreg Clayton     }
43355fdb09bbSGreg Clayton 
4336bd5ae6b4SGreg Clayton     MenuActionResult
4337bd5ae6b4SGreg Clayton     MenuDelegateAction (Menu &menu) override
433844d93782SGreg Clayton     {
433944d93782SGreg Clayton         switch (menu.GetIdentifier())
434044d93782SGreg Clayton         {
434144d93782SGreg Clayton             case eMenuID_ThreadStepIn:
434244d93782SGreg Clayton                 {
434344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
434444d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
434544d93782SGreg Clayton                     {
434644d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
434744d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
43484b4b2478SJim Ingham                             exe_ctx.GetThreadRef().StepIn(true);
434944d93782SGreg Clayton                     }
435044d93782SGreg Clayton                 }
435144d93782SGreg Clayton                 return MenuActionResult::Handled;
435244d93782SGreg Clayton 
435344d93782SGreg Clayton             case eMenuID_ThreadStepOut:
435444d93782SGreg Clayton                 {
435544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
435644d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
435744d93782SGreg Clayton                     {
435844d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
435944d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
436044d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOut();
436144d93782SGreg Clayton                     }
436244d93782SGreg Clayton                 }
436344d93782SGreg Clayton                 return MenuActionResult::Handled;
436444d93782SGreg Clayton 
436544d93782SGreg Clayton             case eMenuID_ThreadStepOver:
436644d93782SGreg Clayton                 {
436744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
436844d93782SGreg Clayton                     if (exe_ctx.HasThreadScope())
436944d93782SGreg Clayton                     {
437044d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
437144d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
437244d93782SGreg Clayton                             exe_ctx.GetThreadRef().StepOver(true);
437344d93782SGreg Clayton                     }
437444d93782SGreg Clayton                 }
437544d93782SGreg Clayton                 return MenuActionResult::Handled;
437644d93782SGreg Clayton 
437744d93782SGreg Clayton             case eMenuID_ProcessContinue:
437844d93782SGreg Clayton                 {
437944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
438044d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
438144d93782SGreg Clayton                     {
438244d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
438344d93782SGreg Clayton                         if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
438444d93782SGreg Clayton                             process->Resume();
438544d93782SGreg Clayton                     }
438644d93782SGreg Clayton                 }
438744d93782SGreg Clayton                 return MenuActionResult::Handled;
438844d93782SGreg Clayton 
438944d93782SGreg Clayton             case eMenuID_ProcessKill:
439044d93782SGreg Clayton                 {
439144d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
439244d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
439344d93782SGreg Clayton                     {
439444d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
439544d93782SGreg Clayton                         if (process && process->IsAlive())
4396ede3193bSJason Molenda                             process->Destroy(false);
439744d93782SGreg Clayton                     }
439844d93782SGreg Clayton                 }
439944d93782SGreg Clayton                 return MenuActionResult::Handled;
440044d93782SGreg Clayton 
440144d93782SGreg Clayton             case eMenuID_ProcessHalt:
440244d93782SGreg Clayton                 {
440344d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
440444d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
440544d93782SGreg Clayton                     {
440644d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
440744d93782SGreg Clayton                         if (process && process->IsAlive())
440844d93782SGreg Clayton                             process->Halt();
440944d93782SGreg Clayton                     }
441044d93782SGreg Clayton                 }
441144d93782SGreg Clayton                 return MenuActionResult::Handled;
441244d93782SGreg Clayton 
441344d93782SGreg Clayton             case eMenuID_ProcessDetach:
441444d93782SGreg Clayton                 {
441544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
441644d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
441744d93782SGreg Clayton                     {
441844d93782SGreg Clayton                         Process *process = exe_ctx.GetProcessPtr();
441944d93782SGreg Clayton                         if (process && process->IsAlive())
442044d93782SGreg Clayton                             process->Detach(false);
442144d93782SGreg Clayton                     }
442244d93782SGreg Clayton                 }
442344d93782SGreg Clayton                 return MenuActionResult::Handled;
442444d93782SGreg Clayton 
442544d93782SGreg Clayton             case eMenuID_Process:
442644d93782SGreg Clayton                 {
442744d93782SGreg Clayton                     // Populate the menu with all of the threads if the process is stopped when
442844d93782SGreg Clayton                     // the Process menu gets selected and is about to display its submenu.
442944d93782SGreg Clayton                     Menus &submenus = menu.GetSubmenus();
443044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
443144d93782SGreg Clayton                     Process *process = exe_ctx.GetProcessPtr();
443244d93782SGreg Clayton                     if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true))
443344d93782SGreg Clayton                     {
443444d93782SGreg Clayton                         if (submenus.size() == 7)
443544d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
443644d93782SGreg Clayton                         else if (submenus.size() > 8)
443744d93782SGreg Clayton                             submenus.erase (submenus.begin() + 8, submenus.end());
443844d93782SGreg Clayton 
443944d93782SGreg Clayton                         ThreadList &threads = process->GetThreadList();
444044d93782SGreg Clayton                         Mutex::Locker locker (threads.GetMutex());
444144d93782SGreg Clayton                         size_t num_threads = threads.GetSize();
444244d93782SGreg Clayton                         for (size_t i=0; i<num_threads; ++i)
444344d93782SGreg Clayton                         {
444444d93782SGreg Clayton                             ThreadSP thread_sp = threads.GetThreadAtIndex(i);
444544d93782SGreg Clayton                             char menu_char = '\0';
444644d93782SGreg Clayton                             if (i < 9)
444744d93782SGreg Clayton                                 menu_char = '1' + i;
444844d93782SGreg Clayton                             StreamString thread_menu_title;
444944d93782SGreg Clayton                             thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID());
445044d93782SGreg Clayton                             const char *thread_name = thread_sp->GetName();
445144d93782SGreg Clayton                             if (thread_name && thread_name[0])
445244d93782SGreg Clayton                                 thread_menu_title.Printf (" %s", thread_name);
445344d93782SGreg Clayton                             else
445444d93782SGreg Clayton                             {
445544d93782SGreg Clayton                                 const char *queue_name = thread_sp->GetQueueName();
445644d93782SGreg Clayton                                 if (queue_name && queue_name[0])
445744d93782SGreg Clayton                                     thread_menu_title.Printf (" %s", queue_name);
445844d93782SGreg Clayton                             }
445944d93782SGreg Clayton                             menu.AddSubmenu (MenuSP (new Menu(thread_menu_title.GetString().c_str(), NULL, menu_char, thread_sp->GetID())));
446044d93782SGreg Clayton                         }
446144d93782SGreg Clayton                     }
446244d93782SGreg Clayton                     else if (submenus.size() > 7)
446344d93782SGreg Clayton                     {
446444d93782SGreg Clayton                         // Remove the separator and any other thread submenu items
446544d93782SGreg Clayton                         // that were previously added
446644d93782SGreg Clayton                         submenus.erase (submenus.begin() + 7, submenus.end());
446744d93782SGreg Clayton                     }
446844d93782SGreg Clayton                     // Since we are adding and removing items we need to recalculate the name lengths
446944d93782SGreg Clayton                     menu.RecalculateNameLengths();
447044d93782SGreg Clayton                 }
447144d93782SGreg Clayton                 return MenuActionResult::Handled;
447244d93782SGreg Clayton 
447344d93782SGreg Clayton             case eMenuID_ViewVariables:
447444d93782SGreg Clayton                 {
447544d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
447644d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
447744d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
447844d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
447944d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
448044d93782SGreg Clayton 
448144d93782SGreg Clayton                     if (variables_window_sp)
448244d93782SGreg Clayton                     {
448344d93782SGreg Clayton                         const Rect variables_bounds = variables_window_sp->GetBounds();
448444d93782SGreg Clayton 
448544d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(variables_window_sp.get());
448644d93782SGreg Clayton 
448744d93782SGreg Clayton                         if (registers_window_sp)
448844d93782SGreg Clayton                         {
448944d93782SGreg Clayton                             // We have a registers window, so give all the area back to the registers window
449044d93782SGreg Clayton                             Rect registers_bounds = variables_bounds;
449144d93782SGreg Clayton                             registers_bounds.size.width = source_bounds.size.width;
449244d93782SGreg Clayton                             registers_window_sp->SetBounds(registers_bounds);
449344d93782SGreg Clayton                         }
449444d93782SGreg Clayton                         else
449544d93782SGreg Clayton                         {
449644d93782SGreg Clayton                             // We have no registers window showing so give the bottom
449744d93782SGreg Clayton                             // area back to the source view
449844d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
449944d93782SGreg Clayton                                                       source_bounds.size.height + variables_bounds.size.height);
450044d93782SGreg Clayton                         }
450144d93782SGreg Clayton                     }
450244d93782SGreg Clayton                     else
450344d93782SGreg Clayton                     {
450444d93782SGreg Clayton                         Rect new_variables_rect;
450544d93782SGreg Clayton                         if (registers_window_sp)
450644d93782SGreg Clayton                         {
450744d93782SGreg Clayton                             // We have a registers window so split the area of the registers
450844d93782SGreg Clayton                             // window into two columns where the left hand side will be the
450944d93782SGreg Clayton                             // variables and the right hand side will be the registers
451044d93782SGreg Clayton                             const Rect variables_bounds = registers_window_sp->GetBounds();
451144d93782SGreg Clayton                             Rect new_registers_rect;
451244d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect);
451344d93782SGreg Clayton                             registers_window_sp->SetBounds (new_registers_rect);
451444d93782SGreg Clayton                         }
451544d93782SGreg Clayton                         else
451644d93782SGreg Clayton                         {
451744d93782SGreg Clayton                             // No variables window, grab the bottom part of the source window
451844d93782SGreg Clayton                             Rect new_source_rect;
451944d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect);
452044d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
452144d93782SGreg Clayton                         }
452244d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables",
452344d93782SGreg Clayton                                                                                   new_variables_rect,
452444d93782SGreg Clayton                                                                                   false);
452544d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
452644d93782SGreg Clayton                     }
452744d93782SGreg Clayton                     touchwin(stdscr);
452844d93782SGreg Clayton                 }
452944d93782SGreg Clayton                 return MenuActionResult::Handled;
453044d93782SGreg Clayton 
453144d93782SGreg Clayton             case eMenuID_ViewRegisters:
453244d93782SGreg Clayton                 {
453344d93782SGreg Clayton                     WindowSP main_window_sp = m_app.GetMainWindow();
453444d93782SGreg Clayton                     WindowSP source_window_sp = main_window_sp->FindSubWindow("Source");
453544d93782SGreg Clayton                     WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables");
453644d93782SGreg Clayton                     WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers");
453744d93782SGreg Clayton                     const Rect source_bounds = source_window_sp->GetBounds();
453844d93782SGreg Clayton 
453944d93782SGreg Clayton                     if (registers_window_sp)
454044d93782SGreg Clayton                     {
454144d93782SGreg Clayton                         if (variables_window_sp)
454244d93782SGreg Clayton                         {
454344d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
454444d93782SGreg Clayton 
454544d93782SGreg Clayton                             // We have a variables window, so give all the area back to the variables window
454644d93782SGreg Clayton                             variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(),
454744d93782SGreg Clayton                                                          variables_bounds.size.height);
454844d93782SGreg Clayton                         }
454944d93782SGreg Clayton                         else
455044d93782SGreg Clayton                         {
455144d93782SGreg Clayton                             // We have no variables window showing so give the bottom
455244d93782SGreg Clayton                             // area back to the source view
455344d93782SGreg Clayton                             source_window_sp->Resize (source_bounds.size.width,
455444d93782SGreg Clayton                                                       source_bounds.size.height + registers_window_sp->GetHeight());
455544d93782SGreg Clayton                         }
455644d93782SGreg Clayton                         main_window_sp->RemoveSubWindow(registers_window_sp.get());
455744d93782SGreg Clayton                     }
455844d93782SGreg Clayton                     else
455944d93782SGreg Clayton                     {
456044d93782SGreg Clayton                         Rect new_regs_rect;
456144d93782SGreg Clayton                         if (variables_window_sp)
456244d93782SGreg Clayton                         {
456344d93782SGreg Clayton                             // We have a variables window, split it into two columns
456444d93782SGreg Clayton                             // where the left hand side will be the variables and the
456544d93782SGreg Clayton                             // right hand side will be the registers
456644d93782SGreg Clayton                             const Rect variables_bounds = variables_window_sp->GetBounds();
456744d93782SGreg Clayton                             Rect new_vars_rect;
456844d93782SGreg Clayton                             variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect);
456944d93782SGreg Clayton                             variables_window_sp->SetBounds (new_vars_rect);
457044d93782SGreg Clayton                         }
457144d93782SGreg Clayton                         else
457244d93782SGreg Clayton                         {
457344d93782SGreg Clayton                             // No registers window, grab the bottom part of the source window
457444d93782SGreg Clayton                             Rect new_source_rect;
457544d93782SGreg Clayton                             source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect);
457644d93782SGreg Clayton                             source_window_sp->SetBounds (new_source_rect);
457744d93782SGreg Clayton                         }
457844d93782SGreg Clayton                         WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers",
457944d93782SGreg Clayton                                                                                   new_regs_rect,
458044d93782SGreg Clayton                                                                                   false);
458144d93782SGreg Clayton                         new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger)));
458244d93782SGreg Clayton                     }
458344d93782SGreg Clayton                     touchwin(stdscr);
458444d93782SGreg Clayton                 }
458544d93782SGreg Clayton                 return MenuActionResult::Handled;
458644d93782SGreg Clayton 
458744d93782SGreg Clayton             case eMenuID_HelpGUIHelp:
45885fdb09bbSGreg Clayton                 m_app.GetMainWindow ()->CreateHelpSubwindow();
458944d93782SGreg Clayton                 return MenuActionResult::Handled;
459044d93782SGreg Clayton 
459144d93782SGreg Clayton             default:
459244d93782SGreg Clayton                 break;
459344d93782SGreg Clayton         }
459444d93782SGreg Clayton 
459544d93782SGreg Clayton         return MenuActionResult::NotHandled;
459644d93782SGreg Clayton     }
459744d93782SGreg Clayton protected:
459844d93782SGreg Clayton     Application &m_app;
459944d93782SGreg Clayton     Debugger &m_debugger;
460044d93782SGreg Clayton };
460144d93782SGreg Clayton 
460244d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate
460344d93782SGreg Clayton {
460444d93782SGreg Clayton public:
460544d93782SGreg Clayton     StatusBarWindowDelegate (Debugger &debugger) :
460644d93782SGreg Clayton         m_debugger (debugger)
460744d93782SGreg Clayton     {
4608554f68d3SGreg Clayton         FormatEntity::Parse("Thread: ${thread.id%tid}",
4609554f68d3SGreg Clayton                             m_format);
461044d93782SGreg Clayton     }
461144d93782SGreg Clayton 
4612315b6884SEugene Zelenko     ~StatusBarWindowDelegate() override = default;
4613bd5ae6b4SGreg Clayton 
4614bd5ae6b4SGreg Clayton     bool
4615bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
461644d93782SGreg Clayton     {
461744d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
461844d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
461944d93782SGreg Clayton         Thread *thread = exe_ctx.GetThreadPtr();
462044d93782SGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
462144d93782SGreg Clayton         window.Erase();
462244d93782SGreg Clayton         window.SetBackground(2);
462344d93782SGreg Clayton         window.MoveCursor (0, 0);
462444d93782SGreg Clayton         if (process)
462544d93782SGreg Clayton         {
462644d93782SGreg Clayton             const StateType state = process->GetState();
462744d93782SGreg Clayton             window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state));
462844d93782SGreg Clayton 
462944d93782SGreg Clayton             if (StateIsStoppedState(state, true))
463044d93782SGreg Clayton             {
46315b031ebcSEd Maste                 StreamString strm;
4632554f68d3SGreg Clayton                 if (thread && FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false))
46335b031ebcSEd Maste                 {
463444d93782SGreg Clayton                     window.MoveCursor (40, 0);
46355b031ebcSEd Maste                     window.PutCStringTruncated(strm.GetString().c_str(), 1);
46365b031ebcSEd Maste                 }
463744d93782SGreg Clayton 
463844d93782SGreg Clayton                 window.MoveCursor (60, 0);
463944d93782SGreg Clayton                 if (frame)
464044d93782SGreg Clayton                     window.Printf ("Frame: %3u  PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr()));
464144d93782SGreg Clayton             }
464244d93782SGreg Clayton             else if (state == eStateExited)
464344d93782SGreg Clayton             {
464444d93782SGreg Clayton                 const char *exit_desc = process->GetExitDescription();
464544d93782SGreg Clayton                 const int exit_status = process->GetExitStatus();
464644d93782SGreg Clayton                 if (exit_desc && exit_desc[0])
464744d93782SGreg Clayton                     window.Printf (" with status = %i (%s)", exit_status, exit_desc);
464844d93782SGreg Clayton                 else
464944d93782SGreg Clayton                     window.Printf (" with status = %i", exit_status);
465044d93782SGreg Clayton             }
465144d93782SGreg Clayton         }
465244d93782SGreg Clayton         window.DeferredRefresh();
465344d93782SGreg Clayton         return true;
465444d93782SGreg Clayton     }
465544d93782SGreg Clayton 
465644d93782SGreg Clayton protected:
465744d93782SGreg Clayton     Debugger &m_debugger;
4658554f68d3SGreg Clayton     FormatEntity::Entry m_format;
465944d93782SGreg Clayton };
466044d93782SGreg Clayton 
466144d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate
466244d93782SGreg Clayton {
466344d93782SGreg Clayton public:
466444d93782SGreg Clayton     SourceFileWindowDelegate (Debugger &debugger) :
466544d93782SGreg Clayton         WindowDelegate (),
466644d93782SGreg Clayton         m_debugger (debugger),
466744d93782SGreg Clayton         m_sc (),
466844d93782SGreg Clayton         m_file_sp (),
466944d93782SGreg Clayton         m_disassembly_scope (NULL),
467044d93782SGreg Clayton         m_disassembly_sp (),
467144d93782SGreg Clayton         m_disassembly_range (),
4672ec990867SGreg Clayton         m_title (),
467344d93782SGreg Clayton         m_line_width (4),
467444d93782SGreg Clayton         m_selected_line (0),
467544d93782SGreg Clayton         m_pc_line (0),
467644d93782SGreg Clayton         m_stop_id (0),
467744d93782SGreg Clayton         m_frame_idx (UINT32_MAX),
467844d93782SGreg Clayton         m_first_visible_line (0),
467944d93782SGreg Clayton         m_min_x (0),
468044d93782SGreg Clayton         m_min_y (0),
468144d93782SGreg Clayton         m_max_x (0),
468244d93782SGreg Clayton         m_max_y (0)
468344d93782SGreg Clayton     {
468444d93782SGreg Clayton     }
468544d93782SGreg Clayton 
4686315b6884SEugene Zelenko     ~SourceFileWindowDelegate() override = default;
468744d93782SGreg Clayton 
468844d93782SGreg Clayton     void
468944d93782SGreg Clayton     Update (const SymbolContext &sc)
469044d93782SGreg Clayton     {
469144d93782SGreg Clayton         m_sc = sc;
469244d93782SGreg Clayton     }
469344d93782SGreg Clayton 
469444d93782SGreg Clayton     uint32_t
469544d93782SGreg Clayton     NumVisibleLines () const
469644d93782SGreg Clayton     {
469744d93782SGreg Clayton         return m_max_y - m_min_y;
469844d93782SGreg Clayton     }
469944d93782SGreg Clayton 
4700bd5ae6b4SGreg Clayton     const char *
4701bd5ae6b4SGreg Clayton     WindowDelegateGetHelpText () override
470244d93782SGreg Clayton     {
470344d93782SGreg Clayton         return "Source/Disassembly window keyboard shortcuts:";
470444d93782SGreg Clayton     }
470544d93782SGreg Clayton 
4706bd5ae6b4SGreg Clayton     KeyHelp *
4707bd5ae6b4SGreg Clayton     WindowDelegateGetKeyHelp () override
470844d93782SGreg Clayton     {
470944d93782SGreg Clayton         static curses::KeyHelp g_source_view_key_help[] = {
471044d93782SGreg Clayton             { KEY_RETURN, "Run to selected line with one shot breakpoint" },
471144d93782SGreg Clayton             { KEY_UP, "Select previous source line" },
471244d93782SGreg Clayton             { KEY_DOWN, "Select next source line" },
471344d93782SGreg Clayton             { KEY_PPAGE, "Page up" },
471444d93782SGreg Clayton             { KEY_NPAGE, "Page down" },
471544d93782SGreg Clayton             { 'b', "Set breakpoint on selected source/disassembly line" },
471644d93782SGreg Clayton             { 'c', "Continue process" },
471744d93782SGreg Clayton             { 'd', "Detach and resume process" },
471844d93782SGreg Clayton             { 'D', "Detach with process suspended" },
471944d93782SGreg Clayton             { 'h', "Show help dialog" },
472044d93782SGreg Clayton             { 'k', "Kill process" },
472144d93782SGreg Clayton             { 'n', "Step over (source line)" },
472244d93782SGreg Clayton             { 'N', "Step over (single instruction)" },
472344d93782SGreg Clayton             { 'o', "Step out" },
472444d93782SGreg Clayton             { 's', "Step in (source line)" },
472544d93782SGreg Clayton             { 'S', "Step in (single instruction)" },
472644d93782SGreg Clayton             { ',', "Page up" },
472744d93782SGreg Clayton             { '.', "Page down" },
472844d93782SGreg Clayton             { '\0', NULL }
472944d93782SGreg Clayton         };
473044d93782SGreg Clayton         return g_source_view_key_help;
473144d93782SGreg Clayton     }
473244d93782SGreg Clayton 
4733bd5ae6b4SGreg Clayton     bool
4734bd5ae6b4SGreg Clayton     WindowDelegateDraw (Window &window, bool force) override
473544d93782SGreg Clayton     {
473644d93782SGreg Clayton         ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
473744d93782SGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
473844d93782SGreg Clayton         Thread *thread = NULL;
473944d93782SGreg Clayton 
474044d93782SGreg Clayton         bool update_location = false;
474144d93782SGreg Clayton         if (process)
474244d93782SGreg Clayton         {
474344d93782SGreg Clayton             StateType state = process->GetState();
474444d93782SGreg Clayton             if (StateIsStoppedState(state, true))
474544d93782SGreg Clayton             {
474644d93782SGreg Clayton                 // We are stopped, so it is ok to
474744d93782SGreg Clayton                 update_location = true;
474844d93782SGreg Clayton             }
474944d93782SGreg Clayton         }
475044d93782SGreg Clayton 
475144d93782SGreg Clayton         m_min_x = 1;
4752ec990867SGreg Clayton         m_min_y = 2;
475344d93782SGreg Clayton         m_max_x = window.GetMaxX()-1;
475444d93782SGreg Clayton         m_max_y = window.GetMaxY()-1;
475544d93782SGreg Clayton 
475644d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
475744d93782SGreg Clayton         StackFrameSP frame_sp;
475844d93782SGreg Clayton         bool set_selected_line_to_pc = false;
475944d93782SGreg Clayton 
476044d93782SGreg Clayton         if (update_location)
476144d93782SGreg Clayton         {
476244d93782SGreg Clayton             const bool process_alive = process ? process->IsAlive() : false;
476344d93782SGreg Clayton             bool thread_changed = false;
476444d93782SGreg Clayton             if (process_alive)
476544d93782SGreg Clayton             {
476644d93782SGreg Clayton                 thread = exe_ctx.GetThreadPtr();
476744d93782SGreg Clayton                 if (thread)
476844d93782SGreg Clayton                 {
476944d93782SGreg Clayton                     frame_sp = thread->GetSelectedFrame();
477044d93782SGreg Clayton                     auto tid = thread->GetID();
477144d93782SGreg Clayton                     thread_changed = tid != m_tid;
477244d93782SGreg Clayton                     m_tid = tid;
477344d93782SGreg Clayton                 }
477444d93782SGreg Clayton                 else
477544d93782SGreg Clayton                 {
477644d93782SGreg Clayton                     if (m_tid != LLDB_INVALID_THREAD_ID)
477744d93782SGreg Clayton                     {
477844d93782SGreg Clayton                         thread_changed = true;
477944d93782SGreg Clayton                         m_tid = LLDB_INVALID_THREAD_ID;
478044d93782SGreg Clayton                     }
478144d93782SGreg Clayton                 }
478244d93782SGreg Clayton             }
478344d93782SGreg Clayton             const uint32_t stop_id = process ? process->GetStopID() : 0;
478444d93782SGreg Clayton             const bool stop_id_changed = stop_id != m_stop_id;
478544d93782SGreg Clayton             bool frame_changed = false;
478644d93782SGreg Clayton             m_stop_id = stop_id;
4787ec990867SGreg Clayton             m_title.Clear();
478844d93782SGreg Clayton             if (frame_sp)
478944d93782SGreg Clayton             {
479044d93782SGreg Clayton                 m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything);
4791ec990867SGreg Clayton                 if (m_sc.module_sp)
4792ec990867SGreg Clayton                 {
4793ec990867SGreg Clayton                     m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString());
4794ec990867SGreg Clayton                     ConstString func_name = m_sc.GetFunctionName();
4795ec990867SGreg Clayton                     if (func_name)
4796ec990867SGreg Clayton                         m_title.Printf("`%s", func_name.GetCString());
4797ec990867SGreg Clayton                 }
479844d93782SGreg Clayton                 const uint32_t frame_idx = frame_sp->GetFrameIndex();
479944d93782SGreg Clayton                 frame_changed = frame_idx != m_frame_idx;
480044d93782SGreg Clayton                 m_frame_idx = frame_idx;
480144d93782SGreg Clayton             }
480244d93782SGreg Clayton             else
480344d93782SGreg Clayton             {
480444d93782SGreg Clayton                 m_sc.Clear(true);
480544d93782SGreg Clayton                 frame_changed = m_frame_idx != UINT32_MAX;
480644d93782SGreg Clayton                 m_frame_idx = UINT32_MAX;
480744d93782SGreg Clayton             }
480844d93782SGreg Clayton 
480944d93782SGreg Clayton             const bool context_changed = thread_changed || frame_changed || stop_id_changed;
481044d93782SGreg Clayton 
481144d93782SGreg Clayton             if (process_alive)
481244d93782SGreg Clayton             {
481344d93782SGreg Clayton                 if (m_sc.line_entry.IsValid())
481444d93782SGreg Clayton                 {
481544d93782SGreg Clayton                     m_pc_line = m_sc.line_entry.line;
481644d93782SGreg Clayton                     if (m_pc_line != UINT32_MAX)
481744d93782SGreg Clayton                         --m_pc_line; // Convert to zero based line number...
481844d93782SGreg Clayton                     // Update the selected line if the stop ID changed...
481944d93782SGreg Clayton                     if (context_changed)
482044d93782SGreg Clayton                         m_selected_line = m_pc_line;
482144d93782SGreg Clayton 
482244d93782SGreg Clayton                     if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file))
482344d93782SGreg Clayton                     {
482444d93782SGreg Clayton                         // Same file, nothing to do, we should either have the
482544d93782SGreg Clayton                         // lines or not (source file missing)
48263985c8c6SSaleem Abdulrasool                         if (m_selected_line >= static_cast<size_t>(m_first_visible_line))
482744d93782SGreg Clayton                         {
482844d93782SGreg Clayton                             if (m_selected_line >= m_first_visible_line + num_visible_lines)
482944d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
483044d93782SGreg Clayton                         }
483144d93782SGreg Clayton                         else
483244d93782SGreg Clayton                         {
483344d93782SGreg Clayton                             if (m_selected_line > 10)
483444d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
483544d93782SGreg Clayton                             else
483644d93782SGreg Clayton                                 m_first_visible_line = 0;
483744d93782SGreg Clayton                         }
483844d93782SGreg Clayton                     }
483944d93782SGreg Clayton                     else
484044d93782SGreg Clayton                     {
484144d93782SGreg Clayton                         // File changed, set selected line to the line with the PC
484244d93782SGreg Clayton                         m_selected_line = m_pc_line;
484344d93782SGreg Clayton                         m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file);
484444d93782SGreg Clayton                         if (m_file_sp)
484544d93782SGreg Clayton                         {
484644d93782SGreg Clayton                             const size_t num_lines = m_file_sp->GetNumLines();
484744d93782SGreg Clayton                             int m_line_width = 1;
484844d93782SGreg Clayton                             for (size_t n = num_lines; n >= 10; n = n / 10)
484944d93782SGreg Clayton                                 ++m_line_width;
485044d93782SGreg Clayton 
485144d93782SGreg Clayton                             snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width);
485244d93782SGreg Clayton                             if (num_lines < num_visible_lines || m_selected_line < num_visible_lines)
485344d93782SGreg Clayton                                 m_first_visible_line = 0;
485444d93782SGreg Clayton                             else
485544d93782SGreg Clayton                                 m_first_visible_line = m_selected_line - 10;
485644d93782SGreg Clayton                         }
485744d93782SGreg Clayton                     }
485844d93782SGreg Clayton                 }
485944d93782SGreg Clayton                 else
486044d93782SGreg Clayton                 {
486144d93782SGreg Clayton                     m_file_sp.reset();
486244d93782SGreg Clayton                 }
486344d93782SGreg Clayton 
486444d93782SGreg Clayton                 if (!m_file_sp || m_file_sp->GetNumLines() == 0)
486544d93782SGreg Clayton                 {
486644d93782SGreg Clayton                     // Show disassembly
486744d93782SGreg Clayton                     bool prefer_file_cache = false;
486844d93782SGreg Clayton                     if (m_sc.function)
486944d93782SGreg Clayton                     {
487044d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.function)
487144d93782SGreg Clayton                         {
487244d93782SGreg Clayton                             m_disassembly_scope = m_sc.function;
487344d93782SGreg Clayton                             m_disassembly_sp = m_sc.function->GetInstructions (exe_ctx, NULL, prefer_file_cache);
487444d93782SGreg Clayton                             if (m_disassembly_sp)
487544d93782SGreg Clayton                             {
487644d93782SGreg Clayton                                 set_selected_line_to_pc = true;
487744d93782SGreg Clayton                                 m_disassembly_range = m_sc.function->GetAddressRange();
487844d93782SGreg Clayton                             }
487944d93782SGreg Clayton                             else
488044d93782SGreg Clayton                             {
488144d93782SGreg Clayton                                 m_disassembly_range.Clear();
488244d93782SGreg Clayton                             }
488344d93782SGreg Clayton                         }
488444d93782SGreg Clayton                         else
488544d93782SGreg Clayton                         {
488644d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
488744d93782SGreg Clayton                         }
488844d93782SGreg Clayton                     }
488944d93782SGreg Clayton                     else if (m_sc.symbol)
489044d93782SGreg Clayton                     {
489144d93782SGreg Clayton                         if (m_disassembly_scope != m_sc.symbol)
489244d93782SGreg Clayton                         {
489344d93782SGreg Clayton                             m_disassembly_scope = m_sc.symbol;
489444d93782SGreg Clayton                             m_disassembly_sp = m_sc.symbol->GetInstructions (exe_ctx, NULL, prefer_file_cache);
489544d93782SGreg Clayton                             if (m_disassembly_sp)
489644d93782SGreg Clayton                             {
489744d93782SGreg Clayton                                 set_selected_line_to_pc = true;
489844d93782SGreg Clayton                                 m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress();
489944d93782SGreg Clayton                                 m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize());
490044d93782SGreg Clayton                             }
490144d93782SGreg Clayton                             else
490244d93782SGreg Clayton                             {
490344d93782SGreg Clayton                                 m_disassembly_range.Clear();
490444d93782SGreg Clayton                             }
490544d93782SGreg Clayton                         }
490644d93782SGreg Clayton                         else
490744d93782SGreg Clayton                         {
490844d93782SGreg Clayton                             set_selected_line_to_pc = context_changed;
490944d93782SGreg Clayton                         }
491044d93782SGreg Clayton                     }
491144d93782SGreg Clayton                 }
491244d93782SGreg Clayton             }
491344d93782SGreg Clayton             else
491444d93782SGreg Clayton             {
491544d93782SGreg Clayton                 m_pc_line = UINT32_MAX;
491644d93782SGreg Clayton             }
491744d93782SGreg Clayton         }
491844d93782SGreg Clayton 
4919ec990867SGreg Clayton         const int window_width = window.GetWidth();
492044d93782SGreg Clayton         window.Erase();
492144d93782SGreg Clayton         window.DrawTitleBox ("Sources");
4922ec990867SGreg Clayton         if (!m_title.GetString().empty())
4923ec990867SGreg Clayton         {
4924ec990867SGreg Clayton             window.AttributeOn(A_REVERSE);
4925ec990867SGreg Clayton             window.MoveCursor(1, 1);
4926ec990867SGreg Clayton             window.PutChar(' ');
4927ec990867SGreg Clayton             window.PutCStringTruncated(m_title.GetString().c_str(), 1);
4928ec990867SGreg Clayton             int x = window.GetCursorX();
4929ec990867SGreg Clayton             if (x < window_width - 1)
4930ec990867SGreg Clayton             {
4931ec990867SGreg Clayton                 window.Printf ("%*s", window_width - x - 1, "");
4932ec990867SGreg Clayton             }
4933ec990867SGreg Clayton             window.AttributeOff(A_REVERSE);
4934ec990867SGreg Clayton         }
493544d93782SGreg Clayton 
493644d93782SGreg Clayton         Target *target = exe_ctx.GetTargetPtr();
493744d93782SGreg Clayton         const size_t num_source_lines = GetNumSourceLines();
493844d93782SGreg Clayton         if (num_source_lines > 0)
493944d93782SGreg Clayton         {
494044d93782SGreg Clayton             // Display source
494144d93782SGreg Clayton             BreakpointLines bp_lines;
494244d93782SGreg Clayton             if (target)
494344d93782SGreg Clayton             {
494444d93782SGreg Clayton                 BreakpointList &bp_list = target->GetBreakpointList();
494544d93782SGreg Clayton                 const size_t num_bps = bp_list.GetSize();
494644d93782SGreg Clayton                 for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
494744d93782SGreg Clayton                 {
494844d93782SGreg Clayton                     BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
494944d93782SGreg Clayton                     const size_t num_bps_locs = bp_sp->GetNumLocations();
495044d93782SGreg Clayton                     for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
495144d93782SGreg Clayton                     {
495244d93782SGreg Clayton                         BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
495344d93782SGreg Clayton                         LineEntry bp_loc_line_entry;
495444d93782SGreg Clayton                         if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry))
495544d93782SGreg Clayton                         {
495644d93782SGreg Clayton                             if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file)
495744d93782SGreg Clayton                             {
495844d93782SGreg Clayton                                 bp_lines.insert(bp_loc_line_entry.line);
495944d93782SGreg Clayton                             }
496044d93782SGreg Clayton                         }
496144d93782SGreg Clayton                     }
496244d93782SGreg Clayton                 }
496344d93782SGreg Clayton             }
496444d93782SGreg Clayton 
496544d93782SGreg Clayton             const attr_t selected_highlight_attr = A_REVERSE;
496644d93782SGreg Clayton             const attr_t pc_highlight_attr = COLOR_PAIR(1);
496744d93782SGreg Clayton 
49683985c8c6SSaleem Abdulrasool             for (size_t i=0; i<num_visible_lines; ++i)
496944d93782SGreg Clayton             {
497044d93782SGreg Clayton                 const uint32_t curr_line = m_first_visible_line + i;
497144d93782SGreg Clayton                 if (curr_line < num_source_lines)
497244d93782SGreg Clayton                 {
4973ec990867SGreg Clayton                     const int line_y = m_min_y+i;
497444d93782SGreg Clayton                     window.MoveCursor(1, line_y);
497544d93782SGreg Clayton                     const bool is_pc_line = curr_line == m_pc_line;
497644d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == curr_line;
497744d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
497844d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
497944d93782SGreg Clayton                     attr_t highlight_attr = 0;
498044d93782SGreg Clayton                     attr_t bp_attr = 0;
498144d93782SGreg Clayton                     if (is_pc_line)
498244d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
498344d93782SGreg Clayton                     else if (line_is_selected)
498444d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
498544d93782SGreg Clayton 
498644d93782SGreg Clayton                     if (bp_lines.find(curr_line+1) != bp_lines.end())
498744d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
498844d93782SGreg Clayton 
498944d93782SGreg Clayton                     if (bp_attr)
499044d93782SGreg Clayton                         window.AttributeOn(bp_attr);
499144d93782SGreg Clayton 
499244d93782SGreg Clayton                     window.Printf (m_line_format, curr_line + 1);
499344d93782SGreg Clayton 
499444d93782SGreg Clayton                     if (bp_attr)
499544d93782SGreg Clayton                         window.AttributeOff(bp_attr);
499644d93782SGreg Clayton 
499744d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
499844d93782SGreg Clayton                     // Mark the line with the PC with a diamond
499944d93782SGreg Clayton                     if (is_pc_line)
500044d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
500144d93782SGreg Clayton                     else
500244d93782SGreg Clayton                         window.PutChar(' ');
500344d93782SGreg Clayton 
500444d93782SGreg Clayton                     if (highlight_attr)
500544d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
500644d93782SGreg Clayton                     const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false);
500744d93782SGreg Clayton                     if (line_len > 0)
500844d93782SGreg Clayton                         window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len);
500944d93782SGreg Clayton 
501044d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
501144d93782SGreg Clayton                     {
501244d93782SGreg Clayton                         StopInfoSP stop_info_sp;
501344d93782SGreg Clayton                         if (thread)
501444d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
501544d93782SGreg Clayton                         if (stop_info_sp)
501644d93782SGreg Clayton                         {
501744d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
501844d93782SGreg Clayton                             if (stop_description && stop_description[0])
501944d93782SGreg Clayton                             {
502044d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5021ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
502244d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5023ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
502444d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
502544d93782SGreg Clayton                             }
502644d93782SGreg Clayton                         }
502744d93782SGreg Clayton                         else
502844d93782SGreg Clayton                         {
5029ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
503044d93782SGreg Clayton                         }
503144d93782SGreg Clayton                     }
503244d93782SGreg Clayton                     if (highlight_attr)
503344d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
503444d93782SGreg Clayton                 }
503544d93782SGreg Clayton                 else
503644d93782SGreg Clayton                 {
503744d93782SGreg Clayton                     break;
503844d93782SGreg Clayton                 }
503944d93782SGreg Clayton             }
504044d93782SGreg Clayton         }
504144d93782SGreg Clayton         else
504244d93782SGreg Clayton         {
504344d93782SGreg Clayton             size_t num_disassembly_lines = GetNumDisassemblyLines();
504444d93782SGreg Clayton             if (num_disassembly_lines > 0)
504544d93782SGreg Clayton             {
504644d93782SGreg Clayton                 // Display disassembly
504744d93782SGreg Clayton                 BreakpointAddrs bp_file_addrs;
504844d93782SGreg Clayton                 Target *target = exe_ctx.GetTargetPtr();
504944d93782SGreg Clayton                 if (target)
505044d93782SGreg Clayton                 {
505144d93782SGreg Clayton                     BreakpointList &bp_list = target->GetBreakpointList();
505244d93782SGreg Clayton                     const size_t num_bps = bp_list.GetSize();
505344d93782SGreg Clayton                     for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx)
505444d93782SGreg Clayton                     {
505544d93782SGreg Clayton                         BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx);
505644d93782SGreg Clayton                         const size_t num_bps_locs = bp_sp->GetNumLocations();
505744d93782SGreg Clayton                         for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx)
505844d93782SGreg Clayton                         {
505944d93782SGreg Clayton                             BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx);
506044d93782SGreg Clayton                             LineEntry bp_loc_line_entry;
506144d93782SGreg Clayton                             const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress();
506244d93782SGreg Clayton                             if (file_addr != LLDB_INVALID_ADDRESS)
506344d93782SGreg Clayton                             {
506444d93782SGreg Clayton                                 if (m_disassembly_range.ContainsFileAddress(file_addr))
506544d93782SGreg Clayton                                     bp_file_addrs.insert(file_addr);
506644d93782SGreg Clayton                             }
506744d93782SGreg Clayton                         }
506844d93782SGreg Clayton                     }
506944d93782SGreg Clayton                 }
507044d93782SGreg Clayton 
507144d93782SGreg Clayton                 const attr_t selected_highlight_attr = A_REVERSE;
507244d93782SGreg Clayton                 const attr_t pc_highlight_attr = COLOR_PAIR(1);
507344d93782SGreg Clayton 
507444d93782SGreg Clayton                 StreamString strm;
507544d93782SGreg Clayton 
507644d93782SGreg Clayton                 InstructionList &insts = m_disassembly_sp->GetInstructionList();
507744d93782SGreg Clayton                 Address pc_address;
507844d93782SGreg Clayton 
507944d93782SGreg Clayton                 if (frame_sp)
508044d93782SGreg Clayton                     pc_address = frame_sp->GetFrameCodeAddress();
508144d93782SGreg Clayton                 const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX;
508244d93782SGreg Clayton                 if (set_selected_line_to_pc)
508344d93782SGreg Clayton                 {
508444d93782SGreg Clayton                     m_selected_line = pc_idx;
508544d93782SGreg Clayton                 }
508644d93782SGreg Clayton 
508744d93782SGreg Clayton                 const uint32_t non_visible_pc_offset = (num_visible_lines / 5);
50883985c8c6SSaleem Abdulrasool                 if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines)
508944d93782SGreg Clayton                     m_first_visible_line = 0;
509044d93782SGreg Clayton 
509144d93782SGreg Clayton                 if (pc_idx < num_disassembly_lines)
509244d93782SGreg Clayton                 {
50933985c8c6SSaleem Abdulrasool                     if (pc_idx < static_cast<uint32_t>(m_first_visible_line) ||
509444d93782SGreg Clayton                         pc_idx >= m_first_visible_line + num_visible_lines)
509544d93782SGreg Clayton                         m_first_visible_line = pc_idx - non_visible_pc_offset;
509644d93782SGreg Clayton                 }
509744d93782SGreg Clayton 
509844d93782SGreg Clayton                 for (size_t i=0; i<num_visible_lines; ++i)
509944d93782SGreg Clayton                 {
510044d93782SGreg Clayton                     const uint32_t inst_idx = m_first_visible_line + i;
510144d93782SGreg Clayton                     Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get();
510244d93782SGreg Clayton                     if (!inst)
510344d93782SGreg Clayton                         break;
510444d93782SGreg Clayton 
5105ec990867SGreg Clayton                     const int line_y = m_min_y+i;
5106ec990867SGreg Clayton                     window.MoveCursor(1, line_y);
510744d93782SGreg Clayton                     const bool is_pc_line = frame_sp && inst_idx == pc_idx;
510844d93782SGreg Clayton                     const bool line_is_selected = m_selected_line == inst_idx;
510944d93782SGreg Clayton                     // Highlight the line as the PC line first, then if the selected line
511044d93782SGreg Clayton                     // isn't the same as the PC line, highlight it differently
511144d93782SGreg Clayton                     attr_t highlight_attr = 0;
511244d93782SGreg Clayton                     attr_t bp_attr = 0;
511344d93782SGreg Clayton                     if (is_pc_line)
511444d93782SGreg Clayton                         highlight_attr = pc_highlight_attr;
511544d93782SGreg Clayton                     else if (line_is_selected)
511644d93782SGreg Clayton                         highlight_attr = selected_highlight_attr;
511744d93782SGreg Clayton 
511844d93782SGreg Clayton                     if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end())
511944d93782SGreg Clayton                         bp_attr = COLOR_PAIR(2);
512044d93782SGreg Clayton 
512144d93782SGreg Clayton                     if (bp_attr)
512244d93782SGreg Clayton                         window.AttributeOn(bp_attr);
512344d93782SGreg Clayton 
5124324a1036SSaleem Abdulrasool                     window.Printf (" 0x%16.16llx ",
5125324a1036SSaleem Abdulrasool                                    static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target)));
512644d93782SGreg Clayton 
512744d93782SGreg Clayton                     if (bp_attr)
512844d93782SGreg Clayton                         window.AttributeOff(bp_attr);
512944d93782SGreg Clayton 
513044d93782SGreg Clayton                     window.PutChar(ACS_VLINE);
513144d93782SGreg Clayton                     // Mark the line with the PC with a diamond
513244d93782SGreg Clayton                     if (is_pc_line)
513344d93782SGreg Clayton                         window.PutChar(ACS_DIAMOND);
513444d93782SGreg Clayton                     else
513544d93782SGreg Clayton                         window.PutChar(' ');
513644d93782SGreg Clayton 
513744d93782SGreg Clayton                     if (highlight_attr)
513844d93782SGreg Clayton                         window.AttributeOn(highlight_attr);
513944d93782SGreg Clayton 
514044d93782SGreg Clayton                     const char *mnemonic = inst->GetMnemonic(&exe_ctx);
514144d93782SGreg Clayton                     const char *operands = inst->GetOperands(&exe_ctx);
514244d93782SGreg Clayton                     const char *comment = inst->GetComment(&exe_ctx);
514344d93782SGreg Clayton 
514444d93782SGreg Clayton                     if (mnemonic && mnemonic[0] == '\0')
514544d93782SGreg Clayton                         mnemonic = NULL;
514644d93782SGreg Clayton                     if (operands && operands[0] == '\0')
514744d93782SGreg Clayton                         operands = NULL;
514844d93782SGreg Clayton                     if (comment && comment[0] == '\0')
514944d93782SGreg Clayton                         comment = NULL;
515044d93782SGreg Clayton 
515144d93782SGreg Clayton                     strm.Clear();
515244d93782SGreg Clayton 
515344d93782SGreg Clayton                     if (mnemonic && operands && comment)
515444d93782SGreg Clayton                         strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment);
515544d93782SGreg Clayton                     else if (mnemonic && operands)
515644d93782SGreg Clayton                         strm.Printf ("%-8s %s", mnemonic, operands);
515744d93782SGreg Clayton                     else if (mnemonic)
515844d93782SGreg Clayton                         strm.Printf ("%s", mnemonic);
515944d93782SGreg Clayton 
516044d93782SGreg Clayton                     int right_pad = 1;
516144d93782SGreg Clayton                     window.PutCStringTruncated(strm.GetString().c_str(), right_pad);
516244d93782SGreg Clayton 
516344d93782SGreg Clayton                     if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0)
516444d93782SGreg Clayton                     {
516544d93782SGreg Clayton                         StopInfoSP stop_info_sp;
516644d93782SGreg Clayton                         if (thread)
516744d93782SGreg Clayton                             stop_info_sp = thread->GetStopInfo();
516844d93782SGreg Clayton                         if (stop_info_sp)
516944d93782SGreg Clayton                         {
517044d93782SGreg Clayton                             const char *stop_description = stop_info_sp->GetDescription();
517144d93782SGreg Clayton                             if (stop_description && stop_description[0])
517244d93782SGreg Clayton                             {
517344d93782SGreg Clayton                                 size_t stop_description_len = strlen(stop_description);
5174ec990867SGreg Clayton                                 int desc_x = window_width - stop_description_len - 16;
517544d93782SGreg Clayton                                 window.Printf ("%*s", desc_x - window.GetCursorX(), "");
5176ec990867SGreg Clayton                                 //window.MoveCursor(window_width - stop_description_len - 15, line_y);
517744d93782SGreg Clayton                                 window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description);
517844d93782SGreg Clayton                             }
517944d93782SGreg Clayton                         }
518044d93782SGreg Clayton                         else
518144d93782SGreg Clayton                         {
5182ec990867SGreg Clayton                             window.Printf ("%*s", window_width - window.GetCursorX() - 1, "");
518344d93782SGreg Clayton                         }
518444d93782SGreg Clayton                     }
518544d93782SGreg Clayton                     if (highlight_attr)
518644d93782SGreg Clayton                         window.AttributeOff(highlight_attr);
518744d93782SGreg Clayton                 }
518844d93782SGreg Clayton             }
518944d93782SGreg Clayton         }
519044d93782SGreg Clayton         window.DeferredRefresh();
519144d93782SGreg Clayton         return true; // Drawing handled
519244d93782SGreg Clayton     }
519344d93782SGreg Clayton 
519444d93782SGreg Clayton     size_t
519544d93782SGreg Clayton     GetNumLines ()
519644d93782SGreg Clayton     {
519744d93782SGreg Clayton         size_t num_lines = GetNumSourceLines();
519844d93782SGreg Clayton         if (num_lines == 0)
519944d93782SGreg Clayton             num_lines = GetNumDisassemblyLines();
520044d93782SGreg Clayton         return num_lines;
520144d93782SGreg Clayton     }
520244d93782SGreg Clayton 
520344d93782SGreg Clayton     size_t
520444d93782SGreg Clayton     GetNumSourceLines () const
520544d93782SGreg Clayton     {
520644d93782SGreg Clayton         if (m_file_sp)
520744d93782SGreg Clayton             return m_file_sp->GetNumLines();
520844d93782SGreg Clayton         return 0;
520944d93782SGreg Clayton     }
5210315b6884SEugene Zelenko 
521144d93782SGreg Clayton     size_t
521244d93782SGreg Clayton     GetNumDisassemblyLines () const
521344d93782SGreg Clayton     {
521444d93782SGreg Clayton         if (m_disassembly_sp)
521544d93782SGreg Clayton             return m_disassembly_sp->GetInstructionList().GetSize();
521644d93782SGreg Clayton         return 0;
521744d93782SGreg Clayton     }
521844d93782SGreg Clayton 
5219bd5ae6b4SGreg Clayton     HandleCharResult
5220bd5ae6b4SGreg Clayton     WindowDelegateHandleChar (Window &window, int c) override
522144d93782SGreg Clayton     {
522244d93782SGreg Clayton         const uint32_t num_visible_lines = NumVisibleLines();
522344d93782SGreg Clayton         const size_t num_lines = GetNumLines ();
522444d93782SGreg Clayton 
522544d93782SGreg Clayton         switch (c)
522644d93782SGreg Clayton         {
522744d93782SGreg Clayton             case ',':
522844d93782SGreg Clayton             case KEY_PPAGE:
522944d93782SGreg Clayton                 // Page up key
52303985c8c6SSaleem Abdulrasool                 if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines)
523144d93782SGreg Clayton                     m_first_visible_line -= num_visible_lines;
523244d93782SGreg Clayton                 else
523344d93782SGreg Clayton                     m_first_visible_line = 0;
523444d93782SGreg Clayton                 m_selected_line = m_first_visible_line;
523544d93782SGreg Clayton                 return eKeyHandled;
523644d93782SGreg Clayton 
523744d93782SGreg Clayton             case '.':
523844d93782SGreg Clayton             case KEY_NPAGE:
523944d93782SGreg Clayton                 // Page down key
524044d93782SGreg Clayton                 {
524144d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < num_lines)
524244d93782SGreg Clayton                         m_first_visible_line += num_visible_lines;
524344d93782SGreg Clayton                     else if (num_lines < num_visible_lines)
524444d93782SGreg Clayton                         m_first_visible_line = 0;
524544d93782SGreg Clayton                     else
524644d93782SGreg Clayton                         m_first_visible_line = num_lines - num_visible_lines;
524744d93782SGreg Clayton                     m_selected_line = m_first_visible_line;
524844d93782SGreg Clayton                 }
524944d93782SGreg Clayton                 return eKeyHandled;
525044d93782SGreg Clayton 
525144d93782SGreg Clayton             case KEY_UP:
525244d93782SGreg Clayton                 if (m_selected_line > 0)
525344d93782SGreg Clayton                 {
525444d93782SGreg Clayton                     m_selected_line--;
52553985c8c6SSaleem Abdulrasool                     if (static_cast<size_t>(m_first_visible_line) > m_selected_line)
525644d93782SGreg Clayton                         m_first_visible_line = m_selected_line;
525744d93782SGreg Clayton                 }
525844d93782SGreg Clayton                 return eKeyHandled;
525944d93782SGreg Clayton 
526044d93782SGreg Clayton             case KEY_DOWN:
526144d93782SGreg Clayton                 if (m_selected_line + 1 < num_lines)
526244d93782SGreg Clayton                 {
526344d93782SGreg Clayton                     m_selected_line++;
526444d93782SGreg Clayton                     if (m_first_visible_line + num_visible_lines < m_selected_line)
526544d93782SGreg Clayton                         m_first_visible_line++;
526644d93782SGreg Clayton                 }
526744d93782SGreg Clayton                 return eKeyHandled;
526844d93782SGreg Clayton 
526944d93782SGreg Clayton             case '\r':
527044d93782SGreg Clayton             case '\n':
527144d93782SGreg Clayton             case KEY_ENTER:
527244d93782SGreg Clayton                 // Set a breakpoint and run to the line using a one shot breakpoint
527344d93782SGreg Clayton                 if (GetNumSourceLines() > 0)
527444d93782SGreg Clayton                 {
527544d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
527644d93782SGreg Clayton                     if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive())
527744d93782SGreg Clayton                     {
527844d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
527944d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
528044d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
528144d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
528244d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
528344d93782SGreg Clayton                                                                                       false,                     // internal
5284055ad9beSIlia K                                                                                       false,                     // request_hardware
5285055ad9beSIlia K                                                                                       eLazyBoolCalculate);       // move_to_nearest_code
528644d93782SGreg Clayton                         // Make breakpoint one shot
528744d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
528844d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
528944d93782SGreg Clayton                     }
529044d93782SGreg Clayton                 }
529144d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
529244d93782SGreg Clayton                 {
529344d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
529444d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
529544d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
529644d93782SGreg Clayton                     {
529744d93782SGreg Clayton                         Address addr = inst->GetAddress();
529844d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
529944d93782SGreg Clayton                                                                                       false,    // internal
530044d93782SGreg Clayton                                                                                       false);   // request_hardware
530144d93782SGreg Clayton                         // Make breakpoint one shot
530244d93782SGreg Clayton                         bp_sp->GetOptions()->SetOneShot(true);
530344d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
530444d93782SGreg Clayton                     }
530544d93782SGreg Clayton                 }
530644d93782SGreg Clayton                 return eKeyHandled;
530744d93782SGreg Clayton 
530844d93782SGreg Clayton             case 'b':   // 'b' == toggle breakpoint on currently selected line
530944d93782SGreg Clayton                 if (m_selected_line < GetNumSourceLines())
531044d93782SGreg Clayton                 {
531144d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
531244d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
531344d93782SGreg Clayton                     {
531444d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL,                      // Don't limit the breakpoint to certain modules
531544d93782SGreg Clayton                                                                                       m_file_sp->GetFileSpec(),  // Source file
531644d93782SGreg Clayton                                                                                       m_selected_line + 1,       // Source line number (m_selected_line is zero based)
531744d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Check inlines using global setting
531844d93782SGreg Clayton                                                                                       eLazyBoolCalculate,        // Skip prologue using global setting,
531944d93782SGreg Clayton                                                                                       false,                     // internal
5320055ad9beSIlia K                                                                                       false,                     // request_hardware
5321055ad9beSIlia K                                                                                       eLazyBoolCalculate);       // move_to_nearest_code
532244d93782SGreg Clayton                     }
532344d93782SGreg Clayton                 }
532444d93782SGreg Clayton                 else if (m_selected_line < GetNumDisassemblyLines())
532544d93782SGreg Clayton                 {
532644d93782SGreg Clayton                     const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get();
532744d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
532844d93782SGreg Clayton                     if (exe_ctx.HasTargetScope())
532944d93782SGreg Clayton                     {
533044d93782SGreg Clayton                         Address addr = inst->GetAddress();
533144d93782SGreg Clayton                         BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr,     // lldb_private::Address
533244d93782SGreg Clayton                                                                                       false,    // internal
533344d93782SGreg Clayton                                                                                       false);   // request_hardware
533444d93782SGreg Clayton                     }
533544d93782SGreg Clayton                 }
533644d93782SGreg Clayton                 return eKeyHandled;
533744d93782SGreg Clayton 
533844d93782SGreg Clayton             case 'd':   // 'd' == detach and let run
533944d93782SGreg Clayton             case 'D':   // 'D' == detach and keep stopped
534044d93782SGreg Clayton                 {
534144d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
534244d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
534344d93782SGreg Clayton                         exe_ctx.GetProcessRef().Detach(c == 'D');
534444d93782SGreg Clayton                 }
534544d93782SGreg Clayton                 return eKeyHandled;
534644d93782SGreg Clayton 
534744d93782SGreg Clayton             case 'k':
534844d93782SGreg Clayton                 // 'k' == kill
534944d93782SGreg Clayton                 {
535044d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
535144d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
5352ede3193bSJason Molenda                         exe_ctx.GetProcessRef().Destroy(false);
535344d93782SGreg Clayton                 }
535444d93782SGreg Clayton                 return eKeyHandled;
535544d93782SGreg Clayton 
535644d93782SGreg Clayton             case 'c':
535744d93782SGreg Clayton                 // 'c' == continue
535844d93782SGreg Clayton                 {
535944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
536044d93782SGreg Clayton                     if (exe_ctx.HasProcessScope())
536144d93782SGreg Clayton                         exe_ctx.GetProcessRef().Resume();
536244d93782SGreg Clayton                 }
536344d93782SGreg Clayton                 return eKeyHandled;
536444d93782SGreg Clayton 
536544d93782SGreg Clayton             case 'o':
536644d93782SGreg Clayton                 // 'o' == step out
536744d93782SGreg Clayton                 {
536844d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
536944d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
537044d93782SGreg Clayton                     {
537144d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOut();
537244d93782SGreg Clayton                     }
537344d93782SGreg Clayton                 }
537444d93782SGreg Clayton                 return eKeyHandled;
5375315b6884SEugene Zelenko 
537644d93782SGreg Clayton             case 'n':   // 'n' == step over
537744d93782SGreg Clayton             case 'N':   // 'N' == step over instruction
537844d93782SGreg Clayton                 {
537944d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
538044d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
538144d93782SGreg Clayton                     {
538244d93782SGreg Clayton                         bool source_step = (c == 'n');
538344d93782SGreg Clayton                         exe_ctx.GetThreadRef().StepOver(source_step);
538444d93782SGreg Clayton                     }
538544d93782SGreg Clayton                 }
538644d93782SGreg Clayton                 return eKeyHandled;
5387315b6884SEugene Zelenko 
538844d93782SGreg Clayton             case 's':   // 's' == step into
538944d93782SGreg Clayton             case 'S':   // 'S' == step into instruction
539044d93782SGreg Clayton                 {
539144d93782SGreg Clayton                     ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext();
539244d93782SGreg Clayton                     if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true))
539344d93782SGreg Clayton                     {
539444d93782SGreg Clayton                         bool source_step = (c == 's');
53954b4b2478SJim Ingham                         exe_ctx.GetThreadRef().StepIn(source_step);
539644d93782SGreg Clayton                     }
539744d93782SGreg Clayton                 }
539844d93782SGreg Clayton                 return eKeyHandled;
539944d93782SGreg Clayton 
540044d93782SGreg Clayton             case 'h':
540144d93782SGreg Clayton                 window.CreateHelpSubwindow ();
540244d93782SGreg Clayton                 return eKeyHandled;
540344d93782SGreg Clayton 
540444d93782SGreg Clayton             default:
540544d93782SGreg Clayton                 break;
540644d93782SGreg Clayton         }
540744d93782SGreg Clayton         return eKeyNotHandled;
540844d93782SGreg Clayton     }
540944d93782SGreg Clayton 
541044d93782SGreg Clayton protected:
541144d93782SGreg Clayton     typedef std::set<uint32_t> BreakpointLines;
541244d93782SGreg Clayton     typedef std::set<lldb::addr_t> BreakpointAddrs;
541344d93782SGreg Clayton 
541444d93782SGreg Clayton     Debugger &m_debugger;
541544d93782SGreg Clayton     SymbolContext m_sc;
541644d93782SGreg Clayton     SourceManager::FileSP m_file_sp;
541744d93782SGreg Clayton     SymbolContextScope *m_disassembly_scope;
541844d93782SGreg Clayton     lldb::DisassemblerSP m_disassembly_sp;
541944d93782SGreg Clayton     AddressRange m_disassembly_range;
5420ec990867SGreg Clayton     StreamString m_title;
542144d93782SGreg Clayton     lldb::user_id_t m_tid;
542244d93782SGreg Clayton     char m_line_format[8];
542344d93782SGreg Clayton     int m_line_width;
542444d93782SGreg Clayton     uint32_t m_selected_line;       // The selected line
542544d93782SGreg Clayton     uint32_t m_pc_line;             // The line with the PC
542644d93782SGreg Clayton     uint32_t m_stop_id;
542744d93782SGreg Clayton     uint32_t m_frame_idx;
542844d93782SGreg Clayton     int m_first_visible_line;
542944d93782SGreg Clayton     int m_min_x;
543044d93782SGreg Clayton     int m_min_y;
543144d93782SGreg Clayton     int m_max_x;
543244d93782SGreg Clayton     int m_max_y;
543344d93782SGreg Clayton };
543444d93782SGreg Clayton 
543544d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true };
543644d93782SGreg Clayton 
543744d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) :
5438e30f11d9SKate Stone     IOHandler (debugger, IOHandler::Type::Curses)
543944d93782SGreg Clayton {
544044d93782SGreg Clayton }
544144d93782SGreg Clayton 
544244d93782SGreg Clayton void
544344d93782SGreg Clayton IOHandlerCursesGUI::Activate ()
544444d93782SGreg Clayton {
544544d93782SGreg Clayton     IOHandler::Activate();
544644d93782SGreg Clayton     if (!m_app_ap)
544744d93782SGreg Clayton     {
544844d93782SGreg Clayton         m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE()));
544944d93782SGreg Clayton 
545044d93782SGreg Clayton         // This is both a window and a menu delegate
545144d93782SGreg Clayton         std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger));
545244d93782SGreg Clayton 
545344d93782SGreg Clayton         MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp);
545444d93782SGreg Clayton         MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB));
545544d93782SGreg Clayton         MenuSP exit_menuitem_sp(new Menu("Exit", NULL, 'x', ApplicationDelegate::eMenuID_LLDBExit));
545644d93782SGreg Clayton         exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit);
545744d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", NULL, 'a', ApplicationDelegate::eMenuID_LLDBAbout)));
545844d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
545944d93782SGreg Clayton         lldb_menu_sp->AddSubmenu (exit_menuitem_sp);
546044d93782SGreg Clayton 
546144d93782SGreg Clayton         MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target));
546244d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", NULL, 'c', ApplicationDelegate::eMenuID_TargetCreate)));
546344d93782SGreg Clayton         target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", NULL, 'd', ApplicationDelegate::eMenuID_TargetDelete)));
546444d93782SGreg Clayton 
546544d93782SGreg Clayton         MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process));
546644d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach"  , NULL, 'a', ApplicationDelegate::eMenuID_ProcessAttach)));
546744d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach"  , NULL, 'd', ApplicationDelegate::eMenuID_ProcessDetach)));
546844d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch"  , NULL, 'l', ApplicationDelegate::eMenuID_ProcessLaunch)));
546944d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator)));
547044d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", NULL, 'c', ApplicationDelegate::eMenuID_ProcessContinue)));
547144d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt"    , NULL, 'h', ApplicationDelegate::eMenuID_ProcessHalt)));
547244d93782SGreg Clayton         process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill"    , NULL, 'k', ApplicationDelegate::eMenuID_ProcessKill)));
547344d93782SGreg Clayton 
547444d93782SGreg Clayton         MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread));
547544d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In"  , NULL, 'i', ApplicationDelegate::eMenuID_ThreadStepIn)));
547644d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", NULL, 'v', ApplicationDelegate::eMenuID_ThreadStepOver)));
547744d93782SGreg Clayton         thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , NULL, 'o', ApplicationDelegate::eMenuID_ThreadStepOut)));
547844d93782SGreg Clayton 
547944d93782SGreg Clayton         MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View));
548044d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", NULL, 'b', ApplicationDelegate::eMenuID_ViewBacktrace)));
548144d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", NULL, 'r', ApplicationDelegate::eMenuID_ViewRegisters)));
548244d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Source"   , NULL, 's', ApplicationDelegate::eMenuID_ViewSource)));
548344d93782SGreg Clayton         view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", NULL, 'v', ApplicationDelegate::eMenuID_ViewVariables)));
548444d93782SGreg Clayton 
548544d93782SGreg Clayton         MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help));
548644d93782SGreg Clayton         help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", NULL, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp)));
548744d93782SGreg Clayton 
548844d93782SGreg Clayton         m_app_ap->Initialize();
548944d93782SGreg Clayton         WindowSP &main_window_sp = m_app_ap->GetMainWindow();
549044d93782SGreg Clayton 
549144d93782SGreg Clayton         MenuSP menubar_sp(new Menu(Menu::Type::Bar));
549244d93782SGreg Clayton         menubar_sp->AddSubmenu (lldb_menu_sp);
549344d93782SGreg Clayton         menubar_sp->AddSubmenu (target_menu_sp);
549444d93782SGreg Clayton         menubar_sp->AddSubmenu (process_menu_sp);
549544d93782SGreg Clayton         menubar_sp->AddSubmenu (thread_menu_sp);
549644d93782SGreg Clayton         menubar_sp->AddSubmenu (view_menu_sp);
549744d93782SGreg Clayton         menubar_sp->AddSubmenu (help_menu_sp);
549844d93782SGreg Clayton         menubar_sp->SetDelegate(app_menu_delegate_sp);
549944d93782SGreg Clayton 
550044d93782SGreg Clayton         Rect content_bounds = main_window_sp->GetFrame();
550144d93782SGreg Clayton         Rect menubar_bounds = content_bounds.MakeMenuBar();
550244d93782SGreg Clayton         Rect status_bounds = content_bounds.MakeStatusBar();
550344d93782SGreg Clayton         Rect source_bounds;
550444d93782SGreg Clayton         Rect variables_bounds;
550544d93782SGreg Clayton         Rect threads_bounds;
550644d93782SGreg Clayton         Rect source_variables_bounds;
550744d93782SGreg Clayton         content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds);
550844d93782SGreg Clayton         source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds);
550944d93782SGreg Clayton 
551044d93782SGreg Clayton         WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false);
551144d93782SGreg Clayton         // Let the menubar get keys if the active window doesn't handle the
551244d93782SGreg Clayton         // keys that are typed so it can respond to menubar key presses.
551344d93782SGreg Clayton         menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window
551444d93782SGreg Clayton         menubar_window_sp->SetDelegate(menubar_sp);
551544d93782SGreg Clayton 
551644d93782SGreg Clayton         WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source",
551744d93782SGreg Clayton                                                                    source_bounds,
551844d93782SGreg Clayton                                                                    true));
551944d93782SGreg Clayton         WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables",
552044d93782SGreg Clayton                                                                       variables_bounds,
552144d93782SGreg Clayton                                                                       false));
552244d93782SGreg Clayton         WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads",
552344d93782SGreg Clayton                                                                       threads_bounds,
552444d93782SGreg Clayton                                                                       false));
552544d93782SGreg Clayton         WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status",
552644d93782SGreg Clayton                                                                    status_bounds,
552744d93782SGreg Clayton                                                                    false));
552844d93782SGreg Clayton         status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window
552944d93782SGreg Clayton         main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp));
553044d93782SGreg Clayton         source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger)));
553144d93782SGreg Clayton         variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger)));
5532ec990867SGreg Clayton         TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger));
553344d93782SGreg Clayton         threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp)));
553444d93782SGreg Clayton         status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger)));
553544d93782SGreg Clayton 
55365fdb09bbSGreg Clayton         // Show the main help window once the first time the curses GUI is launched
55375fdb09bbSGreg Clayton         static bool g_showed_help = false;
55385fdb09bbSGreg Clayton         if (!g_showed_help)
55395fdb09bbSGreg Clayton         {
55405fdb09bbSGreg Clayton             g_showed_help = true;
55415fdb09bbSGreg Clayton             main_window_sp->CreateHelpSubwindow();
55425fdb09bbSGreg Clayton         }
55435fdb09bbSGreg Clayton 
554444d93782SGreg Clayton         init_pair (1, COLOR_WHITE   , COLOR_BLUE  );
554544d93782SGreg Clayton         init_pair (2, COLOR_BLACK   , COLOR_WHITE );
554644d93782SGreg Clayton         init_pair (3, COLOR_MAGENTA , COLOR_WHITE );
554744d93782SGreg Clayton         init_pair (4, COLOR_MAGENTA , COLOR_BLACK );
554844d93782SGreg Clayton         init_pair (5, COLOR_RED     , COLOR_BLACK );
554944d93782SGreg Clayton     }
555044d93782SGreg Clayton }
555144d93782SGreg Clayton 
555244d93782SGreg Clayton void
555344d93782SGreg Clayton IOHandlerCursesGUI::Deactivate ()
555444d93782SGreg Clayton {
555544d93782SGreg Clayton     m_app_ap->Terminate();
555644d93782SGreg Clayton }
555744d93782SGreg Clayton 
555844d93782SGreg Clayton void
555944d93782SGreg Clayton IOHandlerCursesGUI::Run ()
556044d93782SGreg Clayton {
556144d93782SGreg Clayton     m_app_ap->Run(m_debugger);
556244d93782SGreg Clayton     SetIsDone(true);
556344d93782SGreg Clayton }
556444d93782SGreg Clayton 
5565315b6884SEugene Zelenko IOHandlerCursesGUI::~IOHandlerCursesGUI() = default;
556644d93782SGreg Clayton 
556744d93782SGreg Clayton void
5568e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel ()
5569e68f5d6bSGreg Clayton {
5570e68f5d6bSGreg Clayton }
557144d93782SGreg Clayton 
5572f0066ad0SGreg Clayton bool
557344d93782SGreg Clayton IOHandlerCursesGUI::Interrupt ()
557444d93782SGreg Clayton {
5575f0066ad0SGreg Clayton     return false;
557644d93782SGreg Clayton }
557744d93782SGreg Clayton 
557844d93782SGreg Clayton void
557944d93782SGreg Clayton IOHandlerCursesGUI::GotEOF()
558044d93782SGreg Clayton {
558144d93782SGreg Clayton }
558244d93782SGreg Clayton 
5583315b6884SEugene Zelenko #endif // LLDB_DISABLE_CURSES
5584