X-Git-Url: http://code.vuplus.com/gitweb/?p=vuplus_dvbapp;a=blobdiff_plain;f=lib%2Fgui%2Felistbox.cpp;h=f241e6f630cc428c2a9d80ded4d9f75189f5b2c2;hp=c2af3d3287e588967760f9961446d4fe099e3ad5;hb=74f5884fdc5a23465cf40c27b5c069ff7d882746;hpb=8ade23537a682d4b0c9709d533b25702bde2ee23 diff --git a/lib/gui/elistbox.cpp b/lib/gui/elistbox.cpp index c2af3d3..f241e6f 100644 --- a/lib/gui/elistbox.cpp +++ b/lib/gui/elistbox.cpp @@ -1,222 +1,303 @@ #include +#include +#include +#include +#ifdef USE_LIBVUGLES2 +#include "vuplus_gles.h" +#endif -/* - The basic idea is to have an interface which gives all relevant list - processing functions, and can be used by the listbox to browse trough - the list. - - The listbox directly uses the implemented cursor. It tries hard to avoid - iterating trough the (possibly very large) list, so it should be O(1), - i.e. the performance should not be influenced by the size of the list. - - The list interface knows how to draw the current entry to a specified - offset. Different interfaces can be used to adapt different lists, - pre-filter lists on the fly etc. - - cursorSave/Restore is used to avoid re-iterating the list on redraw. - The current selection is always selected as cursor position, the - cursor is then positioned to the start, and then iterated. This gives - at most 2x m_items_per_page cursor movements per redraw, indepenent - of the size of the list. - - Although cursorSet is provided, it should be only used when there is no - other way, as it involves iterating trough the list. - */ - -class eListboxTestContent: public virtual iListboxContent +eListbox::eListbox(eWidget *parent) : + eWidget(parent), m_scrollbar_mode(showNever), m_prev_scrollbar_page(-1), + m_content_changed(false), m_enabled_wrap_around(false), m_top(0), m_selected(0), m_itemheight(25), + m_items_per_page(0), m_selection_enabled(1), m_scrollbar(NULL) { - DECLARE_REF; -public: - void cursorHome(); - void cursorEnd(); - int cursorMove(int count=1); - int cursorValid(); - int cursorSet(int n); - int cursorGet(); - - void cursorSave(); - void cursorRestore(); - int size(); - - RESULT connectItemChanged(const Slot0 &itemChanged, ePtr &connection); - - // void setOutputDevice ? (for allocating colors, ...) .. requires some work, though - void setSize(const eSize &size); - - /* the following functions always refer to the selected item */ - void paint(gPainter &painter, eWindowStyle &style, const ePoint &offset, int selected); -private: - int m_cursor, m_saved_cursor; - eSize m_size; -}; - -DEFINE_REF(eListboxTestContent); - -void eListboxTestContent::cursorHome() -{ - m_cursor = 0; -} - -void eListboxTestContent::cursorEnd() -{ - m_cursor = size(); -} - -int eListboxTestContent::cursorMove(int count) -{ - m_cursor += count; - - if (m_cursor < 0) - cursorHome(); - else if (m_cursor > size()) - cursorEnd(); - return 0; -} + memset(&m_style, 0, sizeof(m_style)); +// setContent(new eListboxStringContent()); -int eListboxTestContent::cursorValid() -{ - return m_cursor < size(); + ePtr ptr; + eActionMap::getInstance(ptr); + ptr->bindAction("ListboxActions", 0, 0, this); } -int eListboxTestContent::cursorSet(int n) +eListbox::~eListbox() { - m_cursor = n; + if (m_scrollbar) + delete m_scrollbar; - if (m_cursor < 0) - cursorHome(); - else if (m_cursor > size()) - cursorEnd(); - return 0; + ePtr ptr; + eActionMap::getInstance(ptr); + ptr->unbindAction(this, 0); } -int eListboxTestContent::cursorGet() +void eListbox::setScrollbarMode(int mode) { - return m_cursor; + m_scrollbar_mode = mode; + if (m_scrollbar) + { + if (m_scrollbar_mode == showNever) + { + delete m_scrollbar; + m_scrollbar=0; + } + } + else + { + m_scrollbar = new eSlider(this); + m_scrollbar->hide(); + m_scrollbar->setBorderWidth(1); + m_scrollbar->setOrientation(eSlider::orVertical); + m_scrollbar->setRange(0,100); + } } -void eListboxTestContent::cursorSave() +void eListbox::setWrapAround(bool state) { - m_saved_cursor = m_cursor; + m_enabled_wrap_around = state; } -void eListboxTestContent::cursorRestore() +void eListbox::setContent(iListboxContent *content) { - m_cursor = m_saved_cursor; + m_content = content; + if (content) + m_content->setListbox(this); + entryReset(); } -int eListboxTestContent::size() -{ - return 10; -} - -RESULT eListboxTestContent::connectItemChanged(const Slot0 &itemChanged, ePtr &connection) +bool eListbox::atBegin() { - return 0; + if (m_content && !m_selected) + return true; + return false; } -void eListboxTestContent::setSize(const eSize &size) +bool eListbox::atEnd() { - m_size = size; + if (m_content && m_content->size() == m_selected+1) + return true; + return false; } -void eListboxTestContent::paint(gPainter &painter, eWindowStyle &style, const ePoint &offset, int selected) +void eListbox::moveToEnd() { - ePtr fnt = new gFont("Arial", 14); - painter.clip(eRect(offset, m_size)); - style.setStyle(painter, selected ? eWindowStyle::styleListboxSelected : eWindowStyle::styleListboxNormal); - painter.clear(); - - if (cursorValid()) + if (!m_content) + return; + /* move to last existing one ("end" is already invalid) */ + m_content->cursorEnd(); m_content->cursorMove(-1); + /* current selection invisible? */ + if (m_top + m_items_per_page <= m_content->cursorGet()) { - painter.setFont(fnt); - char string[10]; - sprintf(string, "%d.)", m_cursor); - - ePoint text_offset = offset + (selected ? ePoint(2, 2) : ePoint(1, 1)); - - painter.renderText(eRect(text_offset, m_size), string); - - if (selected) - style.drawFrame(painter, eRect(offset, m_size), eWindowStyle::frameListboxEntry); + int rest = m_content->size() % m_items_per_page; + if (rest) + m_top = m_content->cursorGet() - rest + 1; + else + m_top = m_content->cursorGet() - m_items_per_page + 1; + if (m_top < 0) + m_top = 0; } - - painter.clippop(); -} - -eListbox::eListbox(eWidget *parent): eWidget(parent) -{ - setContent(new eListboxTestContent()); - m_content->cursorHome(); - m_top = 0; - m_selected = 0; -} - -void eListbox::setContent(iListboxContent *content) -{ - m_content = content; } -void eListbox::moveSelection(int dir) +void eListbox::moveSelection(long dir) { + /* refuse to do anything without a valid list. */ + if (!m_content) + return; + /* if our list does not have one entry, don't do anything. */ + if (!m_items_per_page) + return; /* we need the old top/sel to see what we have to redraw */ int oldtop = m_top; int oldsel = m_selected; - /* first, move cursor */ +#ifdef USE_LIBVUGLES2 + m_dir = dir; +#endif switch (dir) { case moveUp: + { m_content->cursorMove(-1); + if (m_enabled_wrap_around && oldsel == m_content->cursorGet()) // must wrap around ? + moveToEnd(); break; + } case moveDown: m_content->cursorMove(1); - /* ok - we could have reached the end. we just go one back then. */ + /* ok - we could have reached the end. So we do wrap around. */ if (!m_content->cursorValid()) - m_content->cursorMove(-1); + { + if (m_enabled_wrap_around) + { + m_top = 0; + m_content->cursorHome(); + } + else + m_content->cursorMove(-1); + } + break; + case pageUp: + if (m_content->cursorGet() >= m_items_per_page) + { + m_content->cursorMove(-m_items_per_page); + m_top -= m_items_per_page; + if (m_top < 0) + m_top = 0; + } else + { + m_top = 0; + m_content->cursorHome(); + } break; case moveTop: m_content->cursorHome(); m_top = 0; /* align with top, speeds up process */ break; + case pageDown: + m_content->cursorMove(m_items_per_page); + if (m_content->cursorValid()) + break; + /* fall through */ case moveEnd: - /* move to last existing one ("end" is already invalid) */ - m_content->cursorEnd(); m_content->cursorMove(-1); - - m_top = m_content->cursorGet() - m_items_per_page + 1; - if (m_top < 0) - m_top = 0; + moveToEnd(); + break; + case justCheck: break; } + if (m_content->cursorValid() && !m_content->currentCursorSelectable()) + { + /* ok, our cursor position is valid (i.e. in list), but not selectable. */ + + /* when moving up, continue until we found a valid position. */ + if ((dir == moveUp) || (dir == pageDown)) + { + while (m_content->cursorGet()) + { + m_content->cursorMove(-1); + if (m_content->currentCursorSelectable()) + { + break; + } + } + } else + { + /* else move down */ + while (m_content->cursorValid()) + { + m_content->cursorMove(+1); + if (m_content->currentCursorSelectable()) + { + break; + } + } + + if (!m_content->cursorValid()) + m_content->cursorMove(-1); + } + + if (!m_content->currentCursorSelectable()) + m_content->cursorSet(oldsel); + } + /* note that we could be on an invalid cursor position, but we don't care. this only happens on empty lists, and should have almost no side effects. */ /* now, look wether the current selection is out of screen */ m_selected = m_content->cursorGet(); - if (m_selected < m_top) + while (m_selected < m_top) { m_top -= m_items_per_page; if (m_top < 0) m_top = 0; - } else if (m_selected >= m_top + m_items_per_page) - { - /* m_top should be always valid here as it's selected */ - m_top += m_items_per_page; } - + while (m_selected >= m_top + m_items_per_page) + /* m_top should be always valid here as it's selected */ + m_top += m_items_per_page; + + if (oldsel != m_selected) + /* emit */ selectionChanged(); + + updateScrollBar(); + if (m_top != oldtop) invalidate(); - else + else if (m_selected != oldsel) { - /* redraw the old and newly selected */ + /* redraw the old and newly selected */ gRegion inv = eRect(0, m_itemheight * (m_selected-m_top), size().width(), m_itemheight); inv |= eRect(0, m_itemheight * (oldsel-m_top), size().width(), m_itemheight); - invalidate(inv); } } +void eListbox::moveSelectionTo(int index) +{ + if (m_content) + { + m_content->cursorHome(); + m_content->cursorMove(index); + moveSelection(justCheck); + } +} + +int eListbox::getCurrentIndex() +{ + if (m_content && m_content->cursorValid()) + return m_content->cursorGet(); + return 0; +} + +void eListbox::updateScrollBar() +{ + if (!m_content || m_scrollbar_mode == showNever ) + return; + int entries = m_content->size(); + if (m_content_changed) + { + int width = size().width(); + int height = size().height(); + m_content_changed = false; + if (entries > m_items_per_page || m_scrollbar_mode == showAlways) + { + int sbarwidth=width/16; + if (sbarwidth < 18) + sbarwidth = 18; + if (sbarwidth > 22) + sbarwidth = 22; + m_scrollbar->move(ePoint(width-sbarwidth, 0)); + m_scrollbar->resize(eSize(sbarwidth, height)); + m_content->setSize(eSize(width-sbarwidth-5, m_itemheight)); + m_scrollbar->show(); + } + else + { + m_content->setSize(eSize(width, m_itemheight)); + m_scrollbar->hide(); + } + } + if (m_items_per_page && entries) + { + int curVisiblePage = m_top / m_items_per_page; + if (m_prev_scrollbar_page != curVisiblePage) + { + m_prev_scrollbar_page = curVisiblePage; + int pages = entries / m_items_per_page; + if ((pages*m_items_per_page) < entries) + ++pages; + int start=(m_top*100)/(pages*m_items_per_page); + int vis=(m_items_per_page*100+pages*m_items_per_page-1)/(pages*m_items_per_page); + if (vis < 3) + vis=3; + m_scrollbar->setStartEnd(start,start+vis); + } + } +} + +int eListbox::getEntryTop() +{ + return (m_selected - m_top) * m_itemheight; +} + int eListbox::event(int event, void *data, void *data2) { switch (event) @@ -225,8 +306,9 @@ int eListbox::event(int event, void *data, void *data2) { ePtr style; - assert(m_content); - recalcSize(); // move to event + if (!m_content) + return eWidget::event(event, data, data2); + ASSERT(m_content); getStyle(style); @@ -238,16 +320,58 @@ int eListbox::event(int event, void *data, void *data2) m_content->cursorSave(); m_content->cursorMove(m_top - m_selected); - for (int y = 0, i = 0; i < m_items_per_page; y += m_itemheight, ++i) + gRegion entryrect = eRect(0, 0, size().width(), m_itemheight); + const gRegion &paint_region = *(gRegion*)data; + + for (int y = 0, i = 0; i <= m_items_per_page; y += m_itemheight, ++i) { - m_content->paint(painter, *style, ePoint(0, y), m_selected == m_content->cursorGet()); + gRegion entry_clip_rect = paint_region & entryrect; + + if (!entry_clip_rect.empty()) + m_content->paint(painter, *style, ePoint(0, y), m_selected == m_content->cursorGet() && m_content->size() && m_selection_enabled); +#ifdef USE_LIBVUGLES2 + if (m_selected == m_content->cursorGet() && m_content->size() && m_selection_enabled) { + ePoint pos = getAbsolutePosition(); + painter.sendShowItem(m_dir, ePoint(pos.x(), pos.y() + y), eSize(m_scrollbar && m_scrollbar->isVisible() ? size().width() - m_scrollbar->size().width() : size().width(), m_itemheight)); + gles_set_animation_listbox_current(pos.x(), pos.y() + y, m_scrollbar && m_scrollbar->isVisible() ? size().width() - m_scrollbar->size().width() : size().width(), m_itemheight); + m_dir = justCheck; + } +#endif + + /* (we could clip with entry_clip_rect, but + this shouldn't change the behavior of any + well behaving content, so it would just + degrade performance without any gain.) */ + m_content->cursorMove(+1); + entryrect.moveBy(ePoint(0, m_itemheight)); } - + + // clear/repaint empty/unused space between scrollbar and listboxentrys + if (m_scrollbar && m_scrollbar->isVisible()) + { + style->setStyle(painter, eWindowStyle::styleListboxNormal); + painter.clip(eRect(m_scrollbar->position() - ePoint(5,0), eSize(5,m_scrollbar->size().height()))); + painter.clear(); + painter.clippop(); + } + m_content->cursorRestore(); - + return 0; } + + case evtChangedSize: + recalcSize(); + return eWidget::event(event, data, data2); + + case evtAction: + if (isVisible()) + { + moveSelection((long)data2); + return 1; + } + return 0; default: return eWidget::event(event, data, data2); } @@ -255,7 +379,171 @@ int eListbox::event(int event, void *data, void *data2) void eListbox::recalcSize() { - m_itemheight = 20; - m_content->setSize(eSize(size().width(), m_itemheight)); + m_content_changed=true; + m_prev_scrollbar_page=-1; + if (m_content) + m_content->setSize(eSize(size().width(), m_itemheight)); m_items_per_page = size().height() / m_itemheight; + + if (m_items_per_page < 0) /* TODO: whyever - our size could be invalid, or itemheigh could be wrongly specified. */ + m_items_per_page = 0; + + moveSelection(justCheck); +} + +void eListbox::setItemHeight(int h) +{ + if (h) + m_itemheight = h; + else + m_itemheight = 20; + recalcSize(); +} + +void eListbox::setSelectionEnable(int en) +{ + if (m_selection_enabled == en) + return; + m_selection_enabled = en; + entryChanged(m_selected); /* redraw current entry */ +} + +void eListbox::entryAdded(int index) +{ + if (m_content && (m_content->size() % m_items_per_page) == 1) + m_content_changed=true; + /* manage our local pointers. when the entry was added before the current position, we have to advance. */ + + /* we need to check <= - when the new entry has the (old) index of the cursor, the cursor was just moved down. */ + if (index <= m_selected) + ++m_selected; + if (index <= m_top) + ++m_top; + + /* we have to check wether our current cursor is gone out of the screen. */ + /* moveSelection will check for this case */ + moveSelection(justCheck); + + /* now, check if the new index is visible. */ + if ((m_top <= index) && (index < (m_top + m_items_per_page))) + { + /* todo, calc exact invalidation... */ + invalidate(); + } +} + +void eListbox::entryRemoved(int index) +{ + if (m_content && !(m_content->size() % m_items_per_page)) + m_content_changed=true; + + if (index == m_selected && m_content) + m_selected = m_content->cursorGet(); + + if (m_content && m_content->cursorGet() >= m_content->size()) + moveSelection(moveUp); + else + moveSelection(justCheck); + + if ((m_top <= index) && (index < (m_top + m_items_per_page))) + { + /* todo, calc exact invalidation... */ + invalidate(); + } +} + +void eListbox::entryChanged(int index) +{ + if ((m_top <= index) && (index < (m_top + m_items_per_page))) + { + gRegion inv = eRect(0, m_itemheight * (index-m_top), size().width(), m_itemheight); + invalidate(inv); + } +} + +void eListbox::entryReset(bool selectionHome) +{ + m_content_changed = true; + m_prev_scrollbar_page = -1; + int oldsel; + + if (selectionHome) + { + if (m_content) + m_content->cursorHome(); + m_top = 0; + m_selected = 0; + } + + if (m_content && (m_selected >= m_content->size())) + { + if (m_content->size()) + m_selected = m_content->size() - 1; + else + m_selected = 0; + m_content->cursorSet(m_selected); + } + + oldsel = m_selected; + moveSelection(justCheck); + /* if oldsel != m_selected, selectionChanged was already + emitted in moveSelection. we want it in any case, so otherwise, + emit it now. */ + if (oldsel == m_selected) + /* emit */ selectionChanged(); + invalidate(); +} + +void eListbox::setFont(gFont *font) +{ + m_style.m_font = font; +} + +void eListbox::setBackgroundColor(gRGB &col) +{ + m_style.m_background_color = col; + m_style.m_background_color_set = 1; +} + +void eListbox::setBackgroundColorSelected(gRGB &col) +{ + m_style.m_background_color_selected = col; + m_style.m_background_color_selected_set = 1; +} + +void eListbox::setForegroundColor(gRGB &col) +{ + m_style.m_foreground_color = col; + m_style.m_foreground_color_set = 1; +} + +void eListbox::setForegroundColorSelected(gRGB &col) +{ + m_style.m_foreground_color_selected = col; + m_style.m_foreground_color_selected_set = 1; +} + +void eListbox::setBackgroundPicture(ePtr &pm) +{ + m_style.m_background = pm; +} + +void eListbox::setSelectionPicture(ePtr &pm) +{ + m_style.m_selection = pm; +} + +void eListbox::invalidate(const gRegion ®ion) +{ + gRegion tmp(region); + if (m_content) + m_content->updateClip(tmp); + eWidget::invalidate(tmp); +} + +struct eListboxStyle *eListbox::getLocalStyle(void) +{ + /* transparency is set directly in the widget */ + m_style.m_transparent_background = isTransparent(); + return &m_style; }