144d93782SGreg Clayton //===-- IOHandler.cpp -------------------------------------------*- C++ -*-===// 244d93782SGreg Clayton // 344d93782SGreg Clayton // The LLVM Compiler Infrastructure 444d93782SGreg Clayton // 544d93782SGreg Clayton // This file is distributed under the University of Illinois Open Source 644d93782SGreg Clayton // License. See LICENSE.TXT for details. 744d93782SGreg Clayton // 844d93782SGreg Clayton //===----------------------------------------------------------------------===// 944d93782SGreg Clayton 1044d93782SGreg Clayton 1144d93782SGreg Clayton #include "lldb/lldb-python.h" 1244d93782SGreg Clayton 1344d93782SGreg Clayton #include <string> 1444d93782SGreg Clayton 1544d93782SGreg Clayton #include "lldb/Breakpoint/BreakpointLocation.h" 1644d93782SGreg Clayton #include "lldb/Core/IOHandler.h" 1744d93782SGreg Clayton #include "lldb/Core/Debugger.h" 18ec990867SGreg Clayton #include "lldb/Core/Module.h" 1944d93782SGreg Clayton #include "lldb/Core/State.h" 2044d93782SGreg Clayton #include "lldb/Core/StreamFile.h" 2144d93782SGreg Clayton #include "lldb/Core/ValueObjectRegister.h" 22cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 2344d93782SGreg Clayton #include "lldb/Host/Editline.h" 24cacde7dfSTodd Fiala #endif 2544d93782SGreg Clayton #include "lldb/Interpreter/CommandCompletions.h" 2644d93782SGreg Clayton #include "lldb/Interpreter/CommandInterpreter.h" 2744d93782SGreg Clayton #include "lldb/Symbol/Block.h" 2844d93782SGreg Clayton #include "lldb/Symbol/Function.h" 2944d93782SGreg Clayton #include "lldb/Symbol/Symbol.h" 3044d93782SGreg Clayton #include "lldb/Target/RegisterContext.h" 3144d93782SGreg Clayton #include "lldb/Target/ThreadPlan.h" 3244d93782SGreg Clayton 33914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES 3444d93782SGreg Clayton #include <ncurses.h> 3544d93782SGreg Clayton #include <panel.h> 36914b8d98SDeepak Panickal #endif 3744d93782SGreg Clayton 3844d93782SGreg Clayton using namespace lldb; 3944d93782SGreg Clayton using namespace lldb_private; 4044d93782SGreg Clayton 41e30f11d9SKate Stone IOHandler::IOHandler (Debugger &debugger, IOHandler::Type type) : 4244d93782SGreg Clayton IOHandler (debugger, 43e30f11d9SKate Stone type, 4444d93782SGreg Clayton StreamFileSP(), // Adopt STDIN from top input reader 4544d93782SGreg Clayton StreamFileSP(), // Adopt STDOUT from top input reader 46340b0309SGreg Clayton StreamFileSP(), // Adopt STDERR from top input reader 47340b0309SGreg Clayton 0) // Flags 4844d93782SGreg Clayton { 4944d93782SGreg Clayton } 5044d93782SGreg Clayton 5144d93782SGreg Clayton 5244d93782SGreg Clayton IOHandler::IOHandler (Debugger &debugger, 53e30f11d9SKate Stone IOHandler::Type type, 5444d93782SGreg Clayton const lldb::StreamFileSP &input_sp, 5544d93782SGreg Clayton const lldb::StreamFileSP &output_sp, 56340b0309SGreg Clayton const lldb::StreamFileSP &error_sp, 57340b0309SGreg Clayton uint32_t flags) : 5844d93782SGreg Clayton m_debugger (debugger), 5944d93782SGreg Clayton m_input_sp (input_sp), 6044d93782SGreg Clayton m_output_sp (output_sp), 6144d93782SGreg Clayton m_error_sp (error_sp), 62e30f11d9SKate Stone m_popped (false), 63340b0309SGreg Clayton m_flags (flags), 64e30f11d9SKate Stone m_type (type), 6544d93782SGreg Clayton m_user_data (NULL), 6644d93782SGreg Clayton m_done (false), 6744d93782SGreg Clayton m_active (false) 6844d93782SGreg Clayton { 6944d93782SGreg Clayton // If any files are not specified, then adopt them from the top input reader. 7044d93782SGreg Clayton if (!m_input_sp || !m_output_sp || !m_error_sp) 7144d93782SGreg Clayton debugger.AdoptTopIOHandlerFilesIfInvalid (m_input_sp, 7244d93782SGreg Clayton m_output_sp, 7344d93782SGreg Clayton m_error_sp); 7444d93782SGreg Clayton } 7544d93782SGreg Clayton 7644d93782SGreg Clayton IOHandler::~IOHandler() 7744d93782SGreg Clayton { 7844d93782SGreg Clayton } 7944d93782SGreg Clayton 8044d93782SGreg Clayton 8144d93782SGreg Clayton int 8244d93782SGreg Clayton IOHandler::GetInputFD() 8344d93782SGreg Clayton { 8444d93782SGreg Clayton if (m_input_sp) 8544d93782SGreg Clayton return m_input_sp->GetFile().GetDescriptor(); 8644d93782SGreg Clayton return -1; 8744d93782SGreg Clayton } 8844d93782SGreg Clayton 8944d93782SGreg Clayton int 9044d93782SGreg Clayton IOHandler::GetOutputFD() 9144d93782SGreg Clayton { 9244d93782SGreg Clayton if (m_output_sp) 9344d93782SGreg Clayton return m_output_sp->GetFile().GetDescriptor(); 9444d93782SGreg Clayton return -1; 9544d93782SGreg Clayton } 9644d93782SGreg Clayton 9744d93782SGreg Clayton int 9844d93782SGreg Clayton IOHandler::GetErrorFD() 9944d93782SGreg Clayton { 10044d93782SGreg Clayton if (m_error_sp) 10144d93782SGreg Clayton return m_error_sp->GetFile().GetDescriptor(); 10244d93782SGreg Clayton return -1; 10344d93782SGreg Clayton } 10444d93782SGreg Clayton 10544d93782SGreg Clayton FILE * 10644d93782SGreg Clayton IOHandler::GetInputFILE() 10744d93782SGreg Clayton { 10844d93782SGreg Clayton if (m_input_sp) 10944d93782SGreg Clayton return m_input_sp->GetFile().GetStream(); 11044d93782SGreg Clayton return NULL; 11144d93782SGreg Clayton } 11244d93782SGreg Clayton 11344d93782SGreg Clayton FILE * 11444d93782SGreg Clayton IOHandler::GetOutputFILE() 11544d93782SGreg Clayton { 11644d93782SGreg Clayton if (m_output_sp) 11744d93782SGreg Clayton return m_output_sp->GetFile().GetStream(); 11844d93782SGreg Clayton return NULL; 11944d93782SGreg Clayton } 12044d93782SGreg Clayton 12144d93782SGreg Clayton FILE * 12244d93782SGreg Clayton IOHandler::GetErrorFILE() 12344d93782SGreg Clayton { 12444d93782SGreg Clayton if (m_error_sp) 12544d93782SGreg Clayton return m_error_sp->GetFile().GetStream(); 12644d93782SGreg Clayton return NULL; 12744d93782SGreg Clayton } 12844d93782SGreg Clayton 12944d93782SGreg Clayton StreamFileSP & 13044d93782SGreg Clayton IOHandler::GetInputStreamFile() 13144d93782SGreg Clayton { 13244d93782SGreg Clayton return m_input_sp; 13344d93782SGreg Clayton } 13444d93782SGreg Clayton 13544d93782SGreg Clayton StreamFileSP & 13644d93782SGreg Clayton IOHandler::GetOutputStreamFile() 13744d93782SGreg Clayton { 13844d93782SGreg Clayton return m_output_sp; 13944d93782SGreg Clayton } 14044d93782SGreg Clayton 14144d93782SGreg Clayton 14244d93782SGreg Clayton StreamFileSP & 14344d93782SGreg Clayton IOHandler::GetErrorStreamFile() 14444d93782SGreg Clayton { 14544d93782SGreg Clayton return m_error_sp; 14644d93782SGreg Clayton } 14744d93782SGreg Clayton 148340b0309SGreg Clayton bool 149340b0309SGreg Clayton IOHandler::GetIsInteractive () 150340b0309SGreg Clayton { 151340b0309SGreg Clayton return GetInputStreamFile()->GetFile().GetIsInteractive (); 152340b0309SGreg Clayton } 153340b0309SGreg Clayton 154340b0309SGreg Clayton bool 155340b0309SGreg Clayton IOHandler::GetIsRealTerminal () 156340b0309SGreg Clayton { 157340b0309SGreg Clayton return GetInputStreamFile()->GetFile().GetIsRealTerminal(); 158340b0309SGreg Clayton } 15944d93782SGreg Clayton 160e30f11d9SKate Stone void 161e30f11d9SKate Stone IOHandler::SetPopped (bool b) 162e30f11d9SKate Stone { 163e30f11d9SKate Stone m_popped.SetValue(b, eBroadcastOnChange); 164e30f11d9SKate Stone } 165e30f11d9SKate Stone 166e30f11d9SKate Stone void 167e30f11d9SKate Stone IOHandler::WaitForPop () 168e30f11d9SKate Stone { 169e30f11d9SKate Stone m_popped.WaitForValueEqualTo(true); 170e30f11d9SKate Stone } 171e30f11d9SKate Stone 17244d93782SGreg Clayton IOHandlerConfirm::IOHandlerConfirm (Debugger &debugger, 17344d93782SGreg Clayton const char *prompt, 17444d93782SGreg Clayton bool default_response) : 17544d93782SGreg Clayton IOHandlerEditline(debugger, 176e30f11d9SKate Stone IOHandler::Type::Confirm, 17744d93782SGreg Clayton NULL, // NULL editline_name means no history loaded/saved 178e30f11d9SKate Stone NULL, // No prompt 179e30f11d9SKate Stone NULL, // No continuation prompt 18044d93782SGreg Clayton false, // Multi-line 181e30f11d9SKate Stone false, // Don't colorize the prompt (i.e. the confirm message.) 182f6913cd7SGreg Clayton 0, 18344d93782SGreg Clayton *this), 18444d93782SGreg Clayton m_default_response (default_response), 18544d93782SGreg Clayton m_user_response (default_response) 18644d93782SGreg Clayton { 18744d93782SGreg Clayton StreamString prompt_stream; 18844d93782SGreg Clayton prompt_stream.PutCString(prompt); 18944d93782SGreg Clayton if (m_default_response) 19044d93782SGreg Clayton prompt_stream.Printf(": [Y/n] "); 19144d93782SGreg Clayton else 19244d93782SGreg Clayton prompt_stream.Printf(": [y/N] "); 19344d93782SGreg Clayton 19444d93782SGreg Clayton SetPrompt (prompt_stream.GetString().c_str()); 19544d93782SGreg Clayton 19644d93782SGreg Clayton } 19744d93782SGreg Clayton 19844d93782SGreg Clayton 19944d93782SGreg Clayton IOHandlerConfirm::~IOHandlerConfirm () 20044d93782SGreg Clayton { 20144d93782SGreg Clayton } 20244d93782SGreg Clayton 20344d93782SGreg Clayton int 20444d93782SGreg Clayton IOHandlerConfirm::IOHandlerComplete (IOHandler &io_handler, 20544d93782SGreg Clayton const char *current_line, 20644d93782SGreg Clayton const char *cursor, 20744d93782SGreg Clayton const char *last_char, 20844d93782SGreg Clayton int skip_first_n_matches, 20944d93782SGreg Clayton int max_matches, 21044d93782SGreg Clayton StringList &matches) 21144d93782SGreg Clayton { 21244d93782SGreg Clayton if (current_line == cursor) 21344d93782SGreg Clayton { 21444d93782SGreg Clayton if (m_default_response) 21544d93782SGreg Clayton { 21644d93782SGreg Clayton matches.AppendString("y"); 21744d93782SGreg Clayton } 21844d93782SGreg Clayton else 21944d93782SGreg Clayton { 22044d93782SGreg Clayton matches.AppendString("n"); 22144d93782SGreg Clayton } 22244d93782SGreg Clayton } 22344d93782SGreg Clayton return matches.GetSize(); 22444d93782SGreg Clayton } 22544d93782SGreg Clayton 22644d93782SGreg Clayton void 22744d93782SGreg Clayton IOHandlerConfirm::IOHandlerInputComplete (IOHandler &io_handler, std::string &line) 22844d93782SGreg Clayton { 22944d93782SGreg Clayton if (line.empty()) 23044d93782SGreg Clayton { 23144d93782SGreg Clayton // User just hit enter, set the response to the default 23244d93782SGreg Clayton m_user_response = m_default_response; 23344d93782SGreg Clayton io_handler.SetIsDone(true); 23444d93782SGreg Clayton return; 23544d93782SGreg Clayton } 23644d93782SGreg Clayton 23744d93782SGreg Clayton if (line.size() == 1) 23844d93782SGreg Clayton { 23944d93782SGreg Clayton switch (line[0]) 24044d93782SGreg Clayton { 24144d93782SGreg Clayton case 'y': 24244d93782SGreg Clayton case 'Y': 24344d93782SGreg Clayton m_user_response = true; 24444d93782SGreg Clayton io_handler.SetIsDone(true); 24544d93782SGreg Clayton return; 24644d93782SGreg Clayton case 'n': 24744d93782SGreg Clayton case 'N': 24844d93782SGreg Clayton m_user_response = false; 24944d93782SGreg Clayton io_handler.SetIsDone(true); 25044d93782SGreg Clayton return; 25144d93782SGreg Clayton default: 25244d93782SGreg Clayton break; 25344d93782SGreg Clayton } 25444d93782SGreg Clayton } 25544d93782SGreg Clayton 25644d93782SGreg Clayton if (line == "yes" || line == "YES" || line == "Yes") 25744d93782SGreg Clayton { 25844d93782SGreg Clayton m_user_response = true; 25944d93782SGreg Clayton io_handler.SetIsDone(true); 26044d93782SGreg Clayton } 26144d93782SGreg Clayton else if (line == "no" || line == "NO" || line == "No") 26244d93782SGreg Clayton { 26344d93782SGreg Clayton m_user_response = false; 26444d93782SGreg Clayton io_handler.SetIsDone(true); 26544d93782SGreg Clayton } 26644d93782SGreg Clayton } 26744d93782SGreg Clayton 26844d93782SGreg Clayton int 26944d93782SGreg Clayton IOHandlerDelegate::IOHandlerComplete (IOHandler &io_handler, 27044d93782SGreg Clayton const char *current_line, 27144d93782SGreg Clayton const char *cursor, 27244d93782SGreg Clayton const char *last_char, 27344d93782SGreg Clayton int skip_first_n_matches, 27444d93782SGreg Clayton int max_matches, 27544d93782SGreg Clayton StringList &matches) 27644d93782SGreg Clayton { 27744d93782SGreg Clayton switch (m_completion) 27844d93782SGreg Clayton { 27944d93782SGreg Clayton case Completion::None: 28044d93782SGreg Clayton break; 28144d93782SGreg Clayton 28244d93782SGreg Clayton case Completion::LLDBCommand: 28344d93782SGreg Clayton return io_handler.GetDebugger().GetCommandInterpreter().HandleCompletion (current_line, 28444d93782SGreg Clayton cursor, 28544d93782SGreg Clayton last_char, 28644d93782SGreg Clayton skip_first_n_matches, 28744d93782SGreg Clayton max_matches, 28844d93782SGreg Clayton matches); 28944d93782SGreg Clayton 29044d93782SGreg Clayton case Completion::Expression: 29144d93782SGreg Clayton { 29244d93782SGreg Clayton bool word_complete = false; 29344d93782SGreg Clayton const char *word_start = cursor; 29444d93782SGreg Clayton if (cursor > current_line) 29544d93782SGreg Clayton --word_start; 29644d93782SGreg Clayton while (word_start > current_line && !isspace(*word_start)) 29744d93782SGreg Clayton --word_start; 29844d93782SGreg Clayton CommandCompletions::InvokeCommonCompletionCallbacks (io_handler.GetDebugger().GetCommandInterpreter(), 29944d93782SGreg Clayton CommandCompletions::eVariablePathCompletion, 30044d93782SGreg Clayton word_start, 30144d93782SGreg Clayton skip_first_n_matches, 30244d93782SGreg Clayton max_matches, 30344d93782SGreg Clayton NULL, 30444d93782SGreg Clayton word_complete, 30544d93782SGreg Clayton matches); 30644d93782SGreg Clayton 30744d93782SGreg Clayton size_t num_matches = matches.GetSize(); 30844d93782SGreg Clayton if (num_matches > 0) 30944d93782SGreg Clayton { 31044d93782SGreg Clayton std::string common_prefix; 31144d93782SGreg Clayton matches.LongestCommonPrefix (common_prefix); 31244d93782SGreg Clayton const size_t partial_name_len = strlen(word_start); 31344d93782SGreg Clayton 31444d93782SGreg Clayton // If we matched a unique single command, add a space... 31544d93782SGreg Clayton // Only do this if the completer told us this was a complete word, however... 31644d93782SGreg Clayton if (num_matches == 1 && word_complete) 31744d93782SGreg Clayton { 31844d93782SGreg Clayton common_prefix.push_back(' '); 31944d93782SGreg Clayton } 32044d93782SGreg Clayton common_prefix.erase (0, partial_name_len); 32144d93782SGreg Clayton matches.InsertStringAtIndex(0, std::move(common_prefix)); 32244d93782SGreg Clayton } 32344d93782SGreg Clayton return num_matches; 32444d93782SGreg Clayton } 32544d93782SGreg Clayton break; 32644d93782SGreg Clayton } 32744d93782SGreg Clayton 32844d93782SGreg Clayton 32944d93782SGreg Clayton return 0; 33044d93782SGreg Clayton } 33144d93782SGreg Clayton 33244d93782SGreg Clayton 33344d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger, 334e30f11d9SKate Stone IOHandler::Type type, 33544d93782SGreg Clayton const char *editline_name, // Used for saving history files 33644d93782SGreg Clayton const char *prompt, 337e30f11d9SKate Stone const char *continuation_prompt, 33844d93782SGreg Clayton bool multi_line, 339e30f11d9SKate Stone bool color_prompts, 340f6913cd7SGreg Clayton uint32_t line_number_start, 34144d93782SGreg Clayton IOHandlerDelegate &delegate) : 34244d93782SGreg Clayton IOHandlerEditline(debugger, 343e30f11d9SKate Stone type, 34444d93782SGreg Clayton StreamFileSP(), // Inherit input from top input reader 34544d93782SGreg Clayton StreamFileSP(), // Inherit output from top input reader 34644d93782SGreg Clayton StreamFileSP(), // Inherit error from top input reader 347340b0309SGreg Clayton 0, // Flags 34844d93782SGreg Clayton editline_name, // Used for saving history files 34944d93782SGreg Clayton prompt, 350e30f11d9SKate Stone continuation_prompt, 35144d93782SGreg Clayton multi_line, 352e30f11d9SKate Stone color_prompts, 353f6913cd7SGreg Clayton line_number_start, 35444d93782SGreg Clayton delegate) 35544d93782SGreg Clayton { 35644d93782SGreg Clayton } 35744d93782SGreg Clayton 35844d93782SGreg Clayton IOHandlerEditline::IOHandlerEditline (Debugger &debugger, 359e30f11d9SKate Stone IOHandler::Type type, 36044d93782SGreg Clayton const lldb::StreamFileSP &input_sp, 36144d93782SGreg Clayton const lldb::StreamFileSP &output_sp, 36244d93782SGreg Clayton const lldb::StreamFileSP &error_sp, 363340b0309SGreg Clayton uint32_t flags, 36444d93782SGreg Clayton const char *editline_name, // Used for saving history files 36544d93782SGreg Clayton const char *prompt, 366e30f11d9SKate Stone const char *continuation_prompt, 36744d93782SGreg Clayton bool multi_line, 368e30f11d9SKate Stone bool color_prompts, 369f6913cd7SGreg Clayton uint32_t line_number_start, 37044d93782SGreg Clayton IOHandlerDelegate &delegate) : 371e30f11d9SKate Stone IOHandler (debugger, type, input_sp, output_sp, error_sp, flags), 372cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 37344d93782SGreg Clayton m_editline_ap (), 374cacde7dfSTodd Fiala #endif 37544d93782SGreg Clayton m_delegate (delegate), 37644d93782SGreg Clayton m_prompt (), 377e30f11d9SKate Stone m_continuation_prompt(), 378e30f11d9SKate Stone m_current_lines_ptr (NULL), 379f6913cd7SGreg Clayton m_base_line_number (line_number_start), 380e30f11d9SKate Stone m_curr_line_idx (UINT32_MAX), 381e30f11d9SKate Stone m_multi_line (multi_line), 382e30f11d9SKate Stone m_color_prompts (color_prompts), 383e30f11d9SKate Stone m_interrupt_exits (true) 38444d93782SGreg Clayton { 38544d93782SGreg Clayton SetPrompt(prompt); 38644d93782SGreg Clayton 387cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 388914b8d98SDeepak Panickal bool use_editline = false; 389340b0309SGreg Clayton 390340b0309SGreg Clayton use_editline = m_input_sp->GetFile().GetIsRealTerminal(); 39144d93782SGreg Clayton 39244d93782SGreg Clayton if (use_editline) 39344d93782SGreg Clayton { 39444d93782SGreg Clayton m_editline_ap.reset(new Editline (editline_name, 39544d93782SGreg Clayton GetInputFILE (), 39644d93782SGreg Clayton GetOutputFILE (), 397e30f11d9SKate Stone GetErrorFILE (), 398e30f11d9SKate Stone m_color_prompts)); 399e30f11d9SKate Stone m_editline_ap->SetIsInputCompleteCallback (IsInputCompleteCallback, this); 40044d93782SGreg Clayton m_editline_ap->SetAutoCompleteCallback (AutoCompleteCallback, this); 401e30f11d9SKate Stone // See if the delegate supports fixing indentation 402e30f11d9SKate Stone const char *indent_chars = delegate.IOHandlerGetFixIndentationCharacters(); 403e30f11d9SKate Stone if (indent_chars) 404e30f11d9SKate Stone { 405e30f11d9SKate Stone // The delegate does support indentation, hook it up so when any indentation 406e30f11d9SKate Stone // character is typed, the delegate gets a chance to fix it 407e30f11d9SKate Stone m_editline_ap->SetFixIndentationCallback (FixIndentationCallback, this, indent_chars); 408e30f11d9SKate Stone } 40944d93782SGreg Clayton } 410cacde7dfSTodd Fiala #endif 411e30f11d9SKate Stone SetBaseLineNumber (m_base_line_number); 412e30f11d9SKate Stone SetPrompt(prompt ? prompt : ""); 413e30f11d9SKate Stone SetContinuationPrompt(continuation_prompt); 41444d93782SGreg Clayton } 41544d93782SGreg Clayton 41644d93782SGreg Clayton IOHandlerEditline::~IOHandlerEditline () 41744d93782SGreg Clayton { 418cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 41944d93782SGreg Clayton m_editline_ap.reset(); 420cacde7dfSTodd Fiala #endif 42144d93782SGreg Clayton } 42244d93782SGreg Clayton 423e30f11d9SKate Stone void 424e30f11d9SKate Stone IOHandlerEditline::Activate () 425e30f11d9SKate Stone { 426e30f11d9SKate Stone IOHandler::Activate(); 427e30f11d9SKate Stone m_delegate.IOHandlerActivated(*this); 428e30f11d9SKate Stone } 429e30f11d9SKate Stone 430e30f11d9SKate Stone void 431e30f11d9SKate Stone IOHandlerEditline::Deactivate () 432e30f11d9SKate Stone { 433e30f11d9SKate Stone IOHandler::Deactivate(); 434e30f11d9SKate Stone m_delegate.IOHandlerDeactivated(*this); 435e30f11d9SKate Stone } 436e30f11d9SKate Stone 43744d93782SGreg Clayton 43844d93782SGreg Clayton bool 439f0066ad0SGreg Clayton IOHandlerEditline::GetLine (std::string &line, bool &interrupted) 44044d93782SGreg Clayton { 441cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 44244d93782SGreg Clayton if (m_editline_ap) 44344d93782SGreg Clayton { 444e30f11d9SKate Stone return m_editline_ap->GetLine (line, interrupted); 44544d93782SGreg Clayton } 44644d93782SGreg Clayton else 44744d93782SGreg Clayton { 448cacde7dfSTodd Fiala #endif 44944d93782SGreg Clayton line.clear(); 45044d93782SGreg Clayton 45144d93782SGreg Clayton FILE *in = GetInputFILE(); 45244d93782SGreg Clayton if (in) 45344d93782SGreg Clayton { 454340b0309SGreg Clayton if (GetIsInteractive()) 45544d93782SGreg Clayton { 456e30f11d9SKate Stone const char *prompt = NULL; 457e30f11d9SKate Stone 458e30f11d9SKate Stone if (m_multi_line && m_curr_line_idx > 0) 459e30f11d9SKate Stone prompt = GetContinuationPrompt(); 460e30f11d9SKate Stone 461e30f11d9SKate Stone if (prompt == NULL) 462e30f11d9SKate Stone prompt = GetPrompt(); 463e30f11d9SKate Stone 46444d93782SGreg Clayton if (prompt && prompt[0]) 46544d93782SGreg Clayton { 46644d93782SGreg Clayton FILE *out = GetOutputFILE(); 46744d93782SGreg Clayton if (out) 46844d93782SGreg Clayton { 46944d93782SGreg Clayton ::fprintf(out, "%s", prompt); 47044d93782SGreg Clayton ::fflush(out); 47144d93782SGreg Clayton } 47244d93782SGreg Clayton } 47344d93782SGreg Clayton } 47444d93782SGreg Clayton char buffer[256]; 47544d93782SGreg Clayton bool done = false; 4760f86e6e7SGreg Clayton bool got_line = false; 47744d93782SGreg Clayton while (!done) 47844d93782SGreg Clayton { 47944d93782SGreg Clayton if (fgets(buffer, sizeof(buffer), in) == NULL) 480c9cf5798SGreg Clayton { 481c7797accSGreg Clayton const int saved_errno = errno; 482c9cf5798SGreg Clayton if (feof(in)) 48344d93782SGreg Clayton done = true; 484c7797accSGreg Clayton else if (ferror(in)) 485c7797accSGreg Clayton { 486c7797accSGreg Clayton if (saved_errno != EINTR) 487c7797accSGreg Clayton done = true; 488c7797accSGreg Clayton } 489c9cf5798SGreg Clayton } 49044d93782SGreg Clayton else 49144d93782SGreg Clayton { 4920f86e6e7SGreg Clayton got_line = true; 49344d93782SGreg Clayton size_t buffer_len = strlen(buffer); 49444d93782SGreg Clayton assert (buffer[buffer_len] == '\0'); 49544d93782SGreg Clayton char last_char = buffer[buffer_len-1]; 49644d93782SGreg Clayton if (last_char == '\r' || last_char == '\n') 49744d93782SGreg Clayton { 49844d93782SGreg Clayton done = true; 49944d93782SGreg Clayton // Strip trailing newlines 50044d93782SGreg Clayton while (last_char == '\r' || last_char == '\n') 50144d93782SGreg Clayton { 50244d93782SGreg Clayton --buffer_len; 50344d93782SGreg Clayton if (buffer_len == 0) 50444d93782SGreg Clayton break; 50544d93782SGreg Clayton last_char = buffer[buffer_len-1]; 50644d93782SGreg Clayton } 50744d93782SGreg Clayton } 50844d93782SGreg Clayton line.append(buffer, buffer_len); 50944d93782SGreg Clayton } 51044d93782SGreg Clayton } 5110f86e6e7SGreg Clayton // We might have gotten a newline on a line by itself 5120f86e6e7SGreg Clayton // make sure to return true in this case. 5130f86e6e7SGreg Clayton return got_line; 51444d93782SGreg Clayton } 51544d93782SGreg Clayton else 51644d93782SGreg Clayton { 51744d93782SGreg Clayton // No more input file, we are done... 51844d93782SGreg Clayton SetIsDone(true); 51944d93782SGreg Clayton } 520340b0309SGreg Clayton return false; 521cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 52244d93782SGreg Clayton } 523cacde7dfSTodd Fiala #endif 52444d93782SGreg Clayton } 52544d93782SGreg Clayton 52644d93782SGreg Clayton 527cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 528e30f11d9SKate Stone bool 529e30f11d9SKate Stone IOHandlerEditline::IsInputCompleteCallback (Editline *editline, 53044d93782SGreg Clayton StringList &lines, 53144d93782SGreg Clayton void *baton) 53244d93782SGreg Clayton { 53344d93782SGreg Clayton IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton; 534e30f11d9SKate Stone return editline_reader->m_delegate.IOHandlerIsInputComplete(*editline_reader, lines); 535e30f11d9SKate Stone } 536e30f11d9SKate Stone 537e30f11d9SKate Stone int 538e30f11d9SKate Stone IOHandlerEditline::FixIndentationCallback (Editline *editline, 539e30f11d9SKate Stone const StringList &lines, 540e30f11d9SKate Stone int cursor_position, 541e30f11d9SKate Stone void *baton) 542e30f11d9SKate Stone { 543e30f11d9SKate Stone IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton; 544e30f11d9SKate Stone return editline_reader->m_delegate.IOHandlerFixIndentation(*editline_reader, lines, cursor_position); 54544d93782SGreg Clayton } 54644d93782SGreg Clayton 54744d93782SGreg Clayton int 54844d93782SGreg Clayton IOHandlerEditline::AutoCompleteCallback (const char *current_line, 54944d93782SGreg Clayton const char *cursor, 55044d93782SGreg Clayton const char *last_char, 55144d93782SGreg Clayton int skip_first_n_matches, 55244d93782SGreg Clayton int max_matches, 55344d93782SGreg Clayton StringList &matches, 55444d93782SGreg Clayton void *baton) 55544d93782SGreg Clayton { 55644d93782SGreg Clayton IOHandlerEditline *editline_reader = (IOHandlerEditline *) baton; 55744d93782SGreg Clayton if (editline_reader) 55844d93782SGreg Clayton return editline_reader->m_delegate.IOHandlerComplete (*editline_reader, 55944d93782SGreg Clayton current_line, 56044d93782SGreg Clayton cursor, 56144d93782SGreg Clayton last_char, 56244d93782SGreg Clayton skip_first_n_matches, 56344d93782SGreg Clayton max_matches, 56444d93782SGreg Clayton matches); 56544d93782SGreg Clayton return 0; 56644d93782SGreg Clayton } 567cacde7dfSTodd Fiala #endif 56844d93782SGreg Clayton 56944d93782SGreg Clayton const char * 57044d93782SGreg Clayton IOHandlerEditline::GetPrompt () 57144d93782SGreg Clayton { 572cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 57344d93782SGreg Clayton if (m_editline_ap) 574cacde7dfSTodd Fiala { 57544d93782SGreg Clayton return m_editline_ap->GetPrompt (); 576cacde7dfSTodd Fiala } 577cacde7dfSTodd Fiala else 578cacde7dfSTodd Fiala { 579cacde7dfSTodd Fiala #endif 580cacde7dfSTodd Fiala if (m_prompt.empty()) 58144d93782SGreg Clayton return NULL; 582cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 583cacde7dfSTodd Fiala } 584cacde7dfSTodd Fiala #endif 58544d93782SGreg Clayton return m_prompt.c_str(); 58644d93782SGreg Clayton } 58744d93782SGreg Clayton 58844d93782SGreg Clayton bool 58944d93782SGreg Clayton IOHandlerEditline::SetPrompt (const char *p) 59044d93782SGreg Clayton { 59144d93782SGreg Clayton if (p && p[0]) 59244d93782SGreg Clayton m_prompt = p; 59344d93782SGreg Clayton else 59444d93782SGreg Clayton m_prompt.clear(); 595cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 59644d93782SGreg Clayton if (m_editline_ap) 59744d93782SGreg Clayton m_editline_ap->SetPrompt (m_prompt.empty() ? NULL : m_prompt.c_str()); 598cacde7dfSTodd Fiala #endif 59944d93782SGreg Clayton return true; 60044d93782SGreg Clayton } 60144d93782SGreg Clayton 602e30f11d9SKate Stone const char * 603e30f11d9SKate Stone IOHandlerEditline::GetContinuationPrompt () 604e30f11d9SKate Stone { 605e30f11d9SKate Stone if (m_continuation_prompt.empty()) 606e30f11d9SKate Stone return NULL; 607e30f11d9SKate Stone return m_continuation_prompt.c_str(); 608e30f11d9SKate Stone } 609e30f11d9SKate Stone 610e30f11d9SKate Stone 611e30f11d9SKate Stone void 612e30f11d9SKate Stone IOHandlerEditline::SetContinuationPrompt (const char *p) 613e30f11d9SKate Stone { 614e30f11d9SKate Stone if (p && p[0]) 615e30f11d9SKate Stone m_continuation_prompt = p; 616e30f11d9SKate Stone else 617e30f11d9SKate Stone m_continuation_prompt.clear(); 618e30f11d9SKate Stone 619d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT 620e30f11d9SKate Stone if (m_editline_ap) 621e30f11d9SKate Stone m_editline_ap->SetContinuationPrompt (m_continuation_prompt.empty() ? NULL : m_continuation_prompt.c_str()); 622d553d00cSZachary Turner #endif 623e30f11d9SKate Stone } 624e30f11d9SKate Stone 625e30f11d9SKate Stone 626f6913cd7SGreg Clayton void 627f6913cd7SGreg Clayton IOHandlerEditline::SetBaseLineNumber (uint32_t line) 628f6913cd7SGreg Clayton { 629f6913cd7SGreg Clayton m_base_line_number = line; 630f6913cd7SGreg Clayton } 631e30f11d9SKate Stone 632e30f11d9SKate Stone uint32_t 633e30f11d9SKate Stone IOHandlerEditline::GetCurrentLineIndex () const 634e30f11d9SKate Stone { 635d553d00cSZachary Turner #ifndef LLDB_DISABLE_LIBEDIT 636e30f11d9SKate Stone if (m_editline_ap) 637e30f11d9SKate Stone return m_editline_ap->GetCurrentLine(); 638e30f11d9SKate Stone #endif 639e30f11d9SKate Stone return m_curr_line_idx; 640e30f11d9SKate Stone } 641e30f11d9SKate Stone 64244d93782SGreg Clayton bool 643f0066ad0SGreg Clayton IOHandlerEditline::GetLines (StringList &lines, bool &interrupted) 64444d93782SGreg Clayton { 645e30f11d9SKate Stone m_current_lines_ptr = &lines; 646e30f11d9SKate Stone 64744d93782SGreg Clayton bool success = false; 648cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 64944d93782SGreg Clayton if (m_editline_ap) 65044d93782SGreg Clayton { 651e30f11d9SKate Stone return m_editline_ap->GetLines (m_base_line_number, lines, interrupted); 65244d93782SGreg Clayton } 65344d93782SGreg Clayton else 65444d93782SGreg Clayton { 655cacde7dfSTodd Fiala #endif 656e30f11d9SKate Stone bool done = false; 657c3d874a5SGreg Clayton Error error; 65844d93782SGreg Clayton 659e30f11d9SKate Stone while (!done) 66044d93782SGreg Clayton { 661f6913cd7SGreg Clayton // Show line numbers if we are asked to 66244d93782SGreg Clayton std::string line; 663f6913cd7SGreg Clayton if (m_base_line_number > 0 && GetIsInteractive()) 664f6913cd7SGreg Clayton { 665f6913cd7SGreg Clayton FILE *out = GetOutputFILE(); 666f6913cd7SGreg Clayton if (out) 667bc88d938SGreg Clayton ::fprintf(out, "%u%s", m_base_line_number + (uint32_t)lines.GetSize(), GetPrompt() == NULL ? " " : ""); 668f6913cd7SGreg Clayton } 669f6913cd7SGreg Clayton 670e30f11d9SKate Stone m_curr_line_idx = lines.GetSize(); 671e30f11d9SKate Stone 672f0066ad0SGreg Clayton bool interrupted = false; 673e30f11d9SKate Stone if (GetLine(line, interrupted) && !interrupted) 67444d93782SGreg Clayton { 67544d93782SGreg Clayton lines.AppendString(line); 676e30f11d9SKate Stone done = m_delegate.IOHandlerIsInputComplete(*this, lines); 677f0066ad0SGreg Clayton } 67844d93782SGreg Clayton else 67944d93782SGreg Clayton { 680e30f11d9SKate Stone done = true; 68144d93782SGreg Clayton } 68244d93782SGreg Clayton } 68344d93782SGreg Clayton success = lines.GetSize() > 0; 684cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 68544d93782SGreg Clayton } 686cacde7dfSTodd Fiala #endif 68744d93782SGreg Clayton return success; 68844d93782SGreg Clayton } 68944d93782SGreg Clayton 69044d93782SGreg Clayton // Each IOHandler gets to run until it is done. It should read data 69144d93782SGreg Clayton // from the "in" and place output into "out" and "err and return 69244d93782SGreg Clayton // when done. 69344d93782SGreg Clayton void 69444d93782SGreg Clayton IOHandlerEditline::Run () 69544d93782SGreg Clayton { 69644d93782SGreg Clayton std::string line; 69744d93782SGreg Clayton while (IsActive()) 69844d93782SGreg Clayton { 699f0066ad0SGreg Clayton bool interrupted = false; 70044d93782SGreg Clayton if (m_multi_line) 70144d93782SGreg Clayton { 70244d93782SGreg Clayton StringList lines; 703f0066ad0SGreg Clayton if (GetLines (lines, interrupted)) 704f0066ad0SGreg Clayton { 705f0066ad0SGreg Clayton if (interrupted) 706f0066ad0SGreg Clayton { 707e30f11d9SKate Stone m_done = m_interrupt_exits; 708e30f11d9SKate Stone m_delegate.IOHandlerInputInterrupted (*this, line); 709e30f11d9SKate Stone 710f0066ad0SGreg Clayton } 711f0066ad0SGreg Clayton else 71244d93782SGreg Clayton { 71344d93782SGreg Clayton line = lines.CopyList(); 71444d93782SGreg Clayton m_delegate.IOHandlerInputComplete (*this, line); 71544d93782SGreg Clayton } 716f0066ad0SGreg Clayton } 71744d93782SGreg Clayton else 71844d93782SGreg Clayton { 71944d93782SGreg Clayton m_done = true; 72044d93782SGreg Clayton } 72144d93782SGreg Clayton } 72244d93782SGreg Clayton else 72344d93782SGreg Clayton { 724f0066ad0SGreg Clayton if (GetLine(line, interrupted)) 72544d93782SGreg Clayton { 726e30f11d9SKate Stone if (interrupted) 727e30f11d9SKate Stone m_delegate.IOHandlerInputInterrupted (*this, line); 728e30f11d9SKate Stone else 72944d93782SGreg Clayton m_delegate.IOHandlerInputComplete (*this, line); 73044d93782SGreg Clayton } 73144d93782SGreg Clayton else 73244d93782SGreg Clayton { 73344d93782SGreg Clayton m_done = true; 73444d93782SGreg Clayton } 73544d93782SGreg Clayton } 73644d93782SGreg Clayton } 73744d93782SGreg Clayton } 73844d93782SGreg Clayton 73944d93782SGreg Clayton void 74044d93782SGreg Clayton IOHandlerEditline::Hide () 74144d93782SGreg Clayton { 742cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 743b89b7496SGreg Clayton if (m_editline_ap) 74444d93782SGreg Clayton m_editline_ap->Hide(); 745cacde7dfSTodd Fiala #endif 74644d93782SGreg Clayton } 74744d93782SGreg Clayton 74844d93782SGreg Clayton 74944d93782SGreg Clayton void 75044d93782SGreg Clayton IOHandlerEditline::Refresh () 75144d93782SGreg Clayton { 752cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 753b89b7496SGreg Clayton if (m_editline_ap) 754b89b7496SGreg Clayton { 75544d93782SGreg Clayton m_editline_ap->Refresh(); 756b89b7496SGreg Clayton } 75744d93782SGreg Clayton else 75844d93782SGreg Clayton { 759cacde7dfSTodd Fiala #endif 76044d93782SGreg Clayton const char *prompt = GetPrompt(); 76144d93782SGreg Clayton if (prompt && prompt[0]) 76244d93782SGreg Clayton { 76344d93782SGreg Clayton FILE *out = GetOutputFILE(); 76444d93782SGreg Clayton if (out) 76544d93782SGreg Clayton { 76644d93782SGreg Clayton ::fprintf(out, "%s", prompt); 76744d93782SGreg Clayton ::fflush(out); 76844d93782SGreg Clayton } 76944d93782SGreg Clayton } 770cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 77144d93782SGreg Clayton } 772cacde7dfSTodd Fiala #endif 77344d93782SGreg Clayton } 77444d93782SGreg Clayton 77544d93782SGreg Clayton void 776e68f5d6bSGreg Clayton IOHandlerEditline::Cancel () 777e68f5d6bSGreg Clayton { 778cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 779e68f5d6bSGreg Clayton if (m_editline_ap) 780e68f5d6bSGreg Clayton m_editline_ap->Interrupt (); 781cacde7dfSTodd Fiala #endif 782e68f5d6bSGreg Clayton } 783e68f5d6bSGreg Clayton 784f0066ad0SGreg Clayton bool 78544d93782SGreg Clayton IOHandlerEditline::Interrupt () 78644d93782SGreg Clayton { 787f0066ad0SGreg Clayton // Let the delgate handle it first 788f0066ad0SGreg Clayton if (m_delegate.IOHandlerInterrupt(*this)) 789f0066ad0SGreg Clayton return true; 790f0066ad0SGreg Clayton 791cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 79244d93782SGreg Clayton if (m_editline_ap) 793f0066ad0SGreg Clayton return m_editline_ap->Interrupt(); 794cacde7dfSTodd Fiala #endif 795f0066ad0SGreg Clayton return false; 79644d93782SGreg Clayton } 79744d93782SGreg Clayton 79844d93782SGreg Clayton void 79944d93782SGreg Clayton IOHandlerEditline::GotEOF() 80044d93782SGreg Clayton { 801cacde7dfSTodd Fiala #ifndef LLDB_DISABLE_LIBEDIT 80244d93782SGreg Clayton if (m_editline_ap) 80344d93782SGreg Clayton m_editline_ap->Interrupt(); 804cacde7dfSTodd Fiala #endif 80544d93782SGreg Clayton } 80644d93782SGreg Clayton 807914b8d98SDeepak Panickal // we may want curses to be disabled for some builds 808914b8d98SDeepak Panickal // for instance, windows 809914b8d98SDeepak Panickal #ifndef LLDB_DISABLE_CURSES 810914b8d98SDeepak Panickal 81144d93782SGreg Clayton #include "lldb/Core/ValueObject.h" 81244d93782SGreg Clayton #include "lldb/Symbol/VariableList.h" 81344d93782SGreg Clayton #include "lldb/Target/Target.h" 81444d93782SGreg Clayton #include "lldb/Target/Process.h" 81544d93782SGreg Clayton #include "lldb/Target/Thread.h" 81644d93782SGreg Clayton #include "lldb/Target/StackFrame.h" 81744d93782SGreg Clayton 81844d93782SGreg Clayton #define KEY_RETURN 10 81944d93782SGreg Clayton #define KEY_ESCAPE 27 82044d93782SGreg Clayton 82144d93782SGreg Clayton namespace curses 82244d93782SGreg Clayton { 82344d93782SGreg Clayton class Menu; 82444d93782SGreg Clayton class MenuDelegate; 82544d93782SGreg Clayton class Window; 82644d93782SGreg Clayton class WindowDelegate; 82744d93782SGreg Clayton typedef std::shared_ptr<Menu> MenuSP; 82844d93782SGreg Clayton typedef std::shared_ptr<MenuDelegate> MenuDelegateSP; 82944d93782SGreg Clayton typedef std::shared_ptr<Window> WindowSP; 83044d93782SGreg Clayton typedef std::shared_ptr<WindowDelegate> WindowDelegateSP; 83144d93782SGreg Clayton typedef std::vector<MenuSP> Menus; 83244d93782SGreg Clayton typedef std::vector<WindowSP> Windows; 83344d93782SGreg Clayton typedef std::vector<WindowDelegateSP> WindowDelegates; 83444d93782SGreg Clayton 83544d93782SGreg Clayton #if 0 83644d93782SGreg Clayton type summary add -s "x=${var.x}, y=${var.y}" curses::Point 83744d93782SGreg Clayton type summary add -s "w=${var.width}, h=${var.height}" curses::Size 83844d93782SGreg Clayton type summary add -s "${var.origin%S} ${var.size%S}" curses::Rect 83944d93782SGreg Clayton #endif 84044d93782SGreg Clayton struct Point 84144d93782SGreg Clayton { 84244d93782SGreg Clayton int x; 84344d93782SGreg Clayton int y; 84444d93782SGreg Clayton 84544d93782SGreg Clayton Point (int _x = 0, int _y = 0) : 84644d93782SGreg Clayton x(_x), 84744d93782SGreg Clayton y(_y) 84844d93782SGreg Clayton { 84944d93782SGreg Clayton } 85044d93782SGreg Clayton 85144d93782SGreg Clayton void 85244d93782SGreg Clayton Clear () 85344d93782SGreg Clayton { 85444d93782SGreg Clayton x = 0; 85544d93782SGreg Clayton y = 0; 85644d93782SGreg Clayton } 85744d93782SGreg Clayton 85844d93782SGreg Clayton Point & 85944d93782SGreg Clayton operator += (const Point &rhs) 86044d93782SGreg Clayton { 86144d93782SGreg Clayton x += rhs.x; 86244d93782SGreg Clayton y += rhs.y; 86344d93782SGreg Clayton return *this; 86444d93782SGreg Clayton } 86544d93782SGreg Clayton 86644d93782SGreg Clayton void 86744d93782SGreg Clayton Dump () 86844d93782SGreg Clayton { 86944d93782SGreg Clayton printf ("(x=%i, y=%i)\n", x, y); 87044d93782SGreg Clayton } 87144d93782SGreg Clayton 87244d93782SGreg Clayton }; 87344d93782SGreg Clayton 87444d93782SGreg Clayton bool operator == (const Point &lhs, const Point &rhs) 87544d93782SGreg Clayton { 87644d93782SGreg Clayton return lhs.x == rhs.x && lhs.y == rhs.y; 87744d93782SGreg Clayton } 87844d93782SGreg Clayton bool operator != (const Point &lhs, const Point &rhs) 87944d93782SGreg Clayton { 88044d93782SGreg Clayton return lhs.x != rhs.x || lhs.y != rhs.y; 88144d93782SGreg Clayton } 88244d93782SGreg Clayton 88344d93782SGreg Clayton struct Size 88444d93782SGreg Clayton { 88544d93782SGreg Clayton int width; 88644d93782SGreg Clayton int height; 88744d93782SGreg Clayton Size (int w = 0, int h = 0) : 88844d93782SGreg Clayton width (w), 88944d93782SGreg Clayton height (h) 89044d93782SGreg Clayton { 89144d93782SGreg Clayton } 89244d93782SGreg Clayton 89344d93782SGreg Clayton void 89444d93782SGreg Clayton Clear () 89544d93782SGreg Clayton { 89644d93782SGreg Clayton width = 0; 89744d93782SGreg Clayton height = 0; 89844d93782SGreg Clayton } 89944d93782SGreg Clayton 90044d93782SGreg Clayton void 90144d93782SGreg Clayton Dump () 90244d93782SGreg Clayton { 90344d93782SGreg Clayton printf ("(w=%i, h=%i)\n", width, height); 90444d93782SGreg Clayton } 90544d93782SGreg Clayton 90644d93782SGreg Clayton }; 90744d93782SGreg Clayton 90844d93782SGreg Clayton bool operator == (const Size &lhs, const Size &rhs) 90944d93782SGreg Clayton { 91044d93782SGreg Clayton return lhs.width == rhs.width && lhs.height == rhs.height; 91144d93782SGreg Clayton } 91244d93782SGreg Clayton bool operator != (const Size &lhs, const Size &rhs) 91344d93782SGreg Clayton { 91444d93782SGreg Clayton return lhs.width != rhs.width || lhs.height != rhs.height; 91544d93782SGreg Clayton } 91644d93782SGreg Clayton 91744d93782SGreg Clayton struct Rect 91844d93782SGreg Clayton { 91944d93782SGreg Clayton Point origin; 92044d93782SGreg Clayton Size size; 92144d93782SGreg Clayton 92244d93782SGreg Clayton Rect () : 92344d93782SGreg Clayton origin(), 92444d93782SGreg Clayton size() 92544d93782SGreg Clayton { 92644d93782SGreg Clayton } 92744d93782SGreg Clayton 92844d93782SGreg Clayton Rect (const Point &p, const Size &s) : 92944d93782SGreg Clayton origin (p), 93044d93782SGreg Clayton size (s) 93144d93782SGreg Clayton { 93244d93782SGreg Clayton } 93344d93782SGreg Clayton 93444d93782SGreg Clayton void 93544d93782SGreg Clayton Clear () 93644d93782SGreg Clayton { 93744d93782SGreg Clayton origin.Clear(); 93844d93782SGreg Clayton size.Clear(); 93944d93782SGreg Clayton } 94044d93782SGreg Clayton 94144d93782SGreg Clayton void 94244d93782SGreg Clayton Dump () 94344d93782SGreg Clayton { 94444d93782SGreg Clayton printf ("(x=%i, y=%i), w=%i, h=%i)\n", origin.x, origin.y, size.width, size.height); 94544d93782SGreg Clayton } 94644d93782SGreg Clayton 94744d93782SGreg Clayton void 94844d93782SGreg Clayton Inset (int w, int h) 94944d93782SGreg Clayton { 95044d93782SGreg Clayton if (size.width > w*2) 95144d93782SGreg Clayton size.width -= w*2; 95244d93782SGreg Clayton origin.x += w; 95344d93782SGreg Clayton 95444d93782SGreg Clayton if (size.height > h*2) 95544d93782SGreg Clayton size.height -= h*2; 95644d93782SGreg Clayton origin.y += h; 95744d93782SGreg Clayton } 95844d93782SGreg Clayton // Return a status bar rectangle which is the last line of 95944d93782SGreg Clayton // this rectangle. This rectangle will be modified to not 96044d93782SGreg Clayton // include the status bar area. 96144d93782SGreg Clayton Rect 96244d93782SGreg Clayton MakeStatusBar () 96344d93782SGreg Clayton { 96444d93782SGreg Clayton Rect status_bar; 96544d93782SGreg Clayton if (size.height > 1) 96644d93782SGreg Clayton { 96744d93782SGreg Clayton status_bar.origin.x = origin.x; 96844d93782SGreg Clayton status_bar.origin.y = size.height; 96944d93782SGreg Clayton status_bar.size.width = size.width; 97044d93782SGreg Clayton status_bar.size.height = 1; 97144d93782SGreg Clayton --size.height; 97244d93782SGreg Clayton } 97344d93782SGreg Clayton return status_bar; 97444d93782SGreg Clayton } 97544d93782SGreg Clayton 97644d93782SGreg Clayton // Return a menubar rectangle which is the first line of 97744d93782SGreg Clayton // this rectangle. This rectangle will be modified to not 97844d93782SGreg Clayton // include the menubar area. 97944d93782SGreg Clayton Rect 98044d93782SGreg Clayton MakeMenuBar () 98144d93782SGreg Clayton { 98244d93782SGreg Clayton Rect menubar; 98344d93782SGreg Clayton if (size.height > 1) 98444d93782SGreg Clayton { 98544d93782SGreg Clayton menubar.origin.x = origin.x; 98644d93782SGreg Clayton menubar.origin.y = origin.y; 98744d93782SGreg Clayton menubar.size.width = size.width; 98844d93782SGreg Clayton menubar.size.height = 1; 98944d93782SGreg Clayton ++origin.y; 99044d93782SGreg Clayton --size.height; 99144d93782SGreg Clayton } 99244d93782SGreg Clayton return menubar; 99344d93782SGreg Clayton } 99444d93782SGreg Clayton 99544d93782SGreg Clayton void 99644d93782SGreg Clayton HorizontalSplitPercentage (float top_percentage, Rect &top, Rect &bottom) const 99744d93782SGreg Clayton { 99844d93782SGreg Clayton float top_height = top_percentage * size.height; 99944d93782SGreg Clayton HorizontalSplit (top_height, top, bottom); 100044d93782SGreg Clayton } 100144d93782SGreg Clayton 100244d93782SGreg Clayton void 100344d93782SGreg Clayton HorizontalSplit (int top_height, Rect &top, Rect &bottom) const 100444d93782SGreg Clayton { 100544d93782SGreg Clayton top = *this; 100644d93782SGreg Clayton if (top_height < size.height) 100744d93782SGreg Clayton { 100844d93782SGreg Clayton top.size.height = top_height; 100944d93782SGreg Clayton bottom.origin.x = origin.x; 101044d93782SGreg Clayton bottom.origin.y = origin.y + top.size.height; 101144d93782SGreg Clayton bottom.size.width = size.width; 101244d93782SGreg Clayton bottom.size.height = size.height - top.size.height; 101344d93782SGreg Clayton } 101444d93782SGreg Clayton else 101544d93782SGreg Clayton { 101644d93782SGreg Clayton bottom.Clear(); 101744d93782SGreg Clayton } 101844d93782SGreg Clayton } 101944d93782SGreg Clayton 102044d93782SGreg Clayton void 102144d93782SGreg Clayton VerticalSplitPercentage (float left_percentage, Rect &left, Rect &right) const 102244d93782SGreg Clayton { 102344d93782SGreg Clayton float left_width = left_percentage * size.width; 102444d93782SGreg Clayton VerticalSplit (left_width, left, right); 102544d93782SGreg Clayton } 102644d93782SGreg Clayton 102744d93782SGreg Clayton 102844d93782SGreg Clayton void 102944d93782SGreg Clayton VerticalSplit (int left_width, Rect &left, Rect &right) const 103044d93782SGreg Clayton { 103144d93782SGreg Clayton left = *this; 103244d93782SGreg Clayton if (left_width < size.width) 103344d93782SGreg Clayton { 103444d93782SGreg Clayton left.size.width = left_width; 103544d93782SGreg Clayton right.origin.x = origin.x + left.size.width; 103644d93782SGreg Clayton right.origin.y = origin.y; 103744d93782SGreg Clayton right.size.width = size.width - left.size.width; 103844d93782SGreg Clayton right.size.height = size.height; 103944d93782SGreg Clayton } 104044d93782SGreg Clayton else 104144d93782SGreg Clayton { 104244d93782SGreg Clayton right.Clear(); 104344d93782SGreg Clayton } 104444d93782SGreg Clayton } 104544d93782SGreg Clayton }; 104644d93782SGreg Clayton 104744d93782SGreg Clayton bool operator == (const Rect &lhs, const Rect &rhs) 104844d93782SGreg Clayton { 104944d93782SGreg Clayton return lhs.origin == rhs.origin && lhs.size == rhs.size; 105044d93782SGreg Clayton } 105144d93782SGreg Clayton bool operator != (const Rect &lhs, const Rect &rhs) 105244d93782SGreg Clayton { 105344d93782SGreg Clayton return lhs.origin != rhs.origin || lhs.size != rhs.size; 105444d93782SGreg Clayton } 105544d93782SGreg Clayton 105644d93782SGreg Clayton enum HandleCharResult 105744d93782SGreg Clayton { 105844d93782SGreg Clayton eKeyNotHandled = 0, 105944d93782SGreg Clayton eKeyHandled = 1, 106044d93782SGreg Clayton eQuitApplication = 2 106144d93782SGreg Clayton }; 106244d93782SGreg Clayton 106344d93782SGreg Clayton enum class MenuActionResult 106444d93782SGreg Clayton { 106544d93782SGreg Clayton Handled, 106644d93782SGreg Clayton NotHandled, 106744d93782SGreg Clayton Quit // Exit all menus and quit 106844d93782SGreg Clayton }; 106944d93782SGreg Clayton 107044d93782SGreg Clayton struct KeyHelp 107144d93782SGreg Clayton { 107244d93782SGreg Clayton int ch; 107344d93782SGreg Clayton const char *description; 107444d93782SGreg Clayton }; 107544d93782SGreg Clayton 107644d93782SGreg Clayton class WindowDelegate 107744d93782SGreg Clayton { 107844d93782SGreg Clayton public: 107944d93782SGreg Clayton virtual 108044d93782SGreg Clayton ~WindowDelegate() 108144d93782SGreg Clayton { 108244d93782SGreg Clayton } 108344d93782SGreg Clayton 108444d93782SGreg Clayton virtual bool 108544d93782SGreg Clayton WindowDelegateDraw (Window &window, bool force) 108644d93782SGreg Clayton { 108744d93782SGreg Clayton return false; // Drawing not handled 108844d93782SGreg Clayton } 108944d93782SGreg Clayton 109044d93782SGreg Clayton virtual HandleCharResult 109144d93782SGreg Clayton WindowDelegateHandleChar (Window &window, int key) 109244d93782SGreg Clayton { 109344d93782SGreg Clayton return eKeyNotHandled; 109444d93782SGreg Clayton } 109544d93782SGreg Clayton 109644d93782SGreg Clayton virtual const char * 109744d93782SGreg Clayton WindowDelegateGetHelpText () 109844d93782SGreg Clayton { 109944d93782SGreg Clayton return NULL; 110044d93782SGreg Clayton } 110144d93782SGreg Clayton 110244d93782SGreg Clayton virtual KeyHelp * 110344d93782SGreg Clayton WindowDelegateGetKeyHelp () 110444d93782SGreg Clayton { 110544d93782SGreg Clayton return NULL; 110644d93782SGreg Clayton } 110744d93782SGreg Clayton }; 110844d93782SGreg Clayton 110944d93782SGreg Clayton class HelpDialogDelegate : 111044d93782SGreg Clayton public WindowDelegate 111144d93782SGreg Clayton { 111244d93782SGreg Clayton public: 111344d93782SGreg Clayton HelpDialogDelegate (const char *text, KeyHelp *key_help_array); 111444d93782SGreg Clayton 1115bd5ae6b4SGreg Clayton ~HelpDialogDelegate() override; 111644d93782SGreg Clayton 1117bd5ae6b4SGreg Clayton bool 1118bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override; 111944d93782SGreg Clayton 1120bd5ae6b4SGreg Clayton HandleCharResult 1121bd5ae6b4SGreg Clayton WindowDelegateHandleChar (Window &window, int key) override; 112244d93782SGreg Clayton 112344d93782SGreg Clayton size_t 112444d93782SGreg Clayton GetNumLines() const 112544d93782SGreg Clayton { 112644d93782SGreg Clayton return m_text.GetSize(); 112744d93782SGreg Clayton } 112844d93782SGreg Clayton 112944d93782SGreg Clayton size_t 113044d93782SGreg Clayton GetMaxLineLength () const 113144d93782SGreg Clayton { 113244d93782SGreg Clayton return m_text.GetMaxStringLength(); 113344d93782SGreg Clayton } 113444d93782SGreg Clayton 113544d93782SGreg Clayton protected: 113644d93782SGreg Clayton StringList m_text; 113744d93782SGreg Clayton int m_first_visible_line; 113844d93782SGreg Clayton }; 113944d93782SGreg Clayton 114044d93782SGreg Clayton 114144d93782SGreg Clayton class Window 114244d93782SGreg Clayton { 114344d93782SGreg Clayton public: 114444d93782SGreg Clayton 114544d93782SGreg Clayton Window (const char *name) : 114644d93782SGreg Clayton m_name (name), 114744d93782SGreg Clayton m_window (NULL), 114844d93782SGreg Clayton m_panel (NULL), 114944d93782SGreg Clayton m_parent (NULL), 115044d93782SGreg Clayton m_subwindows (), 115144d93782SGreg Clayton m_delegate_sp (), 115244d93782SGreg Clayton m_curr_active_window_idx (UINT32_MAX), 115344d93782SGreg Clayton m_prev_active_window_idx (UINT32_MAX), 115444d93782SGreg Clayton m_delete (false), 115544d93782SGreg Clayton m_needs_update (true), 115644d93782SGreg Clayton m_can_activate (true), 115744d93782SGreg Clayton m_is_subwin (false) 115844d93782SGreg Clayton { 115944d93782SGreg Clayton } 116044d93782SGreg Clayton 116144d93782SGreg Clayton Window (const char *name, WINDOW *w, bool del = true) : 116244d93782SGreg Clayton m_name (name), 116344d93782SGreg Clayton m_window (NULL), 116444d93782SGreg Clayton m_panel (NULL), 116544d93782SGreg Clayton m_parent (NULL), 116644d93782SGreg Clayton m_subwindows (), 116744d93782SGreg Clayton m_delegate_sp (), 116844d93782SGreg Clayton m_curr_active_window_idx (UINT32_MAX), 116944d93782SGreg Clayton m_prev_active_window_idx (UINT32_MAX), 117044d93782SGreg Clayton m_delete (del), 117144d93782SGreg Clayton m_needs_update (true), 117244d93782SGreg Clayton m_can_activate (true), 117344d93782SGreg Clayton m_is_subwin (false) 117444d93782SGreg Clayton { 117544d93782SGreg Clayton if (w) 117644d93782SGreg Clayton Reset(w); 117744d93782SGreg Clayton } 117844d93782SGreg Clayton 117944d93782SGreg Clayton Window (const char *name, const Rect &bounds) : 118044d93782SGreg Clayton m_name (name), 118144d93782SGreg Clayton m_window (NULL), 118244d93782SGreg Clayton m_parent (NULL), 118344d93782SGreg Clayton m_subwindows (), 118444d93782SGreg Clayton m_delegate_sp (), 118544d93782SGreg Clayton m_curr_active_window_idx (UINT32_MAX), 118644d93782SGreg Clayton m_prev_active_window_idx (UINT32_MAX), 118744d93782SGreg Clayton m_delete (true), 118844d93782SGreg Clayton m_needs_update (true), 118944d93782SGreg Clayton m_can_activate (true), 119044d93782SGreg Clayton m_is_subwin (false) 119144d93782SGreg Clayton { 119244d93782SGreg Clayton Reset (::newwin (bounds.size.height, bounds.size.width, bounds.origin.y, bounds.origin.y)); 119344d93782SGreg Clayton } 119444d93782SGreg Clayton 119544d93782SGreg Clayton virtual 119644d93782SGreg Clayton ~Window () 119744d93782SGreg Clayton { 119844d93782SGreg Clayton RemoveSubWindows (); 119944d93782SGreg Clayton Reset (); 120044d93782SGreg Clayton } 120144d93782SGreg Clayton 120244d93782SGreg Clayton void 120344d93782SGreg Clayton Reset (WINDOW *w = NULL, bool del = true) 120444d93782SGreg Clayton { 120544d93782SGreg Clayton if (m_window == w) 120644d93782SGreg Clayton return; 120744d93782SGreg Clayton 120844d93782SGreg Clayton if (m_panel) 120944d93782SGreg Clayton { 121044d93782SGreg Clayton ::del_panel (m_panel); 121144d93782SGreg Clayton m_panel = NULL; 121244d93782SGreg Clayton } 121344d93782SGreg Clayton if (m_window && m_delete) 121444d93782SGreg Clayton { 121544d93782SGreg Clayton ::delwin (m_window); 121644d93782SGreg Clayton m_window = NULL; 121744d93782SGreg Clayton m_delete = false; 121844d93782SGreg Clayton } 121944d93782SGreg Clayton if (w) 122044d93782SGreg Clayton { 122144d93782SGreg Clayton m_window = w; 122244d93782SGreg Clayton m_panel = ::new_panel (m_window); 122344d93782SGreg Clayton m_delete = del; 122444d93782SGreg Clayton } 122544d93782SGreg Clayton } 122644d93782SGreg Clayton 122744d93782SGreg Clayton void AttributeOn (attr_t attr) { ::wattron (m_window, attr); } 122844d93782SGreg Clayton void AttributeOff (attr_t attr) { ::wattroff (m_window, attr); } 122944d93782SGreg Clayton void Box (chtype v_char = ACS_VLINE, chtype h_char = ACS_HLINE) { ::box(m_window, v_char, h_char); } 123044d93782SGreg Clayton void Clear () { ::wclear (m_window); } 123144d93782SGreg Clayton void Erase () { ::werase (m_window); } 123244d93782SGreg Clayton Rect GetBounds () { return Rect (GetParentOrigin(), GetSize()); } // Get the rectangle in our parent window 123344d93782SGreg Clayton int GetChar () { return ::wgetch (m_window); } 123444d93782SGreg Clayton int GetCursorX () { return getcurx (m_window); } 123544d93782SGreg Clayton int GetCursorY () { return getcury (m_window); } 123644d93782SGreg Clayton Rect GetFrame () { return Rect (Point(), GetSize()); } // Get our rectangle in our own coordinate system 123744d93782SGreg Clayton Point GetParentOrigin() { return Point (GetParentX(), GetParentY()); } 123844d93782SGreg Clayton Size GetSize() { return Size (GetWidth(), GetHeight()); } 123944d93782SGreg Clayton int GetParentX () { return getparx (m_window); } 124044d93782SGreg Clayton int GetParentY () { return getpary (m_window); } 124144d93782SGreg Clayton int GetMaxX() { return getmaxx (m_window); } 124244d93782SGreg Clayton int GetMaxY() { return getmaxy (m_window); } 124344d93782SGreg Clayton int GetWidth() { return GetMaxX(); } 124444d93782SGreg Clayton int GetHeight() { return GetMaxY(); } 124544d93782SGreg Clayton void MoveCursor (int x, int y) { ::wmove (m_window, y, x); } 124644d93782SGreg Clayton void MoveWindow (int x, int y) { MoveWindow(Point(x,y)); } 124744d93782SGreg Clayton void Resize (int w, int h) { ::wresize(m_window, h, w); } 124844d93782SGreg Clayton void Resize (const Size &size) { ::wresize(m_window, size.height, size.width); } 124944d93782SGreg Clayton void PutChar (int ch) { ::waddch (m_window, ch); } 125044d93782SGreg Clayton void PutCString (const char *s, int len = -1) { ::waddnstr (m_window, s, len); } 125144d93782SGreg Clayton void Refresh () { ::wrefresh (m_window); } 125244d93782SGreg Clayton void DeferredRefresh () 125344d93782SGreg Clayton { 125444d93782SGreg Clayton // We are using panels, so we don't need to call this... 125544d93782SGreg Clayton //::wnoutrefresh(m_window); 125644d93782SGreg Clayton } 125744d93782SGreg Clayton void SetBackground (int color_pair_idx) { ::wbkgd (m_window,COLOR_PAIR(color_pair_idx)); } 125844d93782SGreg Clayton void UnderlineOn () { AttributeOn(A_UNDERLINE); } 125944d93782SGreg Clayton void UnderlineOff () { AttributeOff(A_UNDERLINE); } 126044d93782SGreg Clayton 126144d93782SGreg Clayton void PutCStringTruncated (const char *s, int right_pad) 126244d93782SGreg Clayton { 126344d93782SGreg Clayton int bytes_left = GetWidth() - GetCursorX(); 126444d93782SGreg Clayton if (bytes_left > right_pad) 126544d93782SGreg Clayton { 126644d93782SGreg Clayton bytes_left -= right_pad; 126744d93782SGreg Clayton ::waddnstr (m_window, s, bytes_left); 126844d93782SGreg Clayton } 126944d93782SGreg Clayton } 127044d93782SGreg Clayton 127144d93782SGreg Clayton void 127244d93782SGreg Clayton MoveWindow (const Point &origin) 127344d93782SGreg Clayton { 127444d93782SGreg Clayton const bool moving_window = 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 Size size = GetSize(); 127944d93782SGreg Clayton Reset (::subwin (m_parent->m_window, 128044d93782SGreg Clayton size.height, 128144d93782SGreg Clayton size.width, 128244d93782SGreg Clayton origin.y, 128344d93782SGreg Clayton origin.x), true); 128444d93782SGreg Clayton } 128544d93782SGreg Clayton else 128644d93782SGreg Clayton { 128744d93782SGreg Clayton ::mvwin (m_window, origin.y, origin.x); 128844d93782SGreg Clayton } 128944d93782SGreg Clayton } 129044d93782SGreg Clayton 129144d93782SGreg Clayton void 129244d93782SGreg Clayton SetBounds (const Rect &bounds) 129344d93782SGreg Clayton { 129444d93782SGreg Clayton const bool moving_window = bounds.origin != GetParentOrigin(); 129544d93782SGreg Clayton if (m_is_subwin && moving_window) 129644d93782SGreg Clayton { 129744d93782SGreg Clayton // Can't move subwindows, must delete and re-create 129844d93782SGreg Clayton Reset (::subwin (m_parent->m_window, 129944d93782SGreg Clayton bounds.size.height, 130044d93782SGreg Clayton bounds.size.width, 130144d93782SGreg Clayton bounds.origin.y, 130244d93782SGreg Clayton bounds.origin.x), true); 130344d93782SGreg Clayton } 130444d93782SGreg Clayton else 130544d93782SGreg Clayton { 130644d93782SGreg Clayton if (moving_window) 130744d93782SGreg Clayton MoveWindow(bounds.origin); 130844d93782SGreg Clayton Resize (bounds.size); 130944d93782SGreg Clayton } 131044d93782SGreg Clayton } 131144d93782SGreg Clayton 131244d93782SGreg Clayton void 131344d93782SGreg Clayton Printf (const char *format, ...) __attribute__ ((format (printf, 2, 3))) 131444d93782SGreg Clayton { 131544d93782SGreg Clayton va_list args; 131644d93782SGreg Clayton va_start (args, format); 131744d93782SGreg Clayton vwprintw(m_window, format, args); 131844d93782SGreg Clayton va_end (args); 131944d93782SGreg Clayton } 132044d93782SGreg Clayton 132144d93782SGreg Clayton void 132244d93782SGreg Clayton Touch () 132344d93782SGreg Clayton { 132444d93782SGreg Clayton ::touchwin (m_window); 132544d93782SGreg Clayton if (m_parent) 132644d93782SGreg Clayton m_parent->Touch(); 132744d93782SGreg Clayton } 132844d93782SGreg Clayton 132944d93782SGreg Clayton WindowSP 133044d93782SGreg Clayton CreateSubWindow (const char *name, const Rect &bounds, bool make_active) 133144d93782SGreg Clayton { 133244d93782SGreg Clayton WindowSP subwindow_sp; 133344d93782SGreg Clayton if (m_window) 133444d93782SGreg Clayton { 133544d93782SGreg Clayton subwindow_sp.reset(new Window(name, ::subwin (m_window, 133644d93782SGreg Clayton bounds.size.height, 133744d93782SGreg Clayton bounds.size.width, 133844d93782SGreg Clayton bounds.origin.y, 133944d93782SGreg Clayton bounds.origin.x), true)); 134044d93782SGreg Clayton subwindow_sp->m_is_subwin = true; 134144d93782SGreg Clayton } 134244d93782SGreg Clayton else 134344d93782SGreg Clayton { 134444d93782SGreg Clayton subwindow_sp.reset(new Window(name, ::newwin (bounds.size.height, 134544d93782SGreg Clayton bounds.size.width, 134644d93782SGreg Clayton bounds.origin.y, 134744d93782SGreg Clayton bounds.origin.x), true)); 134844d93782SGreg Clayton subwindow_sp->m_is_subwin = false; 134944d93782SGreg Clayton } 135044d93782SGreg Clayton subwindow_sp->m_parent = this; 135144d93782SGreg Clayton if (make_active) 135244d93782SGreg Clayton { 135344d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 135444d93782SGreg Clayton m_curr_active_window_idx = m_subwindows.size(); 135544d93782SGreg Clayton } 135644d93782SGreg Clayton m_subwindows.push_back(subwindow_sp); 135744d93782SGreg Clayton ::top_panel (subwindow_sp->m_panel); 135844d93782SGreg Clayton m_needs_update = true; 135944d93782SGreg Clayton return subwindow_sp; 136044d93782SGreg Clayton } 136144d93782SGreg Clayton 136244d93782SGreg Clayton bool 136344d93782SGreg Clayton RemoveSubWindow (Window *window) 136444d93782SGreg Clayton { 136544d93782SGreg Clayton Windows::iterator pos, end = m_subwindows.end(); 136644d93782SGreg Clayton size_t i = 0; 136744d93782SGreg Clayton for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) 136844d93782SGreg Clayton { 136944d93782SGreg Clayton if ((*pos).get() == window) 137044d93782SGreg Clayton { 137144d93782SGreg Clayton if (m_prev_active_window_idx == i) 137244d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 137344d93782SGreg Clayton else if (m_prev_active_window_idx != UINT32_MAX && m_prev_active_window_idx > i) 137444d93782SGreg Clayton --m_prev_active_window_idx; 137544d93782SGreg Clayton 137644d93782SGreg Clayton if (m_curr_active_window_idx == i) 137744d93782SGreg Clayton m_curr_active_window_idx = UINT32_MAX; 137844d93782SGreg Clayton else if (m_curr_active_window_idx != UINT32_MAX && m_curr_active_window_idx > i) 137944d93782SGreg Clayton --m_curr_active_window_idx; 138044d93782SGreg Clayton window->Erase(); 138144d93782SGreg Clayton m_subwindows.erase(pos); 138244d93782SGreg Clayton m_needs_update = true; 138344d93782SGreg Clayton if (m_parent) 138444d93782SGreg Clayton m_parent->Touch(); 138544d93782SGreg Clayton else 138644d93782SGreg Clayton ::touchwin (stdscr); 138744d93782SGreg Clayton return true; 138844d93782SGreg Clayton } 138944d93782SGreg Clayton } 139044d93782SGreg Clayton return false; 139144d93782SGreg Clayton } 139244d93782SGreg Clayton 139344d93782SGreg Clayton WindowSP 139444d93782SGreg Clayton FindSubWindow (const char *name) 139544d93782SGreg Clayton { 139644d93782SGreg Clayton Windows::iterator pos, end = m_subwindows.end(); 139744d93782SGreg Clayton size_t i = 0; 139844d93782SGreg Clayton for (pos = m_subwindows.begin(); pos != end; ++pos, ++i) 139944d93782SGreg Clayton { 140044d93782SGreg Clayton if ((*pos)->m_name.compare(name) == 0) 140144d93782SGreg Clayton return *pos; 140244d93782SGreg Clayton } 140344d93782SGreg Clayton return WindowSP(); 140444d93782SGreg Clayton } 140544d93782SGreg Clayton 140644d93782SGreg Clayton void 140744d93782SGreg Clayton RemoveSubWindows () 140844d93782SGreg Clayton { 140944d93782SGreg Clayton m_curr_active_window_idx = UINT32_MAX; 141044d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 141144d93782SGreg Clayton for (Windows::iterator pos = m_subwindows.begin(); 141244d93782SGreg Clayton pos != m_subwindows.end(); 141344d93782SGreg Clayton pos = m_subwindows.erase(pos)) 141444d93782SGreg Clayton { 141544d93782SGreg Clayton (*pos)->Erase(); 141644d93782SGreg Clayton } 141744d93782SGreg Clayton if (m_parent) 141844d93782SGreg Clayton m_parent->Touch(); 141944d93782SGreg Clayton else 142044d93782SGreg Clayton ::touchwin (stdscr); 142144d93782SGreg Clayton } 142244d93782SGreg Clayton 142344d93782SGreg Clayton WINDOW * 142444d93782SGreg Clayton get() 142544d93782SGreg Clayton { 142644d93782SGreg Clayton return m_window; 142744d93782SGreg Clayton } 142844d93782SGreg Clayton 142944d93782SGreg Clayton operator WINDOW *() 143044d93782SGreg Clayton { 143144d93782SGreg Clayton return m_window; 143244d93782SGreg Clayton } 143344d93782SGreg Clayton 143444d93782SGreg Clayton //---------------------------------------------------------------------- 143544d93782SGreg Clayton // Window drawing utilities 143644d93782SGreg Clayton //---------------------------------------------------------------------- 143744d93782SGreg Clayton void 143844d93782SGreg Clayton DrawTitleBox (const char *title, const char *bottom_message = NULL) 143944d93782SGreg Clayton { 144044d93782SGreg Clayton attr_t attr = 0; 144144d93782SGreg Clayton if (IsActive()) 144244d93782SGreg Clayton attr = A_BOLD | COLOR_PAIR(2); 144344d93782SGreg Clayton else 144444d93782SGreg Clayton attr = 0; 144544d93782SGreg Clayton if (attr) 144644d93782SGreg Clayton AttributeOn(attr); 144744d93782SGreg Clayton 144844d93782SGreg Clayton Box(); 144944d93782SGreg Clayton MoveCursor(3, 0); 145044d93782SGreg Clayton 145144d93782SGreg Clayton if (title && title[0]) 145244d93782SGreg Clayton { 145344d93782SGreg Clayton PutChar ('<'); 145444d93782SGreg Clayton PutCString (title); 145544d93782SGreg Clayton PutChar ('>'); 145644d93782SGreg Clayton } 145744d93782SGreg Clayton 145844d93782SGreg Clayton if (bottom_message && bottom_message[0]) 145944d93782SGreg Clayton { 146044d93782SGreg Clayton int bottom_message_length = strlen(bottom_message); 146144d93782SGreg Clayton int x = GetWidth() - 3 - (bottom_message_length + 2); 146244d93782SGreg Clayton 146344d93782SGreg Clayton if (x > 0) 146444d93782SGreg Clayton { 146544d93782SGreg Clayton MoveCursor (x, GetHeight() - 1); 146644d93782SGreg Clayton PutChar ('['); 146744d93782SGreg Clayton PutCString(bottom_message); 146844d93782SGreg Clayton PutChar (']'); 146944d93782SGreg Clayton } 147044d93782SGreg Clayton else 147144d93782SGreg Clayton { 147244d93782SGreg Clayton MoveCursor (1, GetHeight() - 1); 147344d93782SGreg Clayton PutChar ('['); 147444d93782SGreg Clayton PutCStringTruncated (bottom_message, 1); 147544d93782SGreg Clayton } 147644d93782SGreg Clayton } 147744d93782SGreg Clayton if (attr) 147844d93782SGreg Clayton AttributeOff(attr); 147944d93782SGreg Clayton 148044d93782SGreg Clayton } 148144d93782SGreg Clayton 148244d93782SGreg Clayton virtual void 148344d93782SGreg Clayton Draw (bool force) 148444d93782SGreg Clayton { 148544d93782SGreg Clayton if (m_delegate_sp && m_delegate_sp->WindowDelegateDraw (*this, force)) 148644d93782SGreg Clayton return; 148744d93782SGreg Clayton 148844d93782SGreg Clayton for (auto &subwindow_sp : m_subwindows) 148944d93782SGreg Clayton subwindow_sp->Draw(force); 149044d93782SGreg Clayton } 149144d93782SGreg Clayton 149244d93782SGreg Clayton bool 149344d93782SGreg Clayton CreateHelpSubwindow () 149444d93782SGreg Clayton { 149544d93782SGreg Clayton if (m_delegate_sp) 149644d93782SGreg Clayton { 149744d93782SGreg Clayton const char *text = m_delegate_sp->WindowDelegateGetHelpText (); 149844d93782SGreg Clayton KeyHelp *key_help = m_delegate_sp->WindowDelegateGetKeyHelp (); 149944d93782SGreg Clayton if ((text && text[0]) || key_help) 150044d93782SGreg Clayton { 150144d93782SGreg Clayton std::auto_ptr<HelpDialogDelegate> help_delegate_ap(new HelpDialogDelegate(text, key_help)); 150244d93782SGreg Clayton const size_t num_lines = help_delegate_ap->GetNumLines(); 150344d93782SGreg Clayton const size_t max_length = help_delegate_ap->GetMaxLineLength(); 150444d93782SGreg Clayton Rect bounds = GetBounds(); 150544d93782SGreg Clayton bounds.Inset(1, 1); 15063985c8c6SSaleem Abdulrasool if (max_length + 4 < static_cast<size_t>(bounds.size.width)) 150744d93782SGreg Clayton { 150844d93782SGreg Clayton bounds.origin.x += (bounds.size.width - max_length + 4)/2; 150944d93782SGreg Clayton bounds.size.width = max_length + 4; 151044d93782SGreg Clayton } 151144d93782SGreg Clayton else 151244d93782SGreg Clayton { 151344d93782SGreg Clayton if (bounds.size.width > 100) 151444d93782SGreg Clayton { 151544d93782SGreg Clayton const int inset_w = bounds.size.width / 4; 151644d93782SGreg Clayton bounds.origin.x += inset_w; 151744d93782SGreg Clayton bounds.size.width -= 2*inset_w; 151844d93782SGreg Clayton } 151944d93782SGreg Clayton } 152044d93782SGreg Clayton 15213985c8c6SSaleem Abdulrasool if (num_lines + 2 < static_cast<size_t>(bounds.size.height)) 152244d93782SGreg Clayton { 152344d93782SGreg Clayton bounds.origin.y += (bounds.size.height - num_lines + 2)/2; 152444d93782SGreg Clayton bounds.size.height = num_lines + 2; 152544d93782SGreg Clayton } 152644d93782SGreg Clayton else 152744d93782SGreg Clayton { 152844d93782SGreg Clayton if (bounds.size.height > 100) 152944d93782SGreg Clayton { 153044d93782SGreg Clayton const int inset_h = bounds.size.height / 4; 153144d93782SGreg Clayton bounds.origin.y += inset_h; 153244d93782SGreg Clayton bounds.size.height -= 2*inset_h; 153344d93782SGreg Clayton } 153444d93782SGreg Clayton } 15355fdb09bbSGreg Clayton WindowSP help_window_sp; 15365fdb09bbSGreg Clayton Window *parent_window = GetParent(); 15375fdb09bbSGreg Clayton if (parent_window) 15385fdb09bbSGreg Clayton help_window_sp = parent_window->CreateSubWindow("Help", bounds, true); 15395fdb09bbSGreg Clayton else 15405fdb09bbSGreg Clayton help_window_sp = CreateSubWindow("Help", bounds, true); 154144d93782SGreg Clayton help_window_sp->SetDelegate(WindowDelegateSP(help_delegate_ap.release())); 154244d93782SGreg Clayton return true; 154344d93782SGreg Clayton } 154444d93782SGreg Clayton } 154544d93782SGreg Clayton return false; 154644d93782SGreg Clayton } 154744d93782SGreg Clayton 154844d93782SGreg Clayton virtual HandleCharResult 154944d93782SGreg Clayton HandleChar (int key) 155044d93782SGreg Clayton { 155144d93782SGreg Clayton // Always check the active window first 155244d93782SGreg Clayton HandleCharResult result = eKeyNotHandled; 155344d93782SGreg Clayton WindowSP active_window_sp = GetActiveWindow (); 155444d93782SGreg Clayton if (active_window_sp) 155544d93782SGreg Clayton { 155644d93782SGreg Clayton result = active_window_sp->HandleChar (key); 155744d93782SGreg Clayton if (result != eKeyNotHandled) 155844d93782SGreg Clayton return result; 155944d93782SGreg Clayton } 156044d93782SGreg Clayton 156144d93782SGreg Clayton if (m_delegate_sp) 156244d93782SGreg Clayton { 156344d93782SGreg Clayton result = m_delegate_sp->WindowDelegateHandleChar (*this, key); 156444d93782SGreg Clayton if (result != eKeyNotHandled) 156544d93782SGreg Clayton return result; 156644d93782SGreg Clayton } 156744d93782SGreg Clayton 156844d93782SGreg Clayton // Then check for any windows that want any keys 156944d93782SGreg Clayton // that weren't handled. This is typically only 157044d93782SGreg Clayton // for a menubar. 157144d93782SGreg Clayton // Make a copy of the subwindows in case any HandleChar() 157244d93782SGreg Clayton // functions muck with the subwindows. If we don't do this, 157344d93782SGreg Clayton // we can crash when iterating over the subwindows. 157444d93782SGreg Clayton Windows subwindows (m_subwindows); 157544d93782SGreg Clayton for (auto subwindow_sp : subwindows) 157644d93782SGreg Clayton { 157744d93782SGreg Clayton if (subwindow_sp->m_can_activate == false) 157844d93782SGreg Clayton { 157944d93782SGreg Clayton HandleCharResult result = subwindow_sp->HandleChar(key); 158044d93782SGreg Clayton if (result != eKeyNotHandled) 158144d93782SGreg Clayton return result; 158244d93782SGreg Clayton } 158344d93782SGreg Clayton } 158444d93782SGreg Clayton 158544d93782SGreg Clayton return eKeyNotHandled; 158644d93782SGreg Clayton } 158744d93782SGreg Clayton 158844d93782SGreg Clayton bool 158944d93782SGreg Clayton SetActiveWindow (Window *window) 159044d93782SGreg Clayton { 159144d93782SGreg Clayton const size_t num_subwindows = m_subwindows.size(); 159244d93782SGreg Clayton for (size_t i=0; i<num_subwindows; ++i) 159344d93782SGreg Clayton { 159444d93782SGreg Clayton if (m_subwindows[i].get() == window) 159544d93782SGreg Clayton { 159644d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 159744d93782SGreg Clayton ::top_panel (window->m_panel); 159844d93782SGreg Clayton m_curr_active_window_idx = i; 159944d93782SGreg Clayton return true; 160044d93782SGreg Clayton } 160144d93782SGreg Clayton } 160244d93782SGreg Clayton return false; 160344d93782SGreg Clayton } 160444d93782SGreg Clayton 160544d93782SGreg Clayton WindowSP 160644d93782SGreg Clayton GetActiveWindow () 160744d93782SGreg Clayton { 160844d93782SGreg Clayton if (!m_subwindows.empty()) 160944d93782SGreg Clayton { 161044d93782SGreg Clayton if (m_curr_active_window_idx >= m_subwindows.size()) 161144d93782SGreg Clayton { 161244d93782SGreg Clayton if (m_prev_active_window_idx < m_subwindows.size()) 161344d93782SGreg Clayton { 161444d93782SGreg Clayton m_curr_active_window_idx = m_prev_active_window_idx; 161544d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 161644d93782SGreg Clayton } 161744d93782SGreg Clayton else if (IsActive()) 161844d93782SGreg Clayton { 161944d93782SGreg Clayton m_prev_active_window_idx = UINT32_MAX; 162044d93782SGreg Clayton m_curr_active_window_idx = UINT32_MAX; 162144d93782SGreg Clayton 162244d93782SGreg Clayton // Find first window that wants to be active if this window is active 162344d93782SGreg Clayton const size_t num_subwindows = m_subwindows.size(); 162444d93782SGreg Clayton for (size_t i=0; i<num_subwindows; ++i) 162544d93782SGreg Clayton { 162644d93782SGreg Clayton if (m_subwindows[i]->GetCanBeActive()) 162744d93782SGreg Clayton { 162844d93782SGreg Clayton m_curr_active_window_idx = i; 162944d93782SGreg Clayton break; 163044d93782SGreg Clayton } 163144d93782SGreg Clayton } 163244d93782SGreg Clayton } 163344d93782SGreg Clayton } 163444d93782SGreg Clayton 163544d93782SGreg Clayton if (m_curr_active_window_idx < m_subwindows.size()) 163644d93782SGreg Clayton return m_subwindows[m_curr_active_window_idx]; 163744d93782SGreg Clayton } 163844d93782SGreg Clayton return WindowSP(); 163944d93782SGreg Clayton } 164044d93782SGreg Clayton 164144d93782SGreg Clayton bool 164244d93782SGreg Clayton GetCanBeActive () const 164344d93782SGreg Clayton { 164444d93782SGreg Clayton return m_can_activate; 164544d93782SGreg Clayton } 164644d93782SGreg Clayton 164744d93782SGreg Clayton void 164844d93782SGreg Clayton SetCanBeActive (bool b) 164944d93782SGreg Clayton { 165044d93782SGreg Clayton m_can_activate = b; 165144d93782SGreg Clayton } 165244d93782SGreg Clayton 165344d93782SGreg Clayton const WindowDelegateSP & 165444d93782SGreg Clayton GetDelegate () const 165544d93782SGreg Clayton { 165644d93782SGreg Clayton return m_delegate_sp; 165744d93782SGreg Clayton } 165844d93782SGreg Clayton 165944d93782SGreg Clayton void 166044d93782SGreg Clayton SetDelegate (const WindowDelegateSP &delegate_sp) 166144d93782SGreg Clayton { 166244d93782SGreg Clayton m_delegate_sp = delegate_sp; 166344d93782SGreg Clayton } 166444d93782SGreg Clayton 166544d93782SGreg Clayton Window * 166644d93782SGreg Clayton GetParent () const 166744d93782SGreg Clayton { 166844d93782SGreg Clayton return m_parent; 166944d93782SGreg Clayton } 167044d93782SGreg Clayton 167144d93782SGreg Clayton bool 167244d93782SGreg Clayton IsActive () const 167344d93782SGreg Clayton { 167444d93782SGreg Clayton if (m_parent) 167544d93782SGreg Clayton return m_parent->GetActiveWindow().get() == this; 167644d93782SGreg Clayton else 167744d93782SGreg Clayton return true; // Top level window is always active 167844d93782SGreg Clayton } 167944d93782SGreg Clayton 168044d93782SGreg Clayton void 168144d93782SGreg Clayton SelectNextWindowAsActive () 168244d93782SGreg Clayton { 168344d93782SGreg Clayton // Move active focus to next window 168444d93782SGreg Clayton const size_t num_subwindows = m_subwindows.size(); 168544d93782SGreg Clayton if (m_curr_active_window_idx == UINT32_MAX) 168644d93782SGreg Clayton { 168744d93782SGreg Clayton uint32_t idx = 0; 168844d93782SGreg Clayton for (auto subwindow_sp : m_subwindows) 168944d93782SGreg Clayton { 169044d93782SGreg Clayton if (subwindow_sp->GetCanBeActive()) 169144d93782SGreg Clayton { 169244d93782SGreg Clayton m_curr_active_window_idx = idx; 169344d93782SGreg Clayton break; 169444d93782SGreg Clayton } 169544d93782SGreg Clayton ++idx; 169644d93782SGreg Clayton } 169744d93782SGreg Clayton } 169844d93782SGreg Clayton else if (m_curr_active_window_idx + 1 < num_subwindows) 169944d93782SGreg Clayton { 170044d93782SGreg Clayton bool handled = false; 170144d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 170244d93782SGreg Clayton for (size_t idx=m_curr_active_window_idx + 1; idx<num_subwindows; ++idx) 170344d93782SGreg Clayton { 170444d93782SGreg Clayton if (m_subwindows[idx]->GetCanBeActive()) 170544d93782SGreg Clayton { 170644d93782SGreg Clayton m_curr_active_window_idx = idx; 170744d93782SGreg Clayton handled = true; 170844d93782SGreg Clayton break; 170944d93782SGreg Clayton } 171044d93782SGreg Clayton } 171144d93782SGreg Clayton if (!handled) 171244d93782SGreg Clayton { 171344d93782SGreg Clayton for (size_t idx=0; idx<=m_prev_active_window_idx; ++idx) 171444d93782SGreg Clayton { 171544d93782SGreg Clayton if (m_subwindows[idx]->GetCanBeActive()) 171644d93782SGreg Clayton { 171744d93782SGreg Clayton m_curr_active_window_idx = idx; 171844d93782SGreg Clayton break; 171944d93782SGreg Clayton } 172044d93782SGreg Clayton } 172144d93782SGreg Clayton } 172244d93782SGreg Clayton } 172344d93782SGreg Clayton else 172444d93782SGreg Clayton { 172544d93782SGreg Clayton m_prev_active_window_idx = m_curr_active_window_idx; 172644d93782SGreg Clayton for (size_t idx=0; idx<num_subwindows; ++idx) 172744d93782SGreg Clayton { 172844d93782SGreg Clayton if (m_subwindows[idx]->GetCanBeActive()) 172944d93782SGreg Clayton { 173044d93782SGreg Clayton m_curr_active_window_idx = idx; 173144d93782SGreg Clayton break; 173244d93782SGreg Clayton } 173344d93782SGreg Clayton } 173444d93782SGreg Clayton } 173544d93782SGreg Clayton } 173644d93782SGreg Clayton 173744d93782SGreg Clayton const char * 173844d93782SGreg Clayton GetName () const 173944d93782SGreg Clayton { 174044d93782SGreg Clayton return m_name.c_str(); 174144d93782SGreg Clayton } 174244d93782SGreg Clayton protected: 174344d93782SGreg Clayton std::string m_name; 174444d93782SGreg Clayton WINDOW *m_window; 174544d93782SGreg Clayton PANEL *m_panel; 174644d93782SGreg Clayton Window *m_parent; 174744d93782SGreg Clayton Windows m_subwindows; 174844d93782SGreg Clayton WindowDelegateSP m_delegate_sp; 174944d93782SGreg Clayton uint32_t m_curr_active_window_idx; 175044d93782SGreg Clayton uint32_t m_prev_active_window_idx; 175144d93782SGreg Clayton bool m_delete; 175244d93782SGreg Clayton bool m_needs_update; 175344d93782SGreg Clayton bool m_can_activate; 175444d93782SGreg Clayton bool m_is_subwin; 175544d93782SGreg Clayton 175644d93782SGreg Clayton private: 175744d93782SGreg Clayton DISALLOW_COPY_AND_ASSIGN(Window); 175844d93782SGreg Clayton }; 175944d93782SGreg Clayton 176044d93782SGreg Clayton class MenuDelegate 176144d93782SGreg Clayton { 176244d93782SGreg Clayton public: 176344d93782SGreg Clayton virtual ~MenuDelegate() {} 176444d93782SGreg Clayton 176544d93782SGreg Clayton virtual MenuActionResult 176644d93782SGreg Clayton MenuDelegateAction (Menu &menu) = 0; 176744d93782SGreg Clayton }; 176844d93782SGreg Clayton 176944d93782SGreg Clayton class Menu : public WindowDelegate 177044d93782SGreg Clayton { 177144d93782SGreg Clayton public: 177244d93782SGreg Clayton enum class Type 177344d93782SGreg Clayton { 177444d93782SGreg Clayton Invalid, 177544d93782SGreg Clayton Bar, 177644d93782SGreg Clayton Item, 177744d93782SGreg Clayton Separator 177844d93782SGreg Clayton }; 177944d93782SGreg Clayton 178044d93782SGreg Clayton // Menubar or separator constructor 178144d93782SGreg Clayton Menu (Type type); 178244d93782SGreg Clayton 178344d93782SGreg Clayton // Menuitem constructor 178444d93782SGreg Clayton Menu (const char *name, 178544d93782SGreg Clayton const char *key_name, 178644d93782SGreg Clayton int key_value, 178744d93782SGreg Clayton uint64_t identifier); 178844d93782SGreg Clayton 1789bd5ae6b4SGreg Clayton ~Menu () override 179044d93782SGreg Clayton { 179144d93782SGreg Clayton } 179244d93782SGreg Clayton 179344d93782SGreg Clayton const MenuDelegateSP & 179444d93782SGreg Clayton GetDelegate () const 179544d93782SGreg Clayton { 179644d93782SGreg Clayton return m_delegate_sp; 179744d93782SGreg Clayton } 179844d93782SGreg Clayton 179944d93782SGreg Clayton void 180044d93782SGreg Clayton SetDelegate (const MenuDelegateSP &delegate_sp) 180144d93782SGreg Clayton { 180244d93782SGreg Clayton m_delegate_sp = delegate_sp; 180344d93782SGreg Clayton } 180444d93782SGreg Clayton 180544d93782SGreg Clayton void 180644d93782SGreg Clayton RecalculateNameLengths(); 180744d93782SGreg Clayton 180844d93782SGreg Clayton void 180944d93782SGreg Clayton AddSubmenu (const MenuSP &menu_sp); 181044d93782SGreg Clayton 181144d93782SGreg Clayton int 181244d93782SGreg Clayton DrawAndRunMenu (Window &window); 181344d93782SGreg Clayton 181444d93782SGreg Clayton void 181544d93782SGreg Clayton DrawMenuTitle (Window &window, bool highlight); 181644d93782SGreg Clayton 1817bd5ae6b4SGreg Clayton bool 1818bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override; 181944d93782SGreg Clayton 1820bd5ae6b4SGreg Clayton HandleCharResult 1821bd5ae6b4SGreg Clayton WindowDelegateHandleChar (Window &window, int key) override; 182244d93782SGreg Clayton 182344d93782SGreg Clayton MenuActionResult 182444d93782SGreg Clayton ActionPrivate (Menu &menu) 182544d93782SGreg Clayton { 182644d93782SGreg Clayton MenuActionResult result = MenuActionResult::NotHandled; 182744d93782SGreg Clayton if (m_delegate_sp) 182844d93782SGreg Clayton { 182944d93782SGreg Clayton result = m_delegate_sp->MenuDelegateAction (menu); 183044d93782SGreg Clayton if (result != MenuActionResult::NotHandled) 183144d93782SGreg Clayton return result; 183244d93782SGreg Clayton } 183344d93782SGreg Clayton else if (m_parent) 183444d93782SGreg Clayton { 183544d93782SGreg Clayton result = m_parent->ActionPrivate(menu); 183644d93782SGreg Clayton if (result != MenuActionResult::NotHandled) 183744d93782SGreg Clayton return result; 183844d93782SGreg Clayton } 183944d93782SGreg Clayton return m_canned_result; 184044d93782SGreg Clayton } 184144d93782SGreg Clayton 184244d93782SGreg Clayton MenuActionResult 184344d93782SGreg Clayton Action () 184444d93782SGreg Clayton { 184544d93782SGreg Clayton // Call the recursive action so it can try to handle it 184644d93782SGreg Clayton // with the menu delegate, and if not, try our parent menu 184744d93782SGreg Clayton return ActionPrivate (*this); 184844d93782SGreg Clayton } 184944d93782SGreg Clayton 185044d93782SGreg Clayton void 185144d93782SGreg Clayton SetCannedResult (MenuActionResult result) 185244d93782SGreg Clayton { 185344d93782SGreg Clayton m_canned_result = result; 185444d93782SGreg Clayton } 185544d93782SGreg Clayton 185644d93782SGreg Clayton Menus & 185744d93782SGreg Clayton GetSubmenus() 185844d93782SGreg Clayton { 185944d93782SGreg Clayton return m_submenus; 186044d93782SGreg Clayton } 186144d93782SGreg Clayton 186244d93782SGreg Clayton const Menus & 186344d93782SGreg Clayton GetSubmenus() const 186444d93782SGreg Clayton { 186544d93782SGreg Clayton return m_submenus; 186644d93782SGreg Clayton } 186744d93782SGreg Clayton 186844d93782SGreg Clayton int 186944d93782SGreg Clayton GetSelectedSubmenuIndex () const 187044d93782SGreg Clayton { 187144d93782SGreg Clayton return m_selected; 187244d93782SGreg Clayton } 187344d93782SGreg Clayton 187444d93782SGreg Clayton void 187544d93782SGreg Clayton SetSelectedSubmenuIndex (int idx) 187644d93782SGreg Clayton { 187744d93782SGreg Clayton m_selected = idx; 187844d93782SGreg Clayton } 187944d93782SGreg Clayton 188044d93782SGreg Clayton Type 188144d93782SGreg Clayton GetType () const 188244d93782SGreg Clayton { 188344d93782SGreg Clayton return m_type; 188444d93782SGreg Clayton } 188544d93782SGreg Clayton 188644d93782SGreg Clayton int 188744d93782SGreg Clayton GetStartingColumn() const 188844d93782SGreg Clayton { 188944d93782SGreg Clayton return m_start_col; 189044d93782SGreg Clayton } 189144d93782SGreg Clayton 189244d93782SGreg Clayton void 189344d93782SGreg Clayton SetStartingColumn(int col) 189444d93782SGreg Clayton { 189544d93782SGreg Clayton m_start_col = col; 189644d93782SGreg Clayton } 189744d93782SGreg Clayton 189844d93782SGreg Clayton int 189944d93782SGreg Clayton GetKeyValue() const 190044d93782SGreg Clayton { 190144d93782SGreg Clayton return m_key_value; 190244d93782SGreg Clayton } 190344d93782SGreg Clayton 190444d93782SGreg Clayton void 190544d93782SGreg Clayton SetKeyValue(int key_value) 190644d93782SGreg Clayton { 190744d93782SGreg Clayton m_key_value = key_value; 190844d93782SGreg Clayton } 190944d93782SGreg Clayton 191044d93782SGreg Clayton std::string & 191144d93782SGreg Clayton GetName() 191244d93782SGreg Clayton { 191344d93782SGreg Clayton return m_name; 191444d93782SGreg Clayton } 191544d93782SGreg Clayton 191644d93782SGreg Clayton std::string & 191744d93782SGreg Clayton GetKeyName() 191844d93782SGreg Clayton { 191944d93782SGreg Clayton return m_key_name; 192044d93782SGreg Clayton } 192144d93782SGreg Clayton 192244d93782SGreg Clayton int 192344d93782SGreg Clayton GetDrawWidth () const 192444d93782SGreg Clayton { 192544d93782SGreg Clayton return m_max_submenu_name_length + m_max_submenu_key_name_length + 8; 192644d93782SGreg Clayton } 192744d93782SGreg Clayton 192844d93782SGreg Clayton 192944d93782SGreg Clayton uint64_t 193044d93782SGreg Clayton GetIdentifier() const 193144d93782SGreg Clayton { 193244d93782SGreg Clayton return m_identifier; 193344d93782SGreg Clayton } 193444d93782SGreg Clayton 193544d93782SGreg Clayton void 193644d93782SGreg Clayton SetIdentifier (uint64_t identifier) 193744d93782SGreg Clayton { 193844d93782SGreg Clayton m_identifier = identifier; 193944d93782SGreg Clayton } 194044d93782SGreg Clayton 194144d93782SGreg Clayton protected: 194244d93782SGreg Clayton std::string m_name; 194344d93782SGreg Clayton std::string m_key_name; 194444d93782SGreg Clayton uint64_t m_identifier; 194544d93782SGreg Clayton Type m_type; 194644d93782SGreg Clayton int m_key_value; 194744d93782SGreg Clayton int m_start_col; 194844d93782SGreg Clayton int m_max_submenu_name_length; 194944d93782SGreg Clayton int m_max_submenu_key_name_length; 195044d93782SGreg Clayton int m_selected; 195144d93782SGreg Clayton Menu *m_parent; 195244d93782SGreg Clayton Menus m_submenus; 195344d93782SGreg Clayton WindowSP m_menu_window_sp; 195444d93782SGreg Clayton MenuActionResult m_canned_result; 195544d93782SGreg Clayton MenuDelegateSP m_delegate_sp; 195644d93782SGreg Clayton }; 195744d93782SGreg Clayton 195844d93782SGreg Clayton // Menubar or separator constructor 195944d93782SGreg Clayton Menu::Menu (Type type) : 196044d93782SGreg Clayton m_name (), 196144d93782SGreg Clayton m_key_name (), 196244d93782SGreg Clayton m_identifier (0), 196344d93782SGreg Clayton m_type (type), 196444d93782SGreg Clayton m_key_value (0), 196544d93782SGreg Clayton m_start_col (0), 196644d93782SGreg Clayton m_max_submenu_name_length (0), 196744d93782SGreg Clayton m_max_submenu_key_name_length (0), 196844d93782SGreg Clayton m_selected (0), 196944d93782SGreg Clayton m_parent (NULL), 197044d93782SGreg Clayton m_submenus (), 197144d93782SGreg Clayton m_canned_result (MenuActionResult::NotHandled), 197244d93782SGreg Clayton m_delegate_sp() 197344d93782SGreg Clayton { 197444d93782SGreg Clayton } 197544d93782SGreg Clayton 197644d93782SGreg Clayton // Menuitem constructor 197744d93782SGreg Clayton Menu::Menu (const char *name, 197844d93782SGreg Clayton const char *key_name, 197944d93782SGreg Clayton int key_value, 198044d93782SGreg Clayton uint64_t identifier) : 198144d93782SGreg Clayton m_name (), 198244d93782SGreg Clayton m_key_name (), 198344d93782SGreg Clayton m_identifier (identifier), 198444d93782SGreg Clayton m_type (Type::Invalid), 198544d93782SGreg Clayton m_key_value (key_value), 198644d93782SGreg Clayton m_start_col (0), 198744d93782SGreg Clayton m_max_submenu_name_length (0), 198844d93782SGreg Clayton m_max_submenu_key_name_length (0), 198944d93782SGreg Clayton m_selected (0), 199044d93782SGreg Clayton m_parent (NULL), 199144d93782SGreg Clayton m_submenus (), 199244d93782SGreg Clayton m_canned_result (MenuActionResult::NotHandled), 199344d93782SGreg Clayton m_delegate_sp() 199444d93782SGreg Clayton { 199544d93782SGreg Clayton if (name && name[0]) 199644d93782SGreg Clayton { 199744d93782SGreg Clayton m_name = name; 199844d93782SGreg Clayton m_type = Type::Item; 199944d93782SGreg Clayton if (key_name && key_name[0]) 200044d93782SGreg Clayton m_key_name = key_name; 200144d93782SGreg Clayton } 200244d93782SGreg Clayton else 200344d93782SGreg Clayton { 200444d93782SGreg Clayton m_type = Type::Separator; 200544d93782SGreg Clayton } 200644d93782SGreg Clayton } 200744d93782SGreg Clayton 200844d93782SGreg Clayton void 200944d93782SGreg Clayton Menu::RecalculateNameLengths() 201044d93782SGreg Clayton { 201144d93782SGreg Clayton m_max_submenu_name_length = 0; 201244d93782SGreg Clayton m_max_submenu_key_name_length = 0; 201344d93782SGreg Clayton Menus &submenus = GetSubmenus(); 201444d93782SGreg Clayton const size_t num_submenus = submenus.size(); 201544d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 201644d93782SGreg Clayton { 201744d93782SGreg Clayton Menu *submenu = submenus[i].get(); 20183985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_name_length) < submenu->m_name.size()) 201944d93782SGreg Clayton m_max_submenu_name_length = submenu->m_name.size(); 20203985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_key_name_length) < submenu->m_key_name.size()) 202144d93782SGreg Clayton m_max_submenu_key_name_length = submenu->m_key_name.size(); 202244d93782SGreg Clayton } 202344d93782SGreg Clayton } 202444d93782SGreg Clayton 202544d93782SGreg Clayton void 202644d93782SGreg Clayton Menu::AddSubmenu (const MenuSP &menu_sp) 202744d93782SGreg Clayton { 202844d93782SGreg Clayton menu_sp->m_parent = this; 20293985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_name_length) < menu_sp->m_name.size()) 203044d93782SGreg Clayton m_max_submenu_name_length = menu_sp->m_name.size(); 20313985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_max_submenu_key_name_length) < menu_sp->m_key_name.size()) 203244d93782SGreg Clayton m_max_submenu_key_name_length = menu_sp->m_key_name.size(); 203344d93782SGreg Clayton m_submenus.push_back(menu_sp); 203444d93782SGreg Clayton } 203544d93782SGreg Clayton 203644d93782SGreg Clayton void 203744d93782SGreg Clayton Menu::DrawMenuTitle (Window &window, bool highlight) 203844d93782SGreg Clayton { 203944d93782SGreg Clayton if (m_type == Type::Separator) 204044d93782SGreg Clayton { 204144d93782SGreg Clayton window.MoveCursor(0, window.GetCursorY()); 204244d93782SGreg Clayton window.PutChar(ACS_LTEE); 204344d93782SGreg Clayton int width = window.GetWidth(); 204444d93782SGreg Clayton if (width > 2) 204544d93782SGreg Clayton { 204644d93782SGreg Clayton width -= 2; 20473985c8c6SSaleem Abdulrasool for (int i=0; i< width; ++i) 204844d93782SGreg Clayton window.PutChar(ACS_HLINE); 204944d93782SGreg Clayton } 205044d93782SGreg Clayton window.PutChar(ACS_RTEE); 205144d93782SGreg Clayton } 205244d93782SGreg Clayton else 205344d93782SGreg Clayton { 205444d93782SGreg Clayton const int shortcut_key = m_key_value; 205544d93782SGreg Clayton bool underlined_shortcut = false; 205644d93782SGreg Clayton const attr_t hilgight_attr = A_REVERSE; 205744d93782SGreg Clayton if (highlight) 205844d93782SGreg Clayton window.AttributeOn(hilgight_attr); 205944d93782SGreg Clayton if (isprint(shortcut_key)) 206044d93782SGreg Clayton { 206144d93782SGreg Clayton size_t lower_pos = m_name.find(tolower(shortcut_key)); 206244d93782SGreg Clayton size_t upper_pos = m_name.find(toupper(shortcut_key)); 206344d93782SGreg Clayton const char *name = m_name.c_str(); 206444d93782SGreg Clayton size_t pos = std::min<size_t>(lower_pos, upper_pos); 206544d93782SGreg Clayton if (pos != std::string::npos) 206644d93782SGreg Clayton { 206744d93782SGreg Clayton underlined_shortcut = true; 206844d93782SGreg Clayton if (pos > 0) 206944d93782SGreg Clayton { 207044d93782SGreg Clayton window.PutCString(name, pos); 207144d93782SGreg Clayton name += pos; 207244d93782SGreg Clayton } 207344d93782SGreg Clayton const attr_t shortcut_attr = A_UNDERLINE|A_BOLD; 207444d93782SGreg Clayton window.AttributeOn (shortcut_attr); 207544d93782SGreg Clayton window.PutChar(name[0]); 207644d93782SGreg Clayton window.AttributeOff(shortcut_attr); 207744d93782SGreg Clayton name++; 207844d93782SGreg Clayton if (name[0]) 207944d93782SGreg Clayton window.PutCString(name); 208044d93782SGreg Clayton } 208144d93782SGreg Clayton } 208244d93782SGreg Clayton 208344d93782SGreg Clayton if (!underlined_shortcut) 208444d93782SGreg Clayton { 208544d93782SGreg Clayton window.PutCString(m_name.c_str()); 208644d93782SGreg Clayton } 208744d93782SGreg Clayton 208844d93782SGreg Clayton if (highlight) 208944d93782SGreg Clayton window.AttributeOff(hilgight_attr); 209044d93782SGreg Clayton 209144d93782SGreg Clayton if (m_key_name.empty()) 209244d93782SGreg Clayton { 209344d93782SGreg Clayton if (!underlined_shortcut && isprint(m_key_value)) 209444d93782SGreg Clayton { 209544d93782SGreg Clayton window.AttributeOn (COLOR_PAIR(3)); 209644d93782SGreg Clayton window.Printf (" (%c)", m_key_value); 209744d93782SGreg Clayton window.AttributeOff (COLOR_PAIR(3)); 209844d93782SGreg Clayton } 209944d93782SGreg Clayton } 210044d93782SGreg Clayton else 210144d93782SGreg Clayton { 210244d93782SGreg Clayton window.AttributeOn (COLOR_PAIR(3)); 210344d93782SGreg Clayton window.Printf (" (%s)", m_key_name.c_str()); 210444d93782SGreg Clayton window.AttributeOff (COLOR_PAIR(3)); 210544d93782SGreg Clayton } 210644d93782SGreg Clayton } 210744d93782SGreg Clayton } 210844d93782SGreg Clayton 210944d93782SGreg Clayton bool 211044d93782SGreg Clayton Menu::WindowDelegateDraw (Window &window, bool force) 211144d93782SGreg Clayton { 211244d93782SGreg Clayton Menus &submenus = GetSubmenus(); 211344d93782SGreg Clayton const size_t num_submenus = submenus.size(); 211444d93782SGreg Clayton const int selected_idx = GetSelectedSubmenuIndex(); 211544d93782SGreg Clayton Menu::Type menu_type = GetType (); 211644d93782SGreg Clayton switch (menu_type) 211744d93782SGreg Clayton { 211844d93782SGreg Clayton case Menu::Type::Bar: 211944d93782SGreg Clayton { 212044d93782SGreg Clayton window.SetBackground(2); 212144d93782SGreg Clayton window.MoveCursor(0, 0); 212244d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 212344d93782SGreg Clayton { 212444d93782SGreg Clayton Menu *menu = submenus[i].get(); 212544d93782SGreg Clayton if (i > 0) 212644d93782SGreg Clayton window.PutChar(' '); 212744d93782SGreg Clayton menu->SetStartingColumn (window.GetCursorX()); 212844d93782SGreg Clayton window.PutCString("| "); 212944d93782SGreg Clayton menu->DrawMenuTitle (window, false); 213044d93782SGreg Clayton } 213144d93782SGreg Clayton window.PutCString(" |"); 213244d93782SGreg Clayton window.DeferredRefresh(); 213344d93782SGreg Clayton } 213444d93782SGreg Clayton break; 213544d93782SGreg Clayton 213644d93782SGreg Clayton case Menu::Type::Item: 213744d93782SGreg Clayton { 213844d93782SGreg Clayton int y = 1; 213944d93782SGreg Clayton int x = 3; 214044d93782SGreg Clayton // Draw the menu 214144d93782SGreg Clayton int cursor_x = 0; 214244d93782SGreg Clayton int cursor_y = 0; 214344d93782SGreg Clayton window.Erase(); 214444d93782SGreg Clayton window.SetBackground(2); 214544d93782SGreg Clayton window.Box(); 214644d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 214744d93782SGreg Clayton { 21483985c8c6SSaleem Abdulrasool const bool is_selected = 21493985c8c6SSaleem Abdulrasool (i == static_cast<size_t>(selected_idx)); 215044d93782SGreg Clayton window.MoveCursor(x, y + i); 215144d93782SGreg Clayton if (is_selected) 215244d93782SGreg Clayton { 215344d93782SGreg Clayton // Remember where we want the cursor to be 215444d93782SGreg Clayton cursor_x = x-1; 215544d93782SGreg Clayton cursor_y = y+i; 215644d93782SGreg Clayton } 215744d93782SGreg Clayton submenus[i]->DrawMenuTitle (window, is_selected); 215844d93782SGreg Clayton } 215944d93782SGreg Clayton window.MoveCursor(cursor_x, cursor_y); 216044d93782SGreg Clayton window.DeferredRefresh(); 216144d93782SGreg Clayton } 216244d93782SGreg Clayton break; 216344d93782SGreg Clayton 216444d93782SGreg Clayton default: 216544d93782SGreg Clayton case Menu::Type::Separator: 216644d93782SGreg Clayton break; 216744d93782SGreg Clayton } 216844d93782SGreg Clayton return true; // Drawing handled... 216944d93782SGreg Clayton } 217044d93782SGreg Clayton 217144d93782SGreg Clayton HandleCharResult 217244d93782SGreg Clayton Menu::WindowDelegateHandleChar (Window &window, int key) 217344d93782SGreg Clayton { 217444d93782SGreg Clayton HandleCharResult result = eKeyNotHandled; 217544d93782SGreg Clayton 217644d93782SGreg Clayton Menus &submenus = GetSubmenus(); 217744d93782SGreg Clayton const size_t num_submenus = submenus.size(); 217844d93782SGreg Clayton const int selected_idx = GetSelectedSubmenuIndex(); 217944d93782SGreg Clayton Menu::Type menu_type = GetType (); 218044d93782SGreg Clayton if (menu_type == Menu::Type::Bar) 218144d93782SGreg Clayton { 218244d93782SGreg Clayton MenuSP run_menu_sp; 218344d93782SGreg Clayton switch (key) 218444d93782SGreg Clayton { 218544d93782SGreg Clayton case KEY_DOWN: 218644d93782SGreg Clayton case KEY_UP: 218744d93782SGreg Clayton // Show last menu or first menu 21883985c8c6SSaleem Abdulrasool if (selected_idx < static_cast<int>(num_submenus)) 218944d93782SGreg Clayton run_menu_sp = submenus[selected_idx]; 219044d93782SGreg Clayton else if (!submenus.empty()) 219144d93782SGreg Clayton run_menu_sp = submenus.front(); 219244d93782SGreg Clayton result = eKeyHandled; 219344d93782SGreg Clayton break; 219444d93782SGreg Clayton 219544d93782SGreg Clayton case KEY_RIGHT: 219644d93782SGreg Clayton { 219744d93782SGreg Clayton ++m_selected; 21983985c8c6SSaleem Abdulrasool if (m_selected >= static_cast<int>(num_submenus)) 219944d93782SGreg Clayton m_selected = 0; 22003985c8c6SSaleem Abdulrasool if (m_selected < static_cast<int>(num_submenus)) 220144d93782SGreg Clayton run_menu_sp = submenus[m_selected]; 220244d93782SGreg Clayton else if (!submenus.empty()) 220344d93782SGreg Clayton run_menu_sp = submenus.front(); 220444d93782SGreg Clayton result = eKeyHandled; 220544d93782SGreg Clayton } 220644d93782SGreg Clayton break; 220744d93782SGreg Clayton 220844d93782SGreg Clayton case KEY_LEFT: 220944d93782SGreg Clayton { 221044d93782SGreg Clayton --m_selected; 221144d93782SGreg Clayton if (m_selected < 0) 221244d93782SGreg Clayton m_selected = num_submenus - 1; 22133985c8c6SSaleem Abdulrasool if (m_selected < static_cast<int>(num_submenus)) 221444d93782SGreg Clayton run_menu_sp = submenus[m_selected]; 221544d93782SGreg Clayton else if (!submenus.empty()) 221644d93782SGreg Clayton run_menu_sp = submenus.front(); 221744d93782SGreg Clayton result = eKeyHandled; 221844d93782SGreg Clayton } 221944d93782SGreg Clayton break; 222044d93782SGreg Clayton 222144d93782SGreg Clayton default: 222244d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 222344d93782SGreg Clayton { 222444d93782SGreg Clayton if (submenus[i]->GetKeyValue() == key) 222544d93782SGreg Clayton { 222644d93782SGreg Clayton SetSelectedSubmenuIndex(i); 222744d93782SGreg Clayton run_menu_sp = submenus[i]; 222844d93782SGreg Clayton result = eKeyHandled; 222944d93782SGreg Clayton break; 223044d93782SGreg Clayton } 223144d93782SGreg Clayton } 223244d93782SGreg Clayton break; 223344d93782SGreg Clayton } 223444d93782SGreg Clayton 223544d93782SGreg Clayton if (run_menu_sp) 223644d93782SGreg Clayton { 223744d93782SGreg Clayton // Run the action on this menu in case we need to populate the 223844d93782SGreg Clayton // menu with dynamic content and also in case check marks, and 223944d93782SGreg Clayton // any other menu decorations need to be caclulated 224044d93782SGreg Clayton if (run_menu_sp->Action() == MenuActionResult::Quit) 224144d93782SGreg Clayton return eQuitApplication; 224244d93782SGreg Clayton 224344d93782SGreg Clayton Rect menu_bounds; 224444d93782SGreg Clayton menu_bounds.origin.x = run_menu_sp->GetStartingColumn(); 224544d93782SGreg Clayton menu_bounds.origin.y = 1; 224644d93782SGreg Clayton menu_bounds.size.width = run_menu_sp->GetDrawWidth(); 224744d93782SGreg Clayton menu_bounds.size.height = run_menu_sp->GetSubmenus().size() + 2; 224844d93782SGreg Clayton if (m_menu_window_sp) 224944d93782SGreg Clayton window.GetParent()->RemoveSubWindow(m_menu_window_sp.get()); 225044d93782SGreg Clayton 225144d93782SGreg Clayton m_menu_window_sp = window.GetParent()->CreateSubWindow (run_menu_sp->GetName().c_str(), 225244d93782SGreg Clayton menu_bounds, 225344d93782SGreg Clayton true); 225444d93782SGreg Clayton m_menu_window_sp->SetDelegate (run_menu_sp); 225544d93782SGreg Clayton } 225644d93782SGreg Clayton } 225744d93782SGreg Clayton else if (menu_type == Menu::Type::Item) 225844d93782SGreg Clayton { 225944d93782SGreg Clayton switch (key) 226044d93782SGreg Clayton { 226144d93782SGreg Clayton case KEY_DOWN: 226244d93782SGreg Clayton if (m_submenus.size() > 1) 226344d93782SGreg Clayton { 226444d93782SGreg Clayton const int start_select = m_selected; 226544d93782SGreg Clayton while (++m_selected != start_select) 226644d93782SGreg Clayton { 22673985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_selected) >= num_submenus) 226844d93782SGreg Clayton m_selected = 0; 226944d93782SGreg Clayton if (m_submenus[m_selected]->GetType() == Type::Separator) 227044d93782SGreg Clayton continue; 227144d93782SGreg Clayton else 227244d93782SGreg Clayton break; 227344d93782SGreg Clayton } 227444d93782SGreg Clayton return eKeyHandled; 227544d93782SGreg Clayton } 227644d93782SGreg Clayton break; 227744d93782SGreg Clayton 227844d93782SGreg Clayton case KEY_UP: 227944d93782SGreg Clayton if (m_submenus.size() > 1) 228044d93782SGreg Clayton { 228144d93782SGreg Clayton const int start_select = m_selected; 228244d93782SGreg Clayton while (--m_selected != start_select) 228344d93782SGreg Clayton { 22843985c8c6SSaleem Abdulrasool if (m_selected < static_cast<int>(0)) 228544d93782SGreg Clayton m_selected = num_submenus - 1; 228644d93782SGreg Clayton if (m_submenus[m_selected]->GetType() == Type::Separator) 228744d93782SGreg Clayton continue; 228844d93782SGreg Clayton else 228944d93782SGreg Clayton break; 229044d93782SGreg Clayton } 229144d93782SGreg Clayton return eKeyHandled; 229244d93782SGreg Clayton } 229344d93782SGreg Clayton break; 229444d93782SGreg Clayton 229544d93782SGreg Clayton case KEY_RETURN: 22963985c8c6SSaleem Abdulrasool if (static_cast<size_t>(selected_idx) < num_submenus) 229744d93782SGreg Clayton { 229844d93782SGreg Clayton if (submenus[selected_idx]->Action() == MenuActionResult::Quit) 229944d93782SGreg Clayton return eQuitApplication; 230044d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 230144d93782SGreg Clayton return eKeyHandled; 230244d93782SGreg Clayton } 230344d93782SGreg Clayton break; 230444d93782SGreg Clayton 230544d93782SGreg Clayton case KEY_ESCAPE: // Beware: pressing escape key has 1 to 2 second delay in case other chars are entered for escaped sequences 230644d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 230744d93782SGreg Clayton return eKeyHandled; 230844d93782SGreg Clayton 230944d93782SGreg Clayton default: 231044d93782SGreg Clayton { 231144d93782SGreg Clayton for (size_t i=0; i<num_submenus; ++i) 231244d93782SGreg Clayton { 231344d93782SGreg Clayton Menu *menu = submenus[i].get(); 231444d93782SGreg Clayton if (menu->GetKeyValue() == key) 231544d93782SGreg Clayton { 231644d93782SGreg Clayton SetSelectedSubmenuIndex(i); 231744d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 231844d93782SGreg Clayton if (menu->Action() == MenuActionResult::Quit) 231944d93782SGreg Clayton return eQuitApplication; 232044d93782SGreg Clayton return eKeyHandled; 232144d93782SGreg Clayton } 232244d93782SGreg Clayton } 232344d93782SGreg Clayton } 232444d93782SGreg Clayton break; 232544d93782SGreg Clayton 232644d93782SGreg Clayton } 232744d93782SGreg Clayton } 232844d93782SGreg Clayton else if (menu_type == Menu::Type::Separator) 232944d93782SGreg Clayton { 233044d93782SGreg Clayton 233144d93782SGreg Clayton } 233244d93782SGreg Clayton return result; 233344d93782SGreg Clayton } 233444d93782SGreg Clayton 233544d93782SGreg Clayton 233644d93782SGreg Clayton class Application 233744d93782SGreg Clayton { 233844d93782SGreg Clayton public: 233944d93782SGreg Clayton Application (FILE *in, FILE *out) : 234044d93782SGreg Clayton m_window_sp(), 234144d93782SGreg Clayton m_screen (NULL), 234244d93782SGreg Clayton m_in (in), 234344d93782SGreg Clayton m_out (out) 234444d93782SGreg Clayton { 234544d93782SGreg Clayton 234644d93782SGreg Clayton } 234744d93782SGreg Clayton 234844d93782SGreg Clayton ~Application () 234944d93782SGreg Clayton { 235044d93782SGreg Clayton m_window_delegates.clear(); 235144d93782SGreg Clayton m_window_sp.reset(); 235244d93782SGreg Clayton if (m_screen) 235344d93782SGreg Clayton { 235444d93782SGreg Clayton ::delscreen(m_screen); 235544d93782SGreg Clayton m_screen = NULL; 235644d93782SGreg Clayton } 235744d93782SGreg Clayton } 235844d93782SGreg Clayton 235944d93782SGreg Clayton void 236044d93782SGreg Clayton Initialize () 236144d93782SGreg Clayton { 236244d93782SGreg Clayton ::setlocale(LC_ALL, ""); 236344d93782SGreg Clayton ::setlocale(LC_CTYPE, ""); 236444d93782SGreg Clayton #if 0 236544d93782SGreg Clayton ::initscr(); 236644d93782SGreg Clayton #else 236744d93782SGreg Clayton m_screen = ::newterm(NULL, m_out, m_in); 236844d93782SGreg Clayton #endif 236944d93782SGreg Clayton ::start_color(); 237044d93782SGreg Clayton ::curs_set(0); 237144d93782SGreg Clayton ::noecho(); 237244d93782SGreg Clayton ::keypad(stdscr,TRUE); 237344d93782SGreg Clayton } 237444d93782SGreg Clayton 237544d93782SGreg Clayton void 237644d93782SGreg Clayton Terminate () 237744d93782SGreg Clayton { 237844d93782SGreg Clayton ::endwin(); 237944d93782SGreg Clayton } 238044d93782SGreg Clayton 238144d93782SGreg Clayton void 238244d93782SGreg Clayton Run (Debugger &debugger) 238344d93782SGreg Clayton { 238444d93782SGreg Clayton bool done = false; 238544d93782SGreg Clayton int delay_in_tenths_of_a_second = 1; 238644d93782SGreg Clayton 238744d93782SGreg Clayton // Alas the threading model in curses is a bit lame so we need to 238844d93782SGreg Clayton // resort to polling every 0.5 seconds. We could poll for stdin 238944d93782SGreg Clayton // ourselves and then pass the keys down but then we need to 239044d93782SGreg Clayton // translate all of the escape sequences ourselves. So we resort to 239144d93782SGreg Clayton // polling for input because we need to receive async process events 239244d93782SGreg Clayton // while in this loop. 239344d93782SGreg Clayton 239444d93782SGreg Clayton halfdelay(delay_in_tenths_of_a_second); // Poll using some number of tenths of seconds seconds when calling Window::GetChar() 239544d93782SGreg Clayton 239644d93782SGreg Clayton ListenerSP listener_sp (new Listener ("lldb.IOHandler.curses.Application")); 239744d93782SGreg Clayton ConstString broadcaster_class_target(Target::GetStaticBroadcasterClass()); 239844d93782SGreg Clayton ConstString broadcaster_class_process(Process::GetStaticBroadcasterClass()); 239944d93782SGreg Clayton ConstString broadcaster_class_thread(Thread::GetStaticBroadcasterClass()); 240044d93782SGreg Clayton debugger.EnableForwardEvents (listener_sp); 240144d93782SGreg Clayton 240244d93782SGreg Clayton bool update = true; 240344d93782SGreg Clayton #if defined(__APPLE__) 240444d93782SGreg Clayton std::deque<int> escape_chars; 240544d93782SGreg Clayton #endif 240644d93782SGreg Clayton 240744d93782SGreg Clayton while (!done) 240844d93782SGreg Clayton { 240944d93782SGreg Clayton if (update) 241044d93782SGreg Clayton { 241144d93782SGreg Clayton m_window_sp->Draw(false); 241244d93782SGreg Clayton // All windows should be calling Window::DeferredRefresh() instead 241344d93782SGreg Clayton // of Window::Refresh() so we can do a single update and avoid 241444d93782SGreg Clayton // any screen blinking 241544d93782SGreg Clayton update_panels(); 241644d93782SGreg Clayton 241744d93782SGreg Clayton // Cursor hiding isn't working on MacOSX, so hide it in the top left corner 241844d93782SGreg Clayton m_window_sp->MoveCursor(0, 0); 241944d93782SGreg Clayton 242044d93782SGreg Clayton doupdate(); 242144d93782SGreg Clayton update = false; 242244d93782SGreg Clayton } 242344d93782SGreg Clayton 242444d93782SGreg Clayton #if defined(__APPLE__) 242544d93782SGreg Clayton // Terminal.app doesn't map its function keys correctly, F1-F4 default to: 242644d93782SGreg Clayton // \033OP, \033OQ, \033OR, \033OS, so lets take care of this here if possible 242744d93782SGreg Clayton int ch; 242844d93782SGreg Clayton if (escape_chars.empty()) 242944d93782SGreg Clayton ch = m_window_sp->GetChar(); 243044d93782SGreg Clayton else 243144d93782SGreg Clayton { 243244d93782SGreg Clayton ch = escape_chars.front(); 243344d93782SGreg Clayton escape_chars.pop_front(); 243444d93782SGreg Clayton } 243544d93782SGreg Clayton if (ch == KEY_ESCAPE) 243644d93782SGreg Clayton { 243744d93782SGreg Clayton int ch2 = m_window_sp->GetChar(); 243844d93782SGreg Clayton if (ch2 == 'O') 243944d93782SGreg Clayton { 244044d93782SGreg Clayton int ch3 = m_window_sp->GetChar(); 244144d93782SGreg Clayton switch (ch3) 244244d93782SGreg Clayton { 244344d93782SGreg Clayton case 'P': ch = KEY_F(1); break; 244444d93782SGreg Clayton case 'Q': ch = KEY_F(2); break; 244544d93782SGreg Clayton case 'R': ch = KEY_F(3); break; 244644d93782SGreg Clayton case 'S': ch = KEY_F(4); break; 244744d93782SGreg Clayton default: 244844d93782SGreg Clayton escape_chars.push_back(ch2); 244944d93782SGreg Clayton if (ch3 != -1) 245044d93782SGreg Clayton escape_chars.push_back(ch3); 245144d93782SGreg Clayton break; 245244d93782SGreg Clayton } 245344d93782SGreg Clayton } 245444d93782SGreg Clayton else if (ch2 != -1) 245544d93782SGreg Clayton escape_chars.push_back(ch2); 245644d93782SGreg Clayton } 245744d93782SGreg Clayton #else 245844d93782SGreg Clayton int ch = m_window_sp->GetChar(); 245944d93782SGreg Clayton 246044d93782SGreg Clayton #endif 246144d93782SGreg Clayton if (ch == -1) 246244d93782SGreg Clayton { 246344d93782SGreg Clayton if (feof(m_in) || ferror(m_in)) 246444d93782SGreg Clayton { 246544d93782SGreg Clayton done = true; 246644d93782SGreg Clayton } 246744d93782SGreg Clayton else 246844d93782SGreg Clayton { 246944d93782SGreg Clayton // Just a timeout from using halfdelay(), check for events 247044d93782SGreg Clayton EventSP event_sp; 247144d93782SGreg Clayton while (listener_sp->PeekAtNextEvent()) 247244d93782SGreg Clayton { 247344d93782SGreg Clayton listener_sp->GetNextEvent(event_sp); 247444d93782SGreg Clayton 247544d93782SGreg Clayton if (event_sp) 247644d93782SGreg Clayton { 247744d93782SGreg Clayton Broadcaster *broadcaster = event_sp->GetBroadcaster(); 247844d93782SGreg Clayton if (broadcaster) 247944d93782SGreg Clayton { 248044d93782SGreg Clayton //uint32_t event_type = event_sp->GetType(); 248144d93782SGreg Clayton ConstString broadcaster_class (broadcaster->GetBroadcasterClass()); 248244d93782SGreg Clayton if (broadcaster_class == broadcaster_class_process) 248344d93782SGreg Clayton { 2484ec990867SGreg Clayton debugger.GetCommandInterpreter().UpdateExecutionContext(NULL); 248544d93782SGreg Clayton update = true; 248644d93782SGreg Clayton continue; // Don't get any key, just update our view 248744d93782SGreg Clayton } 248844d93782SGreg Clayton } 248944d93782SGreg Clayton } 249044d93782SGreg Clayton } 249144d93782SGreg Clayton } 249244d93782SGreg Clayton } 249344d93782SGreg Clayton else 249444d93782SGreg Clayton { 249544d93782SGreg Clayton HandleCharResult key_result = m_window_sp->HandleChar(ch); 249644d93782SGreg Clayton switch (key_result) 249744d93782SGreg Clayton { 249844d93782SGreg Clayton case eKeyHandled: 2499ec990867SGreg Clayton debugger.GetCommandInterpreter().UpdateExecutionContext(NULL); 250044d93782SGreg Clayton update = true; 250144d93782SGreg Clayton break; 250244d93782SGreg Clayton case eKeyNotHandled: 250344d93782SGreg Clayton break; 250444d93782SGreg Clayton case eQuitApplication: 250544d93782SGreg Clayton done = true; 250644d93782SGreg Clayton break; 250744d93782SGreg Clayton } 250844d93782SGreg Clayton } 250944d93782SGreg Clayton } 251044d93782SGreg Clayton 251144d93782SGreg Clayton debugger.CancelForwardEvents (listener_sp); 251244d93782SGreg Clayton 251344d93782SGreg Clayton } 251444d93782SGreg Clayton 251544d93782SGreg Clayton WindowSP & 251644d93782SGreg Clayton GetMainWindow () 251744d93782SGreg Clayton { 251844d93782SGreg Clayton if (!m_window_sp) 251944d93782SGreg Clayton m_window_sp.reset (new Window ("main", stdscr, false)); 252044d93782SGreg Clayton return m_window_sp; 252144d93782SGreg Clayton } 252244d93782SGreg Clayton 252344d93782SGreg Clayton WindowDelegates & 252444d93782SGreg Clayton GetWindowDelegates () 252544d93782SGreg Clayton { 252644d93782SGreg Clayton return m_window_delegates; 252744d93782SGreg Clayton } 252844d93782SGreg Clayton 252944d93782SGreg Clayton protected: 253044d93782SGreg Clayton WindowSP m_window_sp; 253144d93782SGreg Clayton WindowDelegates m_window_delegates; 253244d93782SGreg Clayton SCREEN *m_screen; 253344d93782SGreg Clayton FILE *m_in; 253444d93782SGreg Clayton FILE *m_out; 253544d93782SGreg Clayton }; 253644d93782SGreg Clayton 253744d93782SGreg Clayton 253844d93782SGreg Clayton } // namespace curses 253944d93782SGreg Clayton 254044d93782SGreg Clayton 254144d93782SGreg Clayton using namespace curses; 254244d93782SGreg Clayton 254344d93782SGreg Clayton struct Row 254444d93782SGreg Clayton { 254544d93782SGreg Clayton ValueObjectSP valobj; 254644d93782SGreg Clayton Row *parent; 254744d93782SGreg Clayton int row_idx; 254844d93782SGreg Clayton int x; 254944d93782SGreg Clayton int y; 255044d93782SGreg Clayton bool might_have_children; 255144d93782SGreg Clayton bool expanded; 255244d93782SGreg Clayton bool calculated_children; 255344d93782SGreg Clayton std::vector<Row> children; 255444d93782SGreg Clayton 255544d93782SGreg Clayton Row (const ValueObjectSP &v, Row *p) : 255644d93782SGreg Clayton valobj (v), 255744d93782SGreg Clayton parent (p), 255844d93782SGreg Clayton row_idx(0), 255944d93782SGreg Clayton x(1), 256044d93782SGreg Clayton y(1), 256144d93782SGreg Clayton might_have_children (v ? v->MightHaveChildren() : false), 256244d93782SGreg Clayton expanded (false), 256344d93782SGreg Clayton calculated_children (false), 256444d93782SGreg Clayton children() 256544d93782SGreg Clayton { 256644d93782SGreg Clayton } 256744d93782SGreg Clayton 256844d93782SGreg Clayton size_t 256944d93782SGreg Clayton GetDepth () const 257044d93782SGreg Clayton { 257144d93782SGreg Clayton if (parent) 257244d93782SGreg Clayton return 1 + parent->GetDepth(); 257344d93782SGreg Clayton return 0; 257444d93782SGreg Clayton } 257544d93782SGreg Clayton 257644d93782SGreg Clayton void 257744d93782SGreg Clayton Expand() 257844d93782SGreg Clayton { 257944d93782SGreg Clayton expanded = true; 258044d93782SGreg Clayton if (!calculated_children) 258144d93782SGreg Clayton { 258244d93782SGreg Clayton calculated_children = true; 258344d93782SGreg Clayton if (valobj) 258444d93782SGreg Clayton { 258544d93782SGreg Clayton const size_t num_children = valobj->GetNumChildren(); 258644d93782SGreg Clayton for (size_t i=0; i<num_children; ++i) 258744d93782SGreg Clayton { 258844d93782SGreg Clayton children.push_back(Row (valobj->GetChildAtIndex(i, true), this)); 258944d93782SGreg Clayton } 259044d93782SGreg Clayton } 259144d93782SGreg Clayton } 259244d93782SGreg Clayton } 259344d93782SGreg Clayton 259444d93782SGreg Clayton void 259544d93782SGreg Clayton Unexpand () 259644d93782SGreg Clayton { 259744d93782SGreg Clayton expanded = false; 259844d93782SGreg Clayton } 259944d93782SGreg Clayton 260044d93782SGreg Clayton void 260144d93782SGreg Clayton DrawTree (Window &window) 260244d93782SGreg Clayton { 260344d93782SGreg Clayton if (parent) 260444d93782SGreg Clayton parent->DrawTreeForChild (window, this, 0); 260544d93782SGreg Clayton 260644d93782SGreg Clayton if (might_have_children) 260744d93782SGreg Clayton { 260844d93782SGreg Clayton // It we can get UTF8 characters to work we should try to use the "symbol" 260944d93782SGreg Clayton // UTF8 string below 261044d93782SGreg Clayton // const char *symbol = ""; 261144d93782SGreg Clayton // if (row.expanded) 261244d93782SGreg Clayton // symbol = "\xe2\x96\xbd "; 261344d93782SGreg Clayton // else 261444d93782SGreg Clayton // symbol = "\xe2\x96\xb7 "; 261544d93782SGreg Clayton // window.PutCString (symbol); 261644d93782SGreg Clayton 261744d93782SGreg Clayton // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 261844d93782SGreg Clayton // 'v' or '>' character... 261944d93782SGreg Clayton // if (expanded) 262044d93782SGreg Clayton // window.PutChar (ACS_DARROW); 262144d93782SGreg Clayton // else 262244d93782SGreg Clayton // window.PutChar (ACS_RARROW); 262344d93782SGreg Clayton // Since we can't find any good looking right arrow/down arrow 262444d93782SGreg Clayton // symbols, just use a diamond... 262544d93782SGreg Clayton window.PutChar (ACS_DIAMOND); 262644d93782SGreg Clayton window.PutChar (ACS_HLINE); 262744d93782SGreg Clayton } 262844d93782SGreg Clayton } 262944d93782SGreg Clayton 263044d93782SGreg Clayton void 263144d93782SGreg Clayton DrawTreeForChild (Window &window, Row *child, uint32_t reverse_depth) 263244d93782SGreg Clayton { 263344d93782SGreg Clayton if (parent) 263444d93782SGreg Clayton parent->DrawTreeForChild (window, this, reverse_depth + 1); 263544d93782SGreg Clayton 263644d93782SGreg Clayton if (&children.back() == child) 263744d93782SGreg Clayton { 263844d93782SGreg Clayton // Last child 263944d93782SGreg Clayton if (reverse_depth == 0) 264044d93782SGreg Clayton { 264144d93782SGreg Clayton window.PutChar (ACS_LLCORNER); 264244d93782SGreg Clayton window.PutChar (ACS_HLINE); 264344d93782SGreg Clayton } 264444d93782SGreg Clayton else 264544d93782SGreg Clayton { 264644d93782SGreg Clayton window.PutChar (' '); 264744d93782SGreg Clayton window.PutChar (' '); 264844d93782SGreg Clayton } 264944d93782SGreg Clayton } 265044d93782SGreg Clayton else 265144d93782SGreg Clayton { 265244d93782SGreg Clayton if (reverse_depth == 0) 265344d93782SGreg Clayton { 265444d93782SGreg Clayton window.PutChar (ACS_LTEE); 265544d93782SGreg Clayton window.PutChar (ACS_HLINE); 265644d93782SGreg Clayton } 265744d93782SGreg Clayton else 265844d93782SGreg Clayton { 265944d93782SGreg Clayton window.PutChar (ACS_VLINE); 266044d93782SGreg Clayton window.PutChar (' '); 266144d93782SGreg Clayton } 266244d93782SGreg Clayton } 266344d93782SGreg Clayton } 266444d93782SGreg Clayton }; 266544d93782SGreg Clayton 266644d93782SGreg Clayton struct DisplayOptions 266744d93782SGreg Clayton { 266844d93782SGreg Clayton bool show_types; 266944d93782SGreg Clayton }; 267044d93782SGreg Clayton 267144d93782SGreg Clayton class TreeItem; 267244d93782SGreg Clayton 267344d93782SGreg Clayton class TreeDelegate 267444d93782SGreg Clayton { 267544d93782SGreg Clayton public: 267644d93782SGreg Clayton TreeDelegate() {} 267744d93782SGreg Clayton virtual ~TreeDelegate() {} 267844d93782SGreg Clayton virtual void TreeDelegateDrawTreeItem (TreeItem &item, Window &window) = 0; 267944d93782SGreg Clayton virtual void TreeDelegateGenerateChildren (TreeItem &item) = 0; 268044d93782SGreg Clayton virtual bool TreeDelegateItemSelected (TreeItem &item) = 0; // Return true if we need to update views 268144d93782SGreg Clayton }; 268244d93782SGreg Clayton typedef std::shared_ptr<TreeDelegate> TreeDelegateSP; 268344d93782SGreg Clayton 268444d93782SGreg Clayton class TreeItem 268544d93782SGreg Clayton { 268644d93782SGreg Clayton public: 268744d93782SGreg Clayton 268844d93782SGreg Clayton TreeItem (TreeItem *parent, TreeDelegate &delegate, bool might_have_children) : 268944d93782SGreg Clayton m_parent (parent), 269044d93782SGreg Clayton m_delegate (delegate), 2691ec990867SGreg Clayton m_user_data (NULL), 269244d93782SGreg Clayton m_identifier (0), 269344d93782SGreg Clayton m_row_idx (-1), 269444d93782SGreg Clayton m_children (), 269544d93782SGreg Clayton m_might_have_children (might_have_children), 269644d93782SGreg Clayton m_is_expanded (false) 269744d93782SGreg Clayton { 269844d93782SGreg Clayton } 269944d93782SGreg Clayton 270044d93782SGreg Clayton TreeItem & 270144d93782SGreg Clayton operator=(const TreeItem &rhs) 270244d93782SGreg Clayton { 270344d93782SGreg Clayton if (this != &rhs) 270444d93782SGreg Clayton { 270544d93782SGreg Clayton m_parent = rhs.m_parent; 270644d93782SGreg Clayton m_delegate = rhs.m_delegate; 2707ec990867SGreg Clayton m_user_data = rhs.m_user_data; 270844d93782SGreg Clayton m_identifier = rhs.m_identifier; 270944d93782SGreg Clayton m_row_idx = rhs.m_row_idx; 271044d93782SGreg Clayton m_children = rhs.m_children; 271144d93782SGreg Clayton m_might_have_children = rhs.m_might_have_children; 271244d93782SGreg Clayton m_is_expanded = rhs.m_is_expanded; 271344d93782SGreg Clayton } 271444d93782SGreg Clayton return *this; 271544d93782SGreg Clayton } 271644d93782SGreg Clayton 271744d93782SGreg Clayton size_t 271844d93782SGreg Clayton GetDepth () const 271944d93782SGreg Clayton { 272044d93782SGreg Clayton if (m_parent) 272144d93782SGreg Clayton return 1 + m_parent->GetDepth(); 272244d93782SGreg Clayton return 0; 272344d93782SGreg Clayton } 272444d93782SGreg Clayton 272544d93782SGreg Clayton int 272644d93782SGreg Clayton GetRowIndex () const 272744d93782SGreg Clayton { 272844d93782SGreg Clayton return m_row_idx; 272944d93782SGreg Clayton } 273044d93782SGreg Clayton 273144d93782SGreg Clayton void 273244d93782SGreg Clayton ClearChildren () 273344d93782SGreg Clayton { 273444d93782SGreg Clayton m_children.clear(); 273544d93782SGreg Clayton } 273644d93782SGreg Clayton 273744d93782SGreg Clayton void 273844d93782SGreg Clayton Resize (size_t n, const TreeItem &t) 273944d93782SGreg Clayton { 274044d93782SGreg Clayton m_children.resize(n, t); 274144d93782SGreg Clayton } 274244d93782SGreg Clayton 274344d93782SGreg Clayton TreeItem & 274444d93782SGreg Clayton operator [](size_t i) 274544d93782SGreg Clayton { 274644d93782SGreg Clayton return m_children[i]; 274744d93782SGreg Clayton } 274844d93782SGreg Clayton 274944d93782SGreg Clayton void 275044d93782SGreg Clayton SetRowIndex (int row_idx) 275144d93782SGreg Clayton { 275244d93782SGreg Clayton m_row_idx = row_idx; 275344d93782SGreg Clayton } 275444d93782SGreg Clayton 275544d93782SGreg Clayton size_t 275644d93782SGreg Clayton GetNumChildren () 275744d93782SGreg Clayton { 275844d93782SGreg Clayton m_delegate.TreeDelegateGenerateChildren (*this); 275944d93782SGreg Clayton return m_children.size(); 276044d93782SGreg Clayton } 276144d93782SGreg Clayton 276244d93782SGreg Clayton void 276344d93782SGreg Clayton ItemWasSelected () 276444d93782SGreg Clayton { 276544d93782SGreg Clayton m_delegate.TreeDelegateItemSelected(*this); 276644d93782SGreg Clayton } 276744d93782SGreg Clayton void 276844d93782SGreg Clayton CalculateRowIndexes (int &row_idx) 276944d93782SGreg Clayton { 277044d93782SGreg Clayton SetRowIndex(row_idx); 277144d93782SGreg Clayton ++row_idx; 277244d93782SGreg Clayton 2773ec990867SGreg Clayton const bool expanded = IsExpanded(); 2774ec990867SGreg Clayton 2775ec990867SGreg Clayton // The root item must calculate its children, 2776ec990867SGreg Clayton // or we must calculate the number of children 2777ec990867SGreg Clayton // if the item is expanded 2778ec990867SGreg Clayton if (m_parent == NULL || expanded) 277944d93782SGreg Clayton GetNumChildren(); 278044d93782SGreg Clayton 278144d93782SGreg Clayton for (auto &item : m_children) 278244d93782SGreg Clayton { 278344d93782SGreg Clayton if (expanded) 278444d93782SGreg Clayton item.CalculateRowIndexes(row_idx); 278544d93782SGreg Clayton else 278644d93782SGreg Clayton item.SetRowIndex(-1); 278744d93782SGreg Clayton } 278844d93782SGreg Clayton } 278944d93782SGreg Clayton 279044d93782SGreg Clayton TreeItem * 279144d93782SGreg Clayton GetParent () 279244d93782SGreg Clayton { 279344d93782SGreg Clayton return m_parent; 279444d93782SGreg Clayton } 279544d93782SGreg Clayton 279644d93782SGreg Clayton bool 279744d93782SGreg Clayton IsExpanded () const 279844d93782SGreg Clayton { 279944d93782SGreg Clayton return m_is_expanded; 280044d93782SGreg Clayton } 280144d93782SGreg Clayton 280244d93782SGreg Clayton void 280344d93782SGreg Clayton Expand() 280444d93782SGreg Clayton { 280544d93782SGreg Clayton m_is_expanded = true; 280644d93782SGreg Clayton } 280744d93782SGreg Clayton 280844d93782SGreg Clayton void 280944d93782SGreg Clayton Unexpand () 281044d93782SGreg Clayton { 281144d93782SGreg Clayton m_is_expanded = false; 281244d93782SGreg Clayton } 281344d93782SGreg Clayton 281444d93782SGreg Clayton bool 281544d93782SGreg Clayton Draw (Window &window, 281644d93782SGreg Clayton const int first_visible_row, 281744d93782SGreg Clayton const uint32_t selected_row_idx, 281844d93782SGreg Clayton int &row_idx, 281944d93782SGreg Clayton int &num_rows_left) 282044d93782SGreg Clayton { 282144d93782SGreg Clayton if (num_rows_left <= 0) 282244d93782SGreg Clayton return false; 282344d93782SGreg Clayton 282444d93782SGreg Clayton if (m_row_idx >= first_visible_row) 282544d93782SGreg Clayton { 282644d93782SGreg Clayton window.MoveCursor(2, row_idx + 1); 282744d93782SGreg Clayton 282844d93782SGreg Clayton if (m_parent) 282944d93782SGreg Clayton m_parent->DrawTreeForChild (window, this, 0); 283044d93782SGreg Clayton 283144d93782SGreg Clayton if (m_might_have_children) 283244d93782SGreg Clayton { 283344d93782SGreg Clayton // It we can get UTF8 characters to work we should try to use the "symbol" 283444d93782SGreg Clayton // UTF8 string below 283544d93782SGreg Clayton // const char *symbol = ""; 283644d93782SGreg Clayton // if (row.expanded) 283744d93782SGreg Clayton // symbol = "\xe2\x96\xbd "; 283844d93782SGreg Clayton // else 283944d93782SGreg Clayton // symbol = "\xe2\x96\xb7 "; 284044d93782SGreg Clayton // window.PutCString (symbol); 284144d93782SGreg Clayton 284244d93782SGreg Clayton // The ACS_DARROW and ACS_RARROW don't look very nice they are just a 284344d93782SGreg Clayton // 'v' or '>' character... 284444d93782SGreg Clayton // if (expanded) 284544d93782SGreg Clayton // window.PutChar (ACS_DARROW); 284644d93782SGreg Clayton // else 284744d93782SGreg Clayton // window.PutChar (ACS_RARROW); 284844d93782SGreg Clayton // Since we can't find any good looking right arrow/down arrow 284944d93782SGreg Clayton // symbols, just use a diamond... 285044d93782SGreg Clayton window.PutChar (ACS_DIAMOND); 285144d93782SGreg Clayton window.PutChar (ACS_HLINE); 285244d93782SGreg Clayton } 28533985c8c6SSaleem Abdulrasool bool highlight = 28543985c8c6SSaleem Abdulrasool (selected_row_idx == static_cast<size_t>(m_row_idx)) && window.IsActive(); 285544d93782SGreg Clayton 285644d93782SGreg Clayton if (highlight) 285744d93782SGreg Clayton window.AttributeOn(A_REVERSE); 285844d93782SGreg Clayton 285944d93782SGreg Clayton m_delegate.TreeDelegateDrawTreeItem(*this, window); 286044d93782SGreg Clayton 286144d93782SGreg Clayton if (highlight) 286244d93782SGreg Clayton window.AttributeOff(A_REVERSE); 286344d93782SGreg Clayton ++row_idx; 286444d93782SGreg Clayton --num_rows_left; 286544d93782SGreg Clayton } 286644d93782SGreg Clayton 286744d93782SGreg Clayton if (num_rows_left <= 0) 286844d93782SGreg Clayton return false; // We are done drawing... 286944d93782SGreg Clayton 287044d93782SGreg Clayton if (IsExpanded()) 287144d93782SGreg Clayton { 287244d93782SGreg Clayton for (auto &item : m_children) 287344d93782SGreg Clayton { 287444d93782SGreg Clayton // If we displayed all the rows and item.Draw() returns 287544d93782SGreg Clayton // false we are done drawing and can exit this for loop 287644d93782SGreg Clayton if (item.Draw(window, first_visible_row, selected_row_idx, row_idx, num_rows_left) == false) 287744d93782SGreg Clayton break; 287844d93782SGreg Clayton } 287944d93782SGreg Clayton } 288044d93782SGreg Clayton return num_rows_left >= 0; // Return true if not done drawing yet 288144d93782SGreg Clayton } 288244d93782SGreg Clayton 288344d93782SGreg Clayton void 288444d93782SGreg Clayton DrawTreeForChild (Window &window, TreeItem *child, uint32_t reverse_depth) 288544d93782SGreg Clayton { 288644d93782SGreg Clayton if (m_parent) 288744d93782SGreg Clayton m_parent->DrawTreeForChild (window, this, reverse_depth + 1); 288844d93782SGreg Clayton 288944d93782SGreg Clayton if (&m_children.back() == child) 289044d93782SGreg Clayton { 289144d93782SGreg Clayton // Last child 289244d93782SGreg Clayton if (reverse_depth == 0) 289344d93782SGreg Clayton { 289444d93782SGreg Clayton window.PutChar (ACS_LLCORNER); 289544d93782SGreg Clayton window.PutChar (ACS_HLINE); 289644d93782SGreg Clayton } 289744d93782SGreg Clayton else 289844d93782SGreg Clayton { 289944d93782SGreg Clayton window.PutChar (' '); 290044d93782SGreg Clayton window.PutChar (' '); 290144d93782SGreg Clayton } 290244d93782SGreg Clayton } 290344d93782SGreg Clayton else 290444d93782SGreg Clayton { 290544d93782SGreg Clayton if (reverse_depth == 0) 290644d93782SGreg Clayton { 290744d93782SGreg Clayton window.PutChar (ACS_LTEE); 290844d93782SGreg Clayton window.PutChar (ACS_HLINE); 290944d93782SGreg Clayton } 291044d93782SGreg Clayton else 291144d93782SGreg Clayton { 291244d93782SGreg Clayton window.PutChar (ACS_VLINE); 291344d93782SGreg Clayton window.PutChar (' '); 291444d93782SGreg Clayton } 291544d93782SGreg Clayton } 291644d93782SGreg Clayton } 291744d93782SGreg Clayton 291844d93782SGreg Clayton TreeItem * 291944d93782SGreg Clayton GetItemForRowIndex (uint32_t row_idx) 292044d93782SGreg Clayton { 29213985c8c6SSaleem Abdulrasool if (static_cast<uint32_t>(m_row_idx) == row_idx) 292244d93782SGreg Clayton return this; 292344d93782SGreg Clayton if (m_children.empty()) 292444d93782SGreg Clayton return NULL; 292544d93782SGreg Clayton if (IsExpanded()) 292644d93782SGreg Clayton { 292744d93782SGreg Clayton for (auto &item : m_children) 292844d93782SGreg Clayton { 292944d93782SGreg Clayton TreeItem *selected_item_ptr = item.GetItemForRowIndex(row_idx); 293044d93782SGreg Clayton if (selected_item_ptr) 293144d93782SGreg Clayton return selected_item_ptr; 293244d93782SGreg Clayton } 293344d93782SGreg Clayton } 293444d93782SGreg Clayton return NULL; 293544d93782SGreg Clayton } 293644d93782SGreg Clayton 2937ec990867SGreg Clayton void * 2938ec990867SGreg Clayton GetUserData() const 2939ec990867SGreg Clayton { 2940ec990867SGreg Clayton return m_user_data; 2941ec990867SGreg Clayton } 2942ec990867SGreg Clayton 2943ec990867SGreg Clayton void 2944ec990867SGreg Clayton SetUserData (void *user_data) 2945ec990867SGreg Clayton { 2946ec990867SGreg Clayton m_user_data = user_data; 2947ec990867SGreg Clayton } 2948ec990867SGreg Clayton 294944d93782SGreg Clayton uint64_t 295044d93782SGreg Clayton GetIdentifier() const 295144d93782SGreg Clayton { 295244d93782SGreg Clayton return m_identifier; 295344d93782SGreg Clayton } 295444d93782SGreg Clayton 295544d93782SGreg Clayton void 295644d93782SGreg Clayton SetIdentifier (uint64_t identifier) 295744d93782SGreg Clayton { 295844d93782SGreg Clayton m_identifier = identifier; 295944d93782SGreg Clayton } 296044d93782SGreg Clayton 296144d93782SGreg Clayton 2962ec990867SGreg Clayton void 2963ec990867SGreg Clayton SetMightHaveChildren (bool b) 2964ec990867SGreg Clayton { 2965ec990867SGreg Clayton m_might_have_children = b; 2966ec990867SGreg Clayton } 2967ec990867SGreg Clayton 296844d93782SGreg Clayton protected: 296944d93782SGreg Clayton TreeItem *m_parent; 297044d93782SGreg Clayton TreeDelegate &m_delegate; 2971ec990867SGreg Clayton void *m_user_data; 297244d93782SGreg Clayton uint64_t m_identifier; 297344d93782SGreg Clayton int m_row_idx; // Zero based visible row index, -1 if not visible or for the root item 297444d93782SGreg Clayton std::vector<TreeItem> m_children; 297544d93782SGreg Clayton bool m_might_have_children; 297644d93782SGreg Clayton bool m_is_expanded; 297744d93782SGreg Clayton 297844d93782SGreg Clayton }; 297944d93782SGreg Clayton 298044d93782SGreg Clayton class TreeWindowDelegate : public WindowDelegate 298144d93782SGreg Clayton { 298244d93782SGreg Clayton public: 298344d93782SGreg Clayton TreeWindowDelegate (Debugger &debugger, const TreeDelegateSP &delegate_sp) : 298444d93782SGreg Clayton m_debugger (debugger), 298544d93782SGreg Clayton m_delegate_sp (delegate_sp), 298644d93782SGreg Clayton m_root (NULL, *delegate_sp, true), 298744d93782SGreg Clayton m_selected_item (NULL), 298844d93782SGreg Clayton m_num_rows (0), 298944d93782SGreg Clayton m_selected_row_idx (0), 299044d93782SGreg Clayton m_first_visible_row (0), 299144d93782SGreg Clayton m_min_x (0), 299244d93782SGreg Clayton m_min_y (0), 299344d93782SGreg Clayton m_max_x (0), 299444d93782SGreg Clayton m_max_y (0) 299544d93782SGreg Clayton { 299644d93782SGreg Clayton } 299744d93782SGreg Clayton 299844d93782SGreg Clayton int 299944d93782SGreg Clayton NumVisibleRows () const 300044d93782SGreg Clayton { 300144d93782SGreg Clayton return m_max_y - m_min_y; 300244d93782SGreg Clayton } 300344d93782SGreg Clayton 3004bd5ae6b4SGreg Clayton bool 3005bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 300644d93782SGreg Clayton { 300744d93782SGreg Clayton ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext()); 300844d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 300944d93782SGreg Clayton 301044d93782SGreg Clayton bool display_content = false; 301144d93782SGreg Clayton if (process) 301244d93782SGreg Clayton { 301344d93782SGreg Clayton StateType state = process->GetState(); 301444d93782SGreg Clayton if (StateIsStoppedState(state, true)) 301544d93782SGreg Clayton { 301644d93782SGreg Clayton // We are stopped, so it is ok to 301744d93782SGreg Clayton display_content = true; 301844d93782SGreg Clayton } 301944d93782SGreg Clayton else if (StateIsRunningState(state)) 302044d93782SGreg Clayton { 302144d93782SGreg Clayton return true; // Don't do any updating when we are running 302244d93782SGreg Clayton } 302344d93782SGreg Clayton } 302444d93782SGreg Clayton 302544d93782SGreg Clayton m_min_x = 2; 302644d93782SGreg Clayton m_min_y = 1; 302744d93782SGreg Clayton m_max_x = window.GetWidth() - 1; 302844d93782SGreg Clayton m_max_y = window.GetHeight() - 1; 302944d93782SGreg Clayton 303044d93782SGreg Clayton window.Erase(); 303144d93782SGreg Clayton window.DrawTitleBox (window.GetName()); 303244d93782SGreg Clayton 303344d93782SGreg Clayton if (display_content) 303444d93782SGreg Clayton { 303544d93782SGreg Clayton const int num_visible_rows = NumVisibleRows(); 303644d93782SGreg Clayton m_num_rows = 0; 303744d93782SGreg Clayton m_root.CalculateRowIndexes(m_num_rows); 303844d93782SGreg Clayton 303944d93782SGreg Clayton // If we unexpanded while having something selected our 304044d93782SGreg Clayton // total number of rows is less than the num visible rows, 304144d93782SGreg Clayton // then make sure we show all the rows by setting the first 304244d93782SGreg Clayton // visible row accordingly. 304344d93782SGreg Clayton if (m_first_visible_row > 0 && m_num_rows < num_visible_rows) 304444d93782SGreg Clayton m_first_visible_row = 0; 304544d93782SGreg Clayton 304644d93782SGreg Clayton // Make sure the selected row is always visible 304744d93782SGreg Clayton if (m_selected_row_idx < m_first_visible_row) 304844d93782SGreg Clayton m_first_visible_row = m_selected_row_idx; 304944d93782SGreg Clayton else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx) 305044d93782SGreg Clayton m_first_visible_row = m_selected_row_idx - num_visible_rows + 1; 305144d93782SGreg Clayton 305244d93782SGreg Clayton int row_idx = 0; 305344d93782SGreg Clayton int num_rows_left = num_visible_rows; 305444d93782SGreg Clayton m_root.Draw (window, m_first_visible_row, m_selected_row_idx, row_idx, num_rows_left); 305544d93782SGreg Clayton // Get the selected row 305644d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex (m_selected_row_idx); 305744d93782SGreg Clayton } 305844d93782SGreg Clayton else 305944d93782SGreg Clayton { 306044d93782SGreg Clayton m_selected_item = NULL; 306144d93782SGreg Clayton } 306244d93782SGreg Clayton 306344d93782SGreg Clayton window.DeferredRefresh(); 306444d93782SGreg Clayton 306544d93782SGreg Clayton 306644d93782SGreg Clayton return true; // Drawing handled 306744d93782SGreg Clayton } 306844d93782SGreg Clayton 306944d93782SGreg Clayton 3070bd5ae6b4SGreg Clayton const char * 3071bd5ae6b4SGreg Clayton WindowDelegateGetHelpText () override 307244d93782SGreg Clayton { 307344d93782SGreg Clayton return "Thread window keyboard shortcuts:"; 307444d93782SGreg Clayton } 307544d93782SGreg Clayton 3076bd5ae6b4SGreg Clayton KeyHelp * 3077bd5ae6b4SGreg Clayton WindowDelegateGetKeyHelp () override 307844d93782SGreg Clayton { 307944d93782SGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 308044d93782SGreg Clayton { KEY_UP, "Select previous item" }, 308144d93782SGreg Clayton { KEY_DOWN, "Select next item" }, 308244d93782SGreg Clayton { KEY_RIGHT, "Expand the selected item" }, 308344d93782SGreg Clayton { KEY_LEFT, "Unexpand the selected item or select parent if not expanded" }, 308444d93782SGreg Clayton { KEY_PPAGE, "Page up" }, 308544d93782SGreg Clayton { KEY_NPAGE, "Page down" }, 308644d93782SGreg Clayton { 'h', "Show help dialog" }, 308744d93782SGreg Clayton { ' ', "Toggle item expansion" }, 308844d93782SGreg Clayton { ',', "Page up" }, 308944d93782SGreg Clayton { '.', "Page down" }, 309044d93782SGreg Clayton { '\0', NULL } 309144d93782SGreg Clayton }; 309244d93782SGreg Clayton return g_source_view_key_help; 309344d93782SGreg Clayton } 309444d93782SGreg Clayton 3095bd5ae6b4SGreg Clayton HandleCharResult 3096bd5ae6b4SGreg Clayton WindowDelegateHandleChar (Window &window, int c) override 309744d93782SGreg Clayton { 309844d93782SGreg Clayton switch(c) 309944d93782SGreg Clayton { 310044d93782SGreg Clayton case ',': 310144d93782SGreg Clayton case KEY_PPAGE: 310244d93782SGreg Clayton // Page up key 310344d93782SGreg Clayton if (m_first_visible_row > 0) 310444d93782SGreg Clayton { 310544d93782SGreg Clayton if (m_first_visible_row > m_max_y) 310644d93782SGreg Clayton m_first_visible_row -= m_max_y; 310744d93782SGreg Clayton else 310844d93782SGreg Clayton m_first_visible_row = 0; 310944d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 311044d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 311144d93782SGreg Clayton if (m_selected_item) 311244d93782SGreg Clayton m_selected_item->ItemWasSelected (); 311344d93782SGreg Clayton } 311444d93782SGreg Clayton return eKeyHandled; 311544d93782SGreg Clayton 311644d93782SGreg Clayton case '.': 311744d93782SGreg Clayton case KEY_NPAGE: 311844d93782SGreg Clayton // Page down key 311944d93782SGreg Clayton if (m_num_rows > m_max_y) 312044d93782SGreg Clayton { 312144d93782SGreg Clayton if (m_first_visible_row + m_max_y < m_num_rows) 312244d93782SGreg Clayton { 312344d93782SGreg Clayton m_first_visible_row += m_max_y; 312444d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 312544d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 312644d93782SGreg Clayton if (m_selected_item) 312744d93782SGreg Clayton m_selected_item->ItemWasSelected (); 312844d93782SGreg Clayton } 312944d93782SGreg Clayton } 313044d93782SGreg Clayton return eKeyHandled; 313144d93782SGreg Clayton 313244d93782SGreg Clayton case KEY_UP: 313344d93782SGreg Clayton if (m_selected_row_idx > 0) 313444d93782SGreg Clayton { 313544d93782SGreg Clayton --m_selected_row_idx; 313644d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 313744d93782SGreg Clayton if (m_selected_item) 313844d93782SGreg Clayton m_selected_item->ItemWasSelected (); 313944d93782SGreg Clayton } 314044d93782SGreg Clayton return eKeyHandled; 314144d93782SGreg Clayton case KEY_DOWN: 314244d93782SGreg Clayton if (m_selected_row_idx + 1 < m_num_rows) 314344d93782SGreg Clayton { 314444d93782SGreg Clayton ++m_selected_row_idx; 314544d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 314644d93782SGreg Clayton if (m_selected_item) 314744d93782SGreg Clayton m_selected_item->ItemWasSelected (); 314844d93782SGreg Clayton } 314944d93782SGreg Clayton return eKeyHandled; 315044d93782SGreg Clayton 315144d93782SGreg Clayton case KEY_RIGHT: 315244d93782SGreg Clayton if (m_selected_item) 315344d93782SGreg Clayton { 315444d93782SGreg Clayton if (!m_selected_item->IsExpanded()) 315544d93782SGreg Clayton m_selected_item->Expand(); 315644d93782SGreg Clayton } 315744d93782SGreg Clayton return eKeyHandled; 315844d93782SGreg Clayton 315944d93782SGreg Clayton case KEY_LEFT: 316044d93782SGreg Clayton if (m_selected_item) 316144d93782SGreg Clayton { 316244d93782SGreg Clayton if (m_selected_item->IsExpanded()) 316344d93782SGreg Clayton m_selected_item->Unexpand(); 316444d93782SGreg Clayton else if (m_selected_item->GetParent()) 316544d93782SGreg Clayton { 316644d93782SGreg Clayton m_selected_row_idx = m_selected_item->GetParent()->GetRowIndex(); 316744d93782SGreg Clayton m_selected_item = m_root.GetItemForRowIndex(m_selected_row_idx); 316844d93782SGreg Clayton if (m_selected_item) 316944d93782SGreg Clayton m_selected_item->ItemWasSelected (); 317044d93782SGreg Clayton } 317144d93782SGreg Clayton } 317244d93782SGreg Clayton return eKeyHandled; 317344d93782SGreg Clayton 317444d93782SGreg Clayton case ' ': 317544d93782SGreg Clayton // Toggle expansion state when SPACE is pressed 317644d93782SGreg Clayton if (m_selected_item) 317744d93782SGreg Clayton { 317844d93782SGreg Clayton if (m_selected_item->IsExpanded()) 317944d93782SGreg Clayton m_selected_item->Unexpand(); 318044d93782SGreg Clayton else 318144d93782SGreg Clayton m_selected_item->Expand(); 318244d93782SGreg Clayton } 318344d93782SGreg Clayton return eKeyHandled; 318444d93782SGreg Clayton 318544d93782SGreg Clayton case 'h': 318644d93782SGreg Clayton window.CreateHelpSubwindow (); 318744d93782SGreg Clayton return eKeyHandled; 318844d93782SGreg Clayton 318944d93782SGreg Clayton default: 319044d93782SGreg Clayton break; 319144d93782SGreg Clayton } 319244d93782SGreg Clayton return eKeyNotHandled; 319344d93782SGreg Clayton } 319444d93782SGreg Clayton 319544d93782SGreg Clayton protected: 319644d93782SGreg Clayton Debugger &m_debugger; 319744d93782SGreg Clayton TreeDelegateSP m_delegate_sp; 319844d93782SGreg Clayton TreeItem m_root; 319944d93782SGreg Clayton TreeItem *m_selected_item; 320044d93782SGreg Clayton int m_num_rows; 320144d93782SGreg Clayton int m_selected_row_idx; 320244d93782SGreg Clayton int m_first_visible_row; 320344d93782SGreg Clayton int m_min_x; 320444d93782SGreg Clayton int m_min_y; 320544d93782SGreg Clayton int m_max_x; 320644d93782SGreg Clayton int m_max_y; 320744d93782SGreg Clayton 320844d93782SGreg Clayton }; 320944d93782SGreg Clayton 321044d93782SGreg Clayton class FrameTreeDelegate : public TreeDelegate 321144d93782SGreg Clayton { 321244d93782SGreg Clayton public: 3213ec990867SGreg Clayton FrameTreeDelegate () : 3214ec990867SGreg Clayton TreeDelegate() 321544d93782SGreg Clayton { 3216554f68d3SGreg Clayton FormatEntity::Parse ("frame #${frame.index}: {${function.name}${function.pc-offset}}}", 3217554f68d3SGreg Clayton m_format); 321844d93782SGreg Clayton } 321944d93782SGreg Clayton 3220bd5ae6b4SGreg Clayton ~FrameTreeDelegate() override 322144d93782SGreg Clayton { 322244d93782SGreg Clayton } 322344d93782SGreg Clayton 3224bd5ae6b4SGreg Clayton void 3225bd5ae6b4SGreg Clayton TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override 322644d93782SGreg Clayton { 3227ec990867SGreg Clayton Thread* thread = (Thread*)item.GetUserData(); 3228ec990867SGreg Clayton if (thread) 322944d93782SGreg Clayton { 323044d93782SGreg Clayton const uint64_t frame_idx = item.GetIdentifier(); 3231ec990867SGreg Clayton StackFrameSP frame_sp = thread->GetStackFrameAtIndex(frame_idx); 323244d93782SGreg Clayton if (frame_sp) 323344d93782SGreg Clayton { 323444d93782SGreg Clayton StreamString strm; 323544d93782SGreg Clayton const SymbolContext &sc = frame_sp->GetSymbolContext(eSymbolContextEverything); 323644d93782SGreg Clayton ExecutionContext exe_ctx (frame_sp); 3237554f68d3SGreg Clayton if (FormatEntity::Format(m_format, strm, &sc, &exe_ctx, NULL, NULL, false, false)) 323844d93782SGreg Clayton { 323944d93782SGreg Clayton int right_pad = 1; 324044d93782SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 324144d93782SGreg Clayton } 324244d93782SGreg Clayton } 324344d93782SGreg Clayton } 324444d93782SGreg Clayton } 3245bd5ae6b4SGreg Clayton void 3246bd5ae6b4SGreg Clayton TreeDelegateGenerateChildren (TreeItem &item) override 324744d93782SGreg Clayton { 324844d93782SGreg Clayton // No children for frames yet... 324944d93782SGreg Clayton } 325044d93782SGreg Clayton 3251bd5ae6b4SGreg Clayton bool 3252bd5ae6b4SGreg Clayton TreeDelegateItemSelected (TreeItem &item) override 325344d93782SGreg Clayton { 3254ec990867SGreg Clayton Thread* thread = (Thread*)item.GetUserData(); 3255ec990867SGreg Clayton if (thread) 325644d93782SGreg Clayton { 3257ec990867SGreg Clayton thread->GetProcess()->GetThreadList().SetSelectedThreadByID(thread->GetID()); 325844d93782SGreg Clayton const uint64_t frame_idx = item.GetIdentifier(); 3259ec990867SGreg Clayton thread->SetSelectedFrameByIndex(frame_idx); 326044d93782SGreg Clayton return true; 326144d93782SGreg Clayton } 326244d93782SGreg Clayton return false; 326344d93782SGreg Clayton } 3264554f68d3SGreg Clayton protected: 3265554f68d3SGreg Clayton FormatEntity::Entry m_format; 326644d93782SGreg Clayton }; 326744d93782SGreg Clayton 326844d93782SGreg Clayton class ThreadTreeDelegate : public TreeDelegate 326944d93782SGreg Clayton { 327044d93782SGreg Clayton public: 327144d93782SGreg Clayton ThreadTreeDelegate (Debugger &debugger) : 327244d93782SGreg Clayton TreeDelegate(), 327344d93782SGreg Clayton m_debugger (debugger), 327444d93782SGreg Clayton m_tid (LLDB_INVALID_THREAD_ID), 327544d93782SGreg Clayton m_stop_id (UINT32_MAX) 327644d93782SGreg Clayton { 3277554f68d3SGreg Clayton FormatEntity::Parse ("thread #${thread.index}: tid = ${thread.id}{, stop reason = ${thread.stop-reason}}", 3278554f68d3SGreg Clayton m_format); 327944d93782SGreg Clayton } 328044d93782SGreg Clayton 3281bd5ae6b4SGreg Clayton ~ThreadTreeDelegate() override 328244d93782SGreg Clayton { 328344d93782SGreg Clayton } 328444d93782SGreg Clayton 3285ec990867SGreg Clayton ProcessSP 3286ec990867SGreg Clayton GetProcess () 3287ec990867SGreg Clayton { 3288ec990867SGreg Clayton return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP(); 3289ec990867SGreg Clayton } 3290ec990867SGreg Clayton 3291ec990867SGreg Clayton ThreadSP 3292ec990867SGreg Clayton GetThread (const TreeItem &item) 3293ec990867SGreg Clayton { 3294ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3295ec990867SGreg Clayton if (process_sp) 3296ec990867SGreg Clayton return process_sp->GetThreadList().FindThreadByID(item.GetIdentifier()); 3297ec990867SGreg Clayton return ThreadSP(); 3298ec990867SGreg Clayton } 3299ec990867SGreg Clayton 3300bd5ae6b4SGreg Clayton void 3301bd5ae6b4SGreg Clayton TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override 330244d93782SGreg Clayton { 3303ec990867SGreg Clayton ThreadSP thread_sp = GetThread (item); 330444d93782SGreg Clayton if (thread_sp) 330544d93782SGreg Clayton { 330644d93782SGreg Clayton StreamString strm; 330744d93782SGreg Clayton ExecutionContext exe_ctx (thread_sp); 3308554f68d3SGreg Clayton if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false)) 330944d93782SGreg Clayton { 331044d93782SGreg Clayton int right_pad = 1; 331144d93782SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 331244d93782SGreg Clayton } 331344d93782SGreg Clayton } 331444d93782SGreg Clayton } 3315bd5ae6b4SGreg Clayton void 3316bd5ae6b4SGreg Clayton TreeDelegateGenerateChildren (TreeItem &item) override 331744d93782SGreg Clayton { 3318ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 331944d93782SGreg Clayton if (process_sp && process_sp->IsAlive()) 332044d93782SGreg Clayton { 332144d93782SGreg Clayton StateType state = process_sp->GetState(); 332244d93782SGreg Clayton if (StateIsStoppedState(state, true)) 332344d93782SGreg Clayton { 3324ec990867SGreg Clayton ThreadSP thread_sp = GetThread (item); 332544d93782SGreg Clayton if (thread_sp) 332644d93782SGreg Clayton { 332744d93782SGreg Clayton if (m_stop_id == process_sp->GetStopID() && thread_sp->GetID() == m_tid) 332844d93782SGreg Clayton return; // Children are already up to date 3329ec990867SGreg Clayton if (!m_frame_delegate_sp) 333044d93782SGreg Clayton { 333144d93782SGreg Clayton // Always expand the thread item the first time we show it 3332ec990867SGreg Clayton m_frame_delegate_sp.reset (new FrameTreeDelegate()); 333344d93782SGreg Clayton } 333444d93782SGreg Clayton 333544d93782SGreg Clayton m_stop_id = process_sp->GetStopID(); 333644d93782SGreg Clayton m_tid = thread_sp->GetID(); 333744d93782SGreg Clayton 333844d93782SGreg Clayton TreeItem t (&item, *m_frame_delegate_sp, false); 333944d93782SGreg Clayton size_t num_frames = thread_sp->GetStackFrameCount(); 334044d93782SGreg Clayton item.Resize (num_frames, t); 334144d93782SGreg Clayton for (size_t i=0; i<num_frames; ++i) 334244d93782SGreg Clayton { 3343ec990867SGreg Clayton item[i].SetUserData(thread_sp.get()); 334444d93782SGreg Clayton item[i].SetIdentifier(i); 334544d93782SGreg Clayton } 334644d93782SGreg Clayton } 334744d93782SGreg Clayton return; 334844d93782SGreg Clayton } 334944d93782SGreg Clayton } 335044d93782SGreg Clayton item.ClearChildren(); 335144d93782SGreg Clayton } 335244d93782SGreg Clayton 3353bd5ae6b4SGreg Clayton bool 3354bd5ae6b4SGreg Clayton TreeDelegateItemSelected (TreeItem &item) override 335544d93782SGreg Clayton { 3356ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3357ec990867SGreg Clayton if (process_sp && process_sp->IsAlive()) 3358ec990867SGreg Clayton { 3359ec990867SGreg Clayton StateType state = process_sp->GetState(); 3360ec990867SGreg Clayton if (StateIsStoppedState(state, true)) 3361ec990867SGreg Clayton { 3362ec990867SGreg Clayton ThreadSP thread_sp = GetThread (item); 336344d93782SGreg Clayton if (thread_sp) 336444d93782SGreg Clayton { 336544d93782SGreg Clayton ThreadList &thread_list = thread_sp->GetProcess()->GetThreadList(); 336644d93782SGreg Clayton Mutex::Locker locker (thread_list.GetMutex()); 336744d93782SGreg Clayton ThreadSP selected_thread_sp = thread_list.GetSelectedThread(); 336844d93782SGreg Clayton if (selected_thread_sp->GetID() != thread_sp->GetID()) 336944d93782SGreg Clayton { 337044d93782SGreg Clayton thread_list.SetSelectedThreadByID(thread_sp->GetID()); 337144d93782SGreg Clayton return true; 337244d93782SGreg Clayton } 337344d93782SGreg Clayton } 3374ec990867SGreg Clayton } 3375ec990867SGreg Clayton } 337644d93782SGreg Clayton return false; 337744d93782SGreg Clayton } 337844d93782SGreg Clayton 337944d93782SGreg Clayton protected: 338044d93782SGreg Clayton Debugger &m_debugger; 338144d93782SGreg Clayton std::shared_ptr<FrameTreeDelegate> m_frame_delegate_sp; 338244d93782SGreg Clayton lldb::user_id_t m_tid; 338344d93782SGreg Clayton uint32_t m_stop_id; 3384554f68d3SGreg Clayton FormatEntity::Entry m_format; 3385554f68d3SGreg Clayton 338644d93782SGreg Clayton }; 338744d93782SGreg Clayton 3388ec990867SGreg Clayton class ThreadsTreeDelegate : public TreeDelegate 3389ec990867SGreg Clayton { 3390ec990867SGreg Clayton public: 3391ec990867SGreg Clayton ThreadsTreeDelegate (Debugger &debugger) : 3392ec990867SGreg Clayton TreeDelegate(), 3393ec990867SGreg Clayton m_thread_delegate_sp (), 3394ec990867SGreg Clayton m_debugger (debugger), 3395ec990867SGreg Clayton m_stop_id (UINT32_MAX) 3396ec990867SGreg Clayton { 3397554f68d3SGreg Clayton FormatEntity::Parse("process ${process.id}{, name = ${process.name}}", 3398554f68d3SGreg Clayton m_format); 3399ec990867SGreg Clayton } 3400ec990867SGreg Clayton 3401bd5ae6b4SGreg Clayton ~ThreadsTreeDelegate() override 3402ec990867SGreg Clayton { 3403ec990867SGreg Clayton } 3404ec990867SGreg Clayton 3405ec990867SGreg Clayton ProcessSP 3406ec990867SGreg Clayton GetProcess () 3407ec990867SGreg Clayton { 3408ec990867SGreg Clayton return m_debugger.GetCommandInterpreter().GetExecutionContext().GetProcessSP(); 3409ec990867SGreg Clayton } 3410ec990867SGreg Clayton 3411bd5ae6b4SGreg Clayton void 3412bd5ae6b4SGreg Clayton TreeDelegateDrawTreeItem (TreeItem &item, Window &window) override 3413ec990867SGreg Clayton { 3414ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3415ec990867SGreg Clayton if (process_sp && process_sp->IsAlive()) 3416ec990867SGreg Clayton { 3417ec990867SGreg Clayton StreamString strm; 3418ec990867SGreg Clayton ExecutionContext exe_ctx (process_sp); 3419554f68d3SGreg Clayton if (FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false)) 3420ec990867SGreg Clayton { 3421ec990867SGreg Clayton int right_pad = 1; 3422ec990867SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 3423ec990867SGreg Clayton } 3424ec990867SGreg Clayton } 3425ec990867SGreg Clayton } 3426ec990867SGreg Clayton 3427bd5ae6b4SGreg Clayton void 3428bd5ae6b4SGreg Clayton TreeDelegateGenerateChildren (TreeItem &item) override 3429ec990867SGreg Clayton { 3430ec990867SGreg Clayton ProcessSP process_sp = GetProcess (); 3431ec990867SGreg Clayton if (process_sp && process_sp->IsAlive()) 3432ec990867SGreg Clayton { 3433ec990867SGreg Clayton StateType state = process_sp->GetState(); 3434ec990867SGreg Clayton if (StateIsStoppedState(state, true)) 3435ec990867SGreg Clayton { 3436ec990867SGreg Clayton const uint32_t stop_id = process_sp->GetStopID(); 3437ec990867SGreg Clayton if (m_stop_id == stop_id) 3438ec990867SGreg Clayton return; // Children are already up to date 3439ec990867SGreg Clayton 3440ec990867SGreg Clayton m_stop_id = stop_id; 3441ec990867SGreg Clayton 3442ec990867SGreg Clayton if (!m_thread_delegate_sp) 3443ec990867SGreg Clayton { 3444ec990867SGreg Clayton // Always expand the thread item the first time we show it 3445ec990867SGreg Clayton //item.Expand(); 3446ec990867SGreg Clayton m_thread_delegate_sp.reset (new ThreadTreeDelegate(m_debugger)); 3447ec990867SGreg Clayton } 3448ec990867SGreg Clayton 3449ec990867SGreg Clayton TreeItem t (&item, *m_thread_delegate_sp, false); 3450ec990867SGreg Clayton ThreadList &threads = process_sp->GetThreadList(); 3451ec990867SGreg Clayton Mutex::Locker locker (threads.GetMutex()); 3452ec990867SGreg Clayton size_t num_threads = threads.GetSize(); 3453ec990867SGreg Clayton item.Resize (num_threads, t); 3454ec990867SGreg Clayton for (size_t i=0; i<num_threads; ++i) 3455ec990867SGreg Clayton { 3456ec990867SGreg Clayton item[i].SetIdentifier(threads.GetThreadAtIndex(i)->GetID()); 3457ec990867SGreg Clayton item[i].SetMightHaveChildren(true); 3458ec990867SGreg Clayton } 3459ec990867SGreg Clayton return; 3460ec990867SGreg Clayton } 3461ec990867SGreg Clayton } 3462ec990867SGreg Clayton item.ClearChildren(); 3463ec990867SGreg Clayton } 3464ec990867SGreg Clayton 3465bd5ae6b4SGreg Clayton bool 3466bd5ae6b4SGreg Clayton TreeDelegateItemSelected (TreeItem &item) override 3467ec990867SGreg Clayton { 3468ec990867SGreg Clayton return false; 3469ec990867SGreg Clayton } 3470ec990867SGreg Clayton 3471ec990867SGreg Clayton protected: 3472ec990867SGreg Clayton std::shared_ptr<ThreadTreeDelegate> m_thread_delegate_sp; 3473ec990867SGreg Clayton Debugger &m_debugger; 3474ec990867SGreg Clayton uint32_t m_stop_id; 3475554f68d3SGreg Clayton FormatEntity::Entry m_format; 3476554f68d3SGreg Clayton 3477ec990867SGreg Clayton }; 3478ec990867SGreg Clayton 347944d93782SGreg Clayton class ValueObjectListDelegate : public WindowDelegate 348044d93782SGreg Clayton { 348144d93782SGreg Clayton public: 348244d93782SGreg Clayton ValueObjectListDelegate () : 348344d93782SGreg Clayton m_valobj_list (), 348444d93782SGreg Clayton m_rows (), 348544d93782SGreg Clayton m_selected_row (NULL), 348644d93782SGreg Clayton m_selected_row_idx (0), 348744d93782SGreg Clayton m_first_visible_row (0), 348844d93782SGreg Clayton m_num_rows (0), 348944d93782SGreg Clayton m_max_x (0), 349044d93782SGreg Clayton m_max_y (0) 349144d93782SGreg Clayton { 349244d93782SGreg Clayton } 349344d93782SGreg Clayton 349444d93782SGreg Clayton ValueObjectListDelegate (ValueObjectList &valobj_list) : 349544d93782SGreg Clayton m_valobj_list (valobj_list), 349644d93782SGreg Clayton m_rows (), 349744d93782SGreg Clayton m_selected_row (NULL), 349844d93782SGreg Clayton m_selected_row_idx (0), 349944d93782SGreg Clayton m_first_visible_row (0), 350044d93782SGreg Clayton m_num_rows (0), 350144d93782SGreg Clayton m_max_x (0), 350244d93782SGreg Clayton m_max_y (0) 350344d93782SGreg Clayton { 350444d93782SGreg Clayton SetValues (valobj_list); 350544d93782SGreg Clayton } 350644d93782SGreg Clayton 3507bd5ae6b4SGreg Clayton ~ValueObjectListDelegate() override 350844d93782SGreg Clayton { 350944d93782SGreg Clayton } 351044d93782SGreg Clayton 351144d93782SGreg Clayton void 351244d93782SGreg Clayton SetValues (ValueObjectList &valobj_list) 351344d93782SGreg Clayton { 351444d93782SGreg Clayton m_selected_row = NULL; 351544d93782SGreg Clayton m_selected_row_idx = 0; 351644d93782SGreg Clayton m_first_visible_row = 0; 351744d93782SGreg Clayton m_num_rows = 0; 351844d93782SGreg Clayton m_rows.clear(); 351944d93782SGreg Clayton m_valobj_list = valobj_list; 352044d93782SGreg Clayton const size_t num_values = m_valobj_list.GetSize(); 352144d93782SGreg Clayton for (size_t i=0; i<num_values; ++i) 352244d93782SGreg Clayton m_rows.push_back(Row(m_valobj_list.GetValueObjectAtIndex(i), NULL)); 352344d93782SGreg Clayton } 352444d93782SGreg Clayton 3525bd5ae6b4SGreg Clayton bool 3526bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 352744d93782SGreg Clayton { 352844d93782SGreg Clayton m_num_rows = 0; 352944d93782SGreg Clayton m_min_x = 2; 353044d93782SGreg Clayton m_min_y = 1; 353144d93782SGreg Clayton m_max_x = window.GetWidth() - 1; 353244d93782SGreg Clayton m_max_y = window.GetHeight() - 1; 353344d93782SGreg Clayton 353444d93782SGreg Clayton window.Erase(); 353544d93782SGreg Clayton window.DrawTitleBox (window.GetName()); 353644d93782SGreg Clayton 353744d93782SGreg Clayton const int num_visible_rows = NumVisibleRows(); 353844d93782SGreg Clayton const int num_rows = CalculateTotalNumberRows (m_rows); 353944d93782SGreg Clayton 354044d93782SGreg Clayton // If we unexpanded while having something selected our 354144d93782SGreg Clayton // total number of rows is less than the num visible rows, 354244d93782SGreg Clayton // then make sure we show all the rows by setting the first 354344d93782SGreg Clayton // visible row accordingly. 354444d93782SGreg Clayton if (m_first_visible_row > 0 && num_rows < num_visible_rows) 354544d93782SGreg Clayton m_first_visible_row = 0; 354644d93782SGreg Clayton 354744d93782SGreg Clayton // Make sure the selected row is always visible 354844d93782SGreg Clayton if (m_selected_row_idx < m_first_visible_row) 354944d93782SGreg Clayton m_first_visible_row = m_selected_row_idx; 355044d93782SGreg Clayton else if (m_first_visible_row + num_visible_rows <= m_selected_row_idx) 355144d93782SGreg Clayton m_first_visible_row = m_selected_row_idx - num_visible_rows + 1; 355244d93782SGreg Clayton 355344d93782SGreg Clayton DisplayRows (window, m_rows, g_options); 355444d93782SGreg Clayton 355544d93782SGreg Clayton window.DeferredRefresh(); 355644d93782SGreg Clayton 355744d93782SGreg Clayton // Get the selected row 355844d93782SGreg Clayton m_selected_row = GetRowForRowIndex (m_selected_row_idx); 355944d93782SGreg Clayton // Keep the cursor on the selected row so the highlight and the cursor 356044d93782SGreg Clayton // are always on the same line 356144d93782SGreg Clayton if (m_selected_row) 356244d93782SGreg Clayton window.MoveCursor (m_selected_row->x, 356344d93782SGreg Clayton m_selected_row->y); 356444d93782SGreg Clayton 356544d93782SGreg Clayton return true; // Drawing handled 356644d93782SGreg Clayton } 356744d93782SGreg Clayton 3568bd5ae6b4SGreg Clayton KeyHelp * 3569bd5ae6b4SGreg Clayton WindowDelegateGetKeyHelp () override 357044d93782SGreg Clayton { 357144d93782SGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 357244d93782SGreg Clayton { KEY_UP, "Select previous item" }, 357344d93782SGreg Clayton { KEY_DOWN, "Select next item" }, 357444d93782SGreg Clayton { KEY_RIGHT, "Expand selected item" }, 357544d93782SGreg Clayton { KEY_LEFT, "Unexpand selected item or select parent if not expanded" }, 357644d93782SGreg Clayton { KEY_PPAGE, "Page up" }, 357744d93782SGreg Clayton { KEY_NPAGE, "Page down" }, 357844d93782SGreg Clayton { 'A', "Format as annotated address" }, 357944d93782SGreg Clayton { 'b', "Format as binary" }, 358044d93782SGreg Clayton { 'B', "Format as hex bytes with ASCII" }, 358144d93782SGreg Clayton { 'c', "Format as character" }, 358244d93782SGreg Clayton { 'd', "Format as a signed integer" }, 358344d93782SGreg Clayton { 'D', "Format selected value using the default format for the type" }, 358444d93782SGreg Clayton { 'f', "Format as float" }, 358544d93782SGreg Clayton { 'h', "Show help dialog" }, 358644d93782SGreg Clayton { 'i', "Format as instructions" }, 358744d93782SGreg Clayton { 'o', "Format as octal" }, 358844d93782SGreg Clayton { 'p', "Format as pointer" }, 358944d93782SGreg Clayton { 's', "Format as C string" }, 359044d93782SGreg Clayton { 't', "Toggle showing/hiding type names" }, 359144d93782SGreg Clayton { 'u', "Format as an unsigned integer" }, 359244d93782SGreg Clayton { 'x', "Format as hex" }, 359344d93782SGreg Clayton { 'X', "Format as uppercase hex" }, 359444d93782SGreg Clayton { ' ', "Toggle item expansion" }, 359544d93782SGreg Clayton { ',', "Page up" }, 359644d93782SGreg Clayton { '.', "Page down" }, 359744d93782SGreg Clayton { '\0', NULL } 359844d93782SGreg Clayton }; 359944d93782SGreg Clayton return g_source_view_key_help; 360044d93782SGreg Clayton } 360144d93782SGreg Clayton 360244d93782SGreg Clayton 3603bd5ae6b4SGreg Clayton HandleCharResult 3604bd5ae6b4SGreg Clayton WindowDelegateHandleChar (Window &window, int c) override 360544d93782SGreg Clayton { 360644d93782SGreg Clayton switch(c) 360744d93782SGreg Clayton { 360844d93782SGreg Clayton case 'x': 360944d93782SGreg Clayton case 'X': 361044d93782SGreg Clayton case 'o': 361144d93782SGreg Clayton case 's': 361244d93782SGreg Clayton case 'u': 361344d93782SGreg Clayton case 'd': 361444d93782SGreg Clayton case 'D': 361544d93782SGreg Clayton case 'i': 361644d93782SGreg Clayton case 'A': 361744d93782SGreg Clayton case 'p': 361844d93782SGreg Clayton case 'c': 361944d93782SGreg Clayton case 'b': 362044d93782SGreg Clayton case 'B': 362144d93782SGreg Clayton case 'f': 362244d93782SGreg Clayton // Change the format for the currently selected item 362344d93782SGreg Clayton if (m_selected_row) 362444d93782SGreg Clayton m_selected_row->valobj->SetFormat (FormatForChar (c)); 362544d93782SGreg Clayton return eKeyHandled; 362644d93782SGreg Clayton 362744d93782SGreg Clayton case 't': 362844d93782SGreg Clayton // Toggle showing type names 362944d93782SGreg Clayton g_options.show_types = !g_options.show_types; 363044d93782SGreg Clayton return eKeyHandled; 363144d93782SGreg Clayton 363244d93782SGreg Clayton case ',': 363344d93782SGreg Clayton case KEY_PPAGE: 363444d93782SGreg Clayton // Page up key 363544d93782SGreg Clayton if (m_first_visible_row > 0) 363644d93782SGreg Clayton { 36373985c8c6SSaleem Abdulrasool if (static_cast<int>(m_first_visible_row) > m_max_y) 363844d93782SGreg Clayton m_first_visible_row -= m_max_y; 363944d93782SGreg Clayton else 364044d93782SGreg Clayton m_first_visible_row = 0; 364144d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 364244d93782SGreg Clayton } 364344d93782SGreg Clayton return eKeyHandled; 364444d93782SGreg Clayton 364544d93782SGreg Clayton case '.': 364644d93782SGreg Clayton case KEY_NPAGE: 364744d93782SGreg Clayton // Page down key 36483985c8c6SSaleem Abdulrasool if (m_num_rows > static_cast<size_t>(m_max_y)) 364944d93782SGreg Clayton { 365044d93782SGreg Clayton if (m_first_visible_row + m_max_y < m_num_rows) 365144d93782SGreg Clayton { 365244d93782SGreg Clayton m_first_visible_row += m_max_y; 365344d93782SGreg Clayton m_selected_row_idx = m_first_visible_row; 365444d93782SGreg Clayton } 365544d93782SGreg Clayton } 365644d93782SGreg Clayton return eKeyHandled; 365744d93782SGreg Clayton 365844d93782SGreg Clayton case KEY_UP: 365944d93782SGreg Clayton if (m_selected_row_idx > 0) 366044d93782SGreg Clayton --m_selected_row_idx; 366144d93782SGreg Clayton return eKeyHandled; 366244d93782SGreg Clayton case KEY_DOWN: 366344d93782SGreg Clayton if (m_selected_row_idx + 1 < m_num_rows) 366444d93782SGreg Clayton ++m_selected_row_idx; 366544d93782SGreg Clayton return eKeyHandled; 366644d93782SGreg Clayton 366744d93782SGreg Clayton case KEY_RIGHT: 366844d93782SGreg Clayton if (m_selected_row) 366944d93782SGreg Clayton { 367044d93782SGreg Clayton if (!m_selected_row->expanded) 367144d93782SGreg Clayton m_selected_row->Expand(); 367244d93782SGreg Clayton } 367344d93782SGreg Clayton return eKeyHandled; 367444d93782SGreg Clayton 367544d93782SGreg Clayton case KEY_LEFT: 367644d93782SGreg Clayton if (m_selected_row) 367744d93782SGreg Clayton { 367844d93782SGreg Clayton if (m_selected_row->expanded) 367944d93782SGreg Clayton m_selected_row->Unexpand(); 368044d93782SGreg Clayton else if (m_selected_row->parent) 368144d93782SGreg Clayton m_selected_row_idx = m_selected_row->parent->row_idx; 368244d93782SGreg Clayton } 368344d93782SGreg Clayton return eKeyHandled; 368444d93782SGreg Clayton 368544d93782SGreg Clayton case ' ': 368644d93782SGreg Clayton // Toggle expansion state when SPACE is pressed 368744d93782SGreg Clayton if (m_selected_row) 368844d93782SGreg Clayton { 368944d93782SGreg Clayton if (m_selected_row->expanded) 369044d93782SGreg Clayton m_selected_row->Unexpand(); 369144d93782SGreg Clayton else 369244d93782SGreg Clayton m_selected_row->Expand(); 369344d93782SGreg Clayton } 369444d93782SGreg Clayton return eKeyHandled; 369544d93782SGreg Clayton 369644d93782SGreg Clayton case 'h': 369744d93782SGreg Clayton window.CreateHelpSubwindow (); 369844d93782SGreg Clayton return eKeyHandled; 369944d93782SGreg Clayton 370044d93782SGreg Clayton default: 370144d93782SGreg Clayton break; 370244d93782SGreg Clayton } 370344d93782SGreg Clayton return eKeyNotHandled; 370444d93782SGreg Clayton } 370544d93782SGreg Clayton 370644d93782SGreg Clayton protected: 370744d93782SGreg Clayton ValueObjectList m_valobj_list; 370844d93782SGreg Clayton std::vector<Row> m_rows; 370944d93782SGreg Clayton Row *m_selected_row; 371044d93782SGreg Clayton uint32_t m_selected_row_idx; 371144d93782SGreg Clayton uint32_t m_first_visible_row; 371244d93782SGreg Clayton uint32_t m_num_rows; 371344d93782SGreg Clayton int m_min_x; 371444d93782SGreg Clayton int m_min_y; 371544d93782SGreg Clayton int m_max_x; 371644d93782SGreg Clayton int m_max_y; 371744d93782SGreg Clayton 371844d93782SGreg Clayton static Format 371944d93782SGreg Clayton FormatForChar (int c) 372044d93782SGreg Clayton { 372144d93782SGreg Clayton switch (c) 372244d93782SGreg Clayton { 372344d93782SGreg Clayton case 'x': return eFormatHex; 372444d93782SGreg Clayton case 'X': return eFormatHexUppercase; 372544d93782SGreg Clayton case 'o': return eFormatOctal; 372644d93782SGreg Clayton case 's': return eFormatCString; 372744d93782SGreg Clayton case 'u': return eFormatUnsigned; 372844d93782SGreg Clayton case 'd': return eFormatDecimal; 372944d93782SGreg Clayton case 'D': return eFormatDefault; 373044d93782SGreg Clayton case 'i': return eFormatInstruction; 373144d93782SGreg Clayton case 'A': return eFormatAddressInfo; 373244d93782SGreg Clayton case 'p': return eFormatPointer; 373344d93782SGreg Clayton case 'c': return eFormatChar; 373444d93782SGreg Clayton case 'b': return eFormatBinary; 373544d93782SGreg Clayton case 'B': return eFormatBytesWithASCII; 373644d93782SGreg Clayton case 'f': return eFormatFloat; 373744d93782SGreg Clayton } 373844d93782SGreg Clayton return eFormatDefault; 373944d93782SGreg Clayton } 374044d93782SGreg Clayton 374144d93782SGreg Clayton bool 374244d93782SGreg Clayton DisplayRowObject (Window &window, 374344d93782SGreg Clayton Row &row, 374444d93782SGreg Clayton DisplayOptions &options, 374544d93782SGreg Clayton bool highlight, 374644d93782SGreg Clayton bool last_child) 374744d93782SGreg Clayton { 374844d93782SGreg Clayton ValueObject *valobj = row.valobj.get(); 374944d93782SGreg Clayton 375044d93782SGreg Clayton if (valobj == NULL) 375144d93782SGreg Clayton return false; 375244d93782SGreg Clayton 375344d93782SGreg Clayton const char *type_name = options.show_types ? valobj->GetTypeName().GetCString() : NULL; 375444d93782SGreg Clayton const char *name = valobj->GetName().GetCString(); 375544d93782SGreg Clayton const char *value = valobj->GetValueAsCString (); 375644d93782SGreg Clayton const char *summary = valobj->GetSummaryAsCString (); 375744d93782SGreg Clayton 375844d93782SGreg Clayton window.MoveCursor (row.x, row.y); 375944d93782SGreg Clayton 376044d93782SGreg Clayton row.DrawTree (window); 376144d93782SGreg Clayton 376244d93782SGreg Clayton if (highlight) 376344d93782SGreg Clayton window.AttributeOn(A_REVERSE); 376444d93782SGreg Clayton 376544d93782SGreg Clayton if (type_name && type_name[0]) 376644d93782SGreg Clayton window.Printf ("(%s) ", type_name); 376744d93782SGreg Clayton 376844d93782SGreg Clayton if (name && name[0]) 376944d93782SGreg Clayton window.PutCString(name); 377044d93782SGreg Clayton 377144d93782SGreg Clayton attr_t changd_attr = 0; 377244d93782SGreg Clayton if (valobj->GetValueDidChange()) 377344d93782SGreg Clayton changd_attr = COLOR_PAIR(5) | A_BOLD; 377444d93782SGreg Clayton 377544d93782SGreg Clayton if (value && value[0]) 377644d93782SGreg Clayton { 377744d93782SGreg Clayton window.PutCString(" = "); 377844d93782SGreg Clayton if (changd_attr) 377944d93782SGreg Clayton window.AttributeOn(changd_attr); 378044d93782SGreg Clayton window.PutCString (value); 378144d93782SGreg Clayton if (changd_attr) 378244d93782SGreg Clayton window.AttributeOff(changd_attr); 378344d93782SGreg Clayton } 378444d93782SGreg Clayton 378544d93782SGreg Clayton if (summary && summary[0]) 378644d93782SGreg Clayton { 378744d93782SGreg Clayton window.PutChar(' '); 378844d93782SGreg Clayton if (changd_attr) 378944d93782SGreg Clayton window.AttributeOn(changd_attr); 379044d93782SGreg Clayton window.PutCString(summary); 379144d93782SGreg Clayton if (changd_attr) 379244d93782SGreg Clayton window.AttributeOff(changd_attr); 379344d93782SGreg Clayton } 379444d93782SGreg Clayton 379544d93782SGreg Clayton if (highlight) 379644d93782SGreg Clayton window.AttributeOff (A_REVERSE); 379744d93782SGreg Clayton 379844d93782SGreg Clayton return true; 379944d93782SGreg Clayton } 380044d93782SGreg Clayton void 380144d93782SGreg Clayton DisplayRows (Window &window, 380244d93782SGreg Clayton std::vector<Row> &rows, 380344d93782SGreg Clayton DisplayOptions &options) 380444d93782SGreg Clayton { 380544d93782SGreg Clayton // > 0x25B7 380644d93782SGreg Clayton // \/ 0x25BD 380744d93782SGreg Clayton 380844d93782SGreg Clayton bool window_is_active = window.IsActive(); 380944d93782SGreg Clayton for (auto &row : rows) 381044d93782SGreg Clayton { 381144d93782SGreg Clayton const bool last_child = row.parent && &rows[rows.size()-1] == &row; 381244d93782SGreg Clayton // Save the row index in each Row structure 381344d93782SGreg Clayton row.row_idx = m_num_rows; 381444d93782SGreg Clayton if ((m_num_rows >= m_first_visible_row) && 38153985c8c6SSaleem Abdulrasool ((m_num_rows - m_first_visible_row) < static_cast<size_t>(NumVisibleRows()))) 381644d93782SGreg Clayton { 381744d93782SGreg Clayton row.x = m_min_x; 381844d93782SGreg Clayton row.y = m_num_rows - m_first_visible_row + 1; 381944d93782SGreg Clayton if (DisplayRowObject (window, 382044d93782SGreg Clayton row, 382144d93782SGreg Clayton options, 382244d93782SGreg Clayton window_is_active && m_num_rows == m_selected_row_idx, 382344d93782SGreg Clayton last_child)) 382444d93782SGreg Clayton { 382544d93782SGreg Clayton ++m_num_rows; 382644d93782SGreg Clayton } 382744d93782SGreg Clayton else 382844d93782SGreg Clayton { 382944d93782SGreg Clayton row.x = 0; 383044d93782SGreg Clayton row.y = 0; 383144d93782SGreg Clayton } 383244d93782SGreg Clayton } 383344d93782SGreg Clayton else 383444d93782SGreg Clayton { 383544d93782SGreg Clayton row.x = 0; 383644d93782SGreg Clayton row.y = 0; 383744d93782SGreg Clayton ++m_num_rows; 383844d93782SGreg Clayton } 383944d93782SGreg Clayton 384044d93782SGreg Clayton if (row.expanded && !row.children.empty()) 384144d93782SGreg Clayton { 384244d93782SGreg Clayton DisplayRows (window, 384344d93782SGreg Clayton row.children, 384444d93782SGreg Clayton options); 384544d93782SGreg Clayton } 384644d93782SGreg Clayton } 384744d93782SGreg Clayton } 384844d93782SGreg Clayton 384944d93782SGreg Clayton int 385044d93782SGreg Clayton CalculateTotalNumberRows (const std::vector<Row> &rows) 385144d93782SGreg Clayton { 385244d93782SGreg Clayton int row_count = 0; 385344d93782SGreg Clayton for (const auto &row : rows) 385444d93782SGreg Clayton { 385544d93782SGreg Clayton ++row_count; 385644d93782SGreg Clayton if (row.expanded) 385744d93782SGreg Clayton row_count += CalculateTotalNumberRows(row.children); 385844d93782SGreg Clayton } 385944d93782SGreg Clayton return row_count; 386044d93782SGreg Clayton } 386144d93782SGreg Clayton static Row * 386244d93782SGreg Clayton GetRowForRowIndexImpl (std::vector<Row> &rows, size_t &row_index) 386344d93782SGreg Clayton { 386444d93782SGreg Clayton for (auto &row : rows) 386544d93782SGreg Clayton { 386644d93782SGreg Clayton if (row_index == 0) 386744d93782SGreg Clayton return &row; 386844d93782SGreg Clayton else 386944d93782SGreg Clayton { 387044d93782SGreg Clayton --row_index; 387144d93782SGreg Clayton if (row.expanded && !row.children.empty()) 387244d93782SGreg Clayton { 387344d93782SGreg Clayton Row *result = GetRowForRowIndexImpl (row.children, row_index); 387444d93782SGreg Clayton if (result) 387544d93782SGreg Clayton return result; 387644d93782SGreg Clayton } 387744d93782SGreg Clayton } 387844d93782SGreg Clayton } 387944d93782SGreg Clayton return NULL; 388044d93782SGreg Clayton } 388144d93782SGreg Clayton 388244d93782SGreg Clayton Row * 388344d93782SGreg Clayton GetRowForRowIndex (size_t row_index) 388444d93782SGreg Clayton { 388544d93782SGreg Clayton return GetRowForRowIndexImpl (m_rows, row_index); 388644d93782SGreg Clayton } 388744d93782SGreg Clayton 388844d93782SGreg Clayton int 388944d93782SGreg Clayton NumVisibleRows () const 389044d93782SGreg Clayton { 389144d93782SGreg Clayton return m_max_y - m_min_y; 389244d93782SGreg Clayton } 389344d93782SGreg Clayton 389444d93782SGreg Clayton static DisplayOptions g_options; 389544d93782SGreg Clayton }; 389644d93782SGreg Clayton 389744d93782SGreg Clayton class FrameVariablesWindowDelegate : public ValueObjectListDelegate 389844d93782SGreg Clayton { 389944d93782SGreg Clayton public: 390044d93782SGreg Clayton FrameVariablesWindowDelegate (Debugger &debugger) : 390144d93782SGreg Clayton ValueObjectListDelegate (), 390244d93782SGreg Clayton m_debugger (debugger), 390344d93782SGreg Clayton m_frame_block (NULL) 390444d93782SGreg Clayton { 390544d93782SGreg Clayton } 390644d93782SGreg Clayton 3907bd5ae6b4SGreg Clayton ~FrameVariablesWindowDelegate() override 390844d93782SGreg Clayton { 390944d93782SGreg Clayton } 391044d93782SGreg Clayton 3911bd5ae6b4SGreg Clayton const char * 3912bd5ae6b4SGreg Clayton WindowDelegateGetHelpText () override 391344d93782SGreg Clayton { 391444d93782SGreg Clayton return "Frame variable window keyboard shortcuts:"; 391544d93782SGreg Clayton } 391644d93782SGreg Clayton 3917bd5ae6b4SGreg Clayton bool 3918bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 391944d93782SGreg Clayton { 392044d93782SGreg Clayton ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext()); 392144d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 392244d93782SGreg Clayton Block *frame_block = NULL; 392344d93782SGreg Clayton StackFrame *frame = NULL; 392444d93782SGreg Clayton 392544d93782SGreg Clayton if (process) 392644d93782SGreg Clayton { 392744d93782SGreg Clayton StateType state = process->GetState(); 392844d93782SGreg Clayton if (StateIsStoppedState(state, true)) 392944d93782SGreg Clayton { 393044d93782SGreg Clayton frame = exe_ctx.GetFramePtr(); 393144d93782SGreg Clayton if (frame) 393244d93782SGreg Clayton frame_block = frame->GetFrameBlock (); 393344d93782SGreg Clayton } 393444d93782SGreg Clayton else if (StateIsRunningState(state)) 393544d93782SGreg Clayton { 393644d93782SGreg Clayton return true; // Don't do any updating when we are running 393744d93782SGreg Clayton } 393844d93782SGreg Clayton } 393944d93782SGreg Clayton 3940eb72dc7dSGreg Clayton 394144d93782SGreg Clayton ValueObjectList local_values; 394244d93782SGreg Clayton if (frame_block) 394344d93782SGreg Clayton { 394444d93782SGreg Clayton // Only update the variables if they have changed 394544d93782SGreg Clayton if (m_frame_block != frame_block) 394644d93782SGreg Clayton { 394744d93782SGreg Clayton m_frame_block = frame_block; 394844d93782SGreg Clayton 394944d93782SGreg Clayton VariableList *locals = frame->GetVariableList(true); 395044d93782SGreg Clayton if (locals) 395144d93782SGreg Clayton { 395244d93782SGreg Clayton const DynamicValueType use_dynamic = eDynamicDontRunTarget; 395344d93782SGreg Clayton const size_t num_locals = locals->GetSize(); 395444d93782SGreg Clayton for (size_t i=0; i<num_locals; ++i) 3955eb72dc7dSGreg Clayton { 3956eb72dc7dSGreg Clayton ValueObjectSP value_sp = frame->GetValueObjectForFrameVariable (locals->GetVariableAtIndex(i), use_dynamic); 3957eb72dc7dSGreg Clayton if (value_sp) 3958eb72dc7dSGreg Clayton { 3959eb72dc7dSGreg Clayton ValueObjectSP synthetic_value_sp = value_sp->GetSyntheticValue(); 3960eb72dc7dSGreg Clayton if (synthetic_value_sp) 3961eb72dc7dSGreg Clayton local_values.Append(synthetic_value_sp); 3962eb72dc7dSGreg Clayton else 3963eb72dc7dSGreg Clayton local_values.Append(value_sp); 3964eb72dc7dSGreg Clayton 3965eb72dc7dSGreg Clayton } 3966eb72dc7dSGreg Clayton } 396744d93782SGreg Clayton // Update the values 396844d93782SGreg Clayton SetValues(local_values); 396944d93782SGreg Clayton } 397044d93782SGreg Clayton } 397144d93782SGreg Clayton } 397244d93782SGreg Clayton else 397344d93782SGreg Clayton { 397444d93782SGreg Clayton m_frame_block = NULL; 397544d93782SGreg Clayton // Update the values with an empty list if there is no frame 397644d93782SGreg Clayton SetValues(local_values); 397744d93782SGreg Clayton } 397844d93782SGreg Clayton 397944d93782SGreg Clayton return ValueObjectListDelegate::WindowDelegateDraw (window, force); 398044d93782SGreg Clayton 398144d93782SGreg Clayton } 398244d93782SGreg Clayton 398344d93782SGreg Clayton protected: 398444d93782SGreg Clayton Debugger &m_debugger; 398544d93782SGreg Clayton Block *m_frame_block; 398644d93782SGreg Clayton }; 398744d93782SGreg Clayton 398844d93782SGreg Clayton 398944d93782SGreg Clayton class RegistersWindowDelegate : public ValueObjectListDelegate 399044d93782SGreg Clayton { 399144d93782SGreg Clayton public: 399244d93782SGreg Clayton RegistersWindowDelegate (Debugger &debugger) : 399344d93782SGreg Clayton ValueObjectListDelegate (), 399444d93782SGreg Clayton m_debugger (debugger) 399544d93782SGreg Clayton { 399644d93782SGreg Clayton } 399744d93782SGreg Clayton 399844d93782SGreg Clayton ~RegistersWindowDelegate() 399944d93782SGreg Clayton { 400044d93782SGreg Clayton } 400144d93782SGreg Clayton 4002bd5ae6b4SGreg Clayton const char * 4003bd5ae6b4SGreg Clayton WindowDelegateGetHelpText () override 400444d93782SGreg Clayton { 400544d93782SGreg Clayton return "Register window keyboard shortcuts:"; 400644d93782SGreg Clayton } 400744d93782SGreg Clayton 4008bd5ae6b4SGreg Clayton bool 4009bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 401044d93782SGreg Clayton { 401144d93782SGreg Clayton ExecutionContext exe_ctx (m_debugger.GetCommandInterpreter().GetExecutionContext()); 401244d93782SGreg Clayton StackFrame *frame = exe_ctx.GetFramePtr(); 401344d93782SGreg Clayton 401444d93782SGreg Clayton ValueObjectList value_list; 401544d93782SGreg Clayton if (frame) 401644d93782SGreg Clayton { 401744d93782SGreg Clayton if (frame->GetStackID() != m_stack_id) 401844d93782SGreg Clayton { 401944d93782SGreg Clayton m_stack_id = frame->GetStackID(); 402044d93782SGreg Clayton RegisterContextSP reg_ctx (frame->GetRegisterContext()); 402144d93782SGreg Clayton if (reg_ctx) 402244d93782SGreg Clayton { 402344d93782SGreg Clayton const uint32_t num_sets = reg_ctx->GetRegisterSetCount(); 402444d93782SGreg Clayton for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) 402544d93782SGreg Clayton { 402644d93782SGreg Clayton value_list.Append(ValueObjectRegisterSet::Create (frame, reg_ctx, set_idx)); 402744d93782SGreg Clayton } 402844d93782SGreg Clayton } 402944d93782SGreg Clayton SetValues(value_list); 403044d93782SGreg Clayton } 403144d93782SGreg Clayton } 403244d93782SGreg Clayton else 403344d93782SGreg Clayton { 403444d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 403544d93782SGreg Clayton if (process && process->IsAlive()) 403644d93782SGreg Clayton return true; // Don't do any updating if we are running 403744d93782SGreg Clayton else 403844d93782SGreg Clayton { 403944d93782SGreg Clayton // Update the values with an empty list if there 404044d93782SGreg Clayton // is no process or the process isn't alive anymore 404144d93782SGreg Clayton SetValues(value_list); 404244d93782SGreg Clayton } 404344d93782SGreg Clayton } 404444d93782SGreg Clayton return ValueObjectListDelegate::WindowDelegateDraw (window, force); 404544d93782SGreg Clayton } 404644d93782SGreg Clayton 404744d93782SGreg Clayton protected: 404844d93782SGreg Clayton Debugger &m_debugger; 404944d93782SGreg Clayton StackID m_stack_id; 405044d93782SGreg Clayton }; 405144d93782SGreg Clayton 405244d93782SGreg Clayton static const char * 405344d93782SGreg Clayton CursesKeyToCString (int ch) 405444d93782SGreg Clayton { 405544d93782SGreg Clayton static char g_desc[32]; 405644d93782SGreg Clayton if (ch >= KEY_F0 && ch < KEY_F0 + 64) 405744d93782SGreg Clayton { 405844d93782SGreg Clayton snprintf(g_desc, sizeof(g_desc), "F%u", ch - KEY_F0); 405944d93782SGreg Clayton return g_desc; 406044d93782SGreg Clayton } 406144d93782SGreg Clayton switch (ch) 406244d93782SGreg Clayton { 406344d93782SGreg Clayton case KEY_DOWN: return "down"; 406444d93782SGreg Clayton case KEY_UP: return "up"; 406544d93782SGreg Clayton case KEY_LEFT: return "left"; 406644d93782SGreg Clayton case KEY_RIGHT: return "right"; 406744d93782SGreg Clayton case KEY_HOME: return "home"; 406844d93782SGreg Clayton case KEY_BACKSPACE: return "backspace"; 406944d93782SGreg Clayton case KEY_DL: return "delete-line"; 407044d93782SGreg Clayton case KEY_IL: return "insert-line"; 407144d93782SGreg Clayton case KEY_DC: return "delete-char"; 407244d93782SGreg Clayton case KEY_IC: return "insert-char"; 407344d93782SGreg Clayton case KEY_CLEAR: return "clear"; 407444d93782SGreg Clayton case KEY_EOS: return "clear-to-eos"; 407544d93782SGreg Clayton case KEY_EOL: return "clear-to-eol"; 407644d93782SGreg Clayton case KEY_SF: return "scroll-forward"; 407744d93782SGreg Clayton case KEY_SR: return "scroll-backward"; 407844d93782SGreg Clayton case KEY_NPAGE: return "page-down"; 407944d93782SGreg Clayton case KEY_PPAGE: return "page-up"; 408044d93782SGreg Clayton case KEY_STAB: return "set-tab"; 408144d93782SGreg Clayton case KEY_CTAB: return "clear-tab"; 408244d93782SGreg Clayton case KEY_CATAB: return "clear-all-tabs"; 408344d93782SGreg Clayton case KEY_ENTER: return "enter"; 408444d93782SGreg Clayton case KEY_PRINT: return "print"; 408544d93782SGreg Clayton case KEY_LL: return "lower-left key"; 408644d93782SGreg Clayton case KEY_A1: return "upper left of keypad"; 408744d93782SGreg Clayton case KEY_A3: return "upper right of keypad"; 408844d93782SGreg Clayton case KEY_B2: return "center of keypad"; 408944d93782SGreg Clayton case KEY_C1: return "lower left of keypad"; 409044d93782SGreg Clayton case KEY_C3: return "lower right of keypad"; 409144d93782SGreg Clayton case KEY_BTAB: return "back-tab key"; 409244d93782SGreg Clayton case KEY_BEG: return "begin key"; 409344d93782SGreg Clayton case KEY_CANCEL: return "cancel key"; 409444d93782SGreg Clayton case KEY_CLOSE: return "close key"; 409544d93782SGreg Clayton case KEY_COMMAND: return "command key"; 409644d93782SGreg Clayton case KEY_COPY: return "copy key"; 409744d93782SGreg Clayton case KEY_CREATE: return "create key"; 409844d93782SGreg Clayton case KEY_END: return "end key"; 409944d93782SGreg Clayton case KEY_EXIT: return "exit key"; 410044d93782SGreg Clayton case KEY_FIND: return "find key"; 410144d93782SGreg Clayton case KEY_HELP: return "help key"; 410244d93782SGreg Clayton case KEY_MARK: return "mark key"; 410344d93782SGreg Clayton case KEY_MESSAGE: return "message key"; 410444d93782SGreg Clayton case KEY_MOVE: return "move key"; 410544d93782SGreg Clayton case KEY_NEXT: return "next key"; 410644d93782SGreg Clayton case KEY_OPEN: return "open key"; 410744d93782SGreg Clayton case KEY_OPTIONS: return "options key"; 410844d93782SGreg Clayton case KEY_PREVIOUS: return "previous key"; 410944d93782SGreg Clayton case KEY_REDO: return "redo key"; 411044d93782SGreg Clayton case KEY_REFERENCE: return "reference key"; 411144d93782SGreg Clayton case KEY_REFRESH: return "refresh key"; 411244d93782SGreg Clayton case KEY_REPLACE: return "replace key"; 411344d93782SGreg Clayton case KEY_RESTART: return "restart key"; 411444d93782SGreg Clayton case KEY_RESUME: return "resume key"; 411544d93782SGreg Clayton case KEY_SAVE: return "save key"; 411644d93782SGreg Clayton case KEY_SBEG: return "shifted begin key"; 411744d93782SGreg Clayton case KEY_SCANCEL: return "shifted cancel key"; 411844d93782SGreg Clayton case KEY_SCOMMAND: return "shifted command key"; 411944d93782SGreg Clayton case KEY_SCOPY: return "shifted copy key"; 412044d93782SGreg Clayton case KEY_SCREATE: return "shifted create key"; 412144d93782SGreg Clayton case KEY_SDC: return "shifted delete-character key"; 412244d93782SGreg Clayton case KEY_SDL: return "shifted delete-line key"; 412344d93782SGreg Clayton case KEY_SELECT: return "select key"; 412444d93782SGreg Clayton case KEY_SEND: return "shifted end key"; 412544d93782SGreg Clayton case KEY_SEOL: return "shifted clear-to-end-of-line key"; 412644d93782SGreg Clayton case KEY_SEXIT: return "shifted exit key"; 412744d93782SGreg Clayton case KEY_SFIND: return "shifted find key"; 412844d93782SGreg Clayton case KEY_SHELP: return "shifted help key"; 412944d93782SGreg Clayton case KEY_SHOME: return "shifted home key"; 413044d93782SGreg Clayton case KEY_SIC: return "shifted insert-character key"; 413144d93782SGreg Clayton case KEY_SLEFT: return "shifted left-arrow key"; 413244d93782SGreg Clayton case KEY_SMESSAGE: return "shifted message key"; 413344d93782SGreg Clayton case KEY_SMOVE: return "shifted move key"; 413444d93782SGreg Clayton case KEY_SNEXT: return "shifted next key"; 413544d93782SGreg Clayton case KEY_SOPTIONS: return "shifted options key"; 413644d93782SGreg Clayton case KEY_SPREVIOUS: return "shifted previous key"; 413744d93782SGreg Clayton case KEY_SPRINT: return "shifted print key"; 413844d93782SGreg Clayton case KEY_SREDO: return "shifted redo key"; 413944d93782SGreg Clayton case KEY_SREPLACE: return "shifted replace key"; 414044d93782SGreg Clayton case KEY_SRIGHT: return "shifted right-arrow key"; 414144d93782SGreg Clayton case KEY_SRSUME: return "shifted resume key"; 414244d93782SGreg Clayton case KEY_SSAVE: return "shifted save key"; 414344d93782SGreg Clayton case KEY_SSUSPEND: return "shifted suspend key"; 414444d93782SGreg Clayton case KEY_SUNDO: return "shifted undo key"; 414544d93782SGreg Clayton case KEY_SUSPEND: return "suspend key"; 414644d93782SGreg Clayton case KEY_UNDO: return "undo key"; 414744d93782SGreg Clayton case KEY_MOUSE: return "Mouse event has occurred"; 414844d93782SGreg Clayton case KEY_RESIZE: return "Terminal resize event"; 414944d93782SGreg Clayton case KEY_EVENT: return "We were interrupted by an event"; 415044d93782SGreg Clayton case KEY_RETURN: return "return"; 415144d93782SGreg Clayton case ' ': return "space"; 41525fdb09bbSGreg Clayton case '\t': return "tab"; 415344d93782SGreg Clayton case KEY_ESCAPE: return "escape"; 415444d93782SGreg Clayton default: 415544d93782SGreg Clayton if (isprint(ch)) 415644d93782SGreg Clayton snprintf(g_desc, sizeof(g_desc), "%c", ch); 415744d93782SGreg Clayton else 415844d93782SGreg Clayton snprintf(g_desc, sizeof(g_desc), "\\x%2.2x", ch); 415944d93782SGreg Clayton return g_desc; 416044d93782SGreg Clayton } 416144d93782SGreg Clayton return NULL; 416244d93782SGreg Clayton } 416344d93782SGreg Clayton 416444d93782SGreg Clayton HelpDialogDelegate::HelpDialogDelegate (const char *text, KeyHelp *key_help_array) : 416544d93782SGreg Clayton m_text (), 416644d93782SGreg Clayton m_first_visible_line (0) 416744d93782SGreg Clayton { 416844d93782SGreg Clayton if (text && text[0]) 416944d93782SGreg Clayton { 417044d93782SGreg Clayton m_text.SplitIntoLines(text); 417144d93782SGreg Clayton m_text.AppendString(""); 417244d93782SGreg Clayton } 417344d93782SGreg Clayton if (key_help_array) 417444d93782SGreg Clayton { 417544d93782SGreg Clayton for (KeyHelp *key = key_help_array; key->ch; ++key) 417644d93782SGreg Clayton { 417744d93782SGreg Clayton StreamString key_description; 417844d93782SGreg Clayton key_description.Printf("%10s - %s", CursesKeyToCString(key->ch), key->description); 417944d93782SGreg Clayton m_text.AppendString(std::move(key_description.GetString())); 418044d93782SGreg Clayton } 418144d93782SGreg Clayton } 418244d93782SGreg Clayton } 418344d93782SGreg Clayton 418444d93782SGreg Clayton HelpDialogDelegate::~HelpDialogDelegate() 418544d93782SGreg Clayton { 418644d93782SGreg Clayton } 418744d93782SGreg Clayton 418844d93782SGreg Clayton bool 418944d93782SGreg Clayton HelpDialogDelegate::WindowDelegateDraw (Window &window, bool force) 419044d93782SGreg Clayton { 419144d93782SGreg Clayton window.Erase(); 419244d93782SGreg Clayton const int window_height = window.GetHeight(); 419344d93782SGreg Clayton int x = 2; 419444d93782SGreg Clayton int y = 1; 419544d93782SGreg Clayton const int min_y = y; 419644d93782SGreg Clayton const int max_y = window_height - 1 - y; 41973985c8c6SSaleem Abdulrasool const size_t num_visible_lines = max_y - min_y + 1; 419844d93782SGreg Clayton const size_t num_lines = m_text.GetSize(); 419944d93782SGreg Clayton const char *bottom_message; 420044d93782SGreg Clayton if (num_lines <= num_visible_lines) 420144d93782SGreg Clayton bottom_message = "Press any key to exit"; 420244d93782SGreg Clayton else 420344d93782SGreg Clayton bottom_message = "Use arrows to scroll, any other key to exit"; 420444d93782SGreg Clayton window.DrawTitleBox(window.GetName(), bottom_message); 420544d93782SGreg Clayton while (y <= max_y) 420644d93782SGreg Clayton { 420744d93782SGreg Clayton window.MoveCursor(x, y); 420844d93782SGreg Clayton window.PutCStringTruncated(m_text.GetStringAtIndex(m_first_visible_line + y - min_y), 1); 420944d93782SGreg Clayton ++y; 421044d93782SGreg Clayton } 421144d93782SGreg Clayton return true; 421244d93782SGreg Clayton } 421344d93782SGreg Clayton 421444d93782SGreg Clayton HandleCharResult 421544d93782SGreg Clayton HelpDialogDelegate::WindowDelegateHandleChar (Window &window, int key) 421644d93782SGreg Clayton { 421744d93782SGreg Clayton bool done = false; 421844d93782SGreg Clayton const size_t num_lines = m_text.GetSize(); 421944d93782SGreg Clayton const size_t num_visible_lines = window.GetHeight() - 2; 422044d93782SGreg Clayton 422144d93782SGreg Clayton if (num_lines <= num_visible_lines) 422244d93782SGreg Clayton { 422344d93782SGreg Clayton done = true; 422444d93782SGreg Clayton // If we have all lines visible and don't need scrolling, then any 422544d93782SGreg Clayton // key press will cause us to exit 422644d93782SGreg Clayton } 422744d93782SGreg Clayton else 422844d93782SGreg Clayton { 422944d93782SGreg Clayton switch (key) 423044d93782SGreg Clayton { 423144d93782SGreg Clayton case KEY_UP: 423244d93782SGreg Clayton if (m_first_visible_line > 0) 423344d93782SGreg Clayton --m_first_visible_line; 423444d93782SGreg Clayton break; 423544d93782SGreg Clayton 423644d93782SGreg Clayton case KEY_DOWN: 423744d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < num_lines) 423844d93782SGreg Clayton ++m_first_visible_line; 423944d93782SGreg Clayton break; 424044d93782SGreg Clayton 424144d93782SGreg Clayton case KEY_PPAGE: 424244d93782SGreg Clayton case ',': 424344d93782SGreg Clayton if (m_first_visible_line > 0) 424444d93782SGreg Clayton { 42453985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) >= num_visible_lines) 424644d93782SGreg Clayton m_first_visible_line -= num_visible_lines; 424744d93782SGreg Clayton else 424844d93782SGreg Clayton m_first_visible_line = 0; 424944d93782SGreg Clayton } 425044d93782SGreg Clayton break; 425144d93782SGreg Clayton case KEY_NPAGE: 425244d93782SGreg Clayton case '.': 425344d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < num_lines) 425444d93782SGreg Clayton { 425544d93782SGreg Clayton m_first_visible_line += num_visible_lines; 42563985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) > num_lines) 425744d93782SGreg Clayton m_first_visible_line = num_lines - num_visible_lines; 425844d93782SGreg Clayton } 425944d93782SGreg Clayton break; 426044d93782SGreg Clayton default: 426144d93782SGreg Clayton done = true; 426244d93782SGreg Clayton break; 426344d93782SGreg Clayton } 426444d93782SGreg Clayton } 426544d93782SGreg Clayton if (done) 426644d93782SGreg Clayton window.GetParent()->RemoveSubWindow(&window); 426744d93782SGreg Clayton return eKeyHandled; 426844d93782SGreg Clayton } 426944d93782SGreg Clayton 427044d93782SGreg Clayton class ApplicationDelegate : 427144d93782SGreg Clayton public WindowDelegate, 427244d93782SGreg Clayton public MenuDelegate 427344d93782SGreg Clayton { 427444d93782SGreg Clayton public: 427544d93782SGreg Clayton enum { 427644d93782SGreg Clayton eMenuID_LLDB = 1, 427744d93782SGreg Clayton eMenuID_LLDBAbout, 427844d93782SGreg Clayton eMenuID_LLDBExit, 427944d93782SGreg Clayton 428044d93782SGreg Clayton eMenuID_Target, 428144d93782SGreg Clayton eMenuID_TargetCreate, 428244d93782SGreg Clayton eMenuID_TargetDelete, 428344d93782SGreg Clayton 428444d93782SGreg Clayton eMenuID_Process, 428544d93782SGreg Clayton eMenuID_ProcessAttach, 428644d93782SGreg Clayton eMenuID_ProcessDetach, 428744d93782SGreg Clayton eMenuID_ProcessLaunch, 428844d93782SGreg Clayton eMenuID_ProcessContinue, 428944d93782SGreg Clayton eMenuID_ProcessHalt, 429044d93782SGreg Clayton eMenuID_ProcessKill, 429144d93782SGreg Clayton 429244d93782SGreg Clayton eMenuID_Thread, 429344d93782SGreg Clayton eMenuID_ThreadStepIn, 429444d93782SGreg Clayton eMenuID_ThreadStepOver, 429544d93782SGreg Clayton eMenuID_ThreadStepOut, 429644d93782SGreg Clayton 429744d93782SGreg Clayton eMenuID_View, 429844d93782SGreg Clayton eMenuID_ViewBacktrace, 429944d93782SGreg Clayton eMenuID_ViewRegisters, 430044d93782SGreg Clayton eMenuID_ViewSource, 430144d93782SGreg Clayton eMenuID_ViewVariables, 430244d93782SGreg Clayton 430344d93782SGreg Clayton eMenuID_Help, 430444d93782SGreg Clayton eMenuID_HelpGUIHelp 430544d93782SGreg Clayton }; 430644d93782SGreg Clayton 430744d93782SGreg Clayton ApplicationDelegate (Application &app, Debugger &debugger) : 430844d93782SGreg Clayton WindowDelegate (), 430944d93782SGreg Clayton MenuDelegate (), 431044d93782SGreg Clayton m_app (app), 431144d93782SGreg Clayton m_debugger (debugger) 431244d93782SGreg Clayton { 431344d93782SGreg Clayton } 431444d93782SGreg Clayton 431544d93782SGreg Clayton ~ApplicationDelegate () 431644d93782SGreg Clayton { 431744d93782SGreg Clayton } 4318bd5ae6b4SGreg Clayton 4319bd5ae6b4SGreg Clayton bool 4320bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 432144d93782SGreg Clayton { 432244d93782SGreg Clayton return false; // Drawing not handled, let standard window drawing happen 432344d93782SGreg Clayton } 432444d93782SGreg Clayton 4325bd5ae6b4SGreg Clayton HandleCharResult 4326bd5ae6b4SGreg Clayton WindowDelegateHandleChar (Window &window, int key) override 432744d93782SGreg Clayton { 43285fdb09bbSGreg Clayton switch (key) 432944d93782SGreg Clayton { 43305fdb09bbSGreg Clayton case '\t': 433144d93782SGreg Clayton window.SelectNextWindowAsActive(); 433244d93782SGreg Clayton return eKeyHandled; 43335fdb09bbSGreg Clayton 43345fdb09bbSGreg Clayton case 'h': 43355fdb09bbSGreg Clayton window.CreateHelpSubwindow(); 43365fdb09bbSGreg Clayton return eKeyHandled; 43375fdb09bbSGreg Clayton 43385fdb09bbSGreg Clayton case KEY_ESCAPE: 43395fdb09bbSGreg Clayton return eQuitApplication; 43405fdb09bbSGreg Clayton 43415fdb09bbSGreg Clayton default: 43425fdb09bbSGreg Clayton break; 434344d93782SGreg Clayton } 434444d93782SGreg Clayton return eKeyNotHandled; 434544d93782SGreg Clayton } 434644d93782SGreg Clayton 43475fdb09bbSGreg Clayton 4348bd5ae6b4SGreg Clayton const char * 4349bd5ae6b4SGreg Clayton WindowDelegateGetHelpText () override 43505fdb09bbSGreg Clayton { 43515fdb09bbSGreg Clayton return "Welcome to the LLDB curses GUI.\n\n" 43525fdb09bbSGreg Clayton "Press the TAB key to change the selected view.\n" 43535fdb09bbSGreg Clayton "Each view has its own keyboard shortcuts, press 'h' to open a dialog to display them.\n\n" 43545fdb09bbSGreg Clayton "Common key bindings for all views:"; 43555fdb09bbSGreg Clayton } 43565fdb09bbSGreg Clayton 4357bd5ae6b4SGreg Clayton KeyHelp * 4358bd5ae6b4SGreg Clayton WindowDelegateGetKeyHelp () override 43595fdb09bbSGreg Clayton { 43605fdb09bbSGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 43615fdb09bbSGreg Clayton { '\t', "Select next view" }, 43625fdb09bbSGreg Clayton { 'h', "Show help dialog with view specific key bindings" }, 43635fdb09bbSGreg Clayton { ',', "Page up" }, 43645fdb09bbSGreg Clayton { '.', "Page down" }, 43655fdb09bbSGreg Clayton { KEY_UP, "Select previous" }, 43665fdb09bbSGreg Clayton { KEY_DOWN, "Select next" }, 43675fdb09bbSGreg Clayton { KEY_LEFT, "Unexpand or select parent" }, 43685fdb09bbSGreg Clayton { KEY_RIGHT, "Expand" }, 43695fdb09bbSGreg Clayton { KEY_PPAGE, "Page up" }, 43705fdb09bbSGreg Clayton { KEY_NPAGE, "Page down" }, 43715fdb09bbSGreg Clayton { '\0', NULL } 43725fdb09bbSGreg Clayton }; 43735fdb09bbSGreg Clayton return g_source_view_key_help; 43745fdb09bbSGreg Clayton } 43755fdb09bbSGreg Clayton 4376bd5ae6b4SGreg Clayton MenuActionResult 4377bd5ae6b4SGreg Clayton MenuDelegateAction (Menu &menu) override 437844d93782SGreg Clayton { 437944d93782SGreg Clayton switch (menu.GetIdentifier()) 438044d93782SGreg Clayton { 438144d93782SGreg Clayton case eMenuID_ThreadStepIn: 438244d93782SGreg Clayton { 438344d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 438444d93782SGreg Clayton if (exe_ctx.HasThreadScope()) 438544d93782SGreg Clayton { 438644d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 438744d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 43884b4b2478SJim Ingham exe_ctx.GetThreadRef().StepIn(true); 438944d93782SGreg Clayton } 439044d93782SGreg Clayton } 439144d93782SGreg Clayton return MenuActionResult::Handled; 439244d93782SGreg Clayton 439344d93782SGreg Clayton case eMenuID_ThreadStepOut: 439444d93782SGreg Clayton { 439544d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 439644d93782SGreg Clayton if (exe_ctx.HasThreadScope()) 439744d93782SGreg Clayton { 439844d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 439944d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 440044d93782SGreg Clayton exe_ctx.GetThreadRef().StepOut(); 440144d93782SGreg Clayton } 440244d93782SGreg Clayton } 440344d93782SGreg Clayton return MenuActionResult::Handled; 440444d93782SGreg Clayton 440544d93782SGreg Clayton case eMenuID_ThreadStepOver: 440644d93782SGreg Clayton { 440744d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 440844d93782SGreg Clayton if (exe_ctx.HasThreadScope()) 440944d93782SGreg Clayton { 441044d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 441144d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 441244d93782SGreg Clayton exe_ctx.GetThreadRef().StepOver(true); 441344d93782SGreg Clayton } 441444d93782SGreg Clayton } 441544d93782SGreg Clayton return MenuActionResult::Handled; 441644d93782SGreg Clayton 441744d93782SGreg Clayton case eMenuID_ProcessContinue: 441844d93782SGreg Clayton { 441944d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 442044d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 442144d93782SGreg Clayton { 442244d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 442344d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 442444d93782SGreg Clayton process->Resume(); 442544d93782SGreg Clayton } 442644d93782SGreg Clayton } 442744d93782SGreg Clayton return MenuActionResult::Handled; 442844d93782SGreg Clayton 442944d93782SGreg Clayton case eMenuID_ProcessKill: 443044d93782SGreg Clayton { 443144d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 443244d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 443344d93782SGreg Clayton { 443444d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 443544d93782SGreg Clayton if (process && process->IsAlive()) 4436ede3193bSJason Molenda process->Destroy(false); 443744d93782SGreg Clayton } 443844d93782SGreg Clayton } 443944d93782SGreg Clayton return MenuActionResult::Handled; 444044d93782SGreg Clayton 444144d93782SGreg Clayton case eMenuID_ProcessHalt: 444244d93782SGreg Clayton { 444344d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 444444d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 444544d93782SGreg Clayton { 444644d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 444744d93782SGreg Clayton if (process && process->IsAlive()) 444844d93782SGreg Clayton process->Halt(); 444944d93782SGreg Clayton } 445044d93782SGreg Clayton } 445144d93782SGreg Clayton return MenuActionResult::Handled; 445244d93782SGreg Clayton 445344d93782SGreg Clayton case eMenuID_ProcessDetach: 445444d93782SGreg Clayton { 445544d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 445644d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 445744d93782SGreg Clayton { 445844d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 445944d93782SGreg Clayton if (process && process->IsAlive()) 446044d93782SGreg Clayton process->Detach(false); 446144d93782SGreg Clayton } 446244d93782SGreg Clayton } 446344d93782SGreg Clayton return MenuActionResult::Handled; 446444d93782SGreg Clayton 446544d93782SGreg Clayton case eMenuID_Process: 446644d93782SGreg Clayton { 446744d93782SGreg Clayton // Populate the menu with all of the threads if the process is stopped when 446844d93782SGreg Clayton // the Process menu gets selected and is about to display its submenu. 446944d93782SGreg Clayton Menus &submenus = menu.GetSubmenus(); 447044d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 447144d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 447244d93782SGreg Clayton if (process && process->IsAlive() && StateIsStoppedState (process->GetState(), true)) 447344d93782SGreg Clayton { 447444d93782SGreg Clayton if (submenus.size() == 7) 447544d93782SGreg Clayton menu.AddSubmenu (MenuSP (new Menu(Menu::Type::Separator))); 447644d93782SGreg Clayton else if (submenus.size() > 8) 447744d93782SGreg Clayton submenus.erase (submenus.begin() + 8, submenus.end()); 447844d93782SGreg Clayton 447944d93782SGreg Clayton ThreadList &threads = process->GetThreadList(); 448044d93782SGreg Clayton Mutex::Locker locker (threads.GetMutex()); 448144d93782SGreg Clayton size_t num_threads = threads.GetSize(); 448244d93782SGreg Clayton for (size_t i=0; i<num_threads; ++i) 448344d93782SGreg Clayton { 448444d93782SGreg Clayton ThreadSP thread_sp = threads.GetThreadAtIndex(i); 448544d93782SGreg Clayton char menu_char = '\0'; 448644d93782SGreg Clayton if (i < 9) 448744d93782SGreg Clayton menu_char = '1' + i; 448844d93782SGreg Clayton StreamString thread_menu_title; 448944d93782SGreg Clayton thread_menu_title.Printf("Thread %u", thread_sp->GetIndexID()); 449044d93782SGreg Clayton const char *thread_name = thread_sp->GetName(); 449144d93782SGreg Clayton if (thread_name && thread_name[0]) 449244d93782SGreg Clayton thread_menu_title.Printf (" %s", thread_name); 449344d93782SGreg Clayton else 449444d93782SGreg Clayton { 449544d93782SGreg Clayton const char *queue_name = thread_sp->GetQueueName(); 449644d93782SGreg Clayton if (queue_name && queue_name[0]) 449744d93782SGreg Clayton thread_menu_title.Printf (" %s", queue_name); 449844d93782SGreg Clayton } 449944d93782SGreg Clayton menu.AddSubmenu (MenuSP (new Menu(thread_menu_title.GetString().c_str(), NULL, menu_char, thread_sp->GetID()))); 450044d93782SGreg Clayton } 450144d93782SGreg Clayton } 450244d93782SGreg Clayton else if (submenus.size() > 7) 450344d93782SGreg Clayton { 450444d93782SGreg Clayton // Remove the separator and any other thread submenu items 450544d93782SGreg Clayton // that were previously added 450644d93782SGreg Clayton submenus.erase (submenus.begin() + 7, submenus.end()); 450744d93782SGreg Clayton } 450844d93782SGreg Clayton // Since we are adding and removing items we need to recalculate the name lengths 450944d93782SGreg Clayton menu.RecalculateNameLengths(); 451044d93782SGreg Clayton } 451144d93782SGreg Clayton return MenuActionResult::Handled; 451244d93782SGreg Clayton 451344d93782SGreg Clayton case eMenuID_ViewVariables: 451444d93782SGreg Clayton { 451544d93782SGreg Clayton WindowSP main_window_sp = m_app.GetMainWindow(); 451644d93782SGreg Clayton WindowSP source_window_sp = main_window_sp->FindSubWindow("Source"); 451744d93782SGreg Clayton WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables"); 451844d93782SGreg Clayton WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers"); 451944d93782SGreg Clayton const Rect source_bounds = source_window_sp->GetBounds(); 452044d93782SGreg Clayton 452144d93782SGreg Clayton if (variables_window_sp) 452244d93782SGreg Clayton { 452344d93782SGreg Clayton const Rect variables_bounds = variables_window_sp->GetBounds(); 452444d93782SGreg Clayton 452544d93782SGreg Clayton main_window_sp->RemoveSubWindow(variables_window_sp.get()); 452644d93782SGreg Clayton 452744d93782SGreg Clayton if (registers_window_sp) 452844d93782SGreg Clayton { 452944d93782SGreg Clayton // We have a registers window, so give all the area back to the registers window 453044d93782SGreg Clayton Rect registers_bounds = variables_bounds; 453144d93782SGreg Clayton registers_bounds.size.width = source_bounds.size.width; 453244d93782SGreg Clayton registers_window_sp->SetBounds(registers_bounds); 453344d93782SGreg Clayton } 453444d93782SGreg Clayton else 453544d93782SGreg Clayton { 453644d93782SGreg Clayton // We have no registers window showing so give the bottom 453744d93782SGreg Clayton // area back to the source view 453844d93782SGreg Clayton source_window_sp->Resize (source_bounds.size.width, 453944d93782SGreg Clayton source_bounds.size.height + variables_bounds.size.height); 454044d93782SGreg Clayton } 454144d93782SGreg Clayton } 454244d93782SGreg Clayton else 454344d93782SGreg Clayton { 454444d93782SGreg Clayton Rect new_variables_rect; 454544d93782SGreg Clayton if (registers_window_sp) 454644d93782SGreg Clayton { 454744d93782SGreg Clayton // We have a registers window so split the area of the registers 454844d93782SGreg Clayton // window into two columns where the left hand side will be the 454944d93782SGreg Clayton // variables and the right hand side will be the registers 455044d93782SGreg Clayton const Rect variables_bounds = registers_window_sp->GetBounds(); 455144d93782SGreg Clayton Rect new_registers_rect; 455244d93782SGreg Clayton variables_bounds.VerticalSplitPercentage (0.50, new_variables_rect, new_registers_rect); 455344d93782SGreg Clayton registers_window_sp->SetBounds (new_registers_rect); 455444d93782SGreg Clayton } 455544d93782SGreg Clayton else 455644d93782SGreg Clayton { 455744d93782SGreg Clayton // No variables window, grab the bottom part of the source window 455844d93782SGreg Clayton Rect new_source_rect; 455944d93782SGreg Clayton source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_variables_rect); 456044d93782SGreg Clayton source_window_sp->SetBounds (new_source_rect); 456144d93782SGreg Clayton } 456244d93782SGreg Clayton WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Variables", 456344d93782SGreg Clayton new_variables_rect, 456444d93782SGreg Clayton false); 456544d93782SGreg Clayton new_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger))); 456644d93782SGreg Clayton } 456744d93782SGreg Clayton touchwin(stdscr); 456844d93782SGreg Clayton } 456944d93782SGreg Clayton return MenuActionResult::Handled; 457044d93782SGreg Clayton 457144d93782SGreg Clayton case eMenuID_ViewRegisters: 457244d93782SGreg Clayton { 457344d93782SGreg Clayton WindowSP main_window_sp = m_app.GetMainWindow(); 457444d93782SGreg Clayton WindowSP source_window_sp = main_window_sp->FindSubWindow("Source"); 457544d93782SGreg Clayton WindowSP variables_window_sp = main_window_sp->FindSubWindow("Variables"); 457644d93782SGreg Clayton WindowSP registers_window_sp = main_window_sp->FindSubWindow("Registers"); 457744d93782SGreg Clayton const Rect source_bounds = source_window_sp->GetBounds(); 457844d93782SGreg Clayton 457944d93782SGreg Clayton if (registers_window_sp) 458044d93782SGreg Clayton { 458144d93782SGreg Clayton if (variables_window_sp) 458244d93782SGreg Clayton { 458344d93782SGreg Clayton const Rect variables_bounds = variables_window_sp->GetBounds(); 458444d93782SGreg Clayton 458544d93782SGreg Clayton // We have a variables window, so give all the area back to the variables window 458644d93782SGreg Clayton variables_window_sp->Resize (variables_bounds.size.width + registers_window_sp->GetWidth(), 458744d93782SGreg Clayton variables_bounds.size.height); 458844d93782SGreg Clayton } 458944d93782SGreg Clayton else 459044d93782SGreg Clayton { 459144d93782SGreg Clayton // We have no variables window showing so give the bottom 459244d93782SGreg Clayton // area back to the source view 459344d93782SGreg Clayton source_window_sp->Resize (source_bounds.size.width, 459444d93782SGreg Clayton source_bounds.size.height + registers_window_sp->GetHeight()); 459544d93782SGreg Clayton } 459644d93782SGreg Clayton main_window_sp->RemoveSubWindow(registers_window_sp.get()); 459744d93782SGreg Clayton } 459844d93782SGreg Clayton else 459944d93782SGreg Clayton { 460044d93782SGreg Clayton Rect new_regs_rect; 460144d93782SGreg Clayton if (variables_window_sp) 460244d93782SGreg Clayton { 460344d93782SGreg Clayton // We have a variables window, split it into two columns 460444d93782SGreg Clayton // where the left hand side will be the variables and the 460544d93782SGreg Clayton // right hand side will be the registers 460644d93782SGreg Clayton const Rect variables_bounds = variables_window_sp->GetBounds(); 460744d93782SGreg Clayton Rect new_vars_rect; 460844d93782SGreg Clayton variables_bounds.VerticalSplitPercentage (0.50, new_vars_rect, new_regs_rect); 460944d93782SGreg Clayton variables_window_sp->SetBounds (new_vars_rect); 461044d93782SGreg Clayton } 461144d93782SGreg Clayton else 461244d93782SGreg Clayton { 461344d93782SGreg Clayton // No registers window, grab the bottom part of the source window 461444d93782SGreg Clayton Rect new_source_rect; 461544d93782SGreg Clayton source_bounds.HorizontalSplitPercentage (0.70, new_source_rect, new_regs_rect); 461644d93782SGreg Clayton source_window_sp->SetBounds (new_source_rect); 461744d93782SGreg Clayton } 461844d93782SGreg Clayton WindowSP new_window_sp = main_window_sp->CreateSubWindow ("Registers", 461944d93782SGreg Clayton new_regs_rect, 462044d93782SGreg Clayton false); 462144d93782SGreg Clayton new_window_sp->SetDelegate (WindowDelegateSP(new RegistersWindowDelegate(m_debugger))); 462244d93782SGreg Clayton } 462344d93782SGreg Clayton touchwin(stdscr); 462444d93782SGreg Clayton } 462544d93782SGreg Clayton return MenuActionResult::Handled; 462644d93782SGreg Clayton 462744d93782SGreg Clayton case eMenuID_HelpGUIHelp: 46285fdb09bbSGreg Clayton m_app.GetMainWindow ()->CreateHelpSubwindow(); 462944d93782SGreg Clayton return MenuActionResult::Handled; 463044d93782SGreg Clayton 463144d93782SGreg Clayton default: 463244d93782SGreg Clayton break; 463344d93782SGreg Clayton } 463444d93782SGreg Clayton 463544d93782SGreg Clayton return MenuActionResult::NotHandled; 463644d93782SGreg Clayton } 463744d93782SGreg Clayton protected: 463844d93782SGreg Clayton Application &m_app; 463944d93782SGreg Clayton Debugger &m_debugger; 464044d93782SGreg Clayton }; 464144d93782SGreg Clayton 464244d93782SGreg Clayton 464344d93782SGreg Clayton class StatusBarWindowDelegate : public WindowDelegate 464444d93782SGreg Clayton { 464544d93782SGreg Clayton public: 464644d93782SGreg Clayton StatusBarWindowDelegate (Debugger &debugger) : 464744d93782SGreg Clayton m_debugger (debugger) 464844d93782SGreg Clayton { 4649554f68d3SGreg Clayton FormatEntity::Parse("Thread: ${thread.id%tid}", 4650554f68d3SGreg Clayton m_format); 465144d93782SGreg Clayton } 465244d93782SGreg Clayton 465344d93782SGreg Clayton ~StatusBarWindowDelegate () 465444d93782SGreg Clayton { 465544d93782SGreg Clayton } 4656bd5ae6b4SGreg Clayton 4657bd5ae6b4SGreg Clayton bool 4658bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 465944d93782SGreg Clayton { 466044d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 466144d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 466244d93782SGreg Clayton Thread *thread = exe_ctx.GetThreadPtr(); 466344d93782SGreg Clayton StackFrame *frame = exe_ctx.GetFramePtr(); 466444d93782SGreg Clayton window.Erase(); 466544d93782SGreg Clayton window.SetBackground(2); 466644d93782SGreg Clayton window.MoveCursor (0, 0); 466744d93782SGreg Clayton if (process) 466844d93782SGreg Clayton { 466944d93782SGreg Clayton const StateType state = process->GetState(); 467044d93782SGreg Clayton window.Printf ("Process: %5" PRIu64 " %10s", process->GetID(), StateAsCString(state)); 467144d93782SGreg Clayton 467244d93782SGreg Clayton if (StateIsStoppedState(state, true)) 467344d93782SGreg Clayton { 46745b031ebcSEd Maste StreamString strm; 4675554f68d3SGreg Clayton if (thread && FormatEntity::Format (m_format, strm, NULL, &exe_ctx, NULL, NULL, false, false)) 46765b031ebcSEd Maste { 467744d93782SGreg Clayton window.MoveCursor (40, 0); 46785b031ebcSEd Maste window.PutCStringTruncated(strm.GetString().c_str(), 1); 46795b031ebcSEd Maste } 468044d93782SGreg Clayton 468144d93782SGreg Clayton window.MoveCursor (60, 0); 468244d93782SGreg Clayton if (frame) 468344d93782SGreg Clayton window.Printf ("Frame: %3u PC = 0x%16.16" PRIx64, frame->GetFrameIndex(), frame->GetFrameCodeAddress().GetOpcodeLoadAddress (exe_ctx.GetTargetPtr())); 468444d93782SGreg Clayton } 468544d93782SGreg Clayton else if (state == eStateExited) 468644d93782SGreg Clayton { 468744d93782SGreg Clayton const char *exit_desc = process->GetExitDescription(); 468844d93782SGreg Clayton const int exit_status = process->GetExitStatus(); 468944d93782SGreg Clayton if (exit_desc && exit_desc[0]) 469044d93782SGreg Clayton window.Printf (" with status = %i (%s)", exit_status, exit_desc); 469144d93782SGreg Clayton else 469244d93782SGreg Clayton window.Printf (" with status = %i", exit_status); 469344d93782SGreg Clayton } 469444d93782SGreg Clayton } 469544d93782SGreg Clayton window.DeferredRefresh(); 469644d93782SGreg Clayton return true; 469744d93782SGreg Clayton } 469844d93782SGreg Clayton 469944d93782SGreg Clayton protected: 470044d93782SGreg Clayton Debugger &m_debugger; 4701554f68d3SGreg Clayton FormatEntity::Entry m_format; 470244d93782SGreg Clayton }; 470344d93782SGreg Clayton 470444d93782SGreg Clayton class SourceFileWindowDelegate : public WindowDelegate 470544d93782SGreg Clayton { 470644d93782SGreg Clayton public: 470744d93782SGreg Clayton SourceFileWindowDelegate (Debugger &debugger) : 470844d93782SGreg Clayton WindowDelegate (), 470944d93782SGreg Clayton m_debugger (debugger), 471044d93782SGreg Clayton m_sc (), 471144d93782SGreg Clayton m_file_sp (), 471244d93782SGreg Clayton m_disassembly_scope (NULL), 471344d93782SGreg Clayton m_disassembly_sp (), 471444d93782SGreg Clayton m_disassembly_range (), 4715ec990867SGreg Clayton m_title (), 471644d93782SGreg Clayton m_line_width (4), 471744d93782SGreg Clayton m_selected_line (0), 471844d93782SGreg Clayton m_pc_line (0), 471944d93782SGreg Clayton m_stop_id (0), 472044d93782SGreg Clayton m_frame_idx (UINT32_MAX), 472144d93782SGreg Clayton m_first_visible_line (0), 472244d93782SGreg Clayton m_min_x (0), 472344d93782SGreg Clayton m_min_y (0), 472444d93782SGreg Clayton m_max_x (0), 472544d93782SGreg Clayton m_max_y (0) 472644d93782SGreg Clayton { 472744d93782SGreg Clayton } 472844d93782SGreg Clayton 4729bd5ae6b4SGreg Clayton ~SourceFileWindowDelegate() override 473044d93782SGreg Clayton { 473144d93782SGreg Clayton } 473244d93782SGreg Clayton 473344d93782SGreg Clayton void 473444d93782SGreg Clayton Update (const SymbolContext &sc) 473544d93782SGreg Clayton { 473644d93782SGreg Clayton m_sc = sc; 473744d93782SGreg Clayton } 473844d93782SGreg Clayton 473944d93782SGreg Clayton uint32_t 474044d93782SGreg Clayton NumVisibleLines () const 474144d93782SGreg Clayton { 474244d93782SGreg Clayton return m_max_y - m_min_y; 474344d93782SGreg Clayton } 474444d93782SGreg Clayton 4745bd5ae6b4SGreg Clayton const char * 4746bd5ae6b4SGreg Clayton WindowDelegateGetHelpText () override 474744d93782SGreg Clayton { 474844d93782SGreg Clayton return "Source/Disassembly window keyboard shortcuts:"; 474944d93782SGreg Clayton } 475044d93782SGreg Clayton 4751bd5ae6b4SGreg Clayton KeyHelp * 4752bd5ae6b4SGreg Clayton WindowDelegateGetKeyHelp () override 475344d93782SGreg Clayton { 475444d93782SGreg Clayton static curses::KeyHelp g_source_view_key_help[] = { 475544d93782SGreg Clayton { KEY_RETURN, "Run to selected line with one shot breakpoint" }, 475644d93782SGreg Clayton { KEY_UP, "Select previous source line" }, 475744d93782SGreg Clayton { KEY_DOWN, "Select next source line" }, 475844d93782SGreg Clayton { KEY_PPAGE, "Page up" }, 475944d93782SGreg Clayton { KEY_NPAGE, "Page down" }, 476044d93782SGreg Clayton { 'b', "Set breakpoint on selected source/disassembly line" }, 476144d93782SGreg Clayton { 'c', "Continue process" }, 476244d93782SGreg Clayton { 'd', "Detach and resume process" }, 476344d93782SGreg Clayton { 'D', "Detach with process suspended" }, 476444d93782SGreg Clayton { 'h', "Show help dialog" }, 476544d93782SGreg Clayton { 'k', "Kill process" }, 476644d93782SGreg Clayton { 'n', "Step over (source line)" }, 476744d93782SGreg Clayton { 'N', "Step over (single instruction)" }, 476844d93782SGreg Clayton { 'o', "Step out" }, 476944d93782SGreg Clayton { 's', "Step in (source line)" }, 477044d93782SGreg Clayton { 'S', "Step in (single instruction)" }, 477144d93782SGreg Clayton { ',', "Page up" }, 477244d93782SGreg Clayton { '.', "Page down" }, 477344d93782SGreg Clayton { '\0', NULL } 477444d93782SGreg Clayton }; 477544d93782SGreg Clayton return g_source_view_key_help; 477644d93782SGreg Clayton } 477744d93782SGreg Clayton 4778bd5ae6b4SGreg Clayton bool 4779bd5ae6b4SGreg Clayton WindowDelegateDraw (Window &window, bool force) override 478044d93782SGreg Clayton { 478144d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 478244d93782SGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 478344d93782SGreg Clayton Thread *thread = NULL; 478444d93782SGreg Clayton 478544d93782SGreg Clayton bool update_location = false; 478644d93782SGreg Clayton if (process) 478744d93782SGreg Clayton { 478844d93782SGreg Clayton StateType state = process->GetState(); 478944d93782SGreg Clayton if (StateIsStoppedState(state, true)) 479044d93782SGreg Clayton { 479144d93782SGreg Clayton // We are stopped, so it is ok to 479244d93782SGreg Clayton update_location = true; 479344d93782SGreg Clayton } 479444d93782SGreg Clayton } 479544d93782SGreg Clayton 479644d93782SGreg Clayton m_min_x = 1; 4797ec990867SGreg Clayton m_min_y = 2; 479844d93782SGreg Clayton m_max_x = window.GetMaxX()-1; 479944d93782SGreg Clayton m_max_y = window.GetMaxY()-1; 480044d93782SGreg Clayton 480144d93782SGreg Clayton const uint32_t num_visible_lines = NumVisibleLines(); 480244d93782SGreg Clayton StackFrameSP frame_sp; 480344d93782SGreg Clayton bool set_selected_line_to_pc = false; 480444d93782SGreg Clayton 480544d93782SGreg Clayton if (update_location) 480644d93782SGreg Clayton { 480744d93782SGreg Clayton const bool process_alive = process ? process->IsAlive() : false; 480844d93782SGreg Clayton bool thread_changed = false; 480944d93782SGreg Clayton if (process_alive) 481044d93782SGreg Clayton { 481144d93782SGreg Clayton thread = exe_ctx.GetThreadPtr(); 481244d93782SGreg Clayton if (thread) 481344d93782SGreg Clayton { 481444d93782SGreg Clayton frame_sp = thread->GetSelectedFrame(); 481544d93782SGreg Clayton auto tid = thread->GetID(); 481644d93782SGreg Clayton thread_changed = tid != m_tid; 481744d93782SGreg Clayton m_tid = tid; 481844d93782SGreg Clayton } 481944d93782SGreg Clayton else 482044d93782SGreg Clayton { 482144d93782SGreg Clayton if (m_tid != LLDB_INVALID_THREAD_ID) 482244d93782SGreg Clayton { 482344d93782SGreg Clayton thread_changed = true; 482444d93782SGreg Clayton m_tid = LLDB_INVALID_THREAD_ID; 482544d93782SGreg Clayton } 482644d93782SGreg Clayton } 482744d93782SGreg Clayton } 482844d93782SGreg Clayton const uint32_t stop_id = process ? process->GetStopID() : 0; 482944d93782SGreg Clayton const bool stop_id_changed = stop_id != m_stop_id; 483044d93782SGreg Clayton bool frame_changed = false; 483144d93782SGreg Clayton m_stop_id = stop_id; 4832ec990867SGreg Clayton m_title.Clear(); 483344d93782SGreg Clayton if (frame_sp) 483444d93782SGreg Clayton { 483544d93782SGreg Clayton m_sc = frame_sp->GetSymbolContext(eSymbolContextEverything); 4836ec990867SGreg Clayton if (m_sc.module_sp) 4837ec990867SGreg Clayton { 4838ec990867SGreg Clayton m_title.Printf("%s", m_sc.module_sp->GetFileSpec().GetFilename().GetCString()); 4839ec990867SGreg Clayton ConstString func_name = m_sc.GetFunctionName(); 4840ec990867SGreg Clayton if (func_name) 4841ec990867SGreg Clayton m_title.Printf("`%s", func_name.GetCString()); 4842ec990867SGreg Clayton } 484344d93782SGreg Clayton const uint32_t frame_idx = frame_sp->GetFrameIndex(); 484444d93782SGreg Clayton frame_changed = frame_idx != m_frame_idx; 484544d93782SGreg Clayton m_frame_idx = frame_idx; 484644d93782SGreg Clayton } 484744d93782SGreg Clayton else 484844d93782SGreg Clayton { 484944d93782SGreg Clayton m_sc.Clear(true); 485044d93782SGreg Clayton frame_changed = m_frame_idx != UINT32_MAX; 485144d93782SGreg Clayton m_frame_idx = UINT32_MAX; 485244d93782SGreg Clayton } 485344d93782SGreg Clayton 485444d93782SGreg Clayton const bool context_changed = thread_changed || frame_changed || stop_id_changed; 485544d93782SGreg Clayton 485644d93782SGreg Clayton if (process_alive) 485744d93782SGreg Clayton { 485844d93782SGreg Clayton if (m_sc.line_entry.IsValid()) 485944d93782SGreg Clayton { 486044d93782SGreg Clayton m_pc_line = m_sc.line_entry.line; 486144d93782SGreg Clayton if (m_pc_line != UINT32_MAX) 486244d93782SGreg Clayton --m_pc_line; // Convert to zero based line number... 486344d93782SGreg Clayton // Update the selected line if the stop ID changed... 486444d93782SGreg Clayton if (context_changed) 486544d93782SGreg Clayton m_selected_line = m_pc_line; 486644d93782SGreg Clayton 486744d93782SGreg Clayton if (m_file_sp && m_file_sp->FileSpecMatches(m_sc.line_entry.file)) 486844d93782SGreg Clayton { 486944d93782SGreg Clayton // Same file, nothing to do, we should either have the 487044d93782SGreg Clayton // lines or not (source file missing) 48713985c8c6SSaleem Abdulrasool if (m_selected_line >= static_cast<size_t>(m_first_visible_line)) 487244d93782SGreg Clayton { 487344d93782SGreg Clayton if (m_selected_line >= m_first_visible_line + num_visible_lines) 487444d93782SGreg Clayton m_first_visible_line = m_selected_line - 10; 487544d93782SGreg Clayton } 487644d93782SGreg Clayton else 487744d93782SGreg Clayton { 487844d93782SGreg Clayton if (m_selected_line > 10) 487944d93782SGreg Clayton m_first_visible_line = m_selected_line - 10; 488044d93782SGreg Clayton else 488144d93782SGreg Clayton m_first_visible_line = 0; 488244d93782SGreg Clayton } 488344d93782SGreg Clayton } 488444d93782SGreg Clayton else 488544d93782SGreg Clayton { 488644d93782SGreg Clayton // File changed, set selected line to the line with the PC 488744d93782SGreg Clayton m_selected_line = m_pc_line; 488844d93782SGreg Clayton m_file_sp = m_debugger.GetSourceManager().GetFile(m_sc.line_entry.file); 488944d93782SGreg Clayton if (m_file_sp) 489044d93782SGreg Clayton { 489144d93782SGreg Clayton const size_t num_lines = m_file_sp->GetNumLines(); 489244d93782SGreg Clayton int m_line_width = 1; 489344d93782SGreg Clayton for (size_t n = num_lines; n >= 10; n = n / 10) 489444d93782SGreg Clayton ++m_line_width; 489544d93782SGreg Clayton 489644d93782SGreg Clayton snprintf (m_line_format, sizeof(m_line_format), " %%%iu ", m_line_width); 489744d93782SGreg Clayton if (num_lines < num_visible_lines || m_selected_line < num_visible_lines) 489844d93782SGreg Clayton m_first_visible_line = 0; 489944d93782SGreg Clayton else 490044d93782SGreg Clayton m_first_visible_line = m_selected_line - 10; 490144d93782SGreg Clayton } 490244d93782SGreg Clayton } 490344d93782SGreg Clayton } 490444d93782SGreg Clayton else 490544d93782SGreg Clayton { 490644d93782SGreg Clayton m_file_sp.reset(); 490744d93782SGreg Clayton } 490844d93782SGreg Clayton 490944d93782SGreg Clayton if (!m_file_sp || m_file_sp->GetNumLines() == 0) 491044d93782SGreg Clayton { 491144d93782SGreg Clayton // Show disassembly 491244d93782SGreg Clayton bool prefer_file_cache = false; 491344d93782SGreg Clayton if (m_sc.function) 491444d93782SGreg Clayton { 491544d93782SGreg Clayton if (m_disassembly_scope != m_sc.function) 491644d93782SGreg Clayton { 491744d93782SGreg Clayton m_disassembly_scope = m_sc.function; 491844d93782SGreg Clayton m_disassembly_sp = m_sc.function->GetInstructions (exe_ctx, NULL, prefer_file_cache); 491944d93782SGreg Clayton if (m_disassembly_sp) 492044d93782SGreg Clayton { 492144d93782SGreg Clayton set_selected_line_to_pc = true; 492244d93782SGreg Clayton m_disassembly_range = m_sc.function->GetAddressRange(); 492344d93782SGreg Clayton } 492444d93782SGreg Clayton else 492544d93782SGreg Clayton { 492644d93782SGreg Clayton m_disassembly_range.Clear(); 492744d93782SGreg Clayton } 492844d93782SGreg Clayton } 492944d93782SGreg Clayton else 493044d93782SGreg Clayton { 493144d93782SGreg Clayton set_selected_line_to_pc = context_changed; 493244d93782SGreg Clayton } 493344d93782SGreg Clayton } 493444d93782SGreg Clayton else if (m_sc.symbol) 493544d93782SGreg Clayton { 493644d93782SGreg Clayton if (m_disassembly_scope != m_sc.symbol) 493744d93782SGreg Clayton { 493844d93782SGreg Clayton m_disassembly_scope = m_sc.symbol; 493944d93782SGreg Clayton m_disassembly_sp = m_sc.symbol->GetInstructions (exe_ctx, NULL, prefer_file_cache); 494044d93782SGreg Clayton if (m_disassembly_sp) 494144d93782SGreg Clayton { 494244d93782SGreg Clayton set_selected_line_to_pc = true; 494344d93782SGreg Clayton m_disassembly_range.GetBaseAddress() = m_sc.symbol->GetAddress(); 494444d93782SGreg Clayton m_disassembly_range.SetByteSize(m_sc.symbol->GetByteSize()); 494544d93782SGreg Clayton } 494644d93782SGreg Clayton else 494744d93782SGreg Clayton { 494844d93782SGreg Clayton m_disassembly_range.Clear(); 494944d93782SGreg Clayton } 495044d93782SGreg Clayton } 495144d93782SGreg Clayton else 495244d93782SGreg Clayton { 495344d93782SGreg Clayton set_selected_line_to_pc = context_changed; 495444d93782SGreg Clayton } 495544d93782SGreg Clayton } 495644d93782SGreg Clayton } 495744d93782SGreg Clayton } 495844d93782SGreg Clayton else 495944d93782SGreg Clayton { 496044d93782SGreg Clayton m_pc_line = UINT32_MAX; 496144d93782SGreg Clayton } 496244d93782SGreg Clayton } 496344d93782SGreg Clayton 4964ec990867SGreg Clayton const int window_width = window.GetWidth(); 496544d93782SGreg Clayton window.Erase(); 496644d93782SGreg Clayton window.DrawTitleBox ("Sources"); 4967ec990867SGreg Clayton if (!m_title.GetString().empty()) 4968ec990867SGreg Clayton { 4969ec990867SGreg Clayton window.AttributeOn(A_REVERSE); 4970ec990867SGreg Clayton window.MoveCursor(1, 1); 4971ec990867SGreg Clayton window.PutChar(' '); 4972ec990867SGreg Clayton window.PutCStringTruncated(m_title.GetString().c_str(), 1); 4973ec990867SGreg Clayton int x = window.GetCursorX(); 4974ec990867SGreg Clayton if (x < window_width - 1) 4975ec990867SGreg Clayton { 4976ec990867SGreg Clayton window.Printf ("%*s", window_width - x - 1, ""); 4977ec990867SGreg Clayton } 4978ec990867SGreg Clayton window.AttributeOff(A_REVERSE); 4979ec990867SGreg Clayton } 498044d93782SGreg Clayton 498144d93782SGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 498244d93782SGreg Clayton const size_t num_source_lines = GetNumSourceLines(); 498344d93782SGreg Clayton if (num_source_lines > 0) 498444d93782SGreg Clayton { 498544d93782SGreg Clayton // Display source 498644d93782SGreg Clayton BreakpointLines bp_lines; 498744d93782SGreg Clayton if (target) 498844d93782SGreg Clayton { 498944d93782SGreg Clayton BreakpointList &bp_list = target->GetBreakpointList(); 499044d93782SGreg Clayton const size_t num_bps = bp_list.GetSize(); 499144d93782SGreg Clayton for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx) 499244d93782SGreg Clayton { 499344d93782SGreg Clayton BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx); 499444d93782SGreg Clayton const size_t num_bps_locs = bp_sp->GetNumLocations(); 499544d93782SGreg Clayton for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx) 499644d93782SGreg Clayton { 499744d93782SGreg Clayton BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx); 499844d93782SGreg Clayton LineEntry bp_loc_line_entry; 499944d93782SGreg Clayton if (bp_loc_sp->GetAddress().CalculateSymbolContextLineEntry (bp_loc_line_entry)) 500044d93782SGreg Clayton { 500144d93782SGreg Clayton if (m_file_sp->GetFileSpec() == bp_loc_line_entry.file) 500244d93782SGreg Clayton { 500344d93782SGreg Clayton bp_lines.insert(bp_loc_line_entry.line); 500444d93782SGreg Clayton } 500544d93782SGreg Clayton } 500644d93782SGreg Clayton } 500744d93782SGreg Clayton } 500844d93782SGreg Clayton } 500944d93782SGreg Clayton 501044d93782SGreg Clayton const attr_t selected_highlight_attr = A_REVERSE; 501144d93782SGreg Clayton const attr_t pc_highlight_attr = COLOR_PAIR(1); 501244d93782SGreg Clayton 50133985c8c6SSaleem Abdulrasool for (size_t i=0; i<num_visible_lines; ++i) 501444d93782SGreg Clayton { 501544d93782SGreg Clayton const uint32_t curr_line = m_first_visible_line + i; 501644d93782SGreg Clayton if (curr_line < num_source_lines) 501744d93782SGreg Clayton { 5018ec990867SGreg Clayton const int line_y = m_min_y+i; 501944d93782SGreg Clayton window.MoveCursor(1, line_y); 502044d93782SGreg Clayton const bool is_pc_line = curr_line == m_pc_line; 502144d93782SGreg Clayton const bool line_is_selected = m_selected_line == curr_line; 502244d93782SGreg Clayton // Highlight the line as the PC line first, then if the selected line 502344d93782SGreg Clayton // isn't the same as the PC line, highlight it differently 502444d93782SGreg Clayton attr_t highlight_attr = 0; 502544d93782SGreg Clayton attr_t bp_attr = 0; 502644d93782SGreg Clayton if (is_pc_line) 502744d93782SGreg Clayton highlight_attr = pc_highlight_attr; 502844d93782SGreg Clayton else if (line_is_selected) 502944d93782SGreg Clayton highlight_attr = selected_highlight_attr; 503044d93782SGreg Clayton 503144d93782SGreg Clayton if (bp_lines.find(curr_line+1) != bp_lines.end()) 503244d93782SGreg Clayton bp_attr = COLOR_PAIR(2); 503344d93782SGreg Clayton 503444d93782SGreg Clayton if (bp_attr) 503544d93782SGreg Clayton window.AttributeOn(bp_attr); 503644d93782SGreg Clayton 503744d93782SGreg Clayton window.Printf (m_line_format, curr_line + 1); 503844d93782SGreg Clayton 503944d93782SGreg Clayton if (bp_attr) 504044d93782SGreg Clayton window.AttributeOff(bp_attr); 504144d93782SGreg Clayton 504244d93782SGreg Clayton window.PutChar(ACS_VLINE); 504344d93782SGreg Clayton // Mark the line with the PC with a diamond 504444d93782SGreg Clayton if (is_pc_line) 504544d93782SGreg Clayton window.PutChar(ACS_DIAMOND); 504644d93782SGreg Clayton else 504744d93782SGreg Clayton window.PutChar(' '); 504844d93782SGreg Clayton 504944d93782SGreg Clayton if (highlight_attr) 505044d93782SGreg Clayton window.AttributeOn(highlight_attr); 505144d93782SGreg Clayton const uint32_t line_len = m_file_sp->GetLineLength(curr_line + 1, false); 505244d93782SGreg Clayton if (line_len > 0) 505344d93782SGreg Clayton window.PutCString(m_file_sp->PeekLineData(curr_line + 1), line_len); 505444d93782SGreg Clayton 505544d93782SGreg Clayton if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0) 505644d93782SGreg Clayton { 505744d93782SGreg Clayton StopInfoSP stop_info_sp; 505844d93782SGreg Clayton if (thread) 505944d93782SGreg Clayton stop_info_sp = thread->GetStopInfo(); 506044d93782SGreg Clayton if (stop_info_sp) 506144d93782SGreg Clayton { 506244d93782SGreg Clayton const char *stop_description = stop_info_sp->GetDescription(); 506344d93782SGreg Clayton if (stop_description && stop_description[0]) 506444d93782SGreg Clayton { 506544d93782SGreg Clayton size_t stop_description_len = strlen(stop_description); 5066ec990867SGreg Clayton int desc_x = window_width - stop_description_len - 16; 506744d93782SGreg Clayton window.Printf ("%*s", desc_x - window.GetCursorX(), ""); 5068ec990867SGreg Clayton //window.MoveCursor(window_width - stop_description_len - 15, line_y); 506944d93782SGreg Clayton window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description); 507044d93782SGreg Clayton } 507144d93782SGreg Clayton } 507244d93782SGreg Clayton else 507344d93782SGreg Clayton { 5074ec990867SGreg Clayton window.Printf ("%*s", window_width - window.GetCursorX() - 1, ""); 507544d93782SGreg Clayton } 507644d93782SGreg Clayton } 507744d93782SGreg Clayton if (highlight_attr) 507844d93782SGreg Clayton window.AttributeOff(highlight_attr); 507944d93782SGreg Clayton 508044d93782SGreg Clayton } 508144d93782SGreg Clayton else 508244d93782SGreg Clayton { 508344d93782SGreg Clayton break; 508444d93782SGreg Clayton } 508544d93782SGreg Clayton } 508644d93782SGreg Clayton } 508744d93782SGreg Clayton else 508844d93782SGreg Clayton { 508944d93782SGreg Clayton size_t num_disassembly_lines = GetNumDisassemblyLines(); 509044d93782SGreg Clayton if (num_disassembly_lines > 0) 509144d93782SGreg Clayton { 509244d93782SGreg Clayton // Display disassembly 509344d93782SGreg Clayton BreakpointAddrs bp_file_addrs; 509444d93782SGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 509544d93782SGreg Clayton if (target) 509644d93782SGreg Clayton { 509744d93782SGreg Clayton BreakpointList &bp_list = target->GetBreakpointList(); 509844d93782SGreg Clayton const size_t num_bps = bp_list.GetSize(); 509944d93782SGreg Clayton for (size_t bp_idx=0; bp_idx<num_bps; ++bp_idx) 510044d93782SGreg Clayton { 510144d93782SGreg Clayton BreakpointSP bp_sp = bp_list.GetBreakpointAtIndex(bp_idx); 510244d93782SGreg Clayton const size_t num_bps_locs = bp_sp->GetNumLocations(); 510344d93782SGreg Clayton for (size_t bp_loc_idx=0; bp_loc_idx<num_bps_locs; ++bp_loc_idx) 510444d93782SGreg Clayton { 510544d93782SGreg Clayton BreakpointLocationSP bp_loc_sp = bp_sp->GetLocationAtIndex(bp_loc_idx); 510644d93782SGreg Clayton LineEntry bp_loc_line_entry; 510744d93782SGreg Clayton const lldb::addr_t file_addr = bp_loc_sp->GetAddress().GetFileAddress(); 510844d93782SGreg Clayton if (file_addr != LLDB_INVALID_ADDRESS) 510944d93782SGreg Clayton { 511044d93782SGreg Clayton if (m_disassembly_range.ContainsFileAddress(file_addr)) 511144d93782SGreg Clayton bp_file_addrs.insert(file_addr); 511244d93782SGreg Clayton } 511344d93782SGreg Clayton } 511444d93782SGreg Clayton } 511544d93782SGreg Clayton } 511644d93782SGreg Clayton 511744d93782SGreg Clayton const attr_t selected_highlight_attr = A_REVERSE; 511844d93782SGreg Clayton const attr_t pc_highlight_attr = COLOR_PAIR(1); 511944d93782SGreg Clayton 512044d93782SGreg Clayton StreamString strm; 512144d93782SGreg Clayton 512244d93782SGreg Clayton InstructionList &insts = m_disassembly_sp->GetInstructionList(); 512344d93782SGreg Clayton Address pc_address; 512444d93782SGreg Clayton 512544d93782SGreg Clayton if (frame_sp) 512644d93782SGreg Clayton pc_address = frame_sp->GetFrameCodeAddress(); 512744d93782SGreg Clayton const uint32_t pc_idx = pc_address.IsValid() ? insts.GetIndexOfInstructionAtAddress (pc_address) : UINT32_MAX; 512844d93782SGreg Clayton if (set_selected_line_to_pc) 512944d93782SGreg Clayton { 513044d93782SGreg Clayton m_selected_line = pc_idx; 513144d93782SGreg Clayton } 513244d93782SGreg Clayton 513344d93782SGreg Clayton const uint32_t non_visible_pc_offset = (num_visible_lines / 5); 51343985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) >= num_disassembly_lines) 513544d93782SGreg Clayton m_first_visible_line = 0; 513644d93782SGreg Clayton 513744d93782SGreg Clayton if (pc_idx < num_disassembly_lines) 513844d93782SGreg Clayton { 51393985c8c6SSaleem Abdulrasool if (pc_idx < static_cast<uint32_t>(m_first_visible_line) || 514044d93782SGreg Clayton pc_idx >= m_first_visible_line + num_visible_lines) 514144d93782SGreg Clayton m_first_visible_line = pc_idx - non_visible_pc_offset; 514244d93782SGreg Clayton } 514344d93782SGreg Clayton 514444d93782SGreg Clayton for (size_t i=0; i<num_visible_lines; ++i) 514544d93782SGreg Clayton { 514644d93782SGreg Clayton const uint32_t inst_idx = m_first_visible_line + i; 514744d93782SGreg Clayton Instruction *inst = insts.GetInstructionAtIndex(inst_idx).get(); 514844d93782SGreg Clayton if (!inst) 514944d93782SGreg Clayton break; 515044d93782SGreg Clayton 5151ec990867SGreg Clayton const int line_y = m_min_y+i; 5152ec990867SGreg Clayton window.MoveCursor(1, line_y); 515344d93782SGreg Clayton const bool is_pc_line = frame_sp && inst_idx == pc_idx; 515444d93782SGreg Clayton const bool line_is_selected = m_selected_line == inst_idx; 515544d93782SGreg Clayton // Highlight the line as the PC line first, then if the selected line 515644d93782SGreg Clayton // isn't the same as the PC line, highlight it differently 515744d93782SGreg Clayton attr_t highlight_attr = 0; 515844d93782SGreg Clayton attr_t bp_attr = 0; 515944d93782SGreg Clayton if (is_pc_line) 516044d93782SGreg Clayton highlight_attr = pc_highlight_attr; 516144d93782SGreg Clayton else if (line_is_selected) 516244d93782SGreg Clayton highlight_attr = selected_highlight_attr; 516344d93782SGreg Clayton 516444d93782SGreg Clayton if (bp_file_addrs.find(inst->GetAddress().GetFileAddress()) != bp_file_addrs.end()) 516544d93782SGreg Clayton bp_attr = COLOR_PAIR(2); 516644d93782SGreg Clayton 516744d93782SGreg Clayton if (bp_attr) 516844d93782SGreg Clayton window.AttributeOn(bp_attr); 516944d93782SGreg Clayton 5170324a1036SSaleem Abdulrasool window.Printf (" 0x%16.16llx ", 5171324a1036SSaleem Abdulrasool static_cast<unsigned long long>(inst->GetAddress().GetLoadAddress(target))); 517244d93782SGreg Clayton 517344d93782SGreg Clayton if (bp_attr) 517444d93782SGreg Clayton window.AttributeOff(bp_attr); 517544d93782SGreg Clayton 517644d93782SGreg Clayton window.PutChar(ACS_VLINE); 517744d93782SGreg Clayton // Mark the line with the PC with a diamond 517844d93782SGreg Clayton if (is_pc_line) 517944d93782SGreg Clayton window.PutChar(ACS_DIAMOND); 518044d93782SGreg Clayton else 518144d93782SGreg Clayton window.PutChar(' '); 518244d93782SGreg Clayton 518344d93782SGreg Clayton if (highlight_attr) 518444d93782SGreg Clayton window.AttributeOn(highlight_attr); 518544d93782SGreg Clayton 518644d93782SGreg Clayton const char *mnemonic = inst->GetMnemonic(&exe_ctx); 518744d93782SGreg Clayton const char *operands = inst->GetOperands(&exe_ctx); 518844d93782SGreg Clayton const char *comment = inst->GetComment(&exe_ctx); 518944d93782SGreg Clayton 519044d93782SGreg Clayton if (mnemonic && mnemonic[0] == '\0') 519144d93782SGreg Clayton mnemonic = NULL; 519244d93782SGreg Clayton if (operands && operands[0] == '\0') 519344d93782SGreg Clayton operands = NULL; 519444d93782SGreg Clayton if (comment && comment[0] == '\0') 519544d93782SGreg Clayton comment = NULL; 519644d93782SGreg Clayton 519744d93782SGreg Clayton strm.Clear(); 519844d93782SGreg Clayton 519944d93782SGreg Clayton if (mnemonic && operands && comment) 520044d93782SGreg Clayton strm.Printf ("%-8s %-25s ; %s", mnemonic, operands, comment); 520144d93782SGreg Clayton else if (mnemonic && operands) 520244d93782SGreg Clayton strm.Printf ("%-8s %s", mnemonic, operands); 520344d93782SGreg Clayton else if (mnemonic) 520444d93782SGreg Clayton strm.Printf ("%s", mnemonic); 520544d93782SGreg Clayton 520644d93782SGreg Clayton int right_pad = 1; 520744d93782SGreg Clayton window.PutCStringTruncated(strm.GetString().c_str(), right_pad); 520844d93782SGreg Clayton 520944d93782SGreg Clayton if (is_pc_line && frame_sp && frame_sp->GetConcreteFrameIndex() == 0) 521044d93782SGreg Clayton { 521144d93782SGreg Clayton StopInfoSP stop_info_sp; 521244d93782SGreg Clayton if (thread) 521344d93782SGreg Clayton stop_info_sp = thread->GetStopInfo(); 521444d93782SGreg Clayton if (stop_info_sp) 521544d93782SGreg Clayton { 521644d93782SGreg Clayton const char *stop_description = stop_info_sp->GetDescription(); 521744d93782SGreg Clayton if (stop_description && stop_description[0]) 521844d93782SGreg Clayton { 521944d93782SGreg Clayton size_t stop_description_len = strlen(stop_description); 5220ec990867SGreg Clayton int desc_x = window_width - stop_description_len - 16; 522144d93782SGreg Clayton window.Printf ("%*s", desc_x - window.GetCursorX(), ""); 5222ec990867SGreg Clayton //window.MoveCursor(window_width - stop_description_len - 15, line_y); 522344d93782SGreg Clayton window.Printf ("<<< Thread %u: %s ", thread->GetIndexID(), stop_description); 522444d93782SGreg Clayton } 522544d93782SGreg Clayton } 522644d93782SGreg Clayton else 522744d93782SGreg Clayton { 5228ec990867SGreg Clayton window.Printf ("%*s", window_width - window.GetCursorX() - 1, ""); 522944d93782SGreg Clayton } 523044d93782SGreg Clayton } 523144d93782SGreg Clayton if (highlight_attr) 523244d93782SGreg Clayton window.AttributeOff(highlight_attr); 523344d93782SGreg Clayton } 523444d93782SGreg Clayton } 523544d93782SGreg Clayton } 523644d93782SGreg Clayton window.DeferredRefresh(); 523744d93782SGreg Clayton return true; // Drawing handled 523844d93782SGreg Clayton } 523944d93782SGreg Clayton 524044d93782SGreg Clayton size_t 524144d93782SGreg Clayton GetNumLines () 524244d93782SGreg Clayton { 524344d93782SGreg Clayton size_t num_lines = GetNumSourceLines(); 524444d93782SGreg Clayton if (num_lines == 0) 524544d93782SGreg Clayton num_lines = GetNumDisassemblyLines(); 524644d93782SGreg Clayton return num_lines; 524744d93782SGreg Clayton } 524844d93782SGreg Clayton 524944d93782SGreg Clayton size_t 525044d93782SGreg Clayton GetNumSourceLines () const 525144d93782SGreg Clayton { 525244d93782SGreg Clayton if (m_file_sp) 525344d93782SGreg Clayton return m_file_sp->GetNumLines(); 525444d93782SGreg Clayton return 0; 525544d93782SGreg Clayton } 525644d93782SGreg Clayton size_t 525744d93782SGreg Clayton GetNumDisassemblyLines () const 525844d93782SGreg Clayton { 525944d93782SGreg Clayton if (m_disassembly_sp) 526044d93782SGreg Clayton return m_disassembly_sp->GetInstructionList().GetSize(); 526144d93782SGreg Clayton return 0; 526244d93782SGreg Clayton } 526344d93782SGreg Clayton 5264bd5ae6b4SGreg Clayton HandleCharResult 5265bd5ae6b4SGreg Clayton WindowDelegateHandleChar (Window &window, int c) override 526644d93782SGreg Clayton { 526744d93782SGreg Clayton const uint32_t num_visible_lines = NumVisibleLines(); 526844d93782SGreg Clayton const size_t num_lines = GetNumLines (); 526944d93782SGreg Clayton 527044d93782SGreg Clayton switch (c) 527144d93782SGreg Clayton { 527244d93782SGreg Clayton case ',': 527344d93782SGreg Clayton case KEY_PPAGE: 527444d93782SGreg Clayton // Page up key 52753985c8c6SSaleem Abdulrasool if (static_cast<uint32_t>(m_first_visible_line) > num_visible_lines) 527644d93782SGreg Clayton m_first_visible_line -= num_visible_lines; 527744d93782SGreg Clayton else 527844d93782SGreg Clayton m_first_visible_line = 0; 527944d93782SGreg Clayton m_selected_line = m_first_visible_line; 528044d93782SGreg Clayton return eKeyHandled; 528144d93782SGreg Clayton 528244d93782SGreg Clayton case '.': 528344d93782SGreg Clayton case KEY_NPAGE: 528444d93782SGreg Clayton // Page down key 528544d93782SGreg Clayton { 528644d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < num_lines) 528744d93782SGreg Clayton m_first_visible_line += num_visible_lines; 528844d93782SGreg Clayton else if (num_lines < num_visible_lines) 528944d93782SGreg Clayton m_first_visible_line = 0; 529044d93782SGreg Clayton else 529144d93782SGreg Clayton m_first_visible_line = num_lines - num_visible_lines; 529244d93782SGreg Clayton m_selected_line = m_first_visible_line; 529344d93782SGreg Clayton } 529444d93782SGreg Clayton return eKeyHandled; 529544d93782SGreg Clayton 529644d93782SGreg Clayton case KEY_UP: 529744d93782SGreg Clayton if (m_selected_line > 0) 529844d93782SGreg Clayton { 529944d93782SGreg Clayton m_selected_line--; 53003985c8c6SSaleem Abdulrasool if (static_cast<size_t>(m_first_visible_line) > m_selected_line) 530144d93782SGreg Clayton m_first_visible_line = m_selected_line; 530244d93782SGreg Clayton } 530344d93782SGreg Clayton return eKeyHandled; 530444d93782SGreg Clayton 530544d93782SGreg Clayton case KEY_DOWN: 530644d93782SGreg Clayton if (m_selected_line + 1 < num_lines) 530744d93782SGreg Clayton { 530844d93782SGreg Clayton m_selected_line++; 530944d93782SGreg Clayton if (m_first_visible_line + num_visible_lines < m_selected_line) 531044d93782SGreg Clayton m_first_visible_line++; 531144d93782SGreg Clayton } 531244d93782SGreg Clayton return eKeyHandled; 531344d93782SGreg Clayton 531444d93782SGreg Clayton case '\r': 531544d93782SGreg Clayton case '\n': 531644d93782SGreg Clayton case KEY_ENTER: 531744d93782SGreg Clayton // Set a breakpoint and run to the line using a one shot breakpoint 531844d93782SGreg Clayton if (GetNumSourceLines() > 0) 531944d93782SGreg Clayton { 532044d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 532144d93782SGreg Clayton if (exe_ctx.HasProcessScope() && exe_ctx.GetProcessRef().IsAlive()) 532244d93782SGreg Clayton { 532344d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL, // Don't limit the breakpoint to certain modules 532444d93782SGreg Clayton m_file_sp->GetFileSpec(), // Source file 532544d93782SGreg Clayton m_selected_line + 1, // Source line number (m_selected_line is zero based) 532644d93782SGreg Clayton eLazyBoolCalculate, // Check inlines using global setting 532744d93782SGreg Clayton eLazyBoolCalculate, // Skip prologue using global setting, 532844d93782SGreg Clayton false, // internal 5329*055ad9beSIlia K false, // request_hardware 5330*055ad9beSIlia K eLazyBoolCalculate); // move_to_nearest_code 533144d93782SGreg Clayton // Make breakpoint one shot 533244d93782SGreg Clayton bp_sp->GetOptions()->SetOneShot(true); 533344d93782SGreg Clayton exe_ctx.GetProcessRef().Resume(); 533444d93782SGreg Clayton } 533544d93782SGreg Clayton } 533644d93782SGreg Clayton else if (m_selected_line < GetNumDisassemblyLines()) 533744d93782SGreg Clayton { 533844d93782SGreg Clayton const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get(); 533944d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 534044d93782SGreg Clayton if (exe_ctx.HasTargetScope()) 534144d93782SGreg Clayton { 534244d93782SGreg Clayton Address addr = inst->GetAddress(); 534344d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr, // lldb_private::Address 534444d93782SGreg Clayton false, // internal 534544d93782SGreg Clayton false); // request_hardware 534644d93782SGreg Clayton // Make breakpoint one shot 534744d93782SGreg Clayton bp_sp->GetOptions()->SetOneShot(true); 534844d93782SGreg Clayton exe_ctx.GetProcessRef().Resume(); 534944d93782SGreg Clayton } 535044d93782SGreg Clayton } 535144d93782SGreg Clayton return eKeyHandled; 535244d93782SGreg Clayton 535344d93782SGreg Clayton case 'b': // 'b' == toggle breakpoint on currently selected line 535444d93782SGreg Clayton if (m_selected_line < GetNumSourceLines()) 535544d93782SGreg Clayton { 535644d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 535744d93782SGreg Clayton if (exe_ctx.HasTargetScope()) 535844d93782SGreg Clayton { 535944d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (NULL, // Don't limit the breakpoint to certain modules 536044d93782SGreg Clayton m_file_sp->GetFileSpec(), // Source file 536144d93782SGreg Clayton m_selected_line + 1, // Source line number (m_selected_line is zero based) 536244d93782SGreg Clayton eLazyBoolCalculate, // Check inlines using global setting 536344d93782SGreg Clayton eLazyBoolCalculate, // Skip prologue using global setting, 536444d93782SGreg Clayton false, // internal 5365*055ad9beSIlia K false, // request_hardware 5366*055ad9beSIlia K eLazyBoolCalculate); // move_to_nearest_code 536744d93782SGreg Clayton } 536844d93782SGreg Clayton } 536944d93782SGreg Clayton else if (m_selected_line < GetNumDisassemblyLines()) 537044d93782SGreg Clayton { 537144d93782SGreg Clayton const Instruction *inst = m_disassembly_sp->GetInstructionList().GetInstructionAtIndex(m_selected_line).get(); 537244d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 537344d93782SGreg Clayton if (exe_ctx.HasTargetScope()) 537444d93782SGreg Clayton { 537544d93782SGreg Clayton Address addr = inst->GetAddress(); 537644d93782SGreg Clayton BreakpointSP bp_sp = exe_ctx.GetTargetRef().CreateBreakpoint (addr, // lldb_private::Address 537744d93782SGreg Clayton false, // internal 537844d93782SGreg Clayton false); // request_hardware 537944d93782SGreg Clayton } 538044d93782SGreg Clayton } 538144d93782SGreg Clayton return eKeyHandled; 538244d93782SGreg Clayton 538344d93782SGreg Clayton case 'd': // 'd' == detach and let run 538444d93782SGreg Clayton case 'D': // 'D' == detach and keep stopped 538544d93782SGreg Clayton { 538644d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 538744d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 538844d93782SGreg Clayton exe_ctx.GetProcessRef().Detach(c == 'D'); 538944d93782SGreg Clayton } 539044d93782SGreg Clayton return eKeyHandled; 539144d93782SGreg Clayton 539244d93782SGreg Clayton case 'k': 539344d93782SGreg Clayton // 'k' == kill 539444d93782SGreg Clayton { 539544d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 539644d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 5397ede3193bSJason Molenda exe_ctx.GetProcessRef().Destroy(false); 539844d93782SGreg Clayton } 539944d93782SGreg Clayton return eKeyHandled; 540044d93782SGreg Clayton 540144d93782SGreg Clayton case 'c': 540244d93782SGreg Clayton // 'c' == continue 540344d93782SGreg Clayton { 540444d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 540544d93782SGreg Clayton if (exe_ctx.HasProcessScope()) 540644d93782SGreg Clayton exe_ctx.GetProcessRef().Resume(); 540744d93782SGreg Clayton } 540844d93782SGreg Clayton return eKeyHandled; 540944d93782SGreg Clayton 541044d93782SGreg Clayton case 'o': 541144d93782SGreg Clayton // 'o' == step out 541244d93782SGreg Clayton { 541344d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 541444d93782SGreg Clayton if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true)) 541544d93782SGreg Clayton { 541644d93782SGreg Clayton exe_ctx.GetThreadRef().StepOut(); 541744d93782SGreg Clayton } 541844d93782SGreg Clayton } 541944d93782SGreg Clayton return eKeyHandled; 542044d93782SGreg Clayton case 'n': // 'n' == step over 542144d93782SGreg Clayton case 'N': // 'N' == step over instruction 542244d93782SGreg Clayton { 542344d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 542444d93782SGreg Clayton if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true)) 542544d93782SGreg Clayton { 542644d93782SGreg Clayton bool source_step = (c == 'n'); 542744d93782SGreg Clayton exe_ctx.GetThreadRef().StepOver(source_step); 542844d93782SGreg Clayton } 542944d93782SGreg Clayton } 543044d93782SGreg Clayton return eKeyHandled; 543144d93782SGreg Clayton case 's': // 's' == step into 543244d93782SGreg Clayton case 'S': // 'S' == step into instruction 543344d93782SGreg Clayton { 543444d93782SGreg Clayton ExecutionContext exe_ctx = m_debugger.GetCommandInterpreter().GetExecutionContext(); 543544d93782SGreg Clayton if (exe_ctx.HasThreadScope() && StateIsStoppedState (exe_ctx.GetProcessRef().GetState(), true)) 543644d93782SGreg Clayton { 543744d93782SGreg Clayton bool source_step = (c == 's'); 54384b4b2478SJim Ingham exe_ctx.GetThreadRef().StepIn(source_step); 543944d93782SGreg Clayton } 544044d93782SGreg Clayton } 544144d93782SGreg Clayton return eKeyHandled; 544244d93782SGreg Clayton 544344d93782SGreg Clayton case 'h': 544444d93782SGreg Clayton window.CreateHelpSubwindow (); 544544d93782SGreg Clayton return eKeyHandled; 544644d93782SGreg Clayton 544744d93782SGreg Clayton default: 544844d93782SGreg Clayton break; 544944d93782SGreg Clayton } 545044d93782SGreg Clayton return eKeyNotHandled; 545144d93782SGreg Clayton } 545244d93782SGreg Clayton 545344d93782SGreg Clayton protected: 545444d93782SGreg Clayton typedef std::set<uint32_t> BreakpointLines; 545544d93782SGreg Clayton typedef std::set<lldb::addr_t> BreakpointAddrs; 545644d93782SGreg Clayton 545744d93782SGreg Clayton Debugger &m_debugger; 545844d93782SGreg Clayton SymbolContext m_sc; 545944d93782SGreg Clayton SourceManager::FileSP m_file_sp; 546044d93782SGreg Clayton SymbolContextScope *m_disassembly_scope; 546144d93782SGreg Clayton lldb::DisassemblerSP m_disassembly_sp; 546244d93782SGreg Clayton AddressRange m_disassembly_range; 5463ec990867SGreg Clayton StreamString m_title; 546444d93782SGreg Clayton lldb::user_id_t m_tid; 546544d93782SGreg Clayton char m_line_format[8]; 546644d93782SGreg Clayton int m_line_width; 546744d93782SGreg Clayton uint32_t m_selected_line; // The selected line 546844d93782SGreg Clayton uint32_t m_pc_line; // The line with the PC 546944d93782SGreg Clayton uint32_t m_stop_id; 547044d93782SGreg Clayton uint32_t m_frame_idx; 547144d93782SGreg Clayton int m_first_visible_line; 547244d93782SGreg Clayton int m_min_x; 547344d93782SGreg Clayton int m_min_y; 547444d93782SGreg Clayton int m_max_x; 547544d93782SGreg Clayton int m_max_y; 547644d93782SGreg Clayton 547744d93782SGreg Clayton }; 547844d93782SGreg Clayton 547944d93782SGreg Clayton DisplayOptions ValueObjectListDelegate::g_options = { true }; 548044d93782SGreg Clayton 548144d93782SGreg Clayton IOHandlerCursesGUI::IOHandlerCursesGUI (Debugger &debugger) : 5482e30f11d9SKate Stone IOHandler (debugger, IOHandler::Type::Curses) 548344d93782SGreg Clayton { 548444d93782SGreg Clayton } 548544d93782SGreg Clayton 548644d93782SGreg Clayton void 548744d93782SGreg Clayton IOHandlerCursesGUI::Activate () 548844d93782SGreg Clayton { 548944d93782SGreg Clayton IOHandler::Activate(); 549044d93782SGreg Clayton if (!m_app_ap) 549144d93782SGreg Clayton { 549244d93782SGreg Clayton m_app_ap.reset (new Application (GetInputFILE(), GetOutputFILE())); 549344d93782SGreg Clayton 549444d93782SGreg Clayton 549544d93782SGreg Clayton // This is both a window and a menu delegate 549644d93782SGreg Clayton std::shared_ptr<ApplicationDelegate> app_delegate_sp(new ApplicationDelegate(*m_app_ap, m_debugger)); 549744d93782SGreg Clayton 549844d93782SGreg Clayton MenuDelegateSP app_menu_delegate_sp = std::static_pointer_cast<MenuDelegate>(app_delegate_sp); 549944d93782SGreg Clayton MenuSP lldb_menu_sp(new Menu("LLDB" , "F1", KEY_F(1), ApplicationDelegate::eMenuID_LLDB)); 550044d93782SGreg Clayton MenuSP exit_menuitem_sp(new Menu("Exit", NULL, 'x', ApplicationDelegate::eMenuID_LLDBExit)); 550144d93782SGreg Clayton exit_menuitem_sp->SetCannedResult(MenuActionResult::Quit); 550244d93782SGreg Clayton lldb_menu_sp->AddSubmenu (MenuSP (new Menu("About LLDB", NULL, 'a', ApplicationDelegate::eMenuID_LLDBAbout))); 550344d93782SGreg Clayton lldb_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator))); 550444d93782SGreg Clayton lldb_menu_sp->AddSubmenu (exit_menuitem_sp); 550544d93782SGreg Clayton 550644d93782SGreg Clayton MenuSP target_menu_sp(new Menu("Target" ,"F2", KEY_F(2), ApplicationDelegate::eMenuID_Target)); 550744d93782SGreg Clayton target_menu_sp->AddSubmenu (MenuSP (new Menu("Create", NULL, 'c', ApplicationDelegate::eMenuID_TargetCreate))); 550844d93782SGreg Clayton target_menu_sp->AddSubmenu (MenuSP (new Menu("Delete", NULL, 'd', ApplicationDelegate::eMenuID_TargetDelete))); 550944d93782SGreg Clayton 551044d93782SGreg Clayton MenuSP process_menu_sp(new Menu("Process", "F3", KEY_F(3), ApplicationDelegate::eMenuID_Process)); 551144d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Attach" , NULL, 'a', ApplicationDelegate::eMenuID_ProcessAttach))); 551244d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Detach" , NULL, 'd', ApplicationDelegate::eMenuID_ProcessDetach))); 551344d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Launch" , NULL, 'l', ApplicationDelegate::eMenuID_ProcessLaunch))); 551444d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu(Menu::Type::Separator))); 551544d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Continue", NULL, 'c', ApplicationDelegate::eMenuID_ProcessContinue))); 551644d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Halt" , NULL, 'h', ApplicationDelegate::eMenuID_ProcessHalt))); 551744d93782SGreg Clayton process_menu_sp->AddSubmenu (MenuSP (new Menu("Kill" , NULL, 'k', ApplicationDelegate::eMenuID_ProcessKill))); 551844d93782SGreg Clayton 551944d93782SGreg Clayton MenuSP thread_menu_sp(new Menu("Thread", "F4", KEY_F(4), ApplicationDelegate::eMenuID_Thread)); 552044d93782SGreg Clayton thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step In" , NULL, 'i', ApplicationDelegate::eMenuID_ThreadStepIn))); 552144d93782SGreg Clayton thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Over", NULL, 'v', ApplicationDelegate::eMenuID_ThreadStepOver))); 552244d93782SGreg Clayton thread_menu_sp->AddSubmenu (MenuSP (new Menu("Step Out" , NULL, 'o', ApplicationDelegate::eMenuID_ThreadStepOut))); 552344d93782SGreg Clayton 552444d93782SGreg Clayton MenuSP view_menu_sp(new Menu("View", "F5", KEY_F(5), ApplicationDelegate::eMenuID_View)); 552544d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Backtrace", NULL, 'b', ApplicationDelegate::eMenuID_ViewBacktrace))); 552644d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Registers", NULL, 'r', ApplicationDelegate::eMenuID_ViewRegisters))); 552744d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Source" , NULL, 's', ApplicationDelegate::eMenuID_ViewSource))); 552844d93782SGreg Clayton view_menu_sp->AddSubmenu (MenuSP (new Menu("Variables", NULL, 'v', ApplicationDelegate::eMenuID_ViewVariables))); 552944d93782SGreg Clayton 553044d93782SGreg Clayton MenuSP help_menu_sp(new Menu("Help", "F6", KEY_F(6), ApplicationDelegate::eMenuID_Help)); 553144d93782SGreg Clayton help_menu_sp->AddSubmenu (MenuSP (new Menu("GUI Help", NULL, 'g', ApplicationDelegate::eMenuID_HelpGUIHelp))); 553244d93782SGreg Clayton 553344d93782SGreg Clayton m_app_ap->Initialize(); 553444d93782SGreg Clayton WindowSP &main_window_sp = m_app_ap->GetMainWindow(); 553544d93782SGreg Clayton 553644d93782SGreg Clayton MenuSP menubar_sp(new Menu(Menu::Type::Bar)); 553744d93782SGreg Clayton menubar_sp->AddSubmenu (lldb_menu_sp); 553844d93782SGreg Clayton menubar_sp->AddSubmenu (target_menu_sp); 553944d93782SGreg Clayton menubar_sp->AddSubmenu (process_menu_sp); 554044d93782SGreg Clayton menubar_sp->AddSubmenu (thread_menu_sp); 554144d93782SGreg Clayton menubar_sp->AddSubmenu (view_menu_sp); 554244d93782SGreg Clayton menubar_sp->AddSubmenu (help_menu_sp); 554344d93782SGreg Clayton menubar_sp->SetDelegate(app_menu_delegate_sp); 554444d93782SGreg Clayton 554544d93782SGreg Clayton Rect content_bounds = main_window_sp->GetFrame(); 554644d93782SGreg Clayton Rect menubar_bounds = content_bounds.MakeMenuBar(); 554744d93782SGreg Clayton Rect status_bounds = content_bounds.MakeStatusBar(); 554844d93782SGreg Clayton Rect source_bounds; 554944d93782SGreg Clayton Rect variables_bounds; 555044d93782SGreg Clayton Rect threads_bounds; 555144d93782SGreg Clayton Rect source_variables_bounds; 555244d93782SGreg Clayton content_bounds.VerticalSplitPercentage(0.80, source_variables_bounds, threads_bounds); 555344d93782SGreg Clayton source_variables_bounds.HorizontalSplitPercentage(0.70, source_bounds, variables_bounds); 555444d93782SGreg Clayton 555544d93782SGreg Clayton WindowSP menubar_window_sp = main_window_sp->CreateSubWindow("Menubar", menubar_bounds, false); 555644d93782SGreg Clayton // Let the menubar get keys if the active window doesn't handle the 555744d93782SGreg Clayton // keys that are typed so it can respond to menubar key presses. 555844d93782SGreg Clayton menubar_window_sp->SetCanBeActive(false); // Don't let the menubar become the active window 555944d93782SGreg Clayton menubar_window_sp->SetDelegate(menubar_sp); 556044d93782SGreg Clayton 556144d93782SGreg Clayton WindowSP source_window_sp (main_window_sp->CreateSubWindow("Source", 556244d93782SGreg Clayton source_bounds, 556344d93782SGreg Clayton true)); 556444d93782SGreg Clayton WindowSP variables_window_sp (main_window_sp->CreateSubWindow("Variables", 556544d93782SGreg Clayton variables_bounds, 556644d93782SGreg Clayton false)); 556744d93782SGreg Clayton WindowSP threads_window_sp (main_window_sp->CreateSubWindow("Threads", 556844d93782SGreg Clayton threads_bounds, 556944d93782SGreg Clayton false)); 557044d93782SGreg Clayton WindowSP status_window_sp (main_window_sp->CreateSubWindow("Status", 557144d93782SGreg Clayton status_bounds, 557244d93782SGreg Clayton false)); 557344d93782SGreg Clayton status_window_sp->SetCanBeActive(false); // Don't let the status bar become the active window 557444d93782SGreg Clayton main_window_sp->SetDelegate (std::static_pointer_cast<WindowDelegate>(app_delegate_sp)); 557544d93782SGreg Clayton source_window_sp->SetDelegate (WindowDelegateSP(new SourceFileWindowDelegate(m_debugger))); 557644d93782SGreg Clayton variables_window_sp->SetDelegate (WindowDelegateSP(new FrameVariablesWindowDelegate(m_debugger))); 5577ec990867SGreg Clayton TreeDelegateSP thread_delegate_sp (new ThreadsTreeDelegate(m_debugger)); 557844d93782SGreg Clayton threads_window_sp->SetDelegate (WindowDelegateSP(new TreeWindowDelegate(m_debugger, thread_delegate_sp))); 557944d93782SGreg Clayton status_window_sp->SetDelegate (WindowDelegateSP(new StatusBarWindowDelegate(m_debugger))); 558044d93782SGreg Clayton 55815fdb09bbSGreg Clayton // Show the main help window once the first time the curses GUI is launched 55825fdb09bbSGreg Clayton static bool g_showed_help = false; 55835fdb09bbSGreg Clayton if (!g_showed_help) 55845fdb09bbSGreg Clayton { 55855fdb09bbSGreg Clayton g_showed_help = true; 55865fdb09bbSGreg Clayton main_window_sp->CreateHelpSubwindow(); 55875fdb09bbSGreg Clayton } 55885fdb09bbSGreg Clayton 558944d93782SGreg Clayton init_pair (1, COLOR_WHITE , COLOR_BLUE ); 559044d93782SGreg Clayton init_pair (2, COLOR_BLACK , COLOR_WHITE ); 559144d93782SGreg Clayton init_pair (3, COLOR_MAGENTA , COLOR_WHITE ); 559244d93782SGreg Clayton init_pair (4, COLOR_MAGENTA , COLOR_BLACK ); 559344d93782SGreg Clayton init_pair (5, COLOR_RED , COLOR_BLACK ); 559444d93782SGreg Clayton 559544d93782SGreg Clayton } 559644d93782SGreg Clayton } 559744d93782SGreg Clayton 559844d93782SGreg Clayton void 559944d93782SGreg Clayton IOHandlerCursesGUI::Deactivate () 560044d93782SGreg Clayton { 560144d93782SGreg Clayton m_app_ap->Terminate(); 560244d93782SGreg Clayton } 560344d93782SGreg Clayton 560444d93782SGreg Clayton void 560544d93782SGreg Clayton IOHandlerCursesGUI::Run () 560644d93782SGreg Clayton { 560744d93782SGreg Clayton m_app_ap->Run(m_debugger); 560844d93782SGreg Clayton SetIsDone(true); 560944d93782SGreg Clayton } 561044d93782SGreg Clayton 561144d93782SGreg Clayton 561244d93782SGreg Clayton IOHandlerCursesGUI::~IOHandlerCursesGUI () 561344d93782SGreg Clayton { 561444d93782SGreg Clayton 561544d93782SGreg Clayton } 561644d93782SGreg Clayton 561744d93782SGreg Clayton void 561844d93782SGreg Clayton IOHandlerCursesGUI::Hide () 561944d93782SGreg Clayton { 562044d93782SGreg Clayton } 562144d93782SGreg Clayton 562244d93782SGreg Clayton 562344d93782SGreg Clayton void 562444d93782SGreg Clayton IOHandlerCursesGUI::Refresh () 562544d93782SGreg Clayton { 562644d93782SGreg Clayton } 562744d93782SGreg Clayton 5628e68f5d6bSGreg Clayton void 5629e68f5d6bSGreg Clayton IOHandlerCursesGUI::Cancel () 5630e68f5d6bSGreg Clayton { 5631e68f5d6bSGreg Clayton } 563244d93782SGreg Clayton 5633f0066ad0SGreg Clayton bool 563444d93782SGreg Clayton IOHandlerCursesGUI::Interrupt () 563544d93782SGreg Clayton { 5636f0066ad0SGreg Clayton return false; 563744d93782SGreg Clayton } 563844d93782SGreg Clayton 563944d93782SGreg Clayton 564044d93782SGreg Clayton void 564144d93782SGreg Clayton IOHandlerCursesGUI::GotEOF() 564244d93782SGreg Clayton { 564344d93782SGreg Clayton } 564444d93782SGreg Clayton 5645914b8d98SDeepak Panickal #endif // #ifndef LLDB_DISABLE_CURSES 5646