/* * osdbase.c: Basic interface to the On Screen Display * * See the main source file 'vdr.c' for copyright information and * how to reach the author. * * $Id: osdbase.c 1.15 2005/01/07 16:16:41 kls Exp $ */ #include "osdbase.h" #include <string.h> #include "device.h" #include "i18n.h" #include "remote.h" #include "status.h" // --- cOsdItem -------------------------------------------------------------- cOsdItem::cOsdItem(eOSState State) { text = NULL; offset = -1; state = State; selectable = true; fresh = true; } cOsdItem::cOsdItem(const char *Text, eOSState State) { text = NULL; offset = -1; state = State; selectable = true; fresh = true; SetText(Text); } cOsdItem::~cOsdItem() { free(text); } void cOsdItem::SetText(const char *Text, bool Copy) { free(text); text = Copy ? strdup(Text) : (char *)Text; // text assumes ownership! } void cOsdItem::SetSelectable(bool Selectable) { selectable = Selectable; } void cOsdItem::SetFresh(bool Fresh) { fresh = Fresh; } eOSState cOsdItem::ProcessKey(eKeys Key) { return Key == kOk ? state : osUnknown; } // --- cOsdMenu -------------------------------------------------------------- cSkinDisplayMenu *cOsdMenu::displayMenu = NULL; int cOsdMenu::displayMenuCount = 0; int cOsdMenu::displayMenuItems = 0;//XXX dynamic??? cOsdMenu::cOsdMenu(const char *Title, int c0, int c1, int c2, int c3, int c4) { isMenu = true; digit = 0; hasHotkeys = false; title = NULL; SetTitle(Title); cols[0] = c0; cols[1] = c1; cols[2] = c2; cols[3] = c3; cols[4] = c4; first = 0; current = marked = -1; subMenu = NULL; helpRed = helpGreen = helpYellow = helpBlue = NULL; status = NULL; if (!displayMenuCount++) { displayMenu = Skins.Current()->DisplayMenu(); displayMenuItems = displayMenu->MaxItems(); } } cOsdMenu::~cOsdMenu() { free(title); delete subMenu; free(status); displayMenu->Clear(); cStatus::MsgOsdClear(); if (!--displayMenuCount) DELETENULL(displayMenu); } const char *cOsdMenu::hk(const char *s) { static char buffer[64]; if (s && hasHotkeys) { if (digit == 0 && '1' <= *s && *s <= '9' && *(s + 1) == ' ') digit = -1; // prevents automatic hotkeys - input already has them if (digit >= 0) { digit++; snprintf(buffer, sizeof(buffer), " %c %s", (digit < 10) ? '0' + digit : ' ' , s); s = buffer; } } return s; } void cOsdMenu::SetHasHotkeys(void) { hasHotkeys = true; digit = 0; } void cOsdMenu::SetStatus(const char *s) { free(status); status = s ? strdup(s) : NULL; displayMenu->SetMessage(mtStatus, s); } void cOsdMenu::SetTitle(const char *Title) { free(title); title = strdup(Title); } void cOsdMenu::SetHelp(const char *Red, const char *Green, const char *Yellow, const char *Blue) { // strings are NOT copied - must be constants!!! helpRed = Red; helpGreen = Green; helpYellow = Yellow; helpBlue = Blue; displayMenu->SetButtons(helpRed, helpGreen, helpYellow, helpBlue); cStatus::MsgOsdHelpKeys(helpRed, helpGreen, helpYellow, helpBlue); } void cOsdMenu::Del(int Index) { cList<cOsdItem>::Del(Get(Index)); int count = Count(); while (current < count && !SelectableItem(current)) current++; if (current == count) { while (current > 0 && !SelectableItem(current)) current--; } if (Index == first && first > 0) first--; } void cOsdMenu::Add(cOsdItem *Item, bool Current, cOsdItem *After) { cList<cOsdItem>::Add(Item, After); if (Current) current = Item->Index(); } void cOsdMenu::Ins(cOsdItem *Item, bool Current, cOsdItem *Before) { cList<cOsdItem>::Ins(Item, Before); if (Current) current = Item->Index(); } void cOsdMenu::Display(void) { if (subMenu) { subMenu->Display(); return; } displayMenu->SetMessage(mtStatus, NULL); displayMenu->Clear(); cStatus::MsgOsdClear(); displayMenu->SetTabs(cols[0], cols[1], cols[2], cols[3], cols[4]);//XXX displayMenu->SetTitle(title); cStatus::MsgOsdTitle(title); displayMenu->SetButtons(helpRed, helpGreen, helpYellow, helpBlue); cStatus::MsgOsdHelpKeys(helpRed, helpGreen, helpYellow, helpBlue); int count = Count(); if (count > 0) { int ni = 0; for (cOsdItem *item = First(); item; item = Next(item)) cStatus::MsgOsdItem(item->Text(), ni++); if (current < 0) current = 0; // just for safety - there HAS to be a current item! if (current - first >= displayMenuItems || current < first) { first = current - displayMenuItems / 2; if (first + displayMenuItems > count) first = count - displayMenuItems; if (first < 0) first = 0; } int i = first; int n = 0; for (cOsdItem *item = Get(first); item; item = Next(item)) { displayMenu->SetItem(item->Text(), i - first, i == current, item->Selectable()); if (i == current) cStatus::MsgOsdCurrentItem(item->Text()); if (++n == displayMenuItems) break; i++; } } if (!isempty(status)) displayMenu->SetMessage(mtStatus, status); } void cOsdMenu::SetCurrent(cOsdItem *Item) { current = Item ? Item->Index() : -1; } void cOsdMenu::RefreshCurrent(void) { cOsdItem *item = Get(current); if (item) item->Set(); } void cOsdMenu::DisplayCurrent(bool Current) { cOsdItem *item = Get(current); if (item) { displayMenu->SetItem(item->Text(), current - first, Current, item->Selectable()); if (Current) cStatus::MsgOsdCurrentItem(item->Text()); if (!Current) item->SetFresh(true); // leaving the current item resets 'fresh' } } void cOsdMenu::Clear(void) { first = 0; current = marked = -1; cList<cOsdItem>::Clear(); } bool cOsdMenu::SelectableItem(int idx) { cOsdItem *item = Get(idx); return item && item->Selectable(); } void cOsdMenu::CursorUp(void) { if (current > 0) { int tmpCurrent = current; while (--tmpCurrent >= 0 && !SelectableItem(tmpCurrent)); if (tmpCurrent < 0) return; if (tmpCurrent >= first) DisplayCurrent(false); current = tmpCurrent; if (current < first) { first = first > displayMenuItems - 1 ? first - (displayMenuItems - 1) : 0; if (Setup.MenuScrollPage) current = !SelectableItem(first) ? first + 1 : first; Display(); } else DisplayCurrent(true); } } void cOsdMenu::CursorDown(void) { int last = Count() - 1; int lastOnScreen = first + displayMenuItems - 1; if (current < last) { int tmpCurrent = current; while (++tmpCurrent <= last && !SelectableItem(tmpCurrent)); if (tmpCurrent > last) return; if (tmpCurrent <= lastOnScreen) DisplayCurrent(false); current = tmpCurrent; if (current > lastOnScreen) { first += displayMenuItems - 1; lastOnScreen = first + displayMenuItems - 1; if (lastOnScreen > last) { first = last - (displayMenuItems - 1); lastOnScreen = last; } if (Setup.MenuScrollPage) current = !SelectableItem(lastOnScreen) ? lastOnScreen - 1 : lastOnScreen; Display(); } else DisplayCurrent(true); } } void cOsdMenu::PageUp(void) { current -= displayMenuItems; first -= displayMenuItems; if (first < 0) first = current = 0; if (!SelectableItem(current)) { current -= (current > 0) ? 1 : -1; first = min(first, current - 1); } Display(); DisplayCurrent(true); } void cOsdMenu::PageDown(void) { current += displayMenuItems; first += displayMenuItems; int count = Count(); if (current > count - 1) { current = count - 1; first = max(0, count - displayMenuItems); } if (!SelectableItem(current)) { current += (current < count - 1) ? 1 : -1; first = max(first, current - displayMenuItems); } Display(); DisplayCurrent(true); } void cOsdMenu::Mark(void) { if (Count() && marked < 0) { marked = current; SetStatus(tr("Up/Dn for new location - OK to move")); } } eOSState cOsdMenu::HotKey(eKeys Key) { for (cOsdItem *item = First(); item; item = Next(item)) { const char *s = item->Text(); if (s && (s = skipspace(s)) != NULL) { if (*s == Key - k1 + '1') { current = item->Index(); cRemote::Put(kOk, true); break; } } } return osContinue; } eOSState cOsdMenu::AddSubMenu(cOsdMenu *SubMenu) { delete subMenu; subMenu = SubMenu; subMenu->Display(); return osContinue; // convenience return value } eOSState cOsdMenu::CloseSubMenu() { delete subMenu; subMenu = NULL; RefreshCurrent(); Display(); return osContinue; // convenience return value } eOSState cOsdMenu::ProcessKey(eKeys Key) { if (subMenu) { eOSState state = subMenu->ProcessKey(Key); if (state == osBack) return CloseSubMenu(); return state; } cOsdItem *item = Get(current); if (marked < 0 && item) { eOSState state = item->ProcessKey(Key); if (state != osUnknown) { DisplayCurrent(true); return state; } } switch (Key) { case k1...k9: return hasHotkeys ? HotKey(Key) : osUnknown; case kUp|k_Repeat: case kUp: CursorUp(); break; case kDown|k_Repeat: case kDown: CursorDown(); break; case kLeft|k_Repeat: case kLeft: PageUp(); break; case kRight|k_Repeat: case kRight: PageDown(); break; case kBack: return osBack; case kOk: if (marked >= 0) { SetStatus(NULL); if (marked != current) Move(marked, current); marked = -1; break; } // else run into default default: if (marked < 0) return osUnknown; } return osContinue; }