/* Neutrino-GUI - DBoxII-Project Copyright (C) 2001 Steffen Hehn 'McClean' Homepage: http://dbox.cyberphoria.org/ Kommentar: Diese GUI wurde von Grund auf neu programmiert und sollte nun vom Aufbau und auch den Ausbaumoeglichkeiten gut aussehen. Neutrino basiert auf der Client-Server Idee, diese GUI ist also von der direkten DBox- Steuerung getrennt. Diese wird dann von Daemons uebernommen. License: GPL This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include extern CBouquetManager *g_bouquetManager; CBouquetList::CBouquetList(const char * const Name) { frameBuffer = CFrameBuffer::getInstance(); selected = 0; liststart = 0; favonly = false; if(Name == NULL) name = g_Locale->getText(LOCALE_BOUQUETLIST_HEAD); else name = Name; } CBouquetList::~CBouquetList() { for (std::vector::iterator it = Bouquets.begin(); it != Bouquets.end(); ++it) delete (*it); Bouquets.clear(); } CBouquet* CBouquetList::addBouquet(CZapitBouquet * zapitBouquet) { int BouquetKey= Bouquets.size();//FIXME not used ? const char * bname; if (zapitBouquet->bFav) bname = g_Locale->getText(LOCALE_FAVORITES_BOUQUETNAME); else if (zapitBouquet->bOther) bname = g_Locale->getText(LOCALE_BOUQUETNAME_OTHER); else bname = zapitBouquet->Name.c_str(); CBouquet* tmp = new CBouquet(BouquetKey, bname, zapitBouquet->bLocked, !zapitBouquet->bUser); tmp->zapitBouquet = zapitBouquet; Bouquets.push_back(tmp); return tmp; } CBouquet* CBouquetList::addBouquet(const char * const pname, int BouquetKey, bool locked) { if ( BouquetKey==-1 ) BouquetKey= Bouquets.size(); CBouquet* tmp = new CBouquet( BouquetKey, pname, locked, true); Bouquets.push_back(tmp); return(tmp); } void CBouquetList::deleteBouquet(CBouquet*bouquet) { if (bouquet != NULL) { std::vector::iterator it = find(Bouquets.begin(), Bouquets.end(), bouquet); if (it != Bouquets.end()) { Bouquets.erase(it); delete bouquet; } } } t_bouquet_id CBouquetList::getActiveBouquetNumber() { return (t_bouquet_id)selected; } #if 0 void CBouquetList::adjustToChannel( int nChannelNr) { for (uint32_t i=0; ichannelList->hasChannel(nChannelNr); if (nChannelPos > -1) { selected = i; Bouquets[i]->channelList->setSelected(nChannelPos); return; } } } #endif bool CBouquetList::hasChannelID(t_channel_id channel_id) { for (uint32_t i = 0; i < Bouquets.size(); i++) { int nChannelPos = Bouquets[i]->channelList->hasChannelID(channel_id); if (nChannelPos > -1) return true; } return false; } bool CBouquetList::adjustToChannelID(t_channel_id channel_id) { //printf("CBouquetList::adjustToChannelID [%s] to %llx, selected %d size %d\n", name.c_str(), channel_id, selected, Bouquets.size()); if(selected < Bouquets.size()) { int nChannelPos = Bouquets[selected]->channelList->hasChannelID(channel_id); if(nChannelPos > -1) { //printf("CBouquetList::adjustToChannelID [%s] to %llx -> not needed\n", name.c_str(), channel_id); Bouquets[selected]->channelList->setSelected(nChannelPos); return true; } } //printf("CBouquetList::adjustToChannelID [%s] to %llx\n", name.c_str(), channel_id); for (uint32_t i=0; i < Bouquets.size(); i++) { if(i == selected) continue; int nChannelPos = Bouquets[i]->channelList->hasChannelID(channel_id); if (nChannelPos > -1) { selected = i; Bouquets[i]->channelList->setSelected(nChannelPos); return true; } } return false; } /* used in channellist to switch bouquets up/down */ int CBouquetList::showChannelList( int nBouquet) { if ((nBouquet < 0)|| (nBouquet >= (int) Bouquets.size())) nBouquet = selected; int nNewChannel = Bouquets[nBouquet]->channelList->exec(); if (nNewChannel > -1) { selected = nBouquet; nNewChannel = -2; } return nNewChannel; } /* bShowChannelList default to false , return seems not checked anywhere */ int CBouquetList::activateBouquet( int id, bool bShowChannelList) { int res = -1; if((id >= 0) && (id < (int) Bouquets.size())) selected = id; if (bShowChannelList) { res = Bouquets[selected]->channelList->exec(); if(res > -1) res = -2; } return res; } int CBouquetList::exec( bool bShowChannelList) { /* select bouquet to show */ int res = show(bShowChannelList); //printf("Bouquet-exec: res %d bShowChannelList %d\n", res, bShowChannelList); fflush(stdout); if(!bShowChannelList) return res; /* if >= 0, call activateBouquet to show channel list */ if ( res > -1) { return activateBouquet(selected, bShowChannelList); } return res; } int CBouquetList::doMenu() { int i = 0; int select = -1; static int old_selected = 0; signed int bouquet_id; char cnt[5]; CZapitBouquet * tmp, * zapitBouquet; ZapitChannelList* channels; if(Bouquets.empty() || g_settings.minimode) return 0; zapitBouquet = Bouquets[selected]->zapitBouquet; /* zapitBouquet not NULL only on real bouquets, not on virtual SAT or HD */ if(!zapitBouquet) return 0; CMenuWidget* menu = new CMenuWidget(LOCALE_CHANNELLIST_EDIT, NEUTRINO_ICON_SETTINGS); menu->enableFade(false); CMenuSelectorTarget * selector = new CMenuSelectorTarget(&select); sprintf(cnt, "%d", i); if(!zapitBouquet->bUser) { menu->addItem(new CMenuForwarder(LOCALE_FAVORITES_COPY, true, NULL, selector, cnt, CRCInput::RC_blue, NEUTRINO_ICON_BUTTON_BLUE), old_selected == i ++); menu->exec(NULL, ""); delete menu; delete selector; printf("CBouquetList::doMenu: %d selected\n", select); bool added = false; if(select >= 0) { old_selected = select; switch(select) { case 0: hide(); bouquet_id = g_bouquetManager->existsUBouquet(Bouquets[selected]->channelList->getName()); if(bouquet_id < 0) { tmp = g_bouquetManager->addBouquet(Bouquets[selected]->channelList->getName(), true); bouquet_id = g_bouquetManager->existsUBouquet(Bouquets[selected]->channelList->getName()); } else tmp = g_bouquetManager->Bouquets[bouquet_id]; if(bouquet_id < 0) return -1; channels = &zapitBouquet->tvChannels; for(int li = 0; li < (int) channels->size(); li++) { if (!g_bouquetManager->existsChannelInBouquet(bouquet_id, ((*channels)[li])->getChannelID())) { added = true; tmp->addService((*channels)[li]); } } channels = &zapitBouquet->radioChannels; for(int li = 0; li < (int) channels->size(); li++) { if (!g_bouquetManager->existsChannelInBouquet(bouquet_id, ((*channels)[li])->getChannelID())) { added = true; tmp->addService((*channels)[li]); } } return added ? 1 : -1; break; default: break; } } return -1; } else { menu->addItem(new CMenuForwarder(LOCALE_BOUQUETEDITOR_DELETE, true, NULL, selector, cnt, CRCInput::RC_red, NEUTRINO_ICON_BUTTON_RED), old_selected == i ++); menu->exec(NULL, ""); delete menu; delete selector; printf("CBouquetList::doMenu: %d selected\n", select); if(select >= 0) { old_selected = select; switch(select) { case 0: hide(); bouquet_id = g_bouquetManager->existsUBouquet(Bouquets[selected]->channelList->getName()); if(bouquet_id >= 0) { g_bouquetManager->deleteBouquet(bouquet_id); return 1; } break; default: break; } } return -1; } return 0; } const struct button_label CBouquetListButtons[4] = { { NEUTRINO_ICON_BUTTON_RED, LOCALE_CHANNELLIST_FAVS}, { NEUTRINO_ICON_BUTTON_GREEN, LOCALE_CHANNELLIST_PROVS}, { NEUTRINO_ICON_BUTTON_YELLOW, LOCALE_CHANNELLIST_SATS}, { NEUTRINO_ICON_BUTTON_BLUE, LOCALE_CHANNELLIST_HEAD} }; void CBouquetList::updateSelection(int newpos) { if((int) selected != newpos) { int prev_selected = selected; unsigned int oldliststart = liststart; selected = newpos; liststart = (selected/listmaxshow)*listmaxshow; if (oldliststart != liststart) paint(); else { paintItem(prev_selected - liststart); paintItem(selected - liststart); } } } /* bShowChannelList default to true, returns new bouquet or -1/-2 */ int CBouquetList::show(bool bShowChannelList) { neutrino_msg_t msg; neutrino_msg_data_t data; int res = -1; int icol_w, icol_h; int w_max_text = 0; int w_max_icon = 0; int h_max_icon = 0; favonly = !bShowChannelList; for(unsigned int count = 0; count < sizeof(CBouquetListButtons)/sizeof(CBouquetListButtons[0]);count++){ int w_text = g_Font[SNeutrinoSettings::FONT_TYPE_INFOBAR_SMALL]->getRenderWidth(g_Locale->getText (CBouquetListButtons[count].locale),true); w_max_text = std::max(w_max_text, w_text); frameBuffer->getIconSize(CBouquetListButtons[count].button, &icol_w, &icol_h); w_max_icon = std::max(w_max_icon, icol_w); h_max_icon = std::max(h_max_icon, icol_h); } int need_width = sizeof(CBouquetListButtons)/sizeof(CBouquetListButtons[0])*(w_max_icon + w_max_text + 20); CVFD::getInstance()->setMode(CVFD::MODE_MENU_UTF8, ""); fheight = g_Font[SNeutrinoSettings::FONT_TYPE_CHANNELLIST]->getHeight(); width = w_max (need_width, 20);//500 height = h_max (16 * fheight, 40); footerHeight = std::max(h_max_icon+8, g_Font[SNeutrinoSettings::FONT_TYPE_INFOBAR_SMALL]->getHeight()+8); theight = g_Font[SNeutrinoSettings::FONT_TYPE_MENU_TITLE]->getHeight(); listmaxshow = (height - theight - footerHeight)/fheight; height = theight + footerHeight + listmaxshow * fheight; // recalc height x = frameBuffer->getScreenX() + (frameBuffer->getScreenWidth() - width) / 2; y = frameBuffer->getScreenY() + (frameBuffer->getScreenHeight() - height) / 2; int lmaxpos= 1; int i= Bouquets.size(); while ((i= i/10)!=0) lmaxpos++; COSDFader fader(g_settings.menu_Content_alpha); fader.StartFadeIn(); paintHead(); paint(); int oldselected = selected; int firstselected = selected+ 1; int zapOnExit = false; unsigned int chn= 0; int pos= lmaxpos; uint64_t timeoutEnd = CRCInput::calcTimeoutEnd(g_settings.timing[SNeutrinoSettings::TIMING_CHANLIST]); bool loop=true; while (loop) { g_RCInput->getMsgAbsoluteTimeout( &msg, &data, &timeoutEnd ); if ( msg <= CRCInput::RC_MaxRC ) timeoutEnd = CRCInput::calcTimeoutEnd(g_settings.timing[SNeutrinoSettings::TIMING_CHANLIST]); if((msg == NeutrinoMessages::EVT_TIMER) && (data == fader.GetTimer())) { if(fader.Fade()) loop = false; } else if ((msg == CRCInput::RC_timeout ) || (msg == (neutrino_msg_t)g_settings.key_channelList_cancel) || ((msg == CRCInput::RC_favorites) && (CNeutrinoApp::getInstance()->GetChannelMode() == LIST_MODE_FAV))) { selected = oldselected; if(fader.StartFadeOut()) { timeoutEnd = CRCInput::calcTimeoutEnd( 1 ); msg = 0; } else loop=false; } else if(msg == CRCInput::RC_red || msg == CRCInput::RC_favorites) { if (!favonly && CNeutrinoApp::getInstance()->GetChannelMode() != LIST_MODE_FAV) { CNeutrinoApp::getInstance()->SetChannelMode(LIST_MODE_FAV); hide(); return -3; } } else if(msg == CRCInput::RC_green) { if (!favonly && CNeutrinoApp::getInstance()->GetChannelMode() != LIST_MODE_PROV) { CNeutrinoApp::getInstance()->SetChannelMode(LIST_MODE_PROV); hide(); return -3; } } else if(msg == CRCInput::RC_yellow || msg == CRCInput::RC_sat) { if(!favonly && bShowChannelList && CNeutrinoApp::getInstance()->GetChannelMode() != LIST_MODE_SAT) { CNeutrinoApp::getInstance()->SetChannelMode(LIST_MODE_SAT); hide(); return -3; } } else if(msg == CRCInput::RC_blue) { if(!favonly && bShowChannelList && CNeutrinoApp::getInstance()->GetChannelMode() != LIST_MODE_ALL) { CNeutrinoApp::getInstance()->SetChannelMode(LIST_MODE_ALL); hide(); return -3; } } else if ( msg == CRCInput::RC_setup) { if (!favonly && !Bouquets.empty()) { int ret = doMenu(); if(ret > 0) { CNeutrinoApp::getInstance()->MarkChannelListChanged(); res = -4; loop = false; } else if(ret < 0) { paintHead(); paint(); } } } else if ( msg == (neutrino_msg_t) g_settings.key_list_start ) { if (!Bouquets.empty()) updateSelection(0); } else if ( msg == (neutrino_msg_t) g_settings.key_list_end ) { if (!Bouquets.empty()) updateSelection(Bouquets.size()-1); } else if (msg == CRCInput::RC_up || (int) msg == g_settings.key_channelList_pageup) { if (!Bouquets.empty()) { int step = ((int) msg == g_settings.key_channelList_pageup) ? listmaxshow : 1; // browse or step 1 int new_selected = selected - step; if (new_selected < 0) { if (selected != 0 && step != 1) new_selected = 0; else new_selected = Bouquets.size() - 1; } updateSelection(new_selected); } } else if (msg == CRCInput::RC_down || (int) msg == g_settings.key_channelList_pagedown) { if (!Bouquets.empty()) { int step = ((int) msg == g_settings.key_channelList_pagedown) ? listmaxshow : 1; // browse or step 1 int new_selected = selected + step; if (new_selected >= (int) Bouquets.size()) { if ((Bouquets.size() - listmaxshow -1 < selected) && (selected != (Bouquets.size() - 1)) && (step != 1)) new_selected = Bouquets.size() - 1; else if (((Bouquets.size() / listmaxshow) + 1) * listmaxshow == Bouquets.size() + listmaxshow) // last page has full entries new_selected = 0; else new_selected = ((step == (int) listmaxshow) && (new_selected < (int) (((Bouquets.size() / listmaxshow)+1) * listmaxshow))) ? (Bouquets.size() - 1) : 0; } updateSelection(new_selected); } } else if(msg == (neutrino_msg_t)g_settings.key_bouquet_up || msg == (neutrino_msg_t)g_settings.key_bouquet_down) { if(bShowChannelList) { int mode = CNeutrinoApp::getInstance()->GetChannelMode(); mode += (msg == (neutrino_msg_t)g_settings.key_bouquet_down) ? -1 : 1; if(mode < 0) mode = LIST_MODE_LAST - 1; else if(mode >= LIST_MODE_LAST) mode = 0; CNeutrinoApp::getInstance()->SetChannelMode(mode); hide(); return -3; } } else if ( msg == CRCInput::RC_ok ) { if(!Bouquets.empty() && (!bShowChannelList || !Bouquets[selected]->channelList->isEmpty())) { zapOnExit = true; loop=false; } } else if (CRCInput::isNumeric(msg)) { if (!Bouquets.empty()) { if (pos == lmaxpos) { if (msg == CRCInput::RC_0) { chn = firstselected; pos = lmaxpos; } else { chn = CRCInput::getNumericValue(msg); pos = 1; } } else { chn = chn * 10 + CRCInput::getNumericValue(msg); pos++; } if (chn > Bouquets.size()) { chn = firstselected; pos = lmaxpos; } int new_selected = (chn - 1) % Bouquets.size(); // is % necessary (i.e. can firstselected be > Bouquets.size()) ? updateSelection(new_selected); } } else { if ( CNeutrinoApp::getInstance()->handleMsg( msg, data ) & messages_return::cancel_all ) { loop = false; res = -2; } }; } hide(); fader.Stop(); CVFD::getInstance()->setMode(CVFD::MODE_TVRADIO); if(zapOnExit) { return (selected); } else { return (res); } } void CBouquetList::hide() { frameBuffer->paintBackgroundBoxRel(x,y, width,height+10); } void CBouquetList::paintItem(int pos) { int ypos = y+ theight+0 + pos*fheight; fb_pixel_t color; fb_pixel_t bgcolor; bool iscurrent = true; int npos = liststart + pos; const char * lname = NULL; if(npos < (int) Bouquets.size()) lname = (Bouquets[npos]->zapitBouquet && Bouquets[npos]->zapitBouquet->bFav) ? g_Locale->getText(LOCALE_FAVORITES_BOUQUETNAME) : Bouquets[npos]->channelList->getName(); if (npos == (int) selected) { color = COL_MENUCONTENTSELECTED_TEXT; bgcolor = COL_MENUCONTENTSELECTED_PLUS_0; frameBuffer->paintBoxRel(x, ypos, width- 15, fheight, bgcolor, RADIUS_LARGE); if(npos < (int) Bouquets.size()) CVFD::getInstance()->showMenuText(0, lname, -1, true); } else { if(npos < (int) Bouquets.size()) iscurrent = !Bouquets[npos]->channelList->isEmpty(); color = iscurrent ? COL_MENUCONTENT_TEXT : COL_MENUCONTENTINACTIVE_TEXT; bgcolor = iscurrent ? COL_MENUCONTENT_PLUS_0 : COL_MENUCONTENTINACTIVE_PLUS_0; frameBuffer->paintBoxRel(x, ypos, width- 15, fheight, bgcolor); } if(npos < (int) Bouquets.size()) { char tmp[10]; sprintf((char*) tmp, "%d", npos+ 1); int numpos = x+5+numwidth- g_Font[SNeutrinoSettings::FONT_TYPE_CHANNELLIST_NUMBER]->getRenderWidth(tmp); g_Font[SNeutrinoSettings::FONT_TYPE_CHANNELLIST_NUMBER]->RenderString(numpos,ypos+fheight, numwidth+5, tmp, color, fheight); g_Font[SNeutrinoSettings::FONT_TYPE_CHANNELLIST]->RenderString(x+ 5+ numwidth+ 10, ypos+ fheight, width- numwidth- 20- 15, lname, color, 0, true); // UTF-8 //CVFD::getInstance()->showMenuText(0, bouq->channelList->getName(), -1, true); } } void CBouquetList::paintHead() { CComponentsHeader header(x, y, width, theight, name, NULL /*no header icon*/); header.paint(CC_SAVE_SCREEN_NO); } void CBouquetList::paint() { liststart = (selected/listmaxshow)*listmaxshow; int lastnum = liststart + listmaxshow; int bsize = Bouquets.empty() ? 1 : Bouquets.size(); int maxDigitWidth = g_Font[SNeutrinoSettings::FONT_TYPE_CHANNELLIST_NUMBER]->getMaxDigitWidth(); int _lastnum = lastnum; while (_lastnum) { numwidth += maxDigitWidth; _lastnum /= 10; } frameBuffer->paintBoxRel(x, y+theight, width, height - theight - footerHeight, COL_MENUCONTENT_PLUS_0); int numbuttons = sizeof(CBouquetListButtons)/sizeof(CBouquetListButtons[0]); #if 0 if (favonly) /* this actually shows favorites and providers button, but both are active anyway */ numbuttons = 2; ::paintButtons(x, y + (height - footerHeight), width, numbuttons, CBouquetListButtons, width, footerHeight); #endif if (favonly) frameBuffer->paintBoxRel(x, y + (height - footerHeight), width, footerHeight, COL_INFOBAR_SHADOW_PLUS_1, RADIUS_LARGE, CORNER_BOTTOM); //round else ::paintButtons(x, y + (height - footerHeight), width, numbuttons, CBouquetListButtons, width, footerHeight); if(!Bouquets.empty()) { for(unsigned int count=0;countpaintBoxRel(x+ width- 15,ypos, 15, sb, COL_MENUCONTENT_PLUS_1); int sbc= ((bsize - 1)/ listmaxshow)+ 1; /* bsize is > 0, so sbc is also > 0 */ float sbh= (sb - 4)/ sbc; int sbs= (selected/listmaxshow); frameBuffer->paintBoxRel(x+ width- 13, ypos+ 2+ int(sbs* sbh) , 11, int(sbh), COL_MENUCONTENT_PLUS_3); }