mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Removed leftover 'needsBufferReserve' variable from cTransfer (thanks to Marco Schlüßler). - Fixed setting "No title" for broken event data (reported by Ronny Kornexl). - Fixed channel up/down switching on single card systems (reported by Stefan Huelswitt). - Fixed handling "pending" timers that blocked others that actually could record (reported by Thomas Koch). - Speeded up cVideoRepacker (thanks to Reinhard Nissl). - Added an 'Id' parameter to cDevice::PlayAudio() to allow plugins to easier process the audio data (thanks to Marco Schlüßler). - Added Czech language texts (thanks to Vladimír Bárta). Plugin authors may want to add the new entries to their I18N texts and contact the translators to have their texts translated. Note that there are now 21 different OSD languages, so please make sure you have 21 versions for each of your texts. - Updated the Polish OSD texts (thanks to Jaroslaw Swierczynski). - Fixed auto advance in string entry fields when pressing Up/Down in insert mode (reported by Udo Richter). - Fixed handling the "Setup/OSD/Menu button closes" option when set to 'yes' in case a replay is active (thanks to Udo Richter). - Improved cUnbufferedFile; USE_FADVISE is now defined in tools.c by default, so if you don't want to use "fadvise" you need to comment out that line (thanks to Artur Skawina). - Fixed a missing ',' in the Swedish OSD texts (thanks to Arthur Konovalov). - cDevice::Transferring() can now be used to determine whether the (primary) device is currently playing in Transfer Mode (based on a suggestion by Reinhard Nissl). - The 'runvdr' script no longer uses the $VDRUSR environment variable to set the user id under which 'vdr' shall run. Just add the '-u username' option when you call 'runvdr'. - Fixed multiple entries of the same subdirectory in the "Recordings" menu (reported by Christian Jacobsen). - Enabled generating a core dump if VDR is run with a different user id (thanks to Ville Skyttä). - Fixed handling the "Blue" key in the "Schedule" menu for the current channel (thanks to Rolf Ahrenberg). - Renamed the Makefile target 'plugins-clean' to 'clean-plugins' (suggested by Sebastian Frei). - Made all font and image data 'const' (thanks to Darren Salt). - Fixed scrolling with Up/Down in case there are non-selectable items at the beginning or end of the menu (reported by Helmut Auer). - Added cSkin::GetTextAreaWidth() and cSkin::GetTextAreaFont(), so that a plugin that wants to do special text formatting can do so (thanks to Alexander Rieger).
503 lines
12 KiB
C
503 lines
12 KiB
C
/*
|
|
* 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.29 2006/02/05 14:37:03 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;
|
|
state = State;
|
|
selectable = true;
|
|
fresh = true;
|
|
}
|
|
|
|
cOsdItem::cOsdItem(const char *Text, eOSState State, bool Selectable)
|
|
{
|
|
text = NULL;
|
|
state = State;
|
|
selectable = Selectable;
|
|
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;
|
|
}
|
|
|
|
// --- cOsdObject ------------------------------------------------------------
|
|
|
|
void cOsdObject::Show(void)
|
|
{
|
|
if (isMenu)
|
|
((cOsdMenu *)this)->Display();
|
|
}
|
|
|
|
// --- 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);
|
|
SetCols(c0, c1, c2, c3, 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::SetCols(int c0, int c1, int c2, int c3, int c4)
|
|
{
|
|
cols[0] = c0;
|
|
cols[1] = c1;
|
|
cols[2] = c2;
|
|
cols[3] = c3;
|
|
cols[4] = c4;
|
|
}
|
|
|
|
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 && item->Selectable())
|
|
current = item->Index();
|
|
}
|
|
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)) {
|
|
bool CurrentSelectable = (i == current) && item->Selectable();
|
|
displayMenu->SetItem(item->Text(), i - first, CurrentSelectable, item->Selectable());
|
|
if (CurrentSelectable)
|
|
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(), item->Selectable());
|
|
if (Current && item->Selectable())
|
|
cStatus::MsgOsdCurrentItem(item->Text());
|
|
if (!Current)
|
|
item->SetFresh(true); // leaving the current item resets 'fresh'
|
|
}
|
|
}
|
|
|
|
void cOsdMenu::Clear(void)
|
|
{
|
|
if (marked >= 0)
|
|
SetStatus(NULL);
|
|
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)
|
|
{
|
|
int tmpCurrent = current;
|
|
int lastOnScreen = first + displayMenuItems - 1;
|
|
int last = Count() - 1;
|
|
if (last < 0)
|
|
return;
|
|
while (--tmpCurrent != current) {
|
|
if (tmpCurrent < 0) {
|
|
if (first > 0) {
|
|
// make non-selectable items at the beginning visible:
|
|
first = 0;
|
|
Display();
|
|
return;
|
|
}
|
|
if (Setup.MenuScrollWrap)
|
|
tmpCurrent = last + 1;
|
|
else
|
|
return;
|
|
}
|
|
else if (SelectableItem(tmpCurrent))
|
|
break;
|
|
}
|
|
if (first <= tmpCurrent && tmpCurrent <= lastOnScreen)
|
|
DisplayCurrent(false);
|
|
current = tmpCurrent;
|
|
if (current < first) {
|
|
first = Setup.MenuScrollPage ? max(0, current - displayMenuItems + 1) : current;
|
|
Display();
|
|
}
|
|
else if (current > lastOnScreen) {
|
|
first = max(0, current - displayMenuItems + 1);
|
|
Display();
|
|
}
|
|
else
|
|
DisplayCurrent(true);
|
|
}
|
|
|
|
void cOsdMenu::CursorDown(void)
|
|
{
|
|
int tmpCurrent = current;
|
|
int lastOnScreen = first + displayMenuItems - 1;
|
|
int last = Count() - 1;
|
|
if (last < 0)
|
|
return;
|
|
while (++tmpCurrent != current) {
|
|
if (tmpCurrent > last) {
|
|
if (first < last - displayMenuItems) {
|
|
// make non-selectable items at the end visible:
|
|
first = last - displayMenuItems + 1;
|
|
Display();
|
|
return;
|
|
}
|
|
if (Setup.MenuScrollWrap)
|
|
tmpCurrent = -1;
|
|
else
|
|
return;
|
|
}
|
|
else if (SelectableItem(tmpCurrent))
|
|
break;
|
|
}
|
|
if (first <= tmpCurrent && tmpCurrent <= lastOnScreen)
|
|
DisplayCurrent(false);
|
|
current = tmpCurrent;
|
|
if (current > lastOnScreen) {
|
|
first = Setup.MenuScrollPage ? current : max(0, current - displayMenuItems + 1);
|
|
if (first + displayMenuItems > last)
|
|
first = max(0, last - displayMenuItems + 1);
|
|
Display();
|
|
}
|
|
else if (current < first) {
|
|
first = current;
|
|
Display();
|
|
}
|
|
else
|
|
DisplayCurrent(true);
|
|
}
|
|
|
|
void cOsdMenu::PageUp(void)
|
|
{
|
|
int oldCurrent = current;
|
|
int oldFirst = first;
|
|
current -= displayMenuItems;
|
|
first -= displayMenuItems;
|
|
int last = Count() - 1;
|
|
if (current < 0)
|
|
current = 0;
|
|
if (first < 0)
|
|
first = 0;
|
|
int tmpCurrent = current;
|
|
while (!SelectableItem(tmpCurrent) && --tmpCurrent >= 0)
|
|
;
|
|
if (tmpCurrent < 0) {
|
|
tmpCurrent = current;
|
|
while (++tmpCurrent <= last && !SelectableItem(tmpCurrent))
|
|
;
|
|
}
|
|
current = tmpCurrent <= last ? tmpCurrent : -1;
|
|
if (current >= 0) {
|
|
if (current < first)
|
|
first = current;
|
|
else if (current - first >= displayMenuItems)
|
|
first = current - displayMenuItems + 1;
|
|
}
|
|
if (current != oldCurrent || first != oldFirst) {
|
|
Display();
|
|
DisplayCurrent(true);
|
|
}
|
|
else if (Setup.MenuScrollWrap)
|
|
CursorUp();
|
|
}
|
|
|
|
void cOsdMenu::PageDown(void)
|
|
{
|
|
int oldCurrent = current;
|
|
int oldFirst = first;
|
|
current += displayMenuItems;
|
|
first += displayMenuItems;
|
|
int last = Count() - 1;
|
|
if (current > last)
|
|
current = last;
|
|
if (first + displayMenuItems > last)
|
|
first = max(0, last - displayMenuItems + 1);
|
|
int tmpCurrent = current;
|
|
while (!SelectableItem(tmpCurrent) && ++tmpCurrent <= last)
|
|
;
|
|
if (tmpCurrent > last) {
|
|
tmpCurrent = current;
|
|
while (--tmpCurrent >= 0 && !SelectableItem(tmpCurrent))
|
|
;
|
|
}
|
|
current = tmpCurrent > 0 ? tmpCurrent : -1;
|
|
if (current >= 0) {
|
|
if (current < first)
|
|
first = current;
|
|
else if (current - first >= displayMenuItems)
|
|
first = current - displayMenuItems + 1;
|
|
}
|
|
if (current != oldCurrent || first != oldFirst) {
|
|
Display();
|
|
DisplayCurrent(true);
|
|
}
|
|
else if (Setup.MenuScrollWrap)
|
|
CursorDown();
|
|
}
|
|
|
|
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 k0: return osUnknown;
|
|
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;
|
|
}
|
|
|