#include #include #include "imageloader.h" #include "imagecache.h" #include "tools.h" #include "recmenus.h" #include "recmenuitem.h" // --- cRecMenuItem ------------------------------------------------------------- cRecMenuItem::cRecMenuItem(void) { height = 0; action = rmsNotConsumed; defaultBackground = true; drawn = false; font = fontManager.FontRecMenuItem; fontSmall = fontManager.FontRecMenuItemSmall; fontLarge = fontManager.FontRecMenuItemLarge; } cRecMenuItem::~cRecMenuItem(void) { } void cRecMenuItem::SetGeometry(int x, int y, int width) { this->x = x; this->y = y; this->width = width; } void cRecMenuItem::SetPixmaps(void) { if (!pixmap) pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); else pixmap->SetViewPort(cRect(x, y, width, height)); } void cRecMenuItem::setBackground(void) { if (tvguideConfig.style == eStyleGraphical) { if (defaultBackground) { drawBackgroundGraphical(bgButton, active); } colorTextBack = clrTransparent; colorText = (active)?theme.Color(clrFontActive):theme.Color(clrFont); } else { if (active) { color = theme.Color(clrHighlight); colorBlending = theme.Color(clrHighlightBlending); colorText = theme.Color(clrFontActive); } else { color = theme.Color(clrGrid1); colorBlending = theme.Color(clrGrid1Blending); colorText = theme.Color(clrFont); } colorTextBack = (tvguideConfig.style == eStyleFlat)?color:clrTransparent; if (defaultBackground) { drawBackground(); drawBorder(); } } } // --- cRecMenuItemButton ------------------------------------------------------- cRecMenuItemButton::cRecMenuItemButton(const char *text, eRecMenuState action, bool active, bool halfWidth, bool alignLeft, bool largeFont) { selectable = true; this->text = text; this->action = action; this->active = active; fontButtons = (largeFont)?fontLarge:font; height = 3 * fontButtons->Height() / 2; this->halfWidth = halfWidth; this->alignLeft = alignLeft; pixmapText = NULL; } cRecMenuItemButton::~cRecMenuItemButton(void) { if (pixmapText) osdManager.releasePixmap(pixmapText); } int cRecMenuItemButton::GetWidth(void) { return fontButtons->Width(*text); } void cRecMenuItemButton::SetPixmaps(void) { if (halfWidth) { x += width / 4; width = width / 2; } if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapText = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapText->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemButton::Draw(void) { pixmapText->Fill(clrTransparent); int y = (height - fontButtons->Height()) / 2; int x; if (!alignLeft) x = (width - fontButtons->Width(*text)) / 2; else x = 10; pixmapText->DrawText(cPoint(x, y), *text, colorText, colorTextBack, fontButtons); } void cRecMenuItemButton::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapText) pixmapText->SetLayer(-1); } void cRecMenuItemButton::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapText) pixmapText->SetLayer(5); } eRecMenuState cRecMenuItemButton::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kOk: return action; break; default: break; } return rmsNotConsumed; } // --- cRecMenuItemButtonYesNo ------------------------------------------------------- cRecMenuItemButtonYesNo::cRecMenuItemButtonYesNo(cString textYes, cString textNo, eRecMenuState actionYes, eRecMenuState actionNo, bool active) { selectable = true; this->textYes = textYes; this->textNo = textNo; this->action = actionYes; this->actionNo = actionNo; this->active = active; yesActive = true; height = 3 * font->Height() / 2; pixmapNo = NULL; } cRecMenuItemButtonYesNo::~cRecMenuItemButtonYesNo(void) { if (pixmapNo) delete pixmapNo; } void cRecMenuItemButtonYesNo::SetPixmaps(void) { int buttonWidth = 44 * width / 100; int yesX = x + width / 25; int noX = x + 52 * width / 100; int yPixmaps = y + geoManager.borderRecMenus / 2; if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(yesX, yPixmaps, buttonWidth, height)); pixmapNo = new cStyledPixmap(osdManager.requestPixmap(4, cRect(noX, yPixmaps, buttonWidth, height))); } else { pixmap->SetViewPort(cRect(yesX, yPixmaps, buttonWidth, height)); pixmapNo->SetViewPort(cRect(noX, yPixmaps, buttonWidth, height)); } } void cRecMenuItemButtonYesNo::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapNo) pixmapNo->SetLayer(-1); } void cRecMenuItemButtonYesNo::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapNo) pixmapNo->SetLayer(4); } void cRecMenuItemButtonYesNo::setBackground() { if (tvguideConfig.style == eStyleGraphical) { drawBackgroundGraphical(bgButton, yesActive&&active); colorTextBack = clrTransparent; colorTextNoBack = clrTransparent; colorText = (active&&yesActive)?theme.Color(clrFontActive):theme.Color(clrFont); colorTextNo = (active&&!yesActive)?theme.Color(clrFontActive):theme.Color(clrFont); pixmapNo->drawBackgroundGraphical(bgButton, active&&!yesActive); } else { tColor colorNoBack; if (active) { if (yesActive) { color = theme.Color(clrHighlight); colorBlending = theme.Color(clrHighlightBlending); colorText = theme.Color(clrFontActive); colorNoBack = theme.Color(clrGrid1); pixmapNo->setColor( colorNoBack, theme.Color(clrGrid1Blending)); colorTextNo = theme.Color(clrFont); } else { color = theme.Color(clrGrid1); colorBlending = theme.Color(clrGrid1Blending); colorText = theme.Color(clrFont); colorNoBack = theme.Color(clrHighlight); pixmapNo->setColor( colorNoBack, theme.Color(clrHighlightBlending)); colorTextNo = theme.Color(clrFontActive); } } else { color = theme.Color(clrGrid1); colorBlending = theme.Color(clrGrid1Blending); colorText = theme.Color(clrFont); colorNoBack = theme.Color(clrGrid1); pixmapNo->setColor( colorNoBack, theme.Color(clrGrid1Blending)); colorTextNo = theme.Color(clrFont); } colorTextBack = (tvguideConfig.style == eStyleFlat)?color:clrTransparent; colorTextNoBack = (tvguideConfig.style == eStyleFlat)?colorNoBack:clrTransparent; drawBackground(); drawBorder(); pixmapNo->drawBackground(); pixmapNo->drawBorder(); } } void cRecMenuItemButtonYesNo::Draw(void) { int textYesX = (pixmap->ViewPort().Width() - font->Width(*textYes)) / 2; int textNoX = (pixmapNo->Width() - font->Width(*textNo)) / 2; int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(textYesX, textY), *textYes, colorText, colorTextBack, font); pixmapNo->DrawText(cPoint(textNoX, textY), *textNo, colorTextNo, colorTextNoBack, font); } eRecMenuState cRecMenuItemButtonYesNo::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: if (!yesActive) { yesActive = true; setBackground(); Draw(); return rmsConsumed; } else return rmsNotConsumed; break; case kRight: if (yesActive) { yesActive = false; setBackground(); Draw(); return rmsConsumed; } else return rmsNotConsumed; break; case kOk: if (yesActive) return action; return actionNo; break; default: break; } return rmsNotConsumed; } // --- cRecMenuItemInfo ------------------------------------------------------- cRecMenuItemInfo::cRecMenuItemInfo(const char *text, bool largeFont) { selectable = false; active = false; this->text = text; fontInfo = (largeFont) ? fontLarge : font; border = 10; } cRecMenuItemInfo::~cRecMenuItemInfo(void) { } void cRecMenuItemInfo::CalculateHeight(int textWidth) { wrapper.Set(*text, fontInfo, textWidth); height = fontInfo->Height() * wrapper.Lines() + 2*border; } void cRecMenuItemInfo::setBackground(void) { pixmap->Fill(clrTransparent); cRecMenuItem::setBackground(); } void cRecMenuItemInfo::Draw(void) { int lines = wrapper.Lines(); int lineHeight = fontInfo->Height(); int x = 0; int y = border; for (int i = 0; i < lines; i++) { x = (width - fontInfo->Width(wrapper.GetLine(i))) / 2; pixmap->DrawText(cPoint(x,y), wrapper.GetLine(i), theme.Color(clrFont), clrTransparent, fontInfo); y += lineHeight; } } // --- cRecMenuItemInt ------------------------------------------------------- cRecMenuItemInt::cRecMenuItemInt(cString text, int initialVal, int minVal, int maxVal, bool active, int *callback, eRecMenuState action) { selectable = true; this->text = text; this->currentVal = initialVal; this->minVal = minVal; this->maxVal = maxVal; this->active = active; this->callback = callback; this->action = action; height = 3 * font->Height() / 2; pixmapVal = NULL; fresh = true; } cRecMenuItemInt::~cRecMenuItemInt(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); } void cRecMenuItemInt::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemInt::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); } void cRecMenuItemInt::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemInt::setBackground() { cRecMenuItem::setBackground(); fresh = true; } void cRecMenuItemInt::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemInt::DrawValue(void) { pixmapVal->Fill(clrTransparent); cString textVal = cString::sprintf("%d", currentVal); int textX = width - font->Width(*textVal) - 10; int textY = (height - font->Height()) / 2; pixmapVal->DrawText(cPoint(textX, textY), *textVal, colorText, clrTransparent, font); } eRecMenuState cRecMenuItemInt::ProcessKey(eKeys Key) { int oldValue = currentVal; switch (Key & ~k_Repeat) { case kLeft: fresh = true; if (currentVal > minVal) { currentVal--; if (callback) *callback = currentVal; DrawValue(); } return rmsConsumed; break; case kRight: fresh = true; if (currentVal < maxVal) { currentVal++; if (callback) *callback = currentVal; DrawValue(); } return rmsConsumed; break; case k0 ... k9: if (fresh) { currentVal = 0; fresh = false; } currentVal = currentVal * 10 + (Key - k0); if (!((currentVal >= minVal) && (currentVal <= maxVal))) currentVal = oldValue; if (callback) *callback = currentVal; DrawValue(); return rmsConsumed; break; case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemBool ------------------------------------------------------- cRecMenuItemBool::cRecMenuItemBool(cString text, bool initialVal, bool refresh, bool active, bool *callback, eRecMenuState action) { selectable = true; this->text = text; this->yes = initialVal; this->refresh = refresh; this->active = active; this->callback = callback; this->action = action; height = 3 * font->Height() / 2; pixmapVal = NULL; } cRecMenuItemBool::~cRecMenuItemBool(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); } void cRecMenuItemBool::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemBool::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); } void cRecMenuItemBool::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemBool::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemBool::DrawValue(void) { pixmapVal->Fill(clrTransparent); std::string strIcon = yes?"yes":"no"; int iconSize = height - 8; int iconX = width - iconSize - 10; int iconY = (height - iconSize) / 2; cImage *imgYesNo = imgCache.GetIcon(strIcon, iconSize, iconSize); if (imgYesNo) { pixmapVal->DrawImage(cPoint(iconX, iconY), *imgYesNo); } } eRecMenuState cRecMenuItemBool::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: case kRight: yes = !yes; if (callback) *callback = yes; DrawValue(); if (refresh) return rmsRefresh; else return rmsConsumed; break; case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemSelect ------------------------------------------------------- cRecMenuItemSelect::cRecMenuItemSelect(cString text, std::vector Strings, int initialVal, bool active, int *callback, eRecMenuState action, bool refresh) { selectable = true; this->text = text; strings = Strings; numValues = Strings.size(); if ((initialVal < 0) || (initialVal > numValues-1)) this->currentVal = 0; else this->currentVal = initialVal; this->active = active; this->callback = callback; this->refresh = refresh; this->action = action; height = 3 * font->Height() / 2; pixmapVal = NULL; } cRecMenuItemSelect::~cRecMenuItemSelect(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); } void cRecMenuItemSelect::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemSelect::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); } void cRecMenuItemSelect::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemSelect::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemSelect::DrawValue(void) { pixmapVal->Fill(clrTransparent); std::string textVal = strings[currentVal]; int iconSize = min(128, height); int textX = width - font->Width(textVal.c_str()) - iconSize; int textY = (height - font->Height()) / 2; pixmapVal->DrawText(cPoint(textX, textY), textVal.c_str(), colorText, clrTransparent, font); int iconLeftX = textX - iconSize; int iconRightX = width - iconSize; int iconY = (height - iconSize) / 2; cImage *imgLeft = imgCache.GetIcon("arrow_left", iconSize, iconSize); if (imgLeft) { pixmapVal->DrawImage(cPoint(iconLeftX, iconY), *imgLeft); } cImage *imgRight = imgCache.GetIcon("arrow_right", iconSize, iconSize); if (imgRight) { pixmapVal->DrawImage(cPoint(iconRightX, iconY), *imgRight); } } eRecMenuState cRecMenuItemSelect::ProcessKey(eKeys Key) { // int oldValue = currentVal; switch (Key & ~k_Repeat) { case kLeft: currentVal--; if (currentVal<0) currentVal = numValues - 1; if (callback) *callback = currentVal; DrawValue(); if (refresh) return rmsRefresh; else return rmsConsumed; break; case kRight: currentVal = (currentVal+1)%numValues; if (callback) *callback = currentVal; DrawValue(); if (refresh) return rmsRefresh; else return rmsConsumed; break; case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemSelectDirectory ------------------------------------------------------- cRecMenuItemSelectDirectory::cRecMenuItemSelectDirectory(cString text, std::string originalFolder, bool active, char *callback, eRecMenuState action, bool isSearchTimer) { selectable = true; this->text = text; this->originalFolder = originalFolder; this->active = active; this->callback = callback; this->action = action; height = 3 * font->Height() / 2; pixmapVal = NULL; folders.push_back(tr("root video folder")); if (isSearchTimer && tvguideConfig.instRecFixedFolder.size() > 0) folders.push_back(tvguideConfig.instRecFixedFolder); ReadRecordingDirectories(&folders, NULL, ""); numValues = folders.size(); this->currentVal = GetInitial(); } cRecMenuItemSelectDirectory::~cRecMenuItemSelectDirectory(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); } void cRecMenuItemSelectDirectory::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemSelectDirectory::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); } void cRecMenuItemSelectDirectory::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemSelectDirectory::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemSelectDirectory::DrawValue(void) { pixmapVal->Fill(clrTransparent); int iconSize = min(128, height); int textX = width - font->Width(folders[currentVal].c_str()) - iconSize; int textY = (height - font->Height()) / 2; pixmapVal->DrawText(cPoint(textX, textY), folders[currentVal].c_str(), colorText, clrTransparent, font); int iconLeftX = textX - iconSize; int iconRightX = width - iconSize; int iconY = (height - iconSize) / 2; cImage *imgLeft = imgCache.GetIcon("arrow_left", iconSize, iconSize); if (imgLeft) { pixmapVal->DrawImage(cPoint(iconLeftX, iconY), *imgLeft); } cImage *imgRight = imgCache.GetIcon("arrow_right", iconSize, iconSize); if (imgRight) { pixmapVal->DrawImage(cPoint(iconRightX, iconY), *imgRight); } } eRecMenuState cRecMenuItemSelectDirectory::ProcessKey(eKeys Key) { // int oldValue = currentVal; switch (Key & ~k_Repeat) { case kLeft: currentVal--; if (currentVal<0) currentVal = numValues - 1; if (callback) { SetCallback(); } DrawValue(); return rmsConsumed; break; case kRight: { currentVal = (currentVal+1)%numValues; if (callback) { SetCallback(); } DrawValue(); return rmsConsumed; break; } case kOk: return action; default: break; } return rmsNotConsumed; } void cRecMenuItemSelectDirectory::SetCallback(void) { std::string newFolder = folders[currentVal]; if (!newFolder.compare(tr("root video folder"))) newFolder = ""; strncpy(callback, newFolder.c_str(), TEXTINPUTLENGTH); } int cRecMenuItemSelectDirectory::GetInitial(void) { if (originalFolder.size() == 0) return 0; for (int i=0; i < numValues; i++) { if (!folders[i].compare(originalFolder)) { return i; } } return 0; } // --- cRecMenuItemText ------------------------------------------------------- cRecMenuItemText::cRecMenuItemText(cString title, char *initialVal, int length, bool active, char *callback) { selectable = true; this->title = title; value = initialVal; this->active = active; this->callback = callback; height = 3 * font->Height(); pixmapVal = NULL; pixmapKeyboard = NULL; pixmapKeyboardHighlight = NULL; pixmapKeyboardIcons = NULL; keyboardWidth = 50; gridHeight = 3 * fontSmall->Height(); keyboardHeight = 5 * gridHeight; this->length = length; allowed = trVDR(FileNameChars); pos = -1; offset = 0; insert = uppercase = false; newchar = true; lengthUtf8 = 0; valueUtf8 = NULL; allowedUtf8 = NULL; charMapUtf8 = NULL; currentCharUtf8 = NULL; lastKey = kNone; keyboardDrawn = false; } cRecMenuItemText::~cRecMenuItemText(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); if (pixmapKeyboard) delete pixmapKeyboard; if (pixmapKeyboardHighlight) osdManager.releasePixmap(pixmapKeyboardHighlight); if (pixmapKeyboardIcons) osdManager.releasePixmap(pixmapKeyboardIcons); delete[] valueUtf8; delete[] allowedUtf8; delete[] charMapUtf8; } void cRecMenuItemText::SetPixmaps(void) { int xPixmapVal = x + 20; int yPixmapVal = y + height / 2 + 10; int widthPixmapVal = width - 40; int heightPixmapVal = height / 2 - 20; int keyboardX = x + (100 - keyboardWidth)*width / 100; int keyboardY = y + height; if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(xPixmapVal, yPixmapVal, widthPixmapVal, heightPixmapVal)); pixmapKeyboard = new cStyledPixmap(osdManager.requestPixmap(-1, cRect(keyboardX, keyboardY, keyboardWidth*width/100, keyboardHeight))); pixmapKeyboardHighlight = osdManager.requestPixmap(-1, cRect(keyboardX, keyboardY, keyboardWidth*width/100, keyboardHeight)); pixmapKeyboardIcons = osdManager.requestPixmap(-1, cRect(keyboardX, keyboardY, keyboardWidth*width/100, keyboardHeight)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(xPixmapVal, yPixmapVal, widthPixmapVal, heightPixmapVal)); pixmapKeyboard->SetViewPort(cRect(keyboardX, keyboardY, keyboardWidth*width/100, keyboardHeight)); pixmapKeyboardHighlight->SetViewPort(cRect(keyboardX, keyboardY, keyboardWidth*width/100, keyboardHeight)); pixmapKeyboardIcons->SetViewPort(cRect(keyboardX, keyboardY, keyboardWidth*width/100, keyboardHeight)); } pixmapKeyboardIcons->Fill(clrTransparent); } void cRecMenuItemText::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); if (pixmapKeyboard) pixmapKeyboard->SetLayer(-1); if (pixmapKeyboardHighlight) pixmapKeyboardHighlight->SetLayer(-1); if (pixmapKeyboardIcons) pixmapKeyboardIcons->SetLayer(-1); } void cRecMenuItemText::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemText::setBackground() { if (!active) { DeactivateKeyboard(); } cRecMenuItem::setBackground(); } void cRecMenuItemText::Draw(void) { int textY = (height/2 - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *title, colorText, colorTextBack, font); DrawValue(value); } void cRecMenuItemText::DrawValue(char *newValue) { tColor clrBack = InEditMode()?theme.Color(clrRecMenuTextActiveBack):theme.Color(clrRecMenuTextBack); pixmapVal->Fill(clrBack); int textX = pixmapVal->DrawPort().Width() - font->Width(newValue) - 10; int textY = (pixmapVal->DrawPort().Height() - font->Height()) / 2; pixmapVal->DrawText(cPoint(textX, textY), newValue, colorText, clrBack, font); } void cRecMenuItemText::ActivateKeyboard(void) { pixmapKeyboard->SetLayer(6); pixmapKeyboardHighlight->SetLayer(6); pixmapKeyboardIcons->SetLayer(6); pixmapKeyboardHighlight->Fill(clrTransparent); if (keyboardDrawn) return; pixmapKeyboard->setColor(theme.Color(clrRecMenuKeyboardBack), theme.Color(clrRecMenuKeyboardBack)); pixmapKeyboard->drawBackground(); int widthKeyBoard = pixmapKeyboard->Width(); gridWidth = widthKeyBoard / 3; pixmapKeyboard->DrawRectangle(cRect(0, 0, widthKeyBoard, keyboardHeight), theme.Color(clrRecMenuKeyboardBorder)); int num = 1; for (int row = 0; row < 5; row++) { for (int col = 0; col < 3; col++) { int X = col*gridWidth; int Y = row*gridHeight; tColor clrBack = theme.Color(clrRecMenuKeyboardBack); if (num==13) clrBack = theme.Color(clrButtonRedKeyboard); else if (num==14) clrBack = theme.Color(clrButtonGreenKeyboard); else if (num==15) clrBack = theme.Color(clrButtonYellowKeyboard); pixmapKeyboard->DrawRectangle(cRect(X+2, Y+2, gridWidth-4, gridHeight-4), clrBack); pixmapKeyboard->DrawEllipse(cRect(X, Y, 20, 20), theme.Color(clrRecMenuKeyboardBorder),-2); pixmapKeyboard->DrawEllipse(cRect(X, Y+gridHeight-20, 20, 20), theme.Color(clrRecMenuKeyboardBorder),-3); pixmapKeyboard->DrawEllipse(cRect(X+gridWidth-20, Y+gridHeight-20, 20, 20), theme.Color(clrRecMenuKeyboardBorder),-4); pixmapKeyboard->DrawEllipse(cRect(X+gridWidth-20, Y, 20, 20), theme.Color(clrRecMenuKeyboardBorder),-1); bool draw = false; bool drawIcon = false; cString strNum; std::string strIcon; if (num<10) { strNum = *cString::sprintf("%d", num); draw = true; } else if (num == 11) { strNum = *cString::sprintf("%d", 0); draw = true; } else if (num==13) { strIcon = "icon_shift"; drawIcon = true; } else if (num==14) { strIcon = "icon_del_ins"; drawIcon = true; } else if (num==15) { strIcon = "icon_backspace"; drawIcon = true; } if (draw) { int numX = X + (gridWidth - font->Width(*strNum))/2; int numY = Y + font->Height() / 4; pixmapKeyboard->DrawText(cPoint(numX, numY), *strNum, colorText, colorTextBack, font); char *smsKeys = GetSMSKeys(num); int smsKeysX = X + (gridWidth - fontSmall->Width(smsKeys))/2; int smsKeysY = Y + gridHeight - fontSmall->Height() - 10; pixmapKeyboard->DrawText(cPoint(smsKeysX, smsKeysY), smsKeys, theme.Color(clrRecMenuKeyboardBorder), theme.Color(clrRecMenuKeyboardBack), fontSmall); delete[] smsKeys; } if (drawIcon) { int iconSize = gridHeight - 10; cImage *imgIcon = imgCache.GetIcon(strIcon, iconSize, iconSize); if (imgIcon) { int iconX = X + (gridWidth - iconSize) / 2; pixmapKeyboardIcons->DrawImage(cPoint(iconX, Y + 5), *imgIcon); } } num++; } } keyboardDrawn = true; } void cRecMenuItemText::DeactivateKeyboard(void) { pixmapKeyboard->SetLayer(-1); pixmapKeyboardHighlight->SetLayer(-1); pixmapKeyboardIcons->SetLayer(-1); } void cRecMenuItemText::HighlightSMSKey(int num) { int x = 0; int y = 0; if (num == 0) { x = gridWidth; y = 3 * gridHeight; } else { x = (num-1)%3 * gridWidth; y = (num-1)/3 * gridHeight; } pixmapKeyboardHighlight->DrawRectangle(cRect(x, y, gridWidth, gridHeight), theme.Color(clrRecMenuKeyboardHigh)); } void cRecMenuItemText::ClearSMSKey(void) { pixmapKeyboardHighlight->Fill(clrTransparent); } char *cRecMenuItemText::GetSMSKeys(int num) { if (num == 11) num = 0; if (num > 9) return NULL; currentCharUtf8 = charMapUtf8; int pos = num; while (pos > 0 && *currentCharUtf8) { if (*currentCharUtf8++ == '\t') pos--; } while (*currentCharUtf8 && *currentCharUtf8 != '\t' && !IsAllowed(*currentCharUtf8)) { currentCharUtf8++; } uint *startCharUtf8 = currentCharUtf8; int numChars = 0; bool specialChar = false; while(*currentCharUtf8 && *currentCharUtf8 != '\t' && IsAllowed(*currentCharUtf8)) { currentCharUtf8++; if (Utf8CharSet(*currentCharUtf8) > 1) specialChar = true; numChars++; } char buf[100]; char *p = buf; int addition = 0; if (specialChar) addition = 1; Utf8FromArray(startCharUtf8, p, numChars+addition); int maxChars = min(numChars+1+addition, 8); char *smskey = new char[maxChars]; Utf8Strn0Cpy(smskey, p, maxChars); return smskey; } void cRecMenuItemText::EnterEditMode(void) { if (!valueUtf8) { valueUtf8 = new uint[length]; lengthUtf8 = Utf8ToArray(value, valueUtf8, length); int l = strlen(allowed) + 1; allowedUtf8 = new uint[l]; Utf8ToArray(allowed, allowedUtf8, l); const char *charMap = trVDR("CharMap$ 0\t-.,1#~\\^$[]|()*+?{}/:%@&\tabc2\tdef3\tghi4\tjkl5\tmno6\tpqrs7\ttuv8\twxyz9"); l = strlen(charMap) + 1; charMapUtf8 = new uint[l]; Utf8ToArray(charMap, charMapUtf8, l); currentCharUtf8 = charMapUtf8; AdvancePos(); } } void cRecMenuItemText::LeaveEditMode(bool SaveValue) { if (valueUtf8) { if (SaveValue) { Utf8FromArray(valueUtf8, value, length); stripspace(value); if (callback) { strncpy(callback, value, TEXTINPUTLENGTH); } } lengthUtf8 = 0; delete[] valueUtf8; valueUtf8 = NULL; delete[] allowedUtf8; allowedUtf8 = NULL; delete[] charMapUtf8; charMapUtf8 = NULL; pos = -1; offset = 0; newchar = true; } } void cRecMenuItemText::AdvancePos(void) { if (pos < length - 2 && pos < lengthUtf8) { if (++pos >= lengthUtf8) { if (pos >= 2 && valueUtf8[pos - 1] == ' ' && valueUtf8[pos - 2] == ' ') pos--; // allow only two blanks at the end else { valueUtf8[pos] = ' '; valueUtf8[pos + 1] = 0; lengthUtf8++; } } } newchar = true; if (!insert && Utf8is(alpha, valueUtf8[pos])) uppercase = Utf8is(upper, valueUtf8[pos]); } uint cRecMenuItemText::Inc(uint c, bool Up) { uint *p = IsAllowed(c); if (!p) p = allowedUtf8; if (Up) { if (!*++p) p = allowedUtf8; } else if (--p < allowedUtf8) { p = allowedUtf8; while (*p && *(p + 1)) p++; } return *p; } void cRecMenuItemText::Type(uint c) { if (insert && lengthUtf8 < length - 1) Insert(); valueUtf8[pos] = c; if (pos < length - 2) pos++; if (pos >= lengthUtf8) { valueUtf8[pos] = ' '; valueUtf8[pos + 1] = 0; lengthUtf8 = pos + 1; } } void cRecMenuItemText::Insert(void) { memmove(valueUtf8 + pos + 1, valueUtf8 + pos, (lengthUtf8 - pos + 1) * sizeof(*valueUtf8)); lengthUtf8++; valueUtf8[pos] = ' '; } void cRecMenuItemText::Delete(void) { memmove(valueUtf8 + pos, valueUtf8 + pos + 1, (lengthUtf8 - pos) * sizeof(*valueUtf8)); lengthUtf8--; } uint *cRecMenuItemText::IsAllowed(uint c) { if (allowedUtf8) { for (uint *a = allowedUtf8; *a; a++) { if (c == *a) return a; } } return NULL; } void cRecMenuItemText::SetText(void) { if (InEditMode()) { int textAreaWidth = width - 20; textAreaWidth -= font->Width("[]"); textAreaWidth -= font->Width("<>"); // reserving this anyway make the whole thing simpler if (pos < offset) offset = pos; int WidthFromOffset = 0; int EndPos = lengthUtf8; for (int i = offset; i < lengthUtf8; i++) { WidthFromOffset += font->Width(valueUtf8[i]); if (WidthFromOffset > textAreaWidth) { if (pos >= i) { do { WidthFromOffset -= font->Width(valueUtf8[offset]); offset++; } while (WidthFromOffset > textAreaWidth && offset < pos); EndPos = pos + 1; } else { EndPos = i; break; } } } char buf[1000]; char *p = buf; if (offset) *p++ = '<'; p += Utf8FromArray(valueUtf8 + offset, p, sizeof(buf) - (p - buf), pos - offset); *p++ = '['; if (insert && newchar) *p++ = ']'; p += Utf8FromArray(&valueUtf8[pos], p, sizeof(buf) - (p - buf), 1); if (!(insert && newchar)) *p++ = ']'; p += Utf8FromArray(&valueUtf8[pos + 1], p, sizeof(buf) - (p - buf), EndPos - pos - 1); if (EndPos != lengthUtf8) *p++ = '>'; *p = 0; DrawValue(buf); } else { DrawValue(value); } } eRecMenuState cRecMenuItemText::ProcessKey(eKeys Key) { bool consumed = false; bool SameKey = NORMALKEY(Key) == lastKey; ClearSMSKey(); if (Key != kNone) { lastKey = NORMALKEY(Key); } else if (!newchar && k0 <= lastKey && lastKey <= k9 && autoAdvanceTimeout.TimedOut()) { AdvancePos(); newchar = true; currentCharUtf8 = NULL; SetText(); return rmsConsumed; } switch ((int)Key) { case kRed: // Switch between upper- and lowercase characters if (InEditMode()) { if (!insert || !newchar) { uppercase = !uppercase; valueUtf8[pos] = uppercase ? Utf8to(upper, valueUtf8[pos]) : Utf8to(lower, valueUtf8[pos]); } consumed = true; } break; case kGreen: // Toggle insert/overwrite modes if (InEditMode()) { insert = !insert; newchar = true; consumed = true; } break; case kYellow|k_Repeat: case kYellow: // Remove the character at the current position; in insert mode it is the character to the right of the cursor if (InEditMode()) { if (lengthUtf8 > 1) { if (!insert || pos < lengthUtf8 - 1) Delete(); else if (insert && pos == lengthUtf8 - 1) valueUtf8[pos] = ' '; // in insert mode, deleting the last character replaces it with a blank to keep the cursor position // reduce position, if we removed the last character if (pos == lengthUtf8) pos--; } else if (lengthUtf8 == 1) valueUtf8[0] = ' '; // This is the last character in the string, replace it with a blank if (Utf8is(alpha, valueUtf8[pos])) uppercase = Utf8is(upper, valueUtf8[pos]); newchar = true; consumed = true; } break; case kLeft|k_Repeat: case kLeft: if (pos > 0) { if (!insert || newchar) pos--; newchar = true; if (!insert && Utf8is(alpha, valueUtf8[pos])) uppercase = Utf8is(upper, valueUtf8[pos]); } consumed = true; break; case kRight|k_Repeat: case kRight: if (InEditMode()) { AdvancePos(); } else { EnterEditMode(); ActivateKeyboard(); } consumed = true; break; case kUp|k_Repeat: case kUp: case kDown|k_Repeat: case kDown: if (InEditMode()) { if (insert && newchar) { // create a new character in insert mode if (lengthUtf8 < length - 1) Insert(); } if (uppercase) valueUtf8[pos] = Utf8to(upper, Inc(Utf8to(lower, valueUtf8[pos]), NORMALKEY(Key) == kUp)); else valueUtf8[pos] = Inc( valueUtf8[pos], NORMALKEY(Key) == kUp); newchar = false; consumed = true; } break; case k0|k_Repeat ... k9|k_Repeat: case k0 ... k9: { if (InEditMode()) { if (Setup.NumberKeysForChars) { HighlightSMSKey(NORMALKEY(Key) - k0); if (!SameKey) { if (!newchar) AdvancePos(); currentCharUtf8 = NULL; } if (!currentCharUtf8 || !*currentCharUtf8 || *currentCharUtf8 == '\t') { // find the beginning of the character map entry for Key int n = NORMALKEY(Key) - k0; currentCharUtf8 = charMapUtf8; while (n > 0 && *currentCharUtf8) { if (*currentCharUtf8++ == '\t') n--; } // find first allowed character while (*currentCharUtf8 && *currentCharUtf8 != '\t' && !IsAllowed(*currentCharUtf8)) currentCharUtf8++; } if (*currentCharUtf8 && *currentCharUtf8 != '\t') { if (insert && newchar) { // create a new character in insert mode if (lengthUtf8 < length - 1) Insert(); } valueUtf8[pos] = *currentCharUtf8; if (uppercase) valueUtf8[pos] = Utf8to(upper, valueUtf8[pos]); // find next allowed character do { currentCharUtf8++; } while (*currentCharUtf8 && *currentCharUtf8 != '\t' && !IsAllowed(*currentCharUtf8)); newchar = false; autoAdvanceTimeout.Set(AUTO_ADVANCE_TIMEOUT); } } else { Type('0' + NORMALKEY(Key) - k0); } consumed = true; } break; } case kBack: case kOk: if (InEditMode()) { LeaveEditMode(Key == kOk); DeactivateKeyboard(); } else { EnterEditMode(); ActivateKeyboard(); } consumed = true; break; default: if (InEditMode() && BASICKEY(Key) == kKbd) { int c = KEYKBD(Key); if (c <= 0xFF) { if (IsAllowed(Utf8to(lower, c))) Type(c); else { switch (c) { case 0x7F: // backspace if (pos > 0) { pos--; ProcessKey(kYellow); } break; default: ; } } } else { switch (c) { case kfHome: pos = 0; break; case kfEnd: pos = lengthUtf8 - 1; break; case kfIns: ProcessKey(kGreen); case kfDel: ProcessKey(kYellow); default: ; } } consumed = true; } break; } SetText(); if (consumed) return rmsConsumed; return rmsNotConsumed; } // --- cRecMenuItemTime ------------------------------------------------------- cRecMenuItemTime::cRecMenuItemTime(cString text, int initialVal, bool active, int *callback, eRecMenuState action) { selectable = true; this->text = text; this->value = initialVal; hh = value / 100; mm = value % 100; pos = 0; fresh = true; this->active = active; this->callback = callback; this->action = action; height = 3 * font->Height() / 2; pixmapVal = NULL; } cRecMenuItemTime::~cRecMenuItemTime(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); } void cRecMenuItemTime::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemTime::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); } void cRecMenuItemTime::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemTime::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemTime::DrawValue(void) { pixmapVal->Fill(clrTransparent); char buf[10]; switch (pos) { case 1: snprintf(buf, sizeof(buf), "%01d-:--", hh / 10); break; case 2: snprintf(buf, sizeof(buf), "%02d:--", hh); break; case 3: snprintf(buf, sizeof(buf), "%02d:%01d-", hh, mm / 10); break; default: snprintf(buf, sizeof(buf), "%02d:%02d", hh, mm); } int textX = width - font->Width(buf) - 10; int textY = (height - font->Height()) / 2; pixmapVal->DrawText(cPoint(textX, textY), buf, colorText, clrTransparent, font); } eRecMenuState cRecMenuItemTime::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft|k_Repeat: case kLeft: { if (--mm < 0) { mm = 59; if (--hh < 0) hh = 23; } fresh = true; value = hh * 100 + mm; if (callback) *callback = value; DrawValue(); return rmsConsumed; break; } case kRight|k_Repeat: case kRight: { if (++mm > 59) { mm = 0; if (++hh > 23) hh = 0; } fresh = true; value = hh * 100 + mm; DrawValue(); if (callback) *callback = value; return rmsConsumed; break; } case k0|k_Repeat ... k9|k_Repeat: case k0 ... k9: { if (fresh || pos > 3) { pos = 0; fresh = false; } int n = Key - k0; switch (pos) { case 0: if (n <= 2) { hh = n * 10; mm = 0; pos++; } break; case 1: if (hh + n <= 23) { hh += n; pos++; } break; case 2: if (n <= 5) { mm += n * 10; pos++; } break; case 3: if (mm + n <= 59) { mm += n; pos++; } break; default: ; } value = hh * 100 + mm; if (callback) *callback = value; DrawValue(); return rmsConsumed; break; } case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemDay ------------------------------------------------------- cRecMenuItemDay::cRecMenuItemDay(cString text, time_t initialVal, bool active, time_t *callback, eRecMenuState action) { selectable = true; this->text = text; this->currentVal = cTimer::SetTime(initialVal, 0); this->active = active; this->callback = callback; this->action = action; height = 3 * font->Height() / 2; pixmapVal = NULL; } cRecMenuItemDay::~cRecMenuItemDay(void) { if (pixmapVal) osdManager.releasePixmap(pixmapVal); } void cRecMenuItemDay::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapVal = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapVal->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemDay::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapVal) pixmapVal->SetLayer(-1); } void cRecMenuItemDay::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapVal) pixmapVal->SetLayer(5); } void cRecMenuItemDay::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemDay::DrawValue(void) { pixmapVal->Fill(clrTransparent); cString textVal = DateString(currentVal); int textX = width - font->Width(*textVal) - 10; int textY = (height - font->Height()) / 2; pixmapVal->DrawText(cPoint(textX, textY), *textVal, colorText, clrTransparent, font); } eRecMenuState cRecMenuItemDay::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: currentVal -= 60*60*24; if (callback) *callback = currentVal; DrawValue(); return rmsConsumed; break; case kRight: currentVal += 60*60*24; if (callback) *callback = currentVal; DrawValue(); return rmsConsumed; break; case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemTimer ------------------------------------------------------- cRecMenuItemTimer::cRecMenuItemTimer(const cTimer *timer, eRecMenuState action1, eRecMenuState action2, eRecMenuState action3, eRecMenuState action4, time_t conflictStart, time_t conflictStop, time_t overlapStart, time_t overlapStop, bool active) { selectable = true; this->timer = timer; this->action = action1; this->action2 = action2; this->action3 = action3; this->action4 = action4; iconActive = 0; this->conflictStart = conflictStart; this->conflictStop = conflictStop; this->overlapStart = overlapStart; this->overlapStop = overlapStop; this->active = active; height = 3 * font->Height(); pixmapIcons = NULL; } cRecMenuItemTimer::~cRecMenuItemTimer(void) { if (pixmapIcons) osdManager.releasePixmap(pixmapIcons); if (pixmapStatus) osdManager.releasePixmap(pixmapStatus); } void cRecMenuItemTimer::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapStatus = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapIcons = osdManager.requestPixmap(6, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapStatus->SetViewPort(cRect(x, y, width, height)); pixmapIcons->SetViewPort(cRect(x, y, width, height)); } pixmapStatus->Fill(clrTransparent); pixmapIcons->Fill(clrTransparent); } void cRecMenuItemTimer::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapStatus) pixmapStatus->SetLayer(-1); if (pixmapIcons) pixmapIcons->SetLayer(-1); } void cRecMenuItemTimer::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapStatus) pixmapStatus->SetLayer(5); if (pixmapIcons) pixmapIcons->SetLayer(6); } void cRecMenuItemTimer::Draw(void) { const cChannel *channel = timer->Channel(); int channelTransponder = 0; cString channelName = ""; if (channel) { channelTransponder = channel->Transponder(); channelName = channel->Name(); } int logoX = DrawIcons(); int logoWidth = height * tvguideConfig.logoWidthRatio / tvguideConfig.logoHeightRatio; cImageLoader imgLoader; if (!tvguideConfig.hideChannelLogos) { if (imgLoader.LoadLogo(channel, logoWidth, height)) { cImage logo = imgLoader.GetImage(); pixmapIcons->DrawImage(cPoint(logoX, 0), logo); logoX += logoWidth + 5; } } int textX = logoX; int textHeightLine1 = (height/2 - font->Height()) / 2; int textHeightLine2 = height/2 - 5 + (height/4 - fontSmall->Height()) / 2; int textHeightLine3 = 3*height/4 - 5 + (height/4 - fontSmall->Height()) / 2; const cEvent *event = timer->Event(); std::string timerTitle = ""; if (event) { timerTitle = event->Title(); timerTitle = CutText(timerTitle, (70 * width / 100) - textX, font); } cString timeStart = DayDateTime(timer->StartTime()); cString timeEnd = TimeString(timer->StopTime()); cString timerTime = cString::sprintf("%s - %s", *timeStart, *timeEnd); cString channelInfo = cString::sprintf("%s, %s %d", *channelName, tr("Transp."), channelTransponder); pixmapIcons->DrawText(cPoint(textX, textHeightLine1), timerTitle.c_str(), colorText, colorTextBack, font); pixmapIcons->DrawText(cPoint(textX, textHeightLine2), *timerTime, colorText, colorTextBack, fontSmall); pixmapIcons->DrawText(cPoint(textX, textHeightLine3), *channelInfo, colorText, colorTextBack, fontSmall); DrawTimerConflict(); } int cRecMenuItemTimer::DrawIcons(void) { int iconsX = 10; int iconSize = height/2; int iconY = (height - iconSize) / 2; std::string iconInfo, iconDelete, iconEdit, iconSearch; if (active) { iconInfo = (iconActive==0)?"info_active":"info_inactive"; iconDelete = (iconActive==1)?"delete_active":"delete_inactive"; iconEdit = (iconActive==2)?"edit_active":"edit_inactive"; iconSearch = (iconActive==3)?"search_active":"search_inactive"; } else { iconInfo = "info_inactive"; iconDelete = "delete_inactive"; iconEdit = "edit_inactive"; iconSearch = "search_inactive"; } cImage *imgInfo = imgCache.GetIcon(iconInfo, iconSize, iconSize); if (imgInfo) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgInfo); iconsX += iconSize + 5; } cImage *imgDelete = imgCache.GetIcon(iconDelete, iconSize, iconSize); if (imgDelete) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgDelete); iconsX += iconSize + 5; } cImage *imgEdit = imgCache.GetIcon(iconEdit, iconSize, iconSize); if (imgEdit) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgEdit); iconsX += iconSize + 5; } cImage *imgSearch = imgCache.GetIcon(iconSearch, iconSize, iconSize); if (imgSearch) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgSearch); iconsX += iconSize + 5; } return iconsX; } void cRecMenuItemTimer::DrawTimerConflict(void) { int widthConfl = 30 * width / 100; int xConfl = width - widthConfl; int heightConflBar = height / 4; int yConflBar = (height - heightConflBar) / 2; pixmapStatus->DrawRectangle(cRect(xConfl, 0, widthConfl, height), theme.Color(clrRecMenuTimerConflictBackground)); int completeWidthSecs = conflictStop - conflictStart; int xConfBar = xConfl + (timer->StartTime() - conflictStart) * widthConfl / completeWidthSecs; int widthConfBar = (timer->StopTime() - timer->StartTime()) * widthConfl / completeWidthSecs; pixmapStatus->DrawRectangle(cRect(xConfBar, yConflBar, widthConfBar, heightConflBar), theme.Color(clrRecMenuTimerConflictBar)); int xOverlap = xConfl + (overlapStart - conflictStart) * widthConfl / completeWidthSecs; int widthOverlap = (overlapStop - overlapStart) * widthConfl / completeWidthSecs; pixmapIcons->DrawRectangle(cRect(xOverlap, 0, widthOverlap, height), theme.Color(clrRecMenuTimerConflictOverlap)); pixmapStatus->DrawRectangle(cRect(xConfl-3,2,1,height-4), theme.Color(clrBorder)); pixmapStatus->DrawRectangle(cRect(xConfl-2,0,2,height), theme.Color(clrBackground)); } eRecMenuState cRecMenuItemTimer::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: if (iconActive > 0) { iconActive--; DrawIcons(); return rmsConsumed; } else return rmsNotConsumed; break; case kRight: if (iconActive < 3) { iconActive++; DrawIcons(); return rmsConsumed; } else return rmsNotConsumed; break; case kOk: if (iconActive == 0) return action; else if (iconActive == 1) return action2; else if (iconActive == 2) return action3; else if (iconActive == 3) return action4; break; default: break; } return rmsNotConsumed; } // --- cRecMenuItemTimerConflictHeader ------------------------------------------------------- cRecMenuItemTimerConflictHeader::cRecMenuItemTimerConflictHeader(time_t conflictStart, time_t conflictStop, time_t overlapStart, time_t overlapStop) { selectable = false; active = false; this->conflictStart = conflictStart; this->conflictStop = conflictStop; this->overlapStart = overlapStart; this->overlapStop = overlapStop; height = 3*font->Height()/2; pixmapStatus = NULL; } cRecMenuItemTimerConflictHeader::~cRecMenuItemTimerConflictHeader(void) { if (pixmapStatus) osdManager.releasePixmap(pixmapStatus); } void cRecMenuItemTimerConflictHeader::SetPixmaps(void) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapStatus = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapStatus->Fill(clrTransparent); } void cRecMenuItemTimerConflictHeader::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapStatus) pixmapStatus->SetLayer(-1); } void cRecMenuItemTimerConflictHeader::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapStatus) pixmapStatus->SetLayer(5); } void cRecMenuItemTimerConflictHeader::setBackground(void) { pixmap->Fill(clrTransparent); cRecMenuItem::setBackground(); } void cRecMenuItemTimerConflictHeader::Draw(void) { int widthConfl = 30 * width / 100; int xConfl = width - widthConfl; cString headerText = tr("Timer Conflict"); int xHeader = (xConfl - font->Width(*headerText)) / 2; int yHeader = (height - font->Height()) / 2; pixmap->DrawText(cPoint(xHeader, yHeader), *headerText, theme.Color(clrFont), colorTextBack, font); pixmap->DrawRectangle(cRect(xConfl, 0, widthConfl, height), theme.Color(clrRecMenuTimerConflictBackground)); int completeWidthSecs = conflictStop - conflictStart; int xOverlap = xConfl + (overlapStart - conflictStart) * widthConfl / completeWidthSecs; int yOverlap = height - fontSmall->Height(); int widthOverlap = (overlapStop - overlapStart) * widthConfl / completeWidthSecs; pixmapStatus->DrawRectangle(cRect(xOverlap, yOverlap, widthOverlap, height), theme.Color(clrRecMenuTimerConflictOverlap)); cString strConflStart = TimeString(conflictStart); cString strConflStop = TimeString(conflictStop); cString strOverlapStart = TimeString(overlapStart); cString strOverlapStop = TimeString(overlapStop); int y1 = 5; int y2 = yOverlap; int xConflStart = xConfl + 2; int xConflStop = width - fontSmall->Width(*strConflStop) - 2; int xOverlapStart = xOverlap - fontSmall->Width(*strOverlapStart) - 2; int xOverlapStop = xOverlap + widthOverlap + 2; pixmap->DrawText(cPoint(xConflStart, y1), *strConflStart, theme.Color(clrRecMenuTimerConflictBar), theme.Color(clrRecMenuTimerConflictBackground), fontSmall); pixmap->DrawText(cPoint(xConflStop, y1), *strConflStop, theme.Color(clrRecMenuTimerConflictBar), theme.Color(clrRecMenuTimerConflictBackground), fontSmall); pixmap->DrawText(cPoint(xOverlapStart, y2), *strOverlapStart, theme.Color(clrRecMenuTimerConflictOverlap), theme.Color(clrRecMenuTimerConflictBackground), fontSmall); pixmap->DrawText(cPoint(xOverlapStop, y2), *strOverlapStop, theme.Color(clrRecMenuTimerConflictOverlap), theme.Color(clrRecMenuTimerConflictBackground), fontSmall); } // --- cRecMenuItemEvent ------------------------------------------------------- cRecMenuItemEvent::cRecMenuItemEvent(const cEvent *event, eRecMenuState action1, eRecMenuState action2, bool active) { selectable = true; this->event = event; this->action = action1; this->action2 = action2; iconActive = 0; this->active = active; height = font->Height() + 2*fontSmall->Height() + 10; pixmapText = NULL; pixmapIcons = NULL; } cRecMenuItemEvent::~cRecMenuItemEvent(void) { if (pixmapIcons) osdManager.releasePixmap(pixmapIcons); if (pixmapText) osdManager.releasePixmap(pixmapText); } void cRecMenuItemEvent::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapText = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapText->Fill(clrTransparent); pixmapIcons = osdManager.requestPixmap(6, cRect(x, y, width, height)); pixmapIcons->Fill(clrTransparent); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapText->SetViewPort(cRect(x, y, width, height)); pixmapIcons->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemEvent::Draw(void) { if (!event) return; int logoX = DrawIcons(); #if VDRVERSNUM >= 20301 LOCK_CHANNELS_READ; const cChannel *channel = Channels->GetByChannelID(event->ChannelID()); #else const cChannel *channel = Channels.GetByChannelID(event->ChannelID()); #endif cString channelName = ""; if (channel) channelName = channel->Name(); int logoWidth = height * tvguideConfig.logoWidthRatio / tvguideConfig.logoHeightRatio; cImageLoader imgLoader; if (!tvguideConfig.hideChannelLogos) { if (imgLoader.LoadLogo(channel, logoWidth, height)) { cImage logo = imgLoader.GetImage(); pixmapText->DrawImage(cPoint(logoX, (height - logo.Height()) / 2), logo); // logoX += logoWidth + 5; } logoX += logoWidth + 5; } int textX = logoX; int textHeightLine1 = 5; int textHeightLine2 = 5 + fontSmall->Height(); int textHeightLine3 = height - fontSmall->Height() - 5; cString title = event->Title(); cString desc = event->ShortText(); cString start = DayDateTime(event->StartTime()); cString end = event->GetEndTimeString(); colorText = active?theme.Color(clrFontActive):theme.Color(clrFont); cString info = cString::sprintf("%s - %s, %s", *start, *end, *channelName); pixmapText->DrawText(cPoint(textX, textHeightLine1), *info, colorText, clrTransparent, fontSmall); pixmapText->DrawText(cPoint(textX, textHeightLine2), *title, colorText, clrTransparent, font); pixmapText->DrawText(cPoint(textX, textHeightLine3), *desc, colorText, clrTransparent, fontSmall); } int cRecMenuItemEvent::DrawIcons(void) { pixmapIcons->Fill(clrTransparent); int iconsX = 10; int iconSize = height / 2; int iconY = (height - iconSize) / 2; std::string iconInfo; if (active) { iconInfo = (iconActive==0)?"info_active":"info_inactive"; } else { iconInfo = "info_inactive"; } cImage *imgInfo = imgCache.GetIcon(iconInfo, iconSize, iconSize); if (imgInfo) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgInfo); iconsX += iconSize + 5; } iconY = height - iconSize - 10; if (event->HasTimer()) { cImage *imgHasTimer = imgCache.GetIcon("activetimer", iconSize, iconSize); if (imgHasTimer) { pixmapIcons->DrawImage(cPoint(width - iconSize - 10, iconY), *imgHasTimer); } } else if (action2 != rmsDisabled) { std::string iconRec = active ? "record_active" : "record_inactive"; cImage *imgRec = imgCache.GetIcon(iconRec, iconSize, iconSize); if (imgRec) { pixmapIcons->DrawImage(cPoint(width - iconSize - 10, iconY), *imgRec); } } return iconsX; } void cRecMenuItemEvent::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapText) pixmapText->SetLayer(-1); if (pixmapIcons) pixmapIcons->SetLayer(-1); } void cRecMenuItemEvent::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapText) pixmapText->SetLayer(5); if (pixmapIcons) pixmapIcons->SetLayer(6); } eRecMenuState cRecMenuItemEvent::ProcessKey(eKeys Key) { // bool consumed = false; switch (Key & ~k_Repeat) { case kOk: return action; break; case kRed: if (!event->HasTimer()) return action2; break; default: break; } return rmsNotConsumed; } // --- cRecMenuItemChannelChooser ------------------------------------------------------- cRecMenuItemChannelChooser::cRecMenuItemChannelChooser(cString text, const cChannel *initialChannel, bool active, int *callback, eRecMenuState action) { selectable = true; this->text = text; this->channel = initialChannel; if (initialChannel) initialChannelSet = true; else initialChannelSet = false; channelNumber = 0; fresh = true; this->active = active; this->callback = callback; this->action = action; height = 2 * font->Height(); pixmapChannel = NULL; } cRecMenuItemChannelChooser::~cRecMenuItemChannelChooser(void) { if (pixmapChannel) osdManager.releasePixmap(pixmapChannel); } void cRecMenuItemChannelChooser::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapChannel = osdManager.requestPixmap(5, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapChannel->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemChannelChooser::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapChannel) pixmapChannel->SetLayer(-1); } void cRecMenuItemChannelChooser::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapChannel) pixmapChannel->SetLayer(5); } void cRecMenuItemChannelChooser::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawValue(); } void cRecMenuItemChannelChooser::DrawValue(void) { pixmapChannel->Fill(clrTransparent); int textY = (height - font->Height()) / 2; if (channel) { cString textVal = cString::sprintf("%d - %s", channel->Number(), channel->Name()); int textX = width - font->Width(*textVal) - 10; pixmapChannel->DrawText(cPoint(textX, textY), *textVal, colorText, clrTransparent, font); int logoWidth = height * tvguideConfig.logoWidthRatio / tvguideConfig.logoHeightRatio; int logoX = textX - logoWidth - 10; cImageLoader imgLoader; if (imgLoader.LoadLogo(channel, logoWidth, height)) { cImage logo = imgLoader.GetImage(); pixmapChannel->DrawImage(cPoint(logoX, 0), logo); } } else { cString textVal = tr("all Channels"); int textX = width - font->Width(*textVal) - 10; pixmapChannel->DrawText(cPoint(textX, textY), *textVal, colorText, clrTransparent, font); } } int cRecMenuItemChannelChooser::GetIntValue(void) { if (channel) return channel->Number(); return 0; } eRecMenuState cRecMenuItemChannelChooser::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: { fresh = true; if (!channel) return rmsConsumed; const cChannel *prev = channel; #if VDRVERSNUM >= 20301 LOCK_CHANNELS_READ; const cChannels* channels = Channels; #else const cChannels* channels = &Channels; #endif const cChannel *firstChannel = channels->First(); if(firstChannel->GroupSep()) firstChannel = channels->Next(firstChannel); if (prev == firstChannel) { if (!initialChannelSet) channel = NULL; } else { while (prev = channels->Prev(prev)) { if(!prev->GroupSep()) { channel = prev; break; } } } if (callback) { if (channel) *callback = channel->Number(); else *callback = 0; } DrawValue(); return rmsConsumed; break; } case kRight: { fresh = true; #if VDRVERSNUM >= 20301 LOCK_CHANNELS_READ; const cChannels* channels = Channels; #else const cChannels* channels = &Channels; #endif if (!channel) { channel = channels->First(); if(channel->GroupSep()) channel = channels->Next(channel); } else { const cChannel *next = channel; while (next = channels->Next(next)) { if(!next->GroupSep()) { channel = next; break; } } } if (callback) { if (channel) *callback = channel->Number(); else *callback = 0; } DrawValue(); return rmsConsumed; break; } case k0 ... k9: { if (fresh) { channelNumber = 0; fresh = false; } channelNumber = channelNumber * 10 + (Key - k0); #if VDRVERSNUM >= 20301 LOCK_CHANNELS_READ; const cChannel *chanNew = Channels->GetByNumber(channelNumber); #else const cChannel *chanNew = Channels.GetByNumber(channelNumber); #endif if (chanNew) { channel = chanNew; DrawValue(); if (callback) *callback = channel->Number(); } return rmsConsumed; break; } case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemDayChooser ------------------------------------------------------- cRecMenuItemDayChooser::cRecMenuItemDayChooser(cString text, int weekdays, bool active, int *callback) { selectable = true; this->text = text; if (weekdays < 1) weekdays *= -1; this->weekdays = weekdays; this->active = active; this->callback = callback; height = 2 * font->Height(); selectedDay = 0; pixmapWeekdays = NULL; pixmapWeekdaysSelect = NULL; } cRecMenuItemDayChooser::~cRecMenuItemDayChooser(void) { if (pixmapWeekdays) osdManager.releasePixmap(pixmapWeekdays); if (pixmapWeekdaysSelect) osdManager.releasePixmap(pixmapWeekdaysSelect); } void cRecMenuItemDayChooser::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapWeekdays = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapWeekdaysSelect = osdManager.requestPixmap(6, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapWeekdays->SetViewPort(cRect(x, y, width, height)); pixmapWeekdaysSelect->SetViewPort(cRect(x, y, width, height)); } SetSizes(); } void cRecMenuItemDayChooser::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapWeekdays) pixmapWeekdays->SetLayer(-1); if (pixmapWeekdaysSelect) pixmapWeekdaysSelect->SetLayer(-1); } void cRecMenuItemDayChooser::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapWeekdays) pixmapWeekdays->SetLayer(5); if (pixmapWeekdaysSelect) pixmapWeekdaysSelect->SetLayer(6); } void cRecMenuItemDayChooser::SetSizes(void) { days = trVDR("MTWTFSS"); int maxWidth = 0; for (unsigned i=0; iWidth(days.at(i)); if (charWidth > maxWidth) maxWidth = charWidth; } daysSize = min(maxWidth + 15, height-4); daysX = width - 10 - 7*daysSize; daysY = (height - daysSize) / 2; } void cRecMenuItemDayChooser::setBackground() { cRecMenuItem::setBackground(); if (active) { DrawHighlight(selectedDay); } else { DrawHighlight(-1); } } void cRecMenuItemDayChooser::Draw(void) { int textY = (height - font->Height()) / 2; pixmap->DrawText(cPoint(10, textY), *text, colorText, colorTextBack, font); DrawDays(); } void cRecMenuItemDayChooser::DrawDays(void) { pixmapWeekdays->Fill(clrTransparent); int textY = (height - font->Height()) / 2; pixmapWeekdays->DrawRectangle(cRect(daysX, daysY, 7*daysSize, daysSize), theme.Color(clrBorder)); int currentX = daysX; for (unsigned day=0; dayDrawRectangle(cRect(currentX+2, daysY+2, daysSize-4, daysSize-4), theme.Color(clrBackground)); tColor colorDay = WeekDaySet(day)?theme.Color(clrRecMenuDayActive) :theme.Color(clrRecMenuDayInactive); int textX = currentX + (daysSize - font->Width(*strDay)) / 2; pixmapWeekdays->DrawText(cPoint(textX, textY), *strDay, colorDay, clrTransparent, font); currentX += daysSize; } } void cRecMenuItemDayChooser::DrawHighlight(int day) { pixmapWeekdaysSelect->Fill(clrTransparent); if (day > -1) { int currentX = daysX + day*daysSize; pixmapWeekdaysSelect->DrawRectangle(cRect(currentX, daysY, daysSize, daysSize), theme.Color(clrRecMenuDayHighlight)); } } bool cRecMenuItemDayChooser::WeekDaySet(unsigned day) { return weekdays & (1 << day); } void cRecMenuItemDayChooser::ToggleDay(void) { bool dayActive = WeekDaySet(selectedDay); int dayBit = pow(2, selectedDay); if (dayActive) { weekdays -= dayBit; } else { weekdays += dayBit; } if (callback) { *callback = weekdays; } } eRecMenuState cRecMenuItemDayChooser::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: { selectedDay--; if (selectedDay<0) selectedDay += 7; DrawHighlight(selectedDay); return rmsConsumed; break; } case kRight: { selectedDay = (selectedDay+1)%7; DrawHighlight(selectedDay); return rmsConsumed; break; } case kOk: ToggleDay(); DrawDays(); return rmsConsumed; break; default: break; } return rmsNotConsumed; } // --- cRecMenuItemRecording ------------------------------------------------------- cRecMenuItemRecording::cRecMenuItemRecording(const cRecording *recording, bool active) { selectable = true; this->recording = recording; this->active = active; height = font->Height() + 2*fontSmall->Height() + 10; pixmapText = NULL; } cRecMenuItemRecording::~cRecMenuItemRecording(void) { if (pixmapText) osdManager.releasePixmap(pixmapText); } void cRecMenuItemRecording::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapText = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapText->Fill(clrTransparent); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapText->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemRecording::Draw(void) { if (!recording) return; const cRecordingInfo *recInfo = recording->Info(); #if VDRVERSNUM >= 20301 LOCK_CHANNELS_READ; const cChannel *channel = Channels->GetByChannelID(recInfo->ChannelID()); #else const cChannel *channel = Channels.GetByChannelID(recInfo->ChannelID()); #endif cString channelName = tr("unknown channel"); if (channel) channelName = channel->Name(); cString name = recording->Name(); cString dateTime = cString::sprintf("%s, %s", *DateString(recording->Start()), *TimeString(recording->Start())); int recDuration = recording->LengthInSeconds() / 60; cString recDetails = cString::sprintf("%s: %d %s, %s %s %s \"%s\"", tr("Duration"), recDuration, tr("min"), tr("recorded at"), *dateTime, tr("from"), *channelName); recDetails = CutText(*recDetails, width - 40, fontSmall).c_str(); int text1Y = (height/2 - font->Height()) / 2 + 5; int text2Y = height/2 + (height/2 - fontSmall->Height())/2 - 5; colorText = active?theme.Color(clrFontActive):theme.Color(clrFont); pixmapText->DrawText(cPoint(10, text1Y), *name, colorText, clrTransparent, font); pixmapText->DrawText(cPoint(10, text2Y), *recDetails, colorText, clrTransparent, fontSmall); } void cRecMenuItemRecording::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapText) pixmapText->SetLayer(-1); } void cRecMenuItemRecording::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapText) pixmapText->SetLayer(5); } // --- cRecMenuItemTimelineHeader ------------------------------------------------------- cRecMenuItemTimelineHeader::cRecMenuItemTimelineHeader(time_t day, std::vector conflictsToday) { conflicts = conflictsToday; pixmapTimeline = NULL; pixmapTimerInfo = NULL; pixmapTimerConflicts = NULL; timer = NULL; this->day = day; selectable = false; active = false; height = 5 * font->Height(); timelineDrawn = false; } cRecMenuItemTimelineHeader::~cRecMenuItemTimelineHeader(void) { if (pixmapTimeline) osdManager.releasePixmap(pixmapTimeline); if (pixmapTimerInfo) osdManager.releasePixmap(pixmapTimerInfo); if (pixmapTimerConflicts) osdManager.releasePixmap(pixmapTimerConflicts); } void cRecMenuItemTimelineHeader::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapTimeline = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapTimerInfo = osdManager.requestPixmap(6, cRect(x, y + 3 * font->Height() / 2, width, 2 * font->Height())); if (conflicts.size() > 0) { pixmapTimerConflicts = osdManager.requestPixmap(6, cRect(x, y, width, height)); } } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapTimeline->SetViewPort(cRect(x, y, width, height)); pixmapTimerInfo->SetViewPort(cRect(x, y + 3 * font->Height() / 2, width, 2 * font->Height())); if (pixmapTimerConflicts) pixmapTimerConflicts->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemTimelineHeader::RefreshTimerDisplay(void) { if (!pixmapTimerInfo) return; if (timer) DrawCurrentTimer(); else pixmapTimerInfo->Fill(clrTransparent); } void cRecMenuItemTimelineHeader::Draw(void) { if (!timelineDrawn) { DrawTimeline(); timelineDrawn = true; } DrawTimerConflicts(); pixmap->Fill(clrTransparent); cString headerText = tr("Timers for"); cString dateText = DateString(day); cString header = cString::sprintf("%s: %s", *headerText, *dateText); int xText = (width - fontLarge->Width(*header)) / 2; int yText = (height/4 - fontLarge->Height())/2; pixmap->DrawText(cPoint(xText, yText), *header, colorText, clrTransparent, fontLarge); DrawCurrentTimer(); } void cRecMenuItemTimelineHeader::DrawCurrentTimer(void) { pixmapTimerInfo->Fill(clrTransparent); if (!timer) return; int infoHeight = pixmapTimerInfo->ViewPort().Height(); const cEvent *event = timer->Event(); const cChannel *channel = timer->Channel(); int x = 0; if (channel) { int logoWidth = infoHeight * tvguideConfig.logoWidthRatio / tvguideConfig.logoHeightRatio; bool logoDrawn = false; cImageLoader imgLoader; if (!tvguideConfig.hideChannelLogos) { if (imgLoader.LoadLogo(channel, logoWidth, infoHeight)) { cImage logo = imgLoader.GetImage(); pixmapTimerInfo->DrawImage(cPoint(0, 0), logo); x += logoWidth + 10; logoDrawn = true; } } if (tvguideConfig.hideChannelLogos || !logoDrawn) { int channelNameWidth = fontSmall->Width(channel->Name()); pixmapTimerInfo->DrawText(cPoint(10, (infoHeight - fontSmall->Height())/2), channel->Name(), colorText, clrTransparent, fontSmall); x += channelNameWidth + 20; } } cString timerStartTime = TimeString(timer->StartTime()); cString timerStopTime = TimeString(timer->StopTime()); cString channelInfo = cString::sprintf("%s %d", tr("Transp."), timer->Channel()->Transponder()); if (event) { cString title; if (event->ShortText()) { title = cString::sprintf("%s - %s", event->Title(), event->ShortText()); } else { title = event->Title(); } std::string titleCut = CutText(*title, width - x, font); cString infoString = cString::sprintf("%s: %s - %s (%s. %s - %s), %s", *event->GetDateString(), *event->GetTimeString(), *event->GetEndTimeString(), tr("Rec"), *timerStartTime, *timerStopTime, *channelInfo); pixmapTimerInfo->DrawText(cPoint(x, 0), *infoString, colorText, clrTransparent, font); pixmapTimerInfo->DrawText(cPoint(x, font->Height()), titleCut.c_str(), colorText, clrTransparent, font); } else { cString infoString = cString::sprintf("%s. %s - %s, %s", tr("Rec"), *timerStartTime, *timerStopTime, *channelInfo); pixmapTimerInfo->DrawText(cPoint(x, 0), *infoString, colorText, clrTransparent, font); } } void cRecMenuItemTimelineHeader::DrawTimeline(void) { pixmapTimeline->Fill(clrTransparent); width5Mins = (float)width * 5.0 / 24.0 / 60.0; int widthHour = 12 * width5Mins; x0 = (width - 24*widthHour)/2; int barHeight = fontSmall->Height(); int y = height - barHeight; tColor col1 = theme.Color(clrTimeline1); tColor col2 = theme.Color(clrTimeline2); tColor colCurText = col1; tColor colCurBack = col2; int x = x0; for (int hour = 0; hour < 24; hour++) { pixmapTimeline->DrawRectangle(cRect(x, y, widthHour, barHeight), colCurBack); cString hourText = cString::sprintf("%d", hour); int xDelta = (widthHour - fontSmall->Width(*hourText)) / 2; pixmapTimeline->DrawText(cPoint(x + xDelta, y), *hourText, colCurText, colCurBack, fontSmall); x += widthHour; colCurText = (colCurText==col1)?col2:col1; colCurBack = (colCurBack==col1)?col2:col1; } pixmapTimeline->DrawRectangle(cRect(x0, height-2, width - 2*x0, 2), col2); } void cRecMenuItemTimelineHeader::DrawTimerConflicts(void) { if (!pixmapTimerConflicts) return; pixmapTimerConflicts->Fill(clrTransparent); int numConflicts = conflicts.size(); int barHeight = fontSmall->Height(); int y = height - barHeight; for (int conflict = 0; conflict < numConflicts; conflict++) { int confStart = conflicts[conflict]->timeStart - day; int confStop = conflicts[conflict]->timeStop - day; int overlapStart = conflicts[conflict]->overlapStart - day; int overlapStop = conflicts[conflict]->overlapStop - day; if (confStart < 0) confStart = 0; if (confStop > 24*60*60) confStop = 24 * 60 * 60; if (overlapStart < 0) overlapStart = 0; if (overlapStop > 24*60*60) overlapStop = 24 * 60 * 60; confStart = confStart / 60; confStop = confStop / 60; overlapStart = overlapStart / 60; overlapStop = overlapStop / 60; int xStartConflict = x0 + confStart * width5Mins / 5; int xStopConflict = x0 + confStop * width5Mins / 5; int xStartOverlap = x0 + overlapStart * width5Mins / 5; int xStopOverlap = x0 + overlapStop * width5Mins / 5; pixmapTimerConflicts->DrawRectangle(cRect(xStartConflict, y, xStopConflict - xStartConflict, barHeight), theme.Color(clrRecMenuTimelineConflict)); pixmapTimerConflicts->DrawRectangle(cRect(xStartOverlap, y, xStopOverlap - xStartOverlap, barHeight), theme.Color(clrRecMenuTimelineConflictOverlap)); } } void cRecMenuItemTimelineHeader::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapTimeline) pixmapTimeline->SetLayer(-1); if (pixmapTimerInfo) pixmapTimerInfo->SetLayer(-1); if (pixmapTimerConflicts) pixmapTimerConflicts->SetLayer(-1); } void cRecMenuItemTimelineHeader::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapTimeline) pixmapTimeline->SetLayer(5); if (pixmapTimerInfo) pixmapTimerInfo->SetLayer(6); if (pixmapTimerConflicts) pixmapTimerConflicts->SetLayer(6); } // --- cRecMenuItemTimelineTimer ------------------------------------------------------- cRecMenuItemTimelineTimer::cRecMenuItemTimelineTimer(const cTimer *timer, time_t start, time_t stop, std::vector conflictsToday, cRecMenuItemTimelineHeader *header, bool active) { conflicts = conflictsToday; defaultBackground = false; pixmapBack = NULL; pixmapTimerConflicts = NULL; this->timer = timer; this->start = start; this->stop = stop; this->header = header; selectable = true; this->active = active; height = geoManager.osdHeight / 16; } cRecMenuItemTimelineTimer::~cRecMenuItemTimelineTimer(void) { if (pixmapBack) osdManager.releasePixmap(pixmapBack); if (pixmapTimerConflicts) osdManager.releasePixmap(pixmapTimerConflicts); } void cRecMenuItemTimelineTimer::SetPixmaps(void) { if (!pixmap) { pixmapBack = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmap = osdManager.requestPixmap(5, cRect(x, y, width, height)); if (conflicts.size() > 0) { pixmapTimerConflicts = osdManager.requestPixmap(6, cRect(x, y, width, height)); } } else { pixmapBack->SetViewPort(cRect(x, y, width, height)); pixmap->SetViewPort(cRect(x, y, width, height)); if (pixmapTimerConflicts) pixmapTimerConflicts->SetViewPort(cRect(x, y, width, height)); } width5Mins = (float)width * 5.0 / 24.0 / 60.0; x0 = (width - 24*12*width5Mins)/2; } void cRecMenuItemTimelineTimer::Draw(void) { DrawBackground(); if (!timer) { DrawNoTimerInfo(); return; } if (!drawn) { pixmap->Fill(clrTransparent); DrawTimeScale(); DrawTimerBar(); DrawTimerConflicts(); drawn = true; } } void cRecMenuItemTimelineTimer::DrawBackground(void) { tColor backgroundColor = (active)?theme.Color(clrRecMenuTimelineActive):theme.Color(clrRecMenuTimelineBack); pixmapBack->Fill(clrTransparent); pixmapBack->DrawRectangle(cRect(x0, 0, width - 2 * x0, height), backgroundColor); } void cRecMenuItemTimelineTimer::DrawTimeScale(void) { int x = x0; for (int hour = 0; hour < 25; hour++) { int xModified = (hour%2) ? x - 1 : x ; pixmap->DrawRectangle(cRect(xModified,0,1,height), theme.Color(clrTimeline2)); x += width5Mins*12; } } void cRecMenuItemTimelineTimer::DrawTimerBar(void) { time_t timerStart = timer->StartTime() - start; time_t timerStop = timer->StopTime() - start; if (timerStart < 0) timerStart = 0; if (timerStop > 24*60*60) timerStop = 24 * 60 * 60; timerStart = timerStart / 60; timerStop = timerStop / 60; int xStart = x0 + timerStart * width5Mins / 5; int xStop = x0 + timerStop * width5Mins / 5; pixmap->DrawRectangle(cRect(xStart, height / 4, xStop - xStart, height / 2), theme.Color(clrRecMenuTimelineTimer)); } void cRecMenuItemTimelineTimer::DrawTimerConflicts(void) { if (!pixmapTimerConflicts) return; pixmapTimerConflicts->Fill(clrTransparent); int numConflicts = conflicts.size(); for (int conflict = 0; conflict < numConflicts; conflict++) { int confStart = conflicts[conflict]->timeStart - start; int confStop = conflicts[conflict]->timeStop - start; int overlapStart = conflicts[conflict]->overlapStart - start; int overlapStop = conflicts[conflict]->overlapStop - start; if (confStart < 0) confStart = 0; if (confStop > 24*60*60) confStop = 24 * 60 * 60; if (overlapStart < 0) overlapStart = 0; if (overlapStop > 24*60*60) overlapStop = 24 * 60 * 60; confStart = confStart / 60; confStop = confStop / 60; overlapStart = overlapStart / 60; overlapStop = overlapStop / 60; int xStartConflict = x0 + confStart * width5Mins / 5; int xStopConflict = x0 + confStop * width5Mins / 5; int xStartOverlap = x0 + overlapStart * width5Mins / 5; int xStopOverlap = x0 + overlapStop * width5Mins / 5; pixmapTimerConflicts->DrawRectangle(cRect(xStartConflict, 0, xStopConflict - xStartConflict, height), theme.Color(clrRecMenuTimelineConflict)); pixmapTimerConflicts->DrawRectangle(cRect(xStartOverlap, 0, xStopOverlap - xStartOverlap, height), theme.Color(clrRecMenuTimelineConflictOverlap)); } } void cRecMenuItemTimelineTimer::DrawNoTimerInfo(void) { pixmap->Fill(clrTransparent); cString noTimersText = tr("No Timers active"); int widthText = font->Width(*noTimersText); int x = (width - widthText) / 2; int y = (height - font->Height()) / 2; pixmap->DrawText(cPoint(x, y), *noTimersText, colorText, clrTransparent, font); } void cRecMenuItemTimelineTimer::setActive(void) { active = true; header->SetCurrentTimer(timer); header->RefreshTimerDisplay(); } void cRecMenuItemTimelineTimer::setInactive(void) { active = false; header->UnsetCurrentTimer(); header->RefreshTimerDisplay(); } void cRecMenuItemTimelineTimer::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapBack) pixmapBack->SetLayer(-1); if (pixmapTimerConflicts) pixmapTimerConflicts->SetLayer(-1); } void cRecMenuItemTimelineTimer::Show(void) { if (pixmap) pixmap->SetLayer(5); if (pixmapBack) pixmapBack->SetLayer(4); if (pixmapTimerConflicts) pixmapTimerConflicts->SetLayer(6); } const cTimer *cRecMenuItemTimelineTimer::GetTimerValue(void) { return timer; } eRecMenuState cRecMenuItemTimelineTimer::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kOk: return rmsTimelineTimerEdit; default: break; } return rmsNotConsumed; } // --- cRecMenuItemSearchTimer ------------------------------------------------------- cRecMenuItemSearchTimer::cRecMenuItemSearchTimer(cTVGuideSearchTimer timer, eRecMenuState action1, eRecMenuState action2, eRecMenuState action3, bool active) { this->timer = timer; this->action = action1; this->action2 = action2; this->action3 = action3; pixmapText = NULL; pixmapIcons = NULL; selectable = true; this->active = active; iconActive = 0; height = 2 * font->Height();; } cRecMenuItemSearchTimer::~cRecMenuItemSearchTimer(void) { if (pixmapText) osdManager.releasePixmap(pixmapText); if (pixmapIcons) osdManager.releasePixmap(pixmapIcons); } void cRecMenuItemSearchTimer::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapText = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapIcons = osdManager.requestPixmap(6, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapText->SetViewPort(cRect(x, y, width, height)); pixmapIcons->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemSearchTimer::Draw(void) { int textX = DrawIcons(); pixmapText->Fill(clrTransparent); textX += 20; cString label; if (timer.Active()) { label = cString::sprintf("\"%s\"", timer.SearchString().c_str()); } else { label = cString::sprintf("\"%s\" (%s)", timer.SearchString().c_str(), tr("inactive")); } int numTimersActive = timer.GetNumTimers(); int numRecordings = timer.GetNumRecordings(); cString info = cString::sprintf("%s: %d, %s: %d", tr("active timers"), numTimersActive, tr("recordings done"), numRecordings); pixmapText->DrawText(cPoint(textX, 5 + (height/2 - font->Height())/2), *label, colorText, clrTransparent, font); pixmapText->DrawText(cPoint(textX, height/2 + (height/2 - fontSmall->Height())/2), *info, colorText, clrTransparent, fontSmall); } void cRecMenuItemSearchTimer::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapText) pixmapText->SetLayer(-1); if (pixmapIcons) pixmapIcons->SetLayer(-1); } void cRecMenuItemSearchTimer::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapText) pixmapText->SetLayer(5); if (pixmapIcons) pixmapIcons->SetLayer(6); } int cRecMenuItemSearchTimer::DrawIcons(void) { pixmapIcons->Fill(clrTransparent); int iconsX = 10; int iconSize = height / 2; int iconY = (height - iconSize) / 2; std::string iconEdit, iconDelete, iconSearch; if (active) { iconSearch = (iconActive==0)?"search_active":"search_inactive"; iconEdit = (iconActive==1)?"edit_active":"edit_inactive"; iconDelete = (iconActive==2)?"delete_active":"delete_inactive"; } else { iconSearch = "search_inactive"; iconEdit = "edit_inactive"; iconDelete = "delete_inactive"; } cImage *imgSearch = imgCache.GetIcon(iconSearch, iconSize, iconSize); if (imgSearch) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgSearch); iconsX += iconSize + 10; } cImage *imgEdit = imgCache.GetIcon(iconEdit, iconSize, iconSize); if (imgEdit) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgEdit); iconsX += iconSize + 10; } cImage *imgDel = imgCache.GetIcon(iconDelete, iconSize, iconSize); if (imgDel) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgDel); iconsX += iconSize + 10; } return iconsX; } eRecMenuState cRecMenuItemSearchTimer::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kLeft: if (iconActive > 0) { iconActive--; DrawIcons(); return rmsConsumed; } return rmsNotConsumed; break; case kRight: { if (iconActive < 2) { iconActive++; DrawIcons(); return rmsConsumed; } return rmsNotConsumed; break; } case kOk: if (iconActive == 0) return action; else if (iconActive == 1) return action2; else if (iconActive == 2) return action3; break; default: break; } return rmsNotConsumed; } // --- cRecMenuItemFavorite ------------------------------------------------------- cRecMenuItemFavorite::cRecMenuItemFavorite(cTVGuideSearchTimer favorite, eRecMenuState action1, bool active) { this->favorite = favorite; this->action = action1; pixmapText = NULL; pixmapIcons = NULL; selectable = true; this->active = active; height = 2 * font->Height();; } cRecMenuItemFavorite::~cRecMenuItemFavorite(void) { if (pixmapText) osdManager.releasePixmap(pixmapText); if (pixmapIcons) osdManager.releasePixmap(pixmapIcons); } void cRecMenuItemFavorite::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapText = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapIcons = osdManager.requestPixmap(6, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapText->SetViewPort(cRect(x, y, width, height)); pixmapIcons->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemFavorite::Draw(void) { int textX = DrawIcons(); pixmapText->Fill(clrTransparent); textX += 20; cString label = cString::sprintf("\"%s\"", favorite.SearchString().c_str()); pixmapText->DrawText(cPoint(textX, (height - fontLarge->Height())/2), *label, colorText, clrTransparent, fontLarge); } void cRecMenuItemFavorite::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapText) pixmapText->SetLayer(-1); if (pixmapIcons) pixmapIcons->SetLayer(-1); } void cRecMenuItemFavorite::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapText) pixmapText->SetLayer(5); if (pixmapIcons) pixmapIcons->SetLayer(6); } int cRecMenuItemFavorite::DrawIcons(void) { pixmapIcons->Fill(clrTransparent); int iconsX = 10; int iconSize = height / 2; int iconY = (height - iconSize) / 2; std::string iconSearch; iconSearch = active ? "search_active" : "search_inactive" ; cImage *imgSearch = imgCache.GetIcon(iconSearch, iconSize, iconSize); if (imgSearch) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgSearch); iconsX += iconSize + 10; } return iconsX; } eRecMenuState cRecMenuItemFavorite::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kOk: return action; default: break; } return rmsNotConsumed; } // --- cRecMenuItemFavoriteStatic ------------------------------------------------------- cRecMenuItemFavoriteStatic::cRecMenuItemFavoriteStatic(std::string text, eRecMenuState action, bool active) { this->text = text; this->action = action; pixmapText = NULL; pixmapIcons = NULL; selectable = true; this->active = active; height = 2 * font->Height();; } cRecMenuItemFavoriteStatic::~cRecMenuItemFavoriteStatic(void) { if (pixmapText) osdManager.releasePixmap(pixmapText); if (pixmapIcons) osdManager.releasePixmap(pixmapIcons); } void cRecMenuItemFavoriteStatic::SetPixmaps(void) { if (!pixmap) { pixmap = osdManager.requestPixmap(4, cRect(x, y, width, height)); pixmapText = osdManager.requestPixmap(5, cRect(x, y, width, height)); pixmapIcons = osdManager.requestPixmap(6, cRect(x, y, width, height)); } else { pixmap->SetViewPort(cRect(x, y, width, height)); pixmapText->SetViewPort(cRect(x, y, width, height)); pixmapIcons->SetViewPort(cRect(x, y, width, height)); } } void cRecMenuItemFavoriteStatic::Draw(void) { int textX = DrawIcons(); pixmapText->Fill(clrTransparent); textX += 20; pixmapText->DrawText(cPoint(textX, (height - fontLarge->Height())/2), text.c_str(), colorText, clrTransparent, fontLarge); } void cRecMenuItemFavoriteStatic::Hide(void) { if (pixmap) pixmap->SetLayer(-1); if (pixmapText) pixmapText->SetLayer(-1); if (pixmapIcons) pixmapIcons->SetLayer(-1); } void cRecMenuItemFavoriteStatic::Show(void) { if (pixmap) pixmap->SetLayer(4); if (pixmapText) pixmapText->SetLayer(5); if (pixmapIcons) pixmapIcons->SetLayer(6); } int cRecMenuItemFavoriteStatic::DrawIcons(void) { pixmapIcons->Fill(clrTransparent); int iconsX = 10; int iconSize = height / 2; int iconY = (height - iconSize) / 2; std::string iconSearch; iconSearch = active ? "search_active" : "search_inactive" ; cImage *imgSearch = imgCache.GetIcon(iconSearch, iconSize, iconSize); if (imgSearch) { pixmapIcons->DrawImage(cPoint(iconsX, iconY), *imgSearch); iconsX += iconSize + 10; } return iconsX; } eRecMenuState cRecMenuItemFavoriteStatic::ProcessKey(eKeys Key) { switch (Key & ~k_Repeat) { case kOk: return action; default: break; } return rmsNotConsumed; }