/* client/console.cc This file is part of the Osirion project and is distributed under the terms and conditions of the GNU General Public License version 2 */ #include "auxiliary/functions.h" #include "core/core.h" #include "filesystem/filesystem.h" #include "render/render.h" #include "render/textures.h" #include "client/console.h" #include "client/video.h" #include "client/keyboard.h" #include #include #include namespace client { namespace console { //--- private definition ------------------------------------------ /// private client console implementation class Console : public sys::ConsoleInterface { public: /// stream to send normal messages too virtual std::ostream & messagestream(); /// stream to send warning messages too virtual std::ostream & warningstream(); /// stream to send error messages too virtual std::ostream & errorstream(); /// stream to send debug messages too virtual std::ostream & debugstream(); /// flush buffered messages virtual void flush(); /// console text buffer std::stringstream buffer; }; // private client console object Console console; // console text data std::deque text; // input history std::deque history; std::deque::reverse_iterator history_pos; size_t input_pos = 0; // console visibility bool console_visible = false; size_t console_scroll = 0; // notifications size_t notify_pos = 0; std::string notify_text[MAXNOTIFYLINES]; float notify_time[MAXNOTIFYLINES]; //--- engine functions -------------------------------------------- void func_con_toggle(std::string const &args) { std::istringstream argstream(args); int i; if (argstream >> i) { if (i) console_visible = false; else console_visible = true; } toggle(); } //--- public ------------------------------------------------------ void init() { con_print << "^BInitializing console..." << std::endl; console_visible = false; // add engine functions core::Func::add("con_toggle", (core::FuncPtr) func_con_toggle); text.clear(); console_scroll = 0; history.clear(); history.push_back(""); history_pos = history.rbegin(); input_pos = 0; load_history(); } void shutdown() { con_print << "^BShutting down console..." << std::endl; save_history(); // remove engine functions //core::Func::remove("con_toggle"); text.clear(); console_scroll = 0; history.clear(); input_pos = 0; } void clear_notify() { for (size_t i=0; i < MAXNOTIFYLINES; i++) notify_time[i] = 0; } void draw_notify() { using namespace render; // draw notifications Text::setcolor('N'); size_t width = (size_t) ((video::width - 8) / Text::fontwidth()); size_t n = notify_pos % MAXNOTIFYLINES; float h = 4 + 2*Text::fontheight(); for (size_t l = 0; l < MAXNOTIFYLINES; l++) { if (notify_text[n].size() > 2 && notify_time[n] + 4 > core::application()->time()) { std::string linedata(notify_text[n]); linedata += '\n'; std::string word; size_t word_length = 0; std::string line; size_t line_length = 0; const char *c = linedata.c_str(); char pen = 'N'; while (*c) { // color code if (aux::is_color_code(c)) { c++; pen = *c; word += '^'; word += pen; } // new word, wrap if necessary else if ((*c == '\n' ) || ( *c == ' ')) { if (line_length + word_length > width) { if (line.size()) { Text::draw(4, h, line); h += Text::fontheight(); line.clear(); line += '^'; line += pen; line_length = 0; } } line.append(word); line_length += word_length; word.clear(); word_length = 0; // new line if (*c == '\n' ) { Text::draw(4, h, line); h += Text::fontheight(); line.clear(); line_length = 0; // new word } else if (*c == ' ' ) { line += ' '; line_length++; } } // new character else { word += *c; word_length++; if (word_length == width) { if (line.size()) { Text::draw(4, h, line); h += Text::fontheight(); line.clear(); line += '^'; line += pen; line_length = 0; } line.append(word); line_length = word_length; word.clear(); word_length = 0; } } c++; } } n = (n+1) % MAXNOTIFYLINES; } } void draw_console() { using namespace render; float con_height = 0.70f; // draw version below the bottom of the console std::string version(core::name()); version += ' '; version.append(core::version()); gl::color(0.0f, 1.0f, 0.0f, 0.5f); Text::draw(video::width-Text::fontwidth()*(version.size()+1), video::height*con_height-Text::fontheight()-4, version); gl::disable(GL_TEXTURE_2D); // draw the transparent console background gl::color(1.0f, 1.0f, 1.0f, 0.02f); gl::begin(gl::Quads); gl::vertex(0.0f, 0.0f, 0.0f); gl::vertex(video::width, 0.0f,0.0f); gl::vertex(video::width,video::height*con_height,0.0f); gl::vertex(0,video::height*con_height,0.0f); gl::end(); // draw the console text if (console_scroll > text.size()) console_scroll = text.size(); gl::enable(GL_TEXTURE_2D); size_t height = (size_t) (video::height * con_height / Text::fontheight()) -1; size_t width = (size_t) (video::width / Text::fontwidth()) -2; size_t bottom = text.size() - console_scroll; size_t current_line = 0; std::deque lines; for (std::deque::iterator it = text.begin(); it != text.end() && current_line < bottom; it++) { if (current_line >= bottom - height) { std::string linedata(*it); linedata += '\n'; std::string word; size_t word_length = 0; std::string line; size_t line_length = 0; const char *c = linedata.c_str(); char pen = 'N'; while (*c) { // color code if (aux::is_color_code(c)) { c++; pen = *c; word += '^'; word += pen; } // new word, wrap if necessary else if ((*c == '\n' ) || ( *c == ' ')) { if (line_length + word_length > width) { if (line.size()) { lines.push_back(line); line.clear(); line += '^'; line += pen; line_length = 0; } } line.append(word); line_length += word_length; word.clear(); word_length = 0; // new line if (*c == '\n' ) { lines.push_back(line); line.clear(); line_length = 0; // new word } else if (*c == ' ' ) { line += ' '; line_length++; } } // new character else { word += *c; word_length++; if (word_length == width) { if (line.size()) { lines.push_back(line); line.clear(); line += '^'; line += pen; line_length = 0; } line.append(word); line_length = word_length; word.clear(); word_length = 0; } } c++; } } current_line++; } float y = video::height*con_height-2*Text::fontheight()-4; Text::setcolor('N'); for (std::deque::reverse_iterator rit = lines.rbegin(); (y >= 4) && (rit != lines.rend()); ++rit) { Text::draw(4, y, (*rit)); y -= Text::fontheight(); } // draw the console input size_t draw_pos = 0; y = video::height*con_height - Text::fontheight() - 4; while (input_pos - draw_pos > width) draw_pos += 2; Text::draw(4, y, "^B>"); Text::draw(4+Text::fontwidth(), y , (*history_pos).substr(draw_pos, width-1)); // draw cursor if ((core::application()->time() - ::floorf(core::application()->time())) < 0.5f) { std::string cursor("^B_"); Text::draw(4+Text::fontwidth()*(input_pos-draw_pos+1), y , cursor); } } void draw(){ if (visible()) draw_console(); else draw_notify(); } void flush() { char line[MAXCMDSIZE]; while(console.buffer.getline(line, MAXCMDSIZE-1)) { while (text.size() >= MAXCONLINES) { text.pop_front(); } text.push_back(std::string(line)); // save notification notify_text[notify_pos] = line; notify_time[notify_pos] = core::application()->time(); notify_pos = (notify_pos+1) % MAXNOTIFYLINES; // print to stdout std::cout << line << std::endl; } console.buffer.clear(); } void toggle() { console_visible = !console_visible; if (console_visible) { console_scroll = 0; input_pos = 0; history_pos = history.rbegin(); (*history_pos).clear(); SDL_WM_GrabInput(SDL_GRAB_OFF); SDL_ShowCursor(SDL_ENABLE); } else { SDL_WM_GrabInput(SDL_GRAB_ON); SDL_ShowCursor(SDL_DISABLE); clear_notify(); } setkeyboardmode(console::visible()); } void keypressed(int key) { std::deque::reverse_iterator upit; switch( key ) { case SDLK_TAB: core::CommandBuffer::complete( (*history_pos), input_pos); break; case SDLK_RETURN: if ((*history_pos).size()) { // store input into history while (history.size() >= MAXHISTOLINES) { history.pop_front(); } core::cmd() << (*history_pos) << std::endl; con_print << "^B>" << (*history_pos) << std::endl; (*history.rbegin()) = (*history_pos); history.push_back(""); history_pos = history.rbegin(); input_pos = 0; } break; case SDLK_UP: upit = history_pos; ++upit; if (upit != history.rend()) { history_pos = upit; input_pos = (*history_pos).size(); } break; case SDLK_DOWN: if (history_pos != history.rbegin()) { --history_pos; input_pos = (*history_pos).size(); } break; case SDLK_HOME: input_pos = 0; break; case SDLK_END: input_pos = (*history_pos).size(); break; case SDLK_LEFT: if (input_pos > 0) input_pos--; break; case SDLK_RIGHT: if (input_pos < (*history_pos).size()) input_pos++; break; case SDLK_BACKSPACE: if ((*history_pos).size() && input_pos) { (*history_pos).erase(input_pos-1, 1); input_pos--; } break; case SDLK_PAGEUP: console_scroll +=5; if (console_scroll > text.size()) console_scroll = text.size(); break; case SDLK_PAGEDOWN: if (console_scroll > 5) console_scroll -=5; else console_scroll = 0; break; default: if ((key >= 32 ) && (key <175)) { if (input_pos == (*history_pos).size()) (*history_pos) += (char)key; else (*history_pos).insert(input_pos, 1, (char)key); input_pos++; } break; } } bool visible() { return console_visible; } void save_history() { if (history.size() <= 1) return; std::string filename(filesystem::writedir); filename.append("history.txt"); std::ofstream ofs(filename.c_str()); if (!ofs.is_open()) { con_warn << "Could not write " << filename << std::endl; return; } std::deque::iterator it; size_t l = 1; for (it = history.begin(); it != history.end(); it++) { if (l < history.size()) ofs << (*it) << std::endl; l++; } ofs.close(); } void load_history() { std::string filename(filesystem::writedir); filename.append("history.txt"); std::ifstream ifs(filename.c_str(), std::ifstream::in); if (!ifs.is_open()) { con_warn << "Could not read " << filename << std::endl; return; } history.clear(); char line[MAXCMDSIZE]; while (ifs.getline(line, MAXCMDSIZE-1)) { history.push_back(line); } ifs.close(); history.push_back(""); history_pos = history.rbegin(); input_pos = 0; } //--- private ----------------------------------------------------- void Console::flush() { console::flush(); } std::ostream & Console::messagestream() { return (buffer << "^N"); } std::ostream & Console::warningstream() { return (buffer << "^W"); } std::ostream & Console::errorstream() { return (buffer << "^R"); } std::ostream & Console::debugstream() { return (buffer << "^D"); } } // namespace console } // namespace client