#include "displaymenu.h"
#ifndef VDRLOGO
#define VDRLOGO "vdrlogo_default"
#endif
#include "symbols/1080/Cnew.xpm"
#include "symbols/1080/Carrowturn.xpm"
#include "symbols/1080/Crec.xpm"
#include "symbols/1080/Cclock.xpm"
#include "symbols/1080/Cclocksml.xpm"
#include "symbols/1080/Cvpssml.xpm"
cBitmap cFlatDisplayMenu::bmCNew(Cnew_xpm);
cBitmap cFlatDisplayMenu::bmCArrowTurn(Carrowturn_xpm);
cBitmap cFlatDisplayMenu::bmCRec(Crec_xpm);
cBitmap cFlatDisplayMenu::bmCClock(Cclock_xpm);
cBitmap cFlatDisplayMenu::bmCClocksml(Cclocksml_xpm);
cBitmap cFlatDisplayMenu::bmCVPS(Cvpssml_xpm);
/* Possible values of the stream content descriptor according to ETSI EN 300 468 */
enum stream_content
{
sc_reserved = 0x00,
sc_video_MPEG2 = 0x01,
sc_audio_MP2 = 0x02, // MPEG 1 Layer 2 audio
sc_subtitle = 0x03,
sc_audio_AC3 = 0x04,
sc_video_H264_AVC = 0x05,
sc_audio_HEAAC = 0x06,
};
cFlatDisplayMenu::cFlatDisplayMenu(void) {
CreateFullOsd();
TopBarCreate();
ButtonsCreate();
MessageCreate();
VideoDiskUsageState = -1;
itemHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
itemChannelHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
itemTimerHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
scrollBarWidth = ScrollBarWidth() + marginItem;
scrollBarHeight = osdHeight - (topBarHeight + Config.decorBorderTopBarSize*2 + marginItem*3 + buttonsHeight + Config.decorBorderButtonSize*2);
scrollBarTop = topBarHeight + marginItem + Config.decorBorderTopBarSize*2;
isScrolling = false;
ShowEvent = false;
ShowRecording = false;
ShowText = false;
menuWidth = osdWidth;
menuTop = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuItemSize;
menuPixmap = osd->CreatePixmap(1, cRect(0, menuTop, menuWidth, scrollBarHeight ));
menuIconsPixmap = osd->CreatePixmap(2, cRect(0, menuTop, menuWidth, scrollBarHeight ));
chLeft = Config.decorBorderMenuContentHeadSize;
chTop = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuContentHeadSize;
chWidth = menuWidth - Config.decorBorderMenuContentHeadSize*2;
chHeight = fontHeight + fontSmlHeight*2 + marginItem*2;
contentHeadPixmap = osd->CreatePixmap(1, cRect(chLeft, chTop, chWidth, chHeight));
scrollbarPixmap = osd->CreatePixmap(2, cRect(0, scrollBarTop, menuWidth, scrollBarHeight + buttonsHeight + Config.decorBorderButtonSize*2));
menuPixmap->Fill(clrTransparent);
menuIconsPixmap->Fill(clrTransparent);
scrollbarPixmap->Fill(clrTransparent);
menuCategory = mcUndefined;
bmNew = &bmCNew;
bmRec = &bmCRec;
bmArrowTurn = &bmCArrowTurn;
bmClock = &bmCClock;
bmClocksml = &bmCClocksml;
bmVPS = &bmCVPS;
}
cFlatDisplayMenu::~cFlatDisplayMenu() {
osd->DestroyPixmap(menuPixmap);
osd->DestroyPixmap(menuIconsPixmap);
osd->DestroyPixmap(scrollbarPixmap);
osd->DestroyPixmap(contentHeadPixmap);
}
void cFlatDisplayMenu::SetMenuCategory(eMenuCategory MenuCategory) {
ItemBorderClear();
isScrolling = false;
ShowRecording = ShowEvent = ShowText = false;
menuCategory = MenuCategory;
if( menuCategory == mcChannel ) {
if( Config.MenuChannelView == 0 || Config.MenuChannelView == 1 || Config.MenuChannelView == 2 )
itemChannelHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
else if( Config.MenuChannelView == 3 || Config.MenuChannelView == 4 )
itemChannelHeight = fontHeight + fontSmlHeight + marginItem + Config.decorProgressMenuItemSize + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
} else if( menuCategory == mcTimer ) {
if( Config.MenuTimerView == 0 || Config.MenuTimerView == 1 )
itemTimerHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
else if( Config.MenuTimerView == 2 || Config.MenuTimerView == 3 )
itemTimerHeight = fontHeight + fontSmlHeight + marginItem + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
} else if( menuCategory == mcSchedule || menuCategory == mcScheduleNow || menuCategory == mcScheduleNext ) {
if( Config.MenuEventView == 0 || Config.MenuEventView == 1 )
itemEventHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
else if( Config.MenuEventView == 2 || Config.MenuEventView == 3 )
itemEventHeight = fontHeight + fontSmlHeight + marginItem*2 + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2 + Config.decorProgressMenuItemSize/2;
} else if( menuCategory == mcRecording ) {
if( Config.MenuRecordingView == 0 || Config.MenuRecordingView == 1 )
itemRecordingHeight = fontHeight + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
else if( Config.MenuRecordingView == 2 || Config.MenuRecordingView == 3 )
itemRecordingHeight = fontHeight + fontSmlHeight + marginItem + Config.MenuItemPadding + Config.decorBorderMenuItemSize*2;
}
}
void cFlatDisplayMenu::DrawScrollbar(int Total, int Offset, int Shown, int Top, int Height, bool CanScrollUp, bool CanScrollDown, bool isContent) {
if( Total > 0 && Total > Shown ) {
if( isScrolling == false && ShowEvent == false && ShowRecording == false && ShowText == false ) {
isScrolling = true;
DecorBorderClearByFrom(BorderMenuItem);
ItemBorderDrawAllWithScrollbar();
ItemBorderClear();
if( isContent )
menuPixmap->DrawRectangle(cRect(menuItemWidth - scrollBarWidth + Config.decorBorderMenuContentSize, 0, scrollBarWidth + marginItem, scrollBarHeight), clrTransparent);
else
menuPixmap->DrawRectangle(cRect(menuItemWidth - scrollBarWidth + Config.decorBorderMenuItemSize, 0, scrollBarWidth + marginItem, scrollBarHeight), clrTransparent);
}
} else if( ShowEvent == false && ShowRecording == false && ShowText == false ) {
isScrolling = false;
}
if( isContent )
ScrollbarDraw(scrollbarPixmap, menuItemWidth - scrollBarWidth + Config.decorBorderMenuContentSize*2 + marginItem, Top, Height, Total, Offset, Shown, CanScrollUp, CanScrollDown);
else
ScrollbarDraw(scrollbarPixmap, menuItemWidth - scrollBarWidth + Config.decorBorderMenuItemSize*2 + marginItem, Top, Height, Total, Offset, Shown, CanScrollUp, CanScrollDown);
}
void cFlatDisplayMenu::SetScrollbar(int Total, int Offset) {
DrawScrollbar(Total, Offset, MaxItems(), 0, ItemsHeight(), Offset > 0, Offset + MaxItems() < Total);
}
void cFlatDisplayMenu::Scroll(bool Up, bool Page) {
// Wird das Menü gescrollt oder Content?
if( ContentIsShown() ) {
bool scrolled = ContentScroll(Up, Page);
if( scrolled )
DrawScrollbar(ContentScrollTotal(), ContentScrollOffset(), ContentVisibleLines(), contentTop - scrollBarTop, ContentGetHeight(), ContentScrollOffset() > 0, ContentScrollOffset() + ContentVisibleLines() < ContentScrollTotal(), true);
} else {
cSkinDisplayMenu::Scroll(Up, Page);
}
}
int cFlatDisplayMenu::MaxItems(void) {
if( menuCategory == mcChannel )
return scrollBarHeight / itemChannelHeight;
else if( menuCategory == mcTimer )
return scrollBarHeight / itemTimerHeight;
else if( menuCategory == mcSchedule || menuCategory == mcScheduleNow || menuCategory == mcScheduleNext )
return scrollBarHeight / itemEventHeight;
else if( menuCategory == mcRecording )
return scrollBarHeight / itemRecordingHeight;
return scrollBarHeight / itemHeight;
}
int cFlatDisplayMenu::ItemsHeight(void) {
if( menuCategory == mcChannel )
return MaxItems() * itemChannelHeight - Config.MenuItemPadding;
else if( menuCategory == mcTimer )
return MaxItems() * itemTimerHeight - Config.MenuItemPadding;
else if( menuCategory == mcSchedule || menuCategory == mcScheduleNow || menuCategory == mcScheduleNext )
return MaxItems() * itemEventHeight - Config.MenuItemPadding;
else if( menuCategory == mcRecording )
return MaxItems() * itemRecordingHeight - Config.MenuItemPadding;
return MaxItems() * itemHeight - Config.MenuItemPadding;
}
void cFlatDisplayMenu::Clear(void) {
textScroller.Reset();
menuPixmap->Fill(clrTransparent);
menuIconsPixmap->Fill(clrTransparent);
scrollbarPixmap->Fill(clrTransparent);
contentHeadPixmap->Fill(clrTransparent);
ContentClear();
DecorBorderClearByFrom(BorderMenuItem);
DecorBorderClearAll();
isScrolling = false;
ShowRecording = ShowEvent = ShowText = false;
}
void cFlatDisplayMenu::SetTitle(const char *Title) {
if( (menuCategory == mcRecording || menuCategory == mcTimer) && Config.DiskUsageShow ) {
cVideoDiskUsage::HasChanged(VideoDiskUsageState);
int DiskUsage = cVideoDiskUsage::UsedPercent();
double FreeGB = cVideoDiskUsage::FreeMB() / 1024.0;
int FreeMinutes = cVideoDiskUsage::FreeMinutes();
cString extra1 = cString::sprintf("%s: %d%%", tr("disk usage"), DiskUsage);
cString extra2 = cString::sprintf("%s: %.1f GB ~ %02d:%02d", tr("free"), FreeGB, FreeMinutes / 60, FreeMinutes % 60);
cString iconName("chart1");
if( DiskUsage > 14 )
iconName = "chart2";
if( DiskUsage > 28 )
iconName = "chart3";
if( DiskUsage > 42 )
iconName = "chart4";
if( DiskUsage > 56 )
iconName = "chart5";
if( DiskUsage > 70 )
iconName = "chart6";
if( DiskUsage > 84 )
iconName = "chart7";
TopBarSetTitle(Title);
TopBarSetTitleExtra(extra1, extra2);
TopBarSetExtraIcon(iconName);
} else {
TopBarSetTitle(Title);
}
if( Config.TopBarMenuIconShow ) {
cString icon;
switch( menuCategory ) {
case mcMain:
TopBarSetTitle("");
icon = cString::sprintf("menuIcons/%s", VDRLOGO);
break;
case mcSchedule:
case mcScheduleNow:
case mcScheduleNext:
icon = "menuIcons/Schedule";
break;
case mcChannel:
icon = "menuIcons/Channels";
break;
case mcTimer:
icon = "menuIcons/Timers";
break;
case mcRecording:
icon = "menuIcons/Recordings";
break;
case mcSetup:
icon = "menuIcons/Setup";
break;
case mcCommand:
icon = "menuIcons/Commands";
break;
case mcEvent:
icon = "extraIcons/Info";
break;
case mcRecordingInfo:
icon = "extraIcons/PlayInfo";
break;
default:
icon = "";
}
TopBarSetMenuIcon(icon);
}
}
void cFlatDisplayMenu::SetButtons(const char *Red, const char *Green, const char *Yellow, const char *Blue) {
ButtonsSet(Red, Green, Yellow, Blue);
}
void cFlatDisplayMenu::SetMessage(eMessageType Type, const char *Text) {
if (Text)
MessageSet(Type, Text);
else
MessageClear();
}
void cFlatDisplayMenu::SetItem(const char *Text, int Index, bool Current, bool Selectable) {
ShowEvent = false;
ShowRecording = false;
ShowText = false;
int y = Index * itemHeight;
menuItemWidth = menuWidth - Config.decorBorderMenuItemSize*2;
if( menuCategory == mcMain )
menuItemWidth *= Config.MainMenuItemScale;
int AvailableTextWidth = menuItemWidth - scrollBarWidth;
if( isScrolling )
menuItemWidth -= scrollBarWidth;
tColor ColorFg, ColorBg;
if (Current) {
ColorFg = Theme.Color(clrItemCurrentFont);
ColorBg = Theme.Color(clrItemCurrentBg);
}
else {
if( Selectable ) {
ColorFg = Theme.Color(clrItemSelableFont);
ColorBg = Theme.Color(clrItemSelableBg);
} else {
ColorFg = Theme.Color(clrItemFont);
ColorBg = Theme.Color(clrItemBg);
}
}
menuPixmap->DrawRectangle(cRect(Config.decorBorderMenuItemSize, y, menuItemWidth, fontHeight), ColorBg);
int lh = fontHeight;
int x1 = 0;
int xOff = x1;
for (int i = 0; i < MaxTabs; i++) {
const char *s = GetTabbedText(Text, i);
if( s ) {
// from skinelchi
char buffer[9];
bool istimer = false;
bool isnewrecording = false;
bool hasEventtimer = false;
bool haspartEventtimer = false;
bool isRecording = false;
bool hasVPS = false;
bool isRunning = false;
int xt = Tab(i);
if( xt >= menuItemWidth )
continue;
if( true ) {
// check if timer info symbols: " !#>"
if (i == 0 && strlen(s) == 1 && strchr(" !#>", s[0])) {
istimer = true; // update status
} else if ( (strlen(s) == 6 && s[5] == '*' && s[2] == ':' && isdigit(*s)
&& isdigit(*(s + 1)) && isdigit(*(s + 3)) && isdigit(*(s + 4)))
|| (strlen(s) == 5 && s[4] == '*' && s[1] == ':' && isdigit(*s)
&& isdigit(*(s + 2)) && isdigit(*(s + 3)))
|| (strlen(s) == 9 && s[8] == '*' && s[5] == '.' && s[2] == '.'
&& isdigit(*s) && isdigit(*(s + 1)) && isdigit(*(s + 3))
&& isdigit(*(s + 4)) && isdigit(*(s + 6)) && isdigit(*(s + 7)))) {
// check if new recording: "10:10*", "1:10*", "01.01.06*"
isnewrecording = true; // update status
strncpy(buffer, s, strlen(s)); // make a copy
buffer[strlen(s) - 1] = '\0'; // remove the '*' character
} else if ( (strlen(s) == 3) && ( i == 2 || i == 3 || i == 4) ) {
if (s[0] == 'R') isRecording = true;
if (s[0] == 'T') hasEventtimer = true;
if (s[0] == 't') haspartEventtimer = true;
if (s[1] == 'V') hasVPS = true;
if (s[2] == '*') isRunning = true;
} else if ( (strlen(s) == 4) && ( i == 3 ) ) { //epgsearch What's on now default
if (s[1] == 'R') isRecording = true;
if (s[1] == 'T') hasEventtimer = true;
if (s[1] == 't') haspartEventtimer = true;
if (s[2] == 'V') hasVPS = true;
if (s[3] == '*') isRunning = true;
}
}
xOff = x1 + Tab(i) + Config.decorBorderMenuItemSize;
if( istimer ) {
// timer menu
switch( s[0] ) {
case '!':
menuPixmap->DrawBitmap(cPoint(xOff, y + (lh - bmArrowTurn->Height()) / 2), *bmArrowTurn, ColorFg, ColorBg);
break;
case '#':
menuPixmap->DrawBitmap(cPoint(xOff, y + (lh - bmRec->Height()) / 2), *bmRec, ColorFg, ColorBg);
break;
case '>':
menuPixmap->DrawBitmap(cPoint(xOff, y + (lh - bmClock->Height()) / 2), *bmClock, ColorFg, ColorBg);
break;
case ' ':
default:
break;
}
} else if( isRecording || hasEventtimer || haspartEventtimer || hasVPS || isRunning ) {
// program schedule menu
if( isRecording )
menuPixmap->DrawBitmap(cPoint(xOff, y + (lh - bmRec->Height()) / 2), *bmRec, ColorFg, ColorBg);
else {
if( hasEventtimer )
menuPixmap->DrawBitmap(cPoint(xOff, y + (lh - bmClock->Height()) / 2), *bmClock, ColorFg, ColorBg);
if( haspartEventtimer )
menuPixmap->DrawBitmap(cPoint(xOff + (bmClock->Height() - bmClocksml->Height()) / 2, y + (lh - bmClocksml->Height()) / 2), *bmClocksml, ColorFg, ColorBg);
}
xOff += bmClock->Width(); // clock is wider than rec
if( hasVPS )
menuPixmap->DrawBitmap(cPoint(xOff, y + (lh - bmVPS->Height()) / 2), *bmVPS, ColorFg, ColorBg);
xOff += bmVPS->Width();
if( isRunning )
menuPixmap->DrawText(cPoint(xOff, y), "*", ColorFg, ColorBg, font, AvailableTextWidth - xOff);
} else if( isnewrecording ) {
// recordings menu
menuPixmap->DrawText(cPoint(xOff, y), buffer, ColorFg, ColorBg, font, AvailableTextWidth - xOff);
// draw symbol "new" centered
int gap = std::max(0, (Tab(i+1)-Tab(i)- font->Width(buffer) - bmNew->Width()) / 2);
menuPixmap->DrawBitmap(cPoint(xOff + font->Width(buffer) + gap, y + (lh - bmNew->Height()) / 2), *bmNew, ColorFg, ColorBg);
} else if( CheckProgressBar( s ) ) {
int colWidth = Tab(i+1) - Tab(i);
tColor ColorFg = Config.decorProgressMenuItemFg;
tColor ColorBarFg = Config.decorProgressMenuItemBarFg;
tColor ColorBg = Config.decorProgressMenuItemBg;
if( Current ) {
ColorFg = Config.decorProgressMenuItemCurFg;
ColorBarFg = Config.decorProgressMenuItemCurBarFg;
ColorBg = Config.decorProgressMenuItemCurBg;
}
cRect rec = cRect(xt + Config.decorBorderMenuItemSize,
y + (itemHeight-Config.MenuItemPadding)/2 - Config.decorProgressMenuItemSize/2 - Config.decorBorderMenuItemSize,
colWidth, Config.decorProgressMenuItemSize);
cRect recBG = cRect(xt + Config.decorBorderMenuItemSize - marginItem, y,
colWidth + marginItem*2, fontHeight);
DrawProgressBarFromText(rec, recBG, s, ColorFg, ColorBarFg, ColorBg);
} else {
if( (menuCategory == mcMain || menuCategory == mcSetup) && Config.MenuItemIconsShow) {
cString cIcon = GetIconName( MainMenuText(s) );
cImageLoader imgLoader;
cImage *img = imgLoader.LoadIcon(*cIcon, fontHeight - marginItem*2, fontHeight - marginItem*2);
if( img ) {
menuIconsPixmap->DrawImage(cPoint(xt + Config.decorBorderMenuItemSize + marginItem, y + marginItem), *img);
} else {
img = imgLoader.LoadIcon("menuIcons/blank", fontHeight - marginItem*2, fontHeight - marginItem*2);
if( img ) {
menuIconsPixmap->DrawImage(cPoint(xt + Config.decorBorderMenuItemSize + marginItem, y + marginItem), *img);
}
}
menuPixmap->DrawText(cPoint(fontHeight + marginItem*2 + xt + Config.decorBorderMenuItemSize, y), s, ColorFg, ColorBg, font,
AvailableTextWidth - xt - marginItem*2 - fontHeight);
} else {
menuPixmap->DrawText(cPoint(xt + Config.decorBorderMenuItemSize, y), s, ColorFg, ColorBg, font);
}
}
}
if (!Tab(i + 1))
break;
}
sDecorBorder ib;
ib.Left = Config.decorBorderMenuItemSize;
ib.Top = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuItemSize + y;
ib.Width = menuWidth - Config.decorBorderMenuItemSize*2;
if( isScrolling ) {
ib.Width -= scrollBarWidth;
}
ib.Width = menuItemWidth;
ib.Height = fontHeight;
ib.Size = Config.decorBorderMenuItemSize;
ib.Type = Config.decorBorderMenuItemType;
if( Current ) {
ib.ColorFg = Config.decorBorderMenuItemCurFg;
ib.ColorBg = Config.decorBorderMenuItemCurBg;
} else {
if( Selectable ) {
ib.ColorFg = Config.decorBorderMenuItemSelFg;
ib.ColorBg = Config.decorBorderMenuItemSelBg;
} else {
ib.ColorFg = Config.decorBorderMenuItemFg;
ib.ColorBg = Config.decorBorderMenuItemBg;
}
}
DecorBorderDraw(ib.Left, ib.Top, ib.Width, ib.Height,
ib.Size, ib.Type, ib.ColorFg, ib.ColorBg, BorderMenuItem);
if( !isScrolling ) {
ItemBorderInsertUnique(ib);
}
SetEditableWidth(menuWidth - Tab(1));
}
std::string cFlatDisplayMenu::items[16] = { "Schedule", "Channels", "Timers", "Recordings", "Setup", "Commands",
"OSD", "EPG", "DVB", "LNB", "CAM", "Recording", "Replay", "Miscellaneous", "Plugins", "Restart"};
std::string cFlatDisplayMenu::MainMenuText(std::string Text) {
std::string text = skipspace(Text.c_str());
std::string menuEntry;
std::string menuNumber;
bool found = false;
bool doBreak = false;
size_t i = 0;
for (; i < text.length(); i++) {
char s = text.at(i);
if (i==0) {
//if text directly starts with nonnumeric, break
if (!(s >= '0' && s <= '9')) {
break;
}
}
if (found) {
if (!(s >= '0' && s <= '9')) {
doBreak = true;
}
}
if (s >= '0' && s <= '9') {
found = true;
}
if (doBreak)
break;
if (i>4)
break;
}
if (found) {
menuNumber = skipspace(text.substr(0,i).c_str());
menuEntry = skipspace(text.substr(i).c_str());
} else {
menuNumber = "";
menuEntry = text.c_str();
}
return menuEntry;
}
cString cFlatDisplayMenu::GetIconName(std::string element) {
//check for standard menu entries
for (int i=0; i<16; i++) {
std::string s = trVDR(items[i].c_str());
if (s == element) {
cString menuIcon = cString::sprintf("menuIcons/%s", items[i].c_str());
return menuIcon;
}
}
//check for special main menu entries "stop recording", "stop replay"
std::string stopRecording = skipspace(trVDR(" Stop recording "));
std::string stopReplay = skipspace(trVDR(" Stop replaying"));
try {
if (element.substr(0, stopRecording.size()) == stopRecording)
return "menuIcons/StopRecording";
if (element.substr(0, stopReplay.size()) == stopReplay)
return "menuIcons/StopReplay";
} catch (...) {}
//check for Plugins
for (int i = 0; ; i++) {
cPlugin *p = cPluginManager::GetPlugin(i);
if (p) {
const char *mainMenuEntry = p->MainMenuEntry();
if (mainMenuEntry) {
std::string plugMainEntry = mainMenuEntry;
try {
if (element.substr(0, plugMainEntry.size()) == plugMainEntry) {
return cString::sprintf("pluginIcons/%s", p->Name());
}
} catch (...) {}
}
} else
break;
}
return cString::sprintf("extraIcons/%s", element.c_str());
}
bool cFlatDisplayMenu::CheckProgressBar(const char *text) {
if (strlen(text) > 5
&& text[0] == '['
&& ((text[1] == '|')||(text[1] == ' '))
&& ((text[2] == '|')||(text[2] == ' '))
&& text[strlen(text) - 1] == ']')
return true;
return false;
}
void cFlatDisplayMenu::DrawProgressBarFromText(cRect rec, cRect recBg, const char *bar, tColor ColorFg, tColor ColorBarFg, tColor ColorBg) {
const char *p = bar + 1;
bool isProgressbar = true;
int total = 0;
int now = 0;
for (; *p != ']'; ++p) {
if (*p == ' ' || *p == '|') {
++total;
if (*p == '|')
++now;
} else {
isProgressbar = false;
break;
}
}
if (isProgressbar) {
double progress = (double)now / (double)total;
ProgressBarDrawRaw(menuPixmap, menuPixmap, rec, recBg, progress*total, total,
ColorFg, ColorBarFg, ColorBg, Config.decorProgressMenuItemType, true);
}
}
bool cFlatDisplayMenu::SetItemChannel(const cChannel *Channel, int Index, bool Current, bool Selectable, bool WithProvider) {
if( Config.MenuChannelView == 0 || !Channel )
return false;
cSchedulesLock schedulesLock;
const cSchedules *schedules = cSchedules::Schedules(schedulesLock);
const cEvent *Event = NULL;
bool DrawProgress = true;
cString buffer;
int y = Index * itemChannelHeight;
int Height = fontHeight;
if( Config.MenuChannelView == 3 || Config.MenuChannelView == 4 )
Height = fontHeight + fontSmlHeight + marginItem + Config.decorProgressMenuItemSize;
menuItemWidth = menuWidth - Config.decorBorderMenuItemSize*2;
if( Config.MenuChannelView == 3 || Config.MenuChannelView == 4 )
menuItemWidth *= 0.5;
if( isScrolling )
menuItemWidth -= scrollBarWidth;
tColor ColorFg, ColorBg;
if (Current) {
ColorFg = Theme.Color(clrItemCurrentFont);
ColorBg = Theme.Color(clrItemCurrentBg);
}
else {
if( Selectable ) {
ColorFg = Theme.Color(clrItemSelableFont);
ColorBg = Theme.Color(clrItemSelableBg);
} else {
ColorFg = Theme.Color(clrItemFont);
ColorBg = Theme.Color(clrItemBg);
}
}
menuPixmap->DrawRectangle(cRect(Config.decorBorderMenuItemSize, y, menuItemWidth, Height), ColorBg);
int Left, Top, Width;
int LeftName;
Left = Config.decorBorderMenuItemSize + marginItem;
Top = y;
if( Channel->GroupSep() )
DrawProgress = false;
float progress = 0.0;
cString EventTitle = "";
cString ws = cString::sprintf("%d", Channels.MaxNumber());
int w = font->Width(ws);
if( !Channel->GroupSep() )
buffer = cString::sprintf("%d", Channel->Number());
else
buffer = "";
Width = font->Width(buffer);
if( Width < w )
Width = w;
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, Width, fontHeight, taRight);
Left += Width + marginItem;
int imageHeight = fontHeight;
int imageLeft = Left;
int imageTop = Top;
cImage *img = imgLoader.LoadLogo(Channel->Name(), imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
Left += imageHeight + marginItem * 2;
} else {
bool isRadioChannel = ((!Channel->Vpid())&&(Channel->Apid(0))) ? true : false;
if( isRadioChannel ) {
img = imgLoader.LoadIcon("radio", imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
Left += imageHeight + marginItem * 2;
}
} else if( Channel->GroupSep() ) {
img = imgLoader.LoadIcon("changroup", imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
Left += imageHeight + marginItem * 2;
}
} else {
img = imgLoader.LoadIcon("tv", imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
Left += imageHeight + marginItem * 2;
}
}
}
LeftName = Left;
// event from channel
const cSchedule *Schedule = schedules->GetSchedule( Channel->GetChannelID() );
if( Schedule ) {
Event = Schedule->GetPresentEvent();
if( Event ) {
// calculate progress bar
progress = (int)roundf( (float)(time(NULL) - Event->StartTime()) / (float) (Event->Duration()) * 100.0);
if(progress < 0)
progress = 0.;
else if(progress > 100)
progress = 100;
EventTitle = Event->Title();
}
}
if( WithProvider )
buffer = cString::sprintf("%s - %s", Channel->Provider(), Channel->Name());
else
buffer = cString::sprintf("%s", Channel->Name());
if( Config.MenuChannelView == 1 ) {
Width = menuItemWidth - LeftName;
menuPixmap->DrawText(cPoint(LeftName, Top), buffer, ColorFg, ColorBg, font, Width);
} else {
Width = menuItemWidth / 10*2;
if( isScrolling )
Width = (menuItemWidth + scrollBarWidth) / 10*2;
if( Config.MenuChannelView == 3 || Config.MenuChannelView == 4 )
Width = menuItemWidth - LeftName;
menuPixmap->DrawText(cPoint(LeftName, Top), buffer, ColorFg, ColorBg, font, Width);
Left += Width + marginItem;
if( DrawProgress ) {
int PBTop = y + (itemChannelHeight-Config.MenuItemPadding)/2 - Config.decorProgressMenuItemSize/2 - Config.decorBorderMenuItemSize;
int PBLeft = Left;
int PBWidth = menuItemWidth/10;
if( Config.MenuChannelView == 3 || Config.MenuChannelView == 4 ) {
PBTop = Top + fontHeight + fontSmlHeight;
PBLeft = Left - Width - marginItem;
PBWidth = menuItemWidth - LeftName - marginItem*2 - Config.decorBorderMenuItemSize - scrollBarWidth;
if( isScrolling )
PBWidth += scrollBarWidth;
}
Width = menuItemWidth/10;
if( isScrolling )
Width = (menuItemWidth + scrollBarWidth) / 10;
if( Current )
ProgressBarDrawRaw(menuPixmap, menuPixmap, cRect( PBLeft, PBTop, PBWidth, Config.decorProgressMenuItemSize),
cRect( PBLeft, PBTop, PBWidth, Config.decorProgressMenuItemSize), progress, 100,
Config.decorProgressMenuItemCurFg, Config.decorProgressMenuItemCurBarFg, Config.decorProgressMenuItemCurBg, Config.decorProgressMenuItemType, false);
else
ProgressBarDrawRaw(menuPixmap, menuPixmap, cRect( PBLeft, PBTop, PBWidth, Config.decorProgressMenuItemSize),
cRect( PBLeft, PBTop, PBWidth, Config.decorProgressMenuItemSize), progress, 100,
Config.decorProgressMenuItemFg, Config.decorProgressMenuItemBarFg, Config.decorProgressMenuItemBg, Config.decorProgressMenuItemType, false);
Left += Width + marginItem;
}
if( Config.MenuChannelView == 3 || Config.MenuChannelView == 4 ) {
Left = LeftName;
Top += fontHeight;
menuPixmap->DrawText(cPoint(Left, Top), EventTitle, ColorFg, ColorBg, fontSml, menuItemWidth - Left - marginItem );
} else
menuPixmap->DrawText(cPoint(Left, Top), EventTitle, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem );
}
sDecorBorder ib;
ib.Left = Config.decorBorderMenuItemSize;
ib.Top = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuItemSize + y;
ib.Width = menuItemWidth - Config.decorBorderMenuItemSize*2;
if( isScrolling ) {
ib.Width -= scrollBarWidth;
}
ib.Width = menuItemWidth;
ib.Height = Height;
ib.Size = Config.decorBorderMenuItemSize;
ib.Type = Config.decorBorderMenuItemType;
if( Current ) {
ib.ColorFg = Config.decorBorderMenuItemCurFg;
ib.ColorBg = Config.decorBorderMenuItemCurBg;
} else {
if( Selectable ) {
ib.ColorFg = Config.decorBorderMenuItemSelFg;
ib.ColorBg = Config.decorBorderMenuItemSelBg;
} else {
ib.ColorFg = Config.decorBorderMenuItemFg;
ib.ColorBg = Config.decorBorderMenuItemBg;
}
}
DecorBorderDraw(ib.Left, ib.Top, ib.Width, ib.Height,
ib.Size, ib.Type, ib.ColorFg, ib.ColorBg, BorderMenuItem);
if( !isScrolling ) {
ItemBorderInsertUnique(ib);
}
if( Config.MenuChannelView == 4 && Event && Current ) {
DrawItemExtraEvent(Event, "");
}
return true;
}
void cFlatDisplayMenu::DrawItemExtraEvent(const cEvent *Event, cString EmptyText) {
cLeft = menuItemWidth + Config.decorBorderMenuItemSize*2 + Config.decorBorderMenuContentSize + marginItem;
if( isScrolling )
cLeft += scrollBarWidth;
cTop = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuContentSize;
cWidth = menuWidth - cLeft - Config.decorBorderMenuContentSize;
cHeight = osdHeight - (topBarHeight + Config.decorBorderTopBarSize*2 +
buttonsHeight + Config.decorBorderButtonSize*2 + marginItem*3 + Config.decorBorderMenuContentSize*2);
// Description
ostringstream text;
if( Event ) {
if( !isempty(Event->Description()) ) {
text << Event->Description();
}
if( Config.EpgAdditionalInfoShow ) {
text << endl;
const cComponents *Components = Event->Components();
if (Components) {
ostringstream audio;
bool firstAudio = true;
const char *audio_type = NULL;
ostringstream subtitle;
bool firstSubtitle = true;
for (int i = 0; i < Components->NumComponents(); i++) {
const tComponent *p = Components->Component(i);
switch (p->stream) {
case sc_video_MPEG2:
if (p->description)
text << endl << tr("Video") << ": " << p->description << " (MPEG2)";
else
text << endl << tr("Video") << ": MPEG2";
break;
case sc_video_H264_AVC:
if (p->description)
text << endl << tr("Video") << ": " << p->description << " (H.264)";
else
text << endl << tr("Video") << ": H.264";
break;
case sc_audio_MP2:
case sc_audio_AC3:
case sc_audio_HEAAC:
if (firstAudio)
firstAudio = false;
else
audio << ", ";
switch (p->stream) {
case sc_audio_MP2:
// workaround for wrongfully used stream type X 02 05 for AC3
if (p->type == 5)
audio_type = "AC3";
else
audio_type = "MP2";
break;
case sc_audio_AC3:
audio_type = "AC3"; break;
case sc_audio_HEAAC:
audio_type = "HEAAC"; break;
}
if (p->description)
audio << p->description << " (" << audio_type << ", " << p->language << ")";
else
audio << p->language << " (" << audio_type << ")";
break;
case sc_subtitle:
if (firstSubtitle)
firstSubtitle = false;
else
subtitle << ", ";
if (p->description)
subtitle << p->description << " (" << p->language << ")";
else
subtitle << p->language;
break;
}
}
if (audio.str().length() > 0)
text << endl << tr("Audio") << ": "<< audio.str();
if (subtitle.str().length() > 0)
text << endl << tr("Subtitle") << ": "<< subtitle.str();
}
}
} else
text << *EmptyText;
ContentCreate(cLeft, cTop, cWidth, cHeight, 2);
ContentSet( text.str().c_str(), Theme.Color(clrMenuEventFontInfo), Theme.Color(clrMenuEventBg) );
DecorBorderClearByFrom(BorderContent);
if( Config.MenuContentFullSize )
DecorBorderDraw(cLeft, cTop, cWidth, ContentGetHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg, BorderContent);
else
DecorBorderDraw(cLeft, cTop, cWidth, ContentGetTextHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg, BorderContent);
}
bool cFlatDisplayMenu::SetItemTimer(const cTimer *Timer, int Index, bool Current, bool Selectable) {
if( Config.MenuTimerView == 0 || !Timer )
return false;
const cChannel *Channel = Timer->Channel();
const cEvent *Event = Timer->Event();
cString buffer;
int y = Index * itemTimerHeight;
int Height = fontHeight;
if( Config.MenuTimerView == 2 || Config.MenuTimerView == 3 )
Height = fontHeight + fontSmlHeight + marginItem;
menuItemWidth = menuWidth - Config.decorBorderMenuItemSize*2;
if( Config.MenuTimerView == 2 || Config.MenuTimerView == 3 )
menuItemWidth *= 0.5;
if( isScrolling )
menuItemWidth -= scrollBarWidth;
tColor ColorFg, ColorBg;
if (Current) {
ColorFg = Theme.Color(clrItemCurrentFont);
ColorBg = Theme.Color(clrItemCurrentBg);
}
else {
if( Selectable ) {
ColorFg = Theme.Color(clrItemSelableFont);
ColorBg = Theme.Color(clrItemSelableBg);
} else {
ColorFg = Theme.Color(clrItemFont);
ColorBg = Theme.Color(clrItemBg);
}
}
menuPixmap->DrawRectangle(cRect(Config.decorBorderMenuItemSize, y, menuItemWidth, Height), ColorBg);
cImage *img = NULL;
int Left, Top;
Left = Config.decorBorderMenuItemSize + marginItem;
Top = y;
int imageHeight = fontHeight;
int imageLeft = Left;
int imageTop = Top;
cString TimerIconName("");
if (!(Timer->HasFlags(tfActive))) {
TimerIconName = "timerInactive";
ColorFg = Theme.Color( clrMenuTimerItemDisabledFont );
} else if (Timer->Recording()) {
TimerIconName = "timerRecording";
ColorFg = Theme.Color( clrMenuTimerItemRecordingFont );
} else
TimerIconName = "timerActive";
img = imgLoader.LoadIcon(TimerIconName, imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
Left += imageHeight + marginItem * 2;
}
cString ws = cString::sprintf("%d", Channels.MaxNumber());
int w = font->Width(ws);
buffer = cString::sprintf("%d", Channel->Number());
int Width = font->Width(buffer);
if( Width < w )
Width = w;
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, Width, fontHeight, taRight);
Left += Width + marginItem;
imageLeft = Left;
img = imgLoader.LoadLogo(Channel->Name(), imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
} else {
bool isRadioChannel = ((!Channel->Vpid())&&(Channel->Apid(0))) ? true : false;
if( isRadioChannel ) {
img = imgLoader.LoadIcon("radio", imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
}
} else if( Channel->GroupSep() ) {
img = imgLoader.LoadIcon("changroup", imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
}
} else {
img = imgLoader.LoadIcon("tv", imageHeight, imageHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(imageLeft, imageTop), *img );
}
}
}
Left += imageHeight + marginItem * 2;
cString day, name("");
if (Timer->WeekDays())
day = Timer->PrintDay(0, Timer->WeekDays(), false);
else if (Timer->Day() - time(NULL) < 28 * SECSINDAY) {
day = itoa(Timer->GetMDay(Timer->Day()));
name = WeekDayName(Timer->Day());
} else {
struct tm tm_r;
time_t Day = Timer->Day();
localtime_r(&Day, &tm_r);
char buffer[16];
strftime(buffer, sizeof(buffer), "%Y%m%d", &tm_r);
day = buffer;
}
const char *File = Setup.FoldersInTimerMenu ? NULL : strrchr(Timer->File(), FOLDERDELIMCHAR);
if (File && strcmp(File + 1, TIMERMACRO_TITLE) && strcmp(File + 1, TIMERMACRO_EPISODE))
File++;
else
File = Timer->File();
if( Config.MenuTimerView == 1 ) {
buffer = cString::sprintf("%s%s%s. %02d:%02d - %02d:%02d %s",
*name, *name && **name ? " " : "", *day,
Timer->Start() / 100, Timer->Start() % 100,
Timer->Stop() / 100, Timer->Stop() % 100,
File);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
} else if( Config.MenuTimerView == 2 || Config.MenuTimerView == 3 ) {
buffer = cString::sprintf("%s%s%s. %02d:%02d - %02d:%02d",
*name, *name && **name ? " " : "", *day,
Timer->Start() / 100, Timer->Start() % 100,
Timer->Stop() / 100, Timer->Stop() % 100);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
menuPixmap->DrawText(cPoint(Left, Top + fontHeight), File, ColorFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
}
sDecorBorder ib;
ib.Left = Config.decorBorderMenuItemSize;
ib.Top = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuItemSize + y;
ib.Width = menuItemWidth - Config.decorBorderMenuItemSize*2;
if( isScrolling ) {
ib.Width -= scrollBarWidth;
}
ib.Width = menuItemWidth;
ib.Height = Height;
ib.Size = Config.decorBorderMenuItemSize;
ib.Type = Config.decorBorderMenuItemType;
if( Current ) {
ib.ColorFg = Config.decorBorderMenuItemCurFg;
ib.ColorBg = Config.decorBorderMenuItemCurBg;
} else {
if( Selectable ) {
ib.ColorFg = Config.decorBorderMenuItemSelFg;
ib.ColorBg = Config.decorBorderMenuItemSelBg;
} else {
ib.ColorFg = Config.decorBorderMenuItemFg;
ib.ColorBg = Config.decorBorderMenuItemBg;
}
}
DecorBorderDraw(ib.Left, ib.Top, ib.Width, ib.Height,
ib.Size, ib.Type, ib.ColorFg, ib.ColorBg, BorderMenuItem);
if( !isScrolling ) {
ItemBorderInsertUnique(ib);
}
if( Config.MenuTimerView == 3 && Current ) {
DrawItemExtraEvent(Event, tr("timer not enabled"));
}
return true;
}
bool cFlatDisplayMenu::SetItemEvent(const cEvent *Event, int Index, bool Current, bool Selectable, const cChannel *Channel, bool WithDate, eTimerMatch TimerMatch) {
if( Config.MenuEventView == 0 )
return false;
cImage *img = NULL;
cString buffer;
int y = Index * itemEventHeight;
int Height = fontHeight;
if( Config.MenuEventView == 2 || Config.MenuEventView == 3 )
Height = fontHeight + fontSmlHeight + marginItem*2 + Config.decorProgressMenuItemSize/2;
menuItemWidth = menuWidth - Config.decorBorderMenuItemSize*2;
if( Config.MenuEventView == 2 || Config.MenuEventView == 3 )
menuItemWidth *= 0.6;
if( isScrolling )
menuItemWidth -= scrollBarWidth;
tColor ColorFg, ColorBg, ColorShortTextFg;
ColorShortTextFg = Theme.Color(clrMenuItemProgramShortTextFont);
if (Current) {
ColorFg = Theme.Color(clrItemCurrentFont);
ColorBg = Theme.Color(clrItemCurrentBg);
}
else {
if( Selectable ) {
ColorFg = Theme.Color(clrItemSelableFont);
ColorBg = Theme.Color(clrItemSelableBg);
} else {
ColorFg = Theme.Color(clrItemFont);
ColorBg = Theme.Color(clrItemBg);
}
}
menuPixmap->DrawRectangle(cRect(Config.decorBorderMenuItemSize, y, menuItemWidth, Height), ColorBg);
int Left = 0, Top = 0, LeftSecond = 0;
LeftSecond = Left = Config.decorBorderMenuItemSize + marginItem;
Top = y;
int imageTop = Top;
int w = 0;
if( !Channel ) {
int CurrentChannelNr = cDevice::CurrentChannel();
cChannel *ChannelLogo = Channels.GetByNumber(CurrentChannelNr);
cImage *img;
img = imgLoader.LoadLogo(ChannelLogo->Name(), 999, topBarHeight - marginItem*2);
if( img ) {
TopBarSetMenuLogo( ChannelLogo->Name() );
} else {
bool isRadioChannel = ( (!ChannelLogo->Vpid()) && (ChannelLogo->Apid(0)) ) ? true : false;
if( isRadioChannel ) {
img = imgLoader.LoadIcon("radio", 999, topBarHeight - marginItem*2);
if( img ) {
TopBarSetMenuLogo( ChannelLogo->Name() );
}
} else if( ChannelLogo->GroupSep() ) {
img = imgLoader.LoadIcon("changroup", 999, topBarHeight - marginItem*2);
if( img ) {
TopBarSetMenuIcon( ChannelLogo->Name() );
}
} else {
img = imgLoader.LoadIcon("tv", 999, topBarHeight - marginItem*2);
if( img ) {
TopBarSetMenuLogo( ChannelLogo->Name() );
}
}
}
}
if( Channel ) {
cString ws = cString::sprintf("%d", Channels.MaxNumber());
w = font->Width(ws);
if( !Channel->GroupSep() ) {
buffer = cString::sprintf("%d", Channel->Number());
int Width = font->Width(buffer);
if( Width < w )
Width = w;
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, Width, fontHeight, taRight);
}
Left += w + marginItem;
img = imgLoader.LoadLogo(Channel->Name(), fontHeight, fontHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
} else {
bool isRadioChannel = ((!Channel->Vpid())&&(Channel->Apid(0))) ? true : false;
if( isRadioChannel ) {
img = imgLoader.LoadIcon("radio", fontHeight, fontHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
}
} else if( Channel->GroupSep() ) {
img = imgLoader.LoadIcon("changroup", fontHeight, fontHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
}
} else {
img = imgLoader.LoadIcon("tv", fontHeight, fontHeight);
if( img ) {
imageTop = Top + (fontHeight - img->Height()) / 2;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
}
}
}
Left += fontHeight + marginItem * 2;
LeftSecond = Left;
w = menuWidth / 10 * 2;
menuPixmap->DrawText(cPoint(Left, Top), Channel->ShortName(true), ColorFg, ColorBg, font, w);
Left += w + marginItem * 2;
if( Event ) {
int PBWidth = menuItemWidth/20;
time_t now = time(NULL);
if( (now >= Event->StartTime()) && (now <= Event->EndTime()) ) {
int total = Event->EndTime() - Event->StartTime();
if( total > 0 ) {
// calculate progress bar
double progress = (int)roundf( (float)(time(NULL) - Event->StartTime()) / (float) (Event->Duration()) * 100.0);
if(progress < 0)
progress = 0.;
else if(progress > 100)
progress = 100;
int PBTop = y + (itemEventHeight - Config.MenuItemPadding)/2 - Config.decorProgressMenuItemSize/2 - Config.decorBorderMenuItemSize;
int PBLeft = Left;
int PBHeight = Config.decorProgressMenuItemSize;
if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) ) {
PBTop = y + fontHeight + fontSmlHeight + marginItem;
PBWidth = menuItemWidth - LeftSecond - scrollBarWidth - marginItem * 2;
if( isScrolling )
PBWidth += scrollBarWidth;
PBLeft = LeftSecond;
PBHeight = Config.decorProgressMenuItemSize / 2;
}
if( Current )
ProgressBarDrawRaw(menuPixmap, menuPixmap, cRect( PBLeft, PBTop, PBWidth, PBHeight),
cRect( PBLeft, PBTop, PBWidth, PBHeight), progress, 100,
Config.decorProgressMenuItemCurFg, Config.decorProgressMenuItemCurBarFg, Config.decorProgressMenuItemCurBg, Config.decorProgressMenuItemType, false);
else
ProgressBarDrawRaw(menuPixmap, menuPixmap, cRect( PBLeft, PBTop, PBWidth, PBHeight),
cRect( PBLeft, PBTop, PBWidth, PBHeight), progress, 100,
Config.decorProgressMenuItemFg, Config.decorProgressMenuItemBarFg, Config.decorProgressMenuItemBg, Config.decorProgressMenuItemType, false);
}
}
Left += PBWidth + marginItem*2;
}
}
if( WithDate && Event ) {
if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) && Channel )
w = fontSml->Width("XXX 99. ") + marginItem;
else
w = font->Width("XXX 99. ") + marginItem;
struct tm tm_r;
time_t Day = Event->StartTime();
localtime_r(&Day, &tm_r);
char buf[8];
strftime(buf, sizeof(buf), "%2d", &tm_r);
cString DateString = cString::sprintf("%s %s. ", *WeekDayName( (time_t)Event->StartTime()), buf );
if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) && Channel ) {
menuPixmap->DrawText(cPoint(LeftSecond, Top + fontHeight), DateString, ColorFg, ColorBg, fontSml, w);
LeftSecond += w + marginItem;
} else
menuPixmap->DrawText(cPoint(Left, Top), DateString, ColorFg, ColorBg, font, w, fontHeight, taRight);
Left += w + marginItem;
}
int imageHeight = fontHeight;
if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) && Channel && Event ) {
Top += fontHeight;
Left = LeftSecond;
imageHeight = fontSmlHeight;
menuPixmap->DrawText(cPoint(Left, Top), Event->GetTimeString(), ColorFg, ColorBg, fontSml);
Left += fontSml->Width( Event->GetTimeString() ) + marginItem;
} else if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) && Event ){
imageHeight = fontHeight;
menuPixmap->DrawText(cPoint(Left, Top), Event->GetTimeString(), ColorFg, ColorBg, font);
Left += font->Width( Event->GetTimeString() ) + marginItem;
} else if( Event ){
menuPixmap->DrawText(cPoint(Left, Top), Event->GetTimeString(), ColorFg, ColorBg, font);
Left += font->Width( Event->GetTimeString() ) + marginItem;
}
if( TimerMatch == tmFull ) {
img = imgLoader.LoadIcon("timer_full", imageHeight, imageHeight);
if( img ) {
imageTop = Top;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
}
} else if( TimerMatch == tmPartial ) {
img = imgLoader.LoadIcon("timer_partial", imageHeight, imageHeight);
if( img ) {
imageTop = Top;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
}
}
Left += imageHeight + marginItem;
if( Event ) {
if( Event->Vps() && (Event->Vps() - Event->StartTime()) ) {
img = imgLoader.LoadIcon("vps", imageHeight, imageHeight);
if( img ) {
imageTop = Top;
menuIconsPixmap->DrawImage( cPoint(Left, imageTop), *img );
}
}
Left += imageHeight + marginItem;
if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) && Channel ) {
menuPixmap->DrawText(cPoint(Left, Top), Event->Title(), ColorFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
if( Event->ShortText() ) {
Left += fontSml->Width( Event->Title() );
cString ShortText = cString::sprintf(" %s", Event->ShortText());
menuPixmap->DrawText(cPoint(Left, Top), ShortText, ColorShortTextFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
}
} else if( (Config.MenuEventView == 2 || Config.MenuEventView == 3) ) {
menuPixmap->DrawText(cPoint(Left, Top), Event->Title(), ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
if( Event->ShortText() ) {
Top += fontHeight;
menuPixmap->DrawText(cPoint(Left, Top), Event->ShortText(), ColorShortTextFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
}
} else {
menuPixmap->DrawText(cPoint(Left, Top), Event->Title(), ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
if( Event->ShortText() ) {
Left += font->Width(Event->Title());
cString ShortText = cString::sprintf(" %s", Event->ShortText());
menuPixmap->DrawText(cPoint(Left, Top), ShortText, ColorShortTextFg, ColorBg, font, menuItemWidth - Left - marginItem);
}
}
}
sDecorBorder ib;
ib.Left = Config.decorBorderMenuItemSize;
ib.Top = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuItemSize + y;
ib.Width = menuItemWidth - Config.decorBorderMenuItemSize*2;
if( isScrolling ) {
ib.Width -= scrollBarWidth;
}
ib.Width = menuItemWidth;
ib.Height = Height;
ib.Size = Config.decorBorderMenuItemSize;
ib.Type = Config.decorBorderMenuItemType;
if( Current ) {
ib.ColorFg = Config.decorBorderMenuItemCurFg;
ib.ColorBg = Config.decorBorderMenuItemCurBg;
} else {
if( Selectable ) {
ib.ColorFg = Config.decorBorderMenuItemSelFg;
ib.ColorBg = Config.decorBorderMenuItemSelBg;
} else {
ib.ColorFg = Config.decorBorderMenuItemFg;
ib.ColorBg = Config.decorBorderMenuItemBg;
}
}
DecorBorderDraw(ib.Left, ib.Top, ib.Width, ib.Height,
ib.Size, ib.Type, ib.ColorFg, ib.ColorBg, BorderMenuItem);
if( !isScrolling ) {
ItemBorderInsertUnique(ib);
}
if( Config.MenuEventView == 3 ) {
DrawItemExtraEvent(Event, "");
}
return true;
}
bool cFlatDisplayMenu::SetItemRecording(const cRecording *Recording, int Index, bool Current, bool Selectable, int Level, int Total, int New) {
if( Config.MenuRecordingView == 0 )
return false;
cString buffer;
cString RecName = GetRecordingName(Recording, Level, Total == 0);
int y = Index * itemRecordingHeight;
int Height = fontHeight;
if( Config.MenuRecordingView == 2 || Config.MenuRecordingView == 3 )
Height = fontHeight + fontSmlHeight + marginItem;
menuItemWidth = menuWidth - Config.decorBorderMenuItemSize*2;
if( Config.MenuRecordingView == 2 || Config.MenuRecordingView == 3 )
menuItemWidth *= 0.5;
if( isScrolling )
menuItemWidth -= scrollBarWidth;
tColor ColorFg, ColorBg, ColorExtraTextFg;
ColorExtraTextFg = Theme.Color(clrMenuItemReplayExtraTextFont);
if (Current) {
ColorFg = Theme.Color(clrItemCurrentFont);
ColorBg = Theme.Color(clrItemCurrentBg);
}
else {
if( Selectable ) {
ColorFg = Theme.Color(clrItemSelableFont);
ColorBg = Theme.Color(clrItemSelableBg);
} else {
ColorFg = Theme.Color(clrItemFont);
ColorBg = Theme.Color(clrItemBg);
}
}
menuPixmap->DrawRectangle(cRect(Config.decorBorderMenuItemSize, y, menuItemWidth, Height), ColorBg);
cImage *img = NULL;
cImage *imgRecNew = imgLoader.LoadIcon("recording_new", fontHeight, fontHeight);
cImage *imgRecNewSml = imgLoader.LoadIcon("recording_new", fontSmlHeight, fontSmlHeight);
cImage *imgRecCut = imgLoader.LoadIcon("recording_cutted", fontHeight, fontHeight);
int Left, Top;
Left = Config.decorBorderMenuItemSize + marginItem;
Top = y;
if( Config.MenuRecordingView == 1 ) {
int LeftWidth = Left + fontHeight + imgRecNew->Width() + imgRecCut->Width() +
marginItem * 3 + font->Width("99.99.99 99:99 99:99 ");
if( Total == 0 ) {
img = imgLoader.LoadIcon("recording", fontHeight, fontHeight);
if( img ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *img );
Left += fontHeight + marginItem;
}
//int Minutes = max(0, (Recording->LengthInSeconds() + 30) / 60);
int Minutes = (Recording->LengthInSeconds() + 30) / 60;
cString Length = cString::sprintf("%02d:%02d", Minutes / 60, Minutes % 60);
buffer = cString::sprintf("%s %s %s ", *ShortDateString(Recording->Start()), *TimeString(Recording->Start()), *Length);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
Left += font->Width( buffer );
if( Recording->IsNew() ) {
if( imgRecNew ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *imgRecNew );
}
}
Left += imgRecNew->Width() + marginItem;
if (Recording->IsEdited()) {
if( imgRecCut ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *imgRecCut );
}
}
Left += imgRecCut->Width() + marginItem;
menuPixmap->DrawText(cPoint(Left, Top), RecName, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
} else if( Total > 0 ) {
img = imgLoader.LoadIcon("folder", fontHeight, fontHeight);
if( img ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *img );
Left += img->Width() + marginItem;
}
buffer = cString::sprintf(" %d", Total);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, font->Width(" 9999"), fontHeight, taRight);
Left += font->Width(" 9999 ");
if( imgRecNew )
menuIconsPixmap->DrawImage( cPoint(Left, Top), *imgRecNew );
Left += imgRecNew->Width() + marginItem;
buffer = cString::sprintf("%d", New);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
Left += font->Width( buffer );
if( Config.MenuItemRecordingShowFolderDate != 0 ) {
buffer = cString::sprintf(" (%s)", *ShortDateString(GetLastRecTimeFromFolder(Recording, Level)));
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorExtraTextFg, ColorBg, font, menuItemWidth - Left - marginItem);
}
menuPixmap->DrawText(cPoint(LeftWidth, Top), RecName, ColorFg, ColorBg, font, menuItemWidth - LeftWidth - marginItem);
LeftWidth += font->Width(RecName) + marginItem*2;
} else if( Total == -1 ) {
img = imgLoader.LoadIcon("folder", fontHeight, fontHeight);
if( img ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *img );
Left += img->Width() + marginItem;
}
menuPixmap->DrawText(cPoint(Left, Top), Recording->FileName(), ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
}
} else {
if( Total == 0 ) {
img = imgLoader.LoadIcon("recording", fontHeight, fontHeight);
if( img ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *img );
Left += fontHeight + marginItem;
}
int ImagesWidth = imgRecNew->Width() + imgRecCut->Width() + marginItem*2 + scrollBarWidth;
if( isScrolling )
ImagesWidth -= scrollBarWidth;
menuPixmap->DrawText(cPoint(Left, Top), RecName, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem - ImagesWidth);
Top += fontHeight;
int Minutes = (Recording->LengthInSeconds() + 30) / 60;
cString Length = cString::sprintf("%02d:%02d", Minutes / 60, Minutes % 60);
buffer = cString::sprintf("%s %s %s ", *ShortDateString(Recording->Start()), *TimeString(Recording->Start()), *Length);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
Top -= fontHeight;
Left = menuItemWidth - ImagesWidth;
if( Recording->IsNew() ) {
if( imgRecNew ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *imgRecNew );
}
}
Left += imgRecNew->Width() + marginItem;
if (Recording->IsEdited()) {
if( imgRecCut ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *imgRecCut );
}
}
Left += imgRecCut->Width() + marginItem;
} else if( Total > 0 ) {
img = imgLoader.LoadIcon("folder", fontHeight, fontHeight);
if( img ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *img );
Left += img->Width() + marginItem;
}
menuPixmap->DrawText(cPoint(Left, Top), RecName, ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
Top += fontHeight;
buffer = cString::sprintf(" %d", Total);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, fontSml, fontSml->Width(" 9999"), fontSmlHeight, taRight);
Left += fontSml->Width(" 9999 ");
if( imgRecNewSml )
menuIconsPixmap->DrawImage( cPoint(Left, Top), *imgRecNewSml );
Left += imgRecNewSml->Width() + marginItem;
buffer = cString::sprintf("%d", New);
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
Left += fontSml->Width( buffer );
if( Config.MenuItemRecordingShowFolderDate != 0 ) {
buffer = cString::sprintf(" (%s)", *ShortDateString(GetLastRecTimeFromFolder(Recording, Level)));
menuPixmap->DrawText(cPoint(Left, Top), buffer, ColorExtraTextFg, ColorBg, fontSml, menuItemWidth - Left - marginItem);
}
} else if( Total == -1 ) {
img = imgLoader.LoadIcon("folder", fontHeight, fontHeight);
if( img ) {
menuIconsPixmap->DrawImage( cPoint(Left, Top), *img );
Left += img->Width() + marginItem;
}
menuPixmap->DrawText(cPoint(Left, Top), Recording->FileName(), ColorFg, ColorBg, font, menuItemWidth - Left - marginItem);
}
}
sDecorBorder ib;
ib.Left = Config.decorBorderMenuItemSize;
ib.Top = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuItemSize + y;
ib.Width = menuItemWidth - Config.decorBorderMenuItemSize*2;
if( isScrolling ) {
ib.Width -= scrollBarWidth;
}
ib.Width = menuItemWidth;
ib.Height = Height;
ib.Size = Config.decorBorderMenuItemSize;
ib.Type = Config.decorBorderMenuItemType;
if( Current ) {
ib.ColorFg = Config.decorBorderMenuItemCurFg;
ib.ColorBg = Config.decorBorderMenuItemCurBg;
} else {
if( Selectable ) {
ib.ColorFg = Config.decorBorderMenuItemSelFg;
ib.ColorBg = Config.decorBorderMenuItemSelBg;
} else {
ib.ColorFg = Config.decorBorderMenuItemFg;
ib.ColorBg = Config.decorBorderMenuItemBg;
}
}
DecorBorderDraw(ib.Left, ib.Top, ib.Width, ib.Height,
ib.Size, ib.Type, ib.ColorFg, ib.ColorBg, BorderMenuItem);
if( !isScrolling ) {
ItemBorderInsertUnique(ib);
}
if( Config.MenuRecordingView == 3 && Current ) {
DrawItemExtraRecording(Recording, tr("no recording info"));
}
return true;
}
void cFlatDisplayMenu::SetEvent(const cEvent *Event) {
if( !Event )
return;
ShowEvent = true;
ShowRecording = false;
ShowText = false;
ItemBorderClear();
cLeft = Config.decorBorderMenuContentSize;
cTop = chTop + marginItem*3 + fontHeight + fontSmlHeight*2 +
Config.decorBorderMenuContentSize + Config.decorBorderMenuContentHeadSize;
cWidth = menuWidth - Config.decorBorderMenuContentSize*2;
cHeight = osdHeight - (topBarHeight + Config.decorBorderTopBarSize*2 +
buttonsHeight + Config.decorBorderButtonSize*2 + marginItem*3 +
chHeight + Config.decorBorderMenuContentHeadSize*2 + Config.decorBorderMenuContentSize*2);
if( !ButtonsDrawn() )
cHeight += buttonsHeight + Config.decorBorderButtonSize*2;
menuItemWidth = cWidth;
contentHeadPixmap->Fill(clrTransparent);
contentHeadPixmap->DrawRectangle(cRect(0, 0, menuWidth, fontHeight + fontSmlHeight*2 + marginItem*2), Theme.Color(clrScrollbarBg));
cString date = Event->GetDateString();
cString startTime = Event->GetTimeString();
cString endTime = Event->GetEndTimeString();
cString timeString = cString::sprintf("%s %s - %s", *date, *startTime, *endTime);
cString title = Event->Title();
cString shortText = Event->ShortText();
contentHeadPixmap->DrawText(cPoint(marginItem, marginItem), timeString, Theme.Color(clrMenuEventFontInfo), Theme.Color(clrMenuEventBg), fontSml, menuWidth - marginItem*2);
contentHeadPixmap->DrawText(cPoint(marginItem, marginItem + fontSmlHeight), title, Theme.Color(clrMenuEventFontTitle), Theme.Color(clrMenuEventBg), font, menuWidth - marginItem*2);
contentHeadPixmap->DrawText(cPoint(marginItem, marginItem + fontSmlHeight + fontHeight), shortText, Theme.Color(clrMenuEventFontInfo), Theme.Color(clrMenuEventBg), fontSml, menuWidth - marginItem*2);
DecorBorderDraw(chLeft, chTop, chWidth, chHeight, Config.decorBorderMenuContentHeadSize, Config.decorBorderMenuContentHeadType,
Config.decorBorderMenuContentHeadFg, Config.decorBorderMenuContentHeadBg);
// Description
ostringstream text;
if( !isempty(Event->Description()) ) {
text << Event->Description();
}
if( Config.EpgAdditionalInfoShow ) {
text << endl;
const cComponents *Components = Event->Components();
if (Components) {
ostringstream audio;
bool firstAudio = true;
const char *audio_type = NULL;
ostringstream subtitle;
bool firstSubtitle = true;
for (int i = 0; i < Components->NumComponents(); i++) {
const tComponent *p = Components->Component(i);
switch (p->stream) {
case sc_video_MPEG2:
if (p->description)
text << endl << tr("Video") << ": " << p->description << " (MPEG2)";
else
text << endl << tr("Video") << ": MPEG2";
break;
case sc_video_H264_AVC:
if (p->description)
text << endl << tr("Video") << ": " << p->description << " (H.264)";
else
text << endl << tr("Video") << ": H.264";
break;
case sc_audio_MP2:
case sc_audio_AC3:
case sc_audio_HEAAC:
if (firstAudio)
firstAudio = false;
else
audio << ", ";
switch (p->stream) {
case sc_audio_MP2:
// workaround for wrongfully used stream type X 02 05 for AC3
if (p->type == 5)
audio_type = "AC3";
else
audio_type = "MP2";
break;
case sc_audio_AC3:
audio_type = "AC3"; break;
case sc_audio_HEAAC:
audio_type = "HEAAC"; break;
}
if (p->description)
audio << p->description << " (" << audio_type << ", " << p->language << ")";
else
audio << p->language << " (" << audio_type << ")";
break;
case sc_subtitle:
if (firstSubtitle)
firstSubtitle = false;
else
subtitle << ", ";
if (p->description)
subtitle << p->description << " (" << p->language << ")";
else
subtitle << p->language;
break;
}
}
if (audio.str().length() > 0)
text << endl << tr("Audio") << ": "<< audio.str();
if (subtitle.str().length() > 0)
text << endl << tr("Subtitle") << ": "<< subtitle.str();
}
}
bool contentScrollable = ContentWillItBeScrollable(cWidth, cHeight, text.str().c_str(), 0);
if( contentScrollable ) {
cWidth -= scrollBarWidth;
}
ContentCreate(cLeft, cTop, cWidth, cHeight, 0);
ContentSet( text.str().c_str(), Theme.Color(clrMenuEventFontInfo), Theme.Color(clrMenuEventBg) );
if( ContentScrollable() ) {
DrawScrollbar(ContentScrollTotal(), ContentScrollOffset(), ContentVisibleLines(), contentTop - scrollBarTop, ContentGetHeight(), ContentScrollOffset() > 0, ContentScrollOffset() + ContentVisibleLines() < ContentScrollTotal(), true);
}
if( Config.MenuContentFullSize || ContentScrollable() )
DecorBorderDraw(cLeft, cTop, cWidth, ContentGetHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg);
else
DecorBorderDraw(cLeft, cTop, cWidth, ContentGetTextHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg);
}
void cFlatDisplayMenu::DrawItemExtraRecording(const cRecording *Recording, cString EmptyText) {
cLeft = menuItemWidth + Config.decorBorderMenuItemSize*2 + Config.decorBorderMenuContentSize + marginItem;
if( isScrolling )
cLeft += scrollBarWidth;
cTop = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuContentSize;
cWidth = menuWidth - cLeft - Config.decorBorderMenuContentSize;
cHeight = osdHeight - (topBarHeight + Config.decorBorderTopBarSize*2 +
buttonsHeight + Config.decorBorderButtonSize*2 + marginItem*3 + Config.decorBorderMenuContentSize*2);
ostringstream text;
if( Recording ) {
const cRecordingInfo *recInfo = Recording->Info();
text.imbue(std::locale(""));
if (!isempty(recInfo->Description()))
text << recInfo->Description() << endl << endl;
// lent from skinelchi
if( Config.RecordingAdditionalInfoShow ) {
cChannel *channel = Channels.GetByChannelID(((cRecordingInfo *)recInfo)->ChannelID());
if (channel)
text << trVDR("Channel") << ": " << channel->Number() << " - " << channel->Name() << endl;
cMarks marks;
bool hasMarks = marks.Load(Recording->FileName(), Recording->FramesPerSecond(), Recording->IsPesRecording()) && marks.Count();
cIndexFile *index = new cIndexFile(Recording->FileName(), false, Recording->IsPesRecording());
int lastIndex = 0;
int cuttedLength = 0;
long cutinframe = 0;
unsigned long long recsize = 0;
unsigned long long recsizecutted = 0;
unsigned long long cutinoffset = 0;
unsigned long long filesize[100000];
filesize[0] = 0;
int i = 0;
int imax = 999;
struct stat filebuf;
cString filename;
int rc = 0;
do {
if (Recording->IsPesRecording())
filename = cString::sprintf("%s/%03d.vdr", Recording->FileName(), ++i);
else {
filename = cString::sprintf("%s/%05d.ts", Recording->FileName(), ++i);
imax = 99999;
}
rc=stat(filename, &filebuf);
if (rc == 0)
filesize[i] = filesize[i-1] + filebuf.st_size;
else {
if (ENOENT != errno) {
esyslog ("skinflatplus: error determining file size of \"%s\" %d (%s)", (const char *)filename, errno, strerror(errno));
recsize = 0;
}
}
} while( i <= imax && !rc );
recsize = filesize[i-1];
if (hasMarks && index) {
uint16_t FileNumber;
off_t FileOffset;
bool cutin = true;
cMark *mark = marks.First();
while (mark) {
long position = mark->Position();
index->Get(position, &FileNumber, &FileOffset);
if (cutin) {
cutinframe = position;
cutin = false;
cutinoffset = filesize[FileNumber-1] + FileOffset;
} else {
cuttedLength += position - cutinframe;
cutin = true;
recsizecutted += filesize[FileNumber-1] + FileOffset - cutinoffset;
}
cMark *nextmark = marks.Next(mark);
mark = nextmark;
}
if( !cutin ) {
cuttedLength += index->Last() - cutinframe;
index->Get(index->Last() - 1, &FileNumber, &FileOffset);
recsizecutted += filesize[FileNumber-1] + FileOffset - cutinoffset;
}
}
if (index) {
lastIndex = index->Last();
text << tr("Length") << ": " << *IndexToHMSF(lastIndex, false, Recording->FramesPerSecond());
if (hasMarks)
text << " (" << tr("cutted") << ": " << *IndexToHMSF(cuttedLength, false, Recording->FramesPerSecond()) << ")";
text << endl;
}
delete index;
if (recsize > MEGABYTE(1023))
text << tr("Size") << ": " << fixed << setprecision(2) << (float)recsize / MEGABYTE(1024) << " GB";
else
text << tr("Size") << ": " << recsize / MEGABYTE(1) << " MB";
if( hasMarks )
if (recsize > MEGABYTE(1023))
text << " (" << tr("cutted") << ": " << fixed << setprecision(2) << (float)recsizecutted/MEGABYTE(1024) << " GB)";
else
text << " (" << tr("cutted") << ": " << recsizecutted/MEGABYTE(1) << " MB)";
text << endl << trVDR("Priority") << ": " << Recording->Priority() << ", " << trVDR("Lifetime") << ": " << Recording->Lifetime() << endl;
if( lastIndex ) {
text << tr("format") << ": " << (Recording->IsPesRecording() ? "PES" : "TS") << ", " << tr("bit rate") << ": ~ " << fixed << setprecision (2) << (float)recsize/lastIndex*Recording->FramesPerSecond()*8/MEGABYTE(1) << " MBit/s (Video + Audio)";
}
const cComponents *Components = recInfo->Components();
if( Components ) {
ostringstream audio;
bool firstAudio = true;
const char *audio_type = NULL;
ostringstream subtitle;
bool firstSubtitle = true;
for (int i = 0; i < Components->NumComponents(); i++) {
const tComponent *p = Components->Component(i);
switch (p->stream) {
case sc_video_MPEG2:
text << endl << tr("Video") << ": " << p->description << " (MPEG2)";
break;
case sc_video_H264_AVC:
text << endl << tr("Video") << ": " << p->description << " (H.264)";
break;
case sc_audio_MP2:
case sc_audio_AC3:
case sc_audio_HEAAC:
if (firstAudio)
firstAudio = false;
else
audio << ", ";
switch (p->stream) {
case sc_audio_MP2:
// workaround for wrongfully used stream type X 02 05 for AC3
if (p->type == 5)
audio_type = "AC3";
else
audio_type = "MP2";
break;
case sc_audio_AC3:
audio_type = "AC3"; break;
case sc_audio_HEAAC:
audio_type = "HEAAC"; break;
}
if (p->description)
audio << p->description << " (" << audio_type << ", " << p->language << ")";
else
audio << p->language << " (" << audio_type << ")";
break;
case sc_subtitle:
if (firstSubtitle)
firstSubtitle = false;
else
subtitle << ", ";
if (p->description)
subtitle << p->description << " (" << p->language << ")";
else
subtitle << p->language;
break;
}
}
if (audio.str().length() > 0)
text << endl << tr("Audio") << ": "<< audio.str();
if (subtitle.str().length() > 0)
text << endl << tr("Subtitle") << ": "<< subtitle.str();
}
if (recInfo->Aux()) {
string str_epgsearch = xml_substring(recInfo->Aux(), "", "");
string channel, searchtimer, pattern;
if (!str_epgsearch.empty()) {
channel = xml_substring(str_epgsearch, "", "");
searchtimer = xml_substring(str_epgsearch, "", "");
if (searchtimer.empty())
searchtimer = xml_substring(str_epgsearch, "", "");
}
string str_vdradmin = xml_substring(recInfo->Aux(), "", "");
if (!str_vdradmin.empty()) {
pattern = xml_substring(str_vdradmin, "", "");
}
if ((!channel.empty() && !searchtimer.empty()) || !pattern.empty()) {
text << endl << endl << tr("additional information") << ":" << endl;
if (!channel.empty() && !searchtimer.empty()) {
text << "EPGsearch: " << tr("channel") << ": " << channel << ", " << tr("search pattern") << ": " << searchtimer;
}
if (!pattern.empty()) {
text << "VDRadmin-AM: " << tr("search pattern") << ": " << pattern;
}
}
}
}
} else
text << *EmptyText;
ContentCreate(cLeft, cTop, cWidth, cHeight, 2);
ContentSet( text.str().c_str(), Theme.Color(clrMenuEventFontInfo), Theme.Color(clrMenuEventBg) );
DecorBorderClearByFrom(BorderContent);
if( Config.MenuContentFullSize )
DecorBorderDraw(cLeft, cTop, cWidth, ContentGetHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg, BorderContent);
else
DecorBorderDraw(cLeft, cTop, cWidth, ContentGetTextHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg, BorderContent);
}
void cFlatDisplayMenu::SetRecording(const cRecording *Recording) {
if( !Recording )
return;
ShowEvent = false;
ShowRecording = true;
ShowText = false;
ItemBorderClear();
const cRecordingInfo *recInfo = Recording->Info();
chLeft = Config.decorBorderMenuContentHeadSize;
chTop = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuContentHeadSize;
chWidth = menuWidth - Config.decorBorderMenuContentHeadSize*2;
chHeight = fontHeight + fontSmlHeight*2 + marginItem*2;
contentHeadPixmap = osd->CreatePixmap(1, cRect(chLeft, chTop, chWidth, chHeight));
cLeft = Config.decorBorderMenuContentSize;
cTop = chTop + marginItem*3 + fontHeight + fontSmlHeight*2 +
Config.decorBorderMenuContentSize + Config.decorBorderMenuContentHeadSize;
cWidth = menuWidth - Config.decorBorderMenuContentSize*2;
cHeight = osdHeight - (topBarHeight + Config.decorBorderTopBarSize*2 +
buttonsHeight + Config.decorBorderButtonSize*2 + marginItem*3 +
chHeight + Config.decorBorderMenuContentHeadSize*2 + Config.decorBorderMenuContentSize*2);
if( !ButtonsDrawn() )
cHeight += buttonsHeight + Config.decorBorderButtonSize*2;
menuItemWidth = cWidth;
ostringstream text;
text.imbue(std::locale(""));
if (!isempty(recInfo->Description()))
text << recInfo->Description() << endl << endl;
// lent from skinelchi
if( Config.RecordingAdditionalInfoShow ) {
cChannel *channel = Channels.GetByChannelID(((cRecordingInfo *)recInfo)->ChannelID());
if (channel)
text << trVDR("Channel") << ": " << channel->Number() << " - " << channel->Name() << endl;
cMarks marks;
bool hasMarks = marks.Load(Recording->FileName(), Recording->FramesPerSecond(), Recording->IsPesRecording()) && marks.Count();
cIndexFile *index = new cIndexFile(Recording->FileName(), false, Recording->IsPesRecording());
int lastIndex = 0;
int cuttedLength = 0;
long cutinframe = 0;
unsigned long long recsize = 0;
unsigned long long recsizecutted = 0;
unsigned long long cutinoffset = 0;
unsigned long long filesize[100000];
filesize[0] = 0;
int i = 0;
int imax = 999;
struct stat filebuf;
cString filename;
int rc = 0;
do {
if (Recording->IsPesRecording())
filename = cString::sprintf("%s/%03d.vdr", Recording->FileName(), ++i);
else {
filename = cString::sprintf("%s/%05d.ts", Recording->FileName(), ++i);
imax = 99999;
}
rc=stat(filename, &filebuf);
if (rc == 0)
filesize[i] = filesize[i-1] + filebuf.st_size;
else {
if (ENOENT != errno) {
esyslog ("skinflatplus: error determining file size of \"%s\" %d (%s)", (const char *)filename, errno, strerror(errno));
recsize = 0;
}
}
} while( i <= imax && !rc );
recsize = filesize[i-1];
if (hasMarks && index) {
uint16_t FileNumber;
off_t FileOffset;
bool cutin = true;
cMark *mark = marks.First();
while (mark) {
long position = mark->Position();
index->Get(position, &FileNumber, &FileOffset);
if (cutin) {
cutinframe = position;
cutin = false;
cutinoffset = filesize[FileNumber-1] + FileOffset;
} else {
cuttedLength += position - cutinframe;
cutin = true;
recsizecutted += filesize[FileNumber-1] + FileOffset - cutinoffset;
}
cMark *nextmark = marks.Next(mark);
mark = nextmark;
}
if( !cutin ) {
cuttedLength += index->Last() - cutinframe;
index->Get(index->Last() - 1, &FileNumber, &FileOffset);
recsizecutted += filesize[FileNumber-1] + FileOffset - cutinoffset;
}
}
if (index) {
lastIndex = index->Last();
text << tr("Length") << ": " << *IndexToHMSF(lastIndex, false, Recording->FramesPerSecond());
if (hasMarks)
text << " (" << tr("cutted") << ": " << *IndexToHMSF(cuttedLength, false, Recording->FramesPerSecond()) << ")";
text << endl;
}
delete index;
if (recsize > MEGABYTE(1023))
text << tr("Size") << ": " << fixed << setprecision(2) << (float)recsize / MEGABYTE(1024) << " GB";
else
text << tr("Size") << ": " << recsize / MEGABYTE(1) << " MB";
if( hasMarks )
if (recsize > MEGABYTE(1023))
text << " (" << tr("cutted") << ": " << fixed << setprecision(2) << (float)recsizecutted/MEGABYTE(1024) << " GB)";
else
text << " (" << tr("cutted") << ": " << recsizecutted/MEGABYTE(1) << " MB)";
text << endl << trVDR("Priority") << ": " << Recording->Priority() << ", " << trVDR("Lifetime") << ": " << Recording->Lifetime() << endl;
if( lastIndex ) {
text << tr("format") << ": " << (Recording->IsPesRecording() ? "PES" : "TS") << ", " << tr("bit rate") << ": ~ " << fixed << setprecision (2) << (float)recsize/lastIndex*Recording->FramesPerSecond()*8/MEGABYTE(1) << " MBit/s (Video + Audio)";
}
const cComponents *Components = recInfo->Components();
if( Components ) {
ostringstream audio;
bool firstAudio = true;
const char *audio_type = NULL;
ostringstream subtitle;
bool firstSubtitle = true;
for (int i = 0; i < Components->NumComponents(); i++) {
const tComponent *p = Components->Component(i);
switch (p->stream) {
case sc_video_MPEG2:
text << endl << tr("Video") << ": " << p->description << " (MPEG2)";
break;
case sc_video_H264_AVC:
text << endl << tr("Video") << ": " << p->description << " (H.264)";
break;
case sc_audio_MP2:
case sc_audio_AC3:
case sc_audio_HEAAC:
if (firstAudio)
firstAudio = false;
else
audio << ", ";
switch (p->stream) {
case sc_audio_MP2:
// workaround for wrongfully used stream type X 02 05 for AC3
if (p->type == 5)
audio_type = "AC3";
else
audio_type = "MP2";
break;
case sc_audio_AC3:
audio_type = "AC3"; break;
case sc_audio_HEAAC:
audio_type = "HEAAC"; break;
}
if (p->description)
audio << p->description << " (" << audio_type << ", " << p->language << ")";
else
audio << p->language << " (" << audio_type << ")";
break;
case sc_subtitle:
if (firstSubtitle)
firstSubtitle = false;
else
subtitle << ", ";
if (p->description)
subtitle << p->description << " (" << p->language << ")";
else
subtitle << p->language;
break;
}
}
if (audio.str().length() > 0)
text << endl << tr("Audio") << ": "<< audio.str();
if (subtitle.str().length() > 0)
text << endl << tr("Subtitle") << ": "<< subtitle.str();
}
if (recInfo->Aux()) {
string str_epgsearch = xml_substring(recInfo->Aux(), "", "");
string channel, searchtimer, pattern;
if (!str_epgsearch.empty()) {
channel = xml_substring(str_epgsearch, "", "");
searchtimer = xml_substring(str_epgsearch, "", "");
if (searchtimer.empty())
searchtimer = xml_substring(str_epgsearch, "", "");
}
string str_vdradmin = xml_substring(recInfo->Aux(), "", "");
if (!str_vdradmin.empty()) {
pattern = xml_substring(str_vdradmin, "", "");
}
if ((!channel.empty() && !searchtimer.empty()) || !pattern.empty()) {
text << endl << endl << tr("additional information") << ":" << endl;
if (!channel.empty() && !searchtimer.empty()) {
text << "EPGsearch: " << tr("channel") << ": " << channel << ", " << tr("search pattern") << ": " << searchtimer;
}
if (!pattern.empty()) {
text << "VDRadmin-AM: " << tr("search pattern") << ": " << pattern;
}
}
}
}
bool contentScrollable = ContentWillItBeScrollable(cWidth, cHeight, text.str().c_str(), 0);
if( contentScrollable ) {
cWidth -= scrollBarWidth;
}
ContentCreate(cLeft, cTop, cWidth, cHeight, 0);
contentHeadPixmap->Fill(clrTransparent);
contentHeadPixmap->DrawRectangle(cRect(0, 0, menuWidth, fontHeight + fontSmlHeight*2 + marginItem*2), Theme.Color(clrScrollbarBg));
cString timeString = cString::sprintf("%s %s %s", *DateString(Recording->Start()), *TimeString(Recording->Start()), recInfo->ChannelName() ? recInfo->ChannelName() : "");
cString title = recInfo->Title();
if( isempty(title) )
title = Recording->Name();
cString shortText = recInfo->ShortText();
contentHeadPixmap->DrawText(cPoint(marginItem, marginItem), timeString, Theme.Color(clrMenuRecFontInfo), Theme.Color(clrMenuRecBg), fontSml, menuWidth - marginItem*2);
contentHeadPixmap->DrawText(cPoint(marginItem, marginItem + fontSmlHeight), title, Theme.Color(clrMenuRecFontTitle), Theme.Color(clrMenuRecBg), font, menuWidth - marginItem*2);
contentHeadPixmap->DrawText(cPoint(marginItem, marginItem + fontSmlHeight + fontHeight), shortText, Theme.Color(clrMenuRecFontInfo), Theme.Color(clrMenuRecBg), fontSml, menuWidth - marginItem*2);
DecorBorderDraw(chLeft, chTop, chWidth, chHeight, Config.decorBorderMenuContentHeadSize, Config.decorBorderMenuContentHeadType,
Config.decorBorderMenuContentHeadFg, Config.decorBorderMenuContentHeadBg);
ContentSet( text.str().c_str(), Theme.Color(clrMenuRecFontInfo), Theme.Color(clrMenuRecBg) );
if( ContentScrollable() ) {
DrawScrollbar(ContentScrollTotal(), ContentScrollOffset(), ContentVisibleLines(), contentTop - scrollBarTop, ContentGetHeight(), ContentScrollOffset() > 0, ContentScrollOffset() + ContentVisibleLines() < ContentScrollTotal(), true);
}
RecordingBorder.Left = cLeft;
RecordingBorder.Top = cTop;
RecordingBorder.Width = cWidth;
RecordingBorder.Height = ContentGetHeight();
RecordingBorder.Size = Config.decorBorderMenuContentSize;
RecordingBorder.Type = Config.decorBorderMenuContentType;
RecordingBorder.ColorFg = Config.decorBorderMenuContentFg;
RecordingBorder.ColorBg = Config.decorBorderMenuContentBg;
RecordingBorder.From = BorderMenuRecord;
if( Config.MenuContentFullSize || ContentScrollable() )
DecorBorderDraw(RecordingBorder.Left, RecordingBorder.Top, RecordingBorder.Width, RecordingBorder.Height,
RecordingBorder.Size, RecordingBorder.Type,
RecordingBorder.ColorFg, RecordingBorder.ColorBg, RecordingBorder.From);
else
DecorBorderDraw(RecordingBorder.Left, RecordingBorder.Top, RecordingBorder.Width, ContentGetTextHeight(),
RecordingBorder.Size, RecordingBorder.Type,
RecordingBorder.ColorFg, RecordingBorder.ColorBg, RecordingBorder.From);
}
void cFlatDisplayMenu::SetText(const char *Text, bool FixedFont) {
if( !Text )
return;
ShowEvent = false;
ShowRecording = false;
ShowText = true;
ItemBorderClear();
contentHeadPixmap->Fill(clrTransparent);
int Left = Config.decorBorderMenuContentSize;
int Top = topBarHeight + marginItem + Config.decorBorderTopBarSize*2 + Config.decorBorderMenuContentSize;
int Width = menuWidth - Config.decorBorderMenuContentSize*2;
int Height = osdHeight - (topBarHeight + Config.decorBorderTopBarSize*2 +
buttonsHeight + Config.decorBorderButtonSize*2 + Config.decorBorderMenuContentSize*2 + marginItem);
if( !ButtonsDrawn() )
Height += buttonsHeight + Config.decorBorderButtonSize*2;
menuItemWidth = Width;
bool contentScrollable = ContentWillItBeScrollable(Width, Height, Text, FixedFont);
if( contentScrollable ) {
Width -= scrollBarWidth;
}
if( FixedFont ) {
ContentCreate(Left, Top, Width, Height, 1);
ContentSet( Text, Theme.Color(clrMenuTextFixedFont), Theme.Color(clrMenuTextBg) );
} else {
ContentCreate(Left, Top, Width, Height, 1);
ContentSet( Text, Theme.Color(clrMenuTextFont), Theme.Color(clrMenuTextBg) );
}
if( ContentScrollable() )
DrawScrollbar(ContentScrollTotal(), ContentScrollOffset(), ContentVisibleLines(), contentTop - scrollBarTop, ContentGetHeight(), ContentScrollOffset() > 0, ContentScrollOffset() + ContentVisibleLines() < ContentScrollTotal(), true);
if( Config.MenuContentFullSize || ContentScrollable() )
DecorBorderDraw(Left, Top, Width, ContentGetHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg);
else
DecorBorderDraw(Left, Top, Width, ContentGetTextHeight(), Config.decorBorderMenuContentSize, Config.decorBorderMenuContentType,
Config.decorBorderMenuContentFg, Config.decorBorderMenuContentBg);
}
int cFlatDisplayMenu::GetTextAreaWidth(void) const {
return menuWidth - (marginItem*2);
}
const cFont *cFlatDisplayMenu::GetTextAreaFont(bool FixedFont) const {
const cFont *rfont = FixedFont ? fontFixed : font;
return rfont;
}
void cFlatDisplayMenu::Flush(void) {
TopBarUpdate();
osd->Flush();
}
void cFlatDisplayMenu::ItemBorderInsertUnique(sDecorBorder ib) {
std::list::iterator it;
for( it = ItemsBorder.begin(); it != ItemsBorder.end(); it++ ) {
if( (*it).Left == ib.Left && (*it).Top == ib.Top ) {
(*it).Left = ib.Left;
(*it).Top = ib.Top;
(*it).Width = ib.Width;
(*it).Height = ib.Height;
(*it).Size = ib.Size;
(*it).Type = ib.Type;
(*it).ColorFg = ib.ColorFg;
(*it).ColorBg = ib.ColorBg;
(*it).From = ib.From;
return;
}
}
ItemsBorder.push_back(ib);
}
void cFlatDisplayMenu::ItemBorderDrawAllWithScrollbar(void) {
std::list::iterator it;
for( it = ItemsBorder.begin(); it != ItemsBorder.end(); it++ ) {
DecorBorderDraw((*it).Left, (*it).Top, (*it).Width - scrollBarWidth, (*it).Height, (*it).Size, (*it).Type,
(*it).ColorFg, (*it).ColorBg, BorderMenuItem);
}
}
void cFlatDisplayMenu::ItemBorderDrawAllWithoutScrollbar(void) {
std::list::iterator it;
for( it = ItemsBorder.begin(); it != ItemsBorder.end(); it++ ) {
DecorBorderDraw((*it).Left, (*it).Top, (*it).Width + scrollBarWidth, (*it).Height, (*it).Size, (*it).Type,
(*it).ColorFg, (*it).ColorBg, BorderMenuItem);
}
}
void cFlatDisplayMenu::ItemBorderClear(void) {
ItemsBorder.clear();
}
time_t cFlatDisplayMenu::GetLastRecTimeFromFolder(const cRecording *Recording, int Level) {
std::string RecFolder = GetRecordingName(Recording, Level, true);
time_t RecStart = Recording->Start();
for(cRecording *rec = Recordings.First(); rec; rec = Recordings.Next(rec)) {
std::string RecFolder2 = GetRecordingName(rec, Level, true);
if( RecFolder == RecFolder2 ) { // recordings must be in the same folder
time_t RecStart2 = rec->Start();
if( Config.MenuItemRecordingShowFolderDate == 1) { // newest
if( RecStart2 > RecStart )
RecStart = RecStart2;
} else if( Config.MenuItemRecordingShowFolderDate == 2 ) // oldest
if( RecStart2 < RecStart )
RecStart = RecStart2;
}
}
return RecStart;
}
// returns the string between start and end or an empty string if not found
string cFlatDisplayMenu::xml_substring(string source, const char* str_start, const char* str_end) {
size_t start = source.find(str_start);
size_t end = source.find(str_end);
if (string::npos != start && string::npos != end) {
return (source.substr(start + strlen(str_start), end - start - strlen(str_start)));
}
return string();
}
const char * cFlatDisplayMenu::GetRecordingName(const cRecording *Recording, int Level, bool isFolder) {
std::string recNamePart;
std::string recName = Recording->Name();
try {
std::vector tokens;
std::istringstream f(recName.c_str());
std::string s;
while (std::getline(f, s, FOLDERDELIMCHAR)) {
tokens.push_back(s);
}
recNamePart = tokens.at(Level);
if(!isFolder && Recording->IsEdited() ) {
recNamePart = recNamePart.substr(1);
}
} catch (...) {
recNamePart = recName.c_str();
}
if( Config.MenuItemRecordingClearPercent && isFolder ) {
if( recNamePart[0] == '%' ) {
recNamePart.erase(0, 1);
}
}
return recNamePart.c_str();
}