/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /*************************************************************************** * lineedit.cc * * Sun Oct 9 13:01:52 CEST 2011 * Copyright 2011 Bent Bisballe Nyeng * deva@aasimon.org ****************************************************************************/ /* * This file is part of DrumGizmo. * * DrumGizmo is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * DrumGizmo is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with DrumGizmo; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ #include "lineedit.h" #include #include "window.h" #include #include #define BORDER 10 GUI::LineEdit::LineEdit(Widget *parent) : GUI::Widget(parent) { pos = 0; offsetpos = 0; setReadOnly(false); box.topLeft = new Image(":widget_tl.png"); box.top = new Image(":widget_t.png"); box.topRight = new Image(":widget_tr.png"); box.left = new Image(":widget_l.png"); box.right = new Image(":widget_r.png"); box.bottomLeft = new Image(":widget_bl.png"); box.bottom = new Image(":widget_b.png"); box.bottomRight = new Image(":widget_br.png"); box.center = new Image(":widget_c.png"); handler = NULL; } void GUI::LineEdit::registerEnterPressedHandler(void (*handler)(void *), void *ptr) { this->handler = handler; this->ptr = ptr; } void GUI::LineEdit::setReadOnly(bool ro) { readonly = ro; } bool GUI::LineEdit::readOnly() { return readonly; } void GUI::LineEdit::setText(std::string text) { _text = text; pos = text.size(); // if(_text.size() < pos) pos = text.size(); repaintEvent(NULL); textChanged(); } std::string GUI::LineEdit::text() { return _text; } void GUI::LineEdit::buttonEvent(ButtonEvent *e) { if(readOnly()) return; if(e->direction == 1) { for(int i = 0; i < (int)_visibletext.length(); i++) { if(e->x < (int)(font.textWidth(_visibletext.substr(0, i)) + BORDER)) { printf("i, Offset: %d, %d\n", i, offsetpos); pos = i + offsetpos; break; } } // if(e->x >= (int)(font.textWidth(_visibletext) + BORDER)) pos = _visibletext.length(); repaintEvent(NULL); } } void GUI::LineEdit::keyEvent(GUI::KeyEvent *e) { if(readOnly()) return; DEBUG(lineedit, "Handling keyevent with text '%s'\n", _text.c_str()); bool change = false; if(e->direction == -1) { if(e->keycode == GUI::KeyEvent::KEY_LEFT) { if(pos) pos--; if(offsetpos >= pos) walkstate = WALK_LEFT; // else walkstate = NOOP; } else if(e->keycode == GUI::KeyEvent::KEY_HOME) { pos = 0; } else if(e->keycode == GUI::KeyEvent::KEY_END) { pos = _text.length(); } else if(e->keycode == GUI::KeyEvent::KEY_RIGHT) { if(pos < _text.length()) pos++; if(offsetpos + _visibletext.length() <= pos) walkstate = WALK_RIGHT; // else walkstate = NOOP; } else if(e->keycode == GUI::KeyEvent::KEY_DELETE) { if(pos < _text.length()) { std::string t = _text.substr(0, pos); t += _text.substr(pos + 1, std::string::npos); _text = t; change = true; } } else if(e->keycode == GUI::KeyEvent::KEY_BACKSPACE) { if(pos > 0) { std::string t = _text.substr(0, pos - 1); t += _text.substr(pos, std::string::npos); _text = t; pos--; change = true; } } else if(e->keycode == GUI::KeyEvent::KEY_CHARACTER) { std::string pre = _text.substr(0, pos); std::string post = _text.substr(pos, std::string::npos); _text = pre + e->text + post; change = true; pos++; } else if(e->keycode == GUI::KeyEvent::KEY_ENTER) { if(handler) handler(ptr); } repaintEvent(NULL); } if(change) textChanged(); } void GUI::LineEdit::repaintEvent(GUI::RepaintEvent *e) { Painter p(this); p.clear(); int w = width(); int h = height(); if(w == 0 || h == 0) return; p.drawBox(0, 0, &box, w, h); p.setColour(GUI::Colour(183.0/255.0, 219.0/255.0 , 255.0/255.0, 1)); if(walkstate == WALK_LEFT) { _visibletext = _text.substr(pos, std::string::npos); offsetpos = pos; } else if(walkstate == WALK_RIGHT) { int d = (offsetpos < _text.length()) ? 1 : 0; _visibletext = _text.substr(offsetpos + d); offsetpos = offsetpos + d; } else { _visibletext = _text; offsetpos = 0; } while(true) { int textwidth = font.textWidth(_visibletext); if(textwidth > w - BORDER - 4 + 3) { if(walkstate == WALK_LEFT) { _visibletext = _visibletext.substr(0, _visibletext.length()-1); } else if(walkstate == WALK_RIGHT) { _visibletext = _visibletext.substr(0, _visibletext.length()-1); } else { if(offsetpos < pos) { _visibletext = _visibletext.substr(1); offsetpos++; } else { _visibletext = _visibletext.substr(0, _visibletext.length() - 1); } } } else { DEBUG(lienedit, "Full text: '%s'\n", _text.c_str()); DEBUG(lineedit, "Drawing text in lineedit '%s'\n", _visibletext.c_str()); DEBUG(lineedit, "Offset, pos: %d, %d\n", offsetpos, pos); break; } } walkstate = NOOP; p.drawText(BORDER - 4 + 3, height()/2+5 + 1 + 1 + 1, font, _visibletext); if(readOnly()) return; if(hasKeyboardFocus()) { // size_t px = font.textWidth(_text.substr(0, pos - offsetpos)); size_t px = font.textWidth(_visibletext.substr(0, pos - offsetpos)); //p.setColour(Colour(0.8)); p.drawLine(px + BORDER - 1 - 4 + 3, 6, px + BORDER - 1 - 4 + 3, height() - 7); } } #ifdef TEST_LINEEDIT //Additional dependency files //deps: //Required cflags (autoconf vars may be used) //cflags: //Required link options (autoconf vars may be used) //libs: #include "test.h" TEST_BEGIN; // TODO: Put some testcode here (see test.h for usable macros). TEST_END; #endif/*TEST_LINEEDIT*/