mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- EPG events from epg.data or SVDRP's PUTE command now have their Title set to "No Title" if none was set. - Fixed checking toFile in cCuttingThread::Action() (found in a larger patch from Artur Skawina). - Fixed a crash when pressing '0' in the "Schedule" menu on a channel that doesn't have any EPG data (reported ny Alexander Hans). - Updated the Danish OSD texts (thanks to Mogens Elneff). - Fixed a missing ',' in the Greek OSD texts (thanks to Arthur Konovalov). - Updated the Estonian OSD texts (thanks to Arthur Konovalov). - Fixed handling the tfRecording flag when reading timers (bug reported by Andreas Mair). - Now checking whether the channel exists before setting the PMT filter in cPatFilter::Process() (thanks to Thomas Bergwinkl). - Now trying to reestablish the connection to the LIRC daemon in case it breaks (thanks to Ville Skyttä). - Updated the Finnish OSD texts (thanks to Rolf Ahrenberg). - Fixed channel switching with the Down (Up) key in case the current channel is already the first (last) in the list (reported by Frank Krömmelbein). - Removed the "buffer reserve" in Transfer Mode - it's no longer necessary with recent driver/firmware versions. - The epg.data file is now written when VDR exits (suggested by Daniel Karsubka). - Fixed cTimers::GetNextActiveTimer() so that it won't return an expired timer (reported by Rolf Ahrenberg). - Changed DVBS_TUNE_TIMEOUT and DVBC_TUNE_TIMEOUT to 9000ms to avoid problems with channels that have low symbol rates (reported by Suur Karu). - Fixed displaying the current audio track in the channel display. - When reading epg.data (or data from PUTE), the version number of events with table IDs smaller than 0x50 is now ignored because otherwise the current running status would not be set after a restart of VDR. - Implemented a timeout for remote controls that don't deliver "repeat" keypresses very fast (based on a suggestion by Luca Olivetti; problem with the new handling of k_Repeat keypresses in channel switching reported by Udo Richter). - When looking for the present or following EPG event, the running status is now always taken into account. - Now initializing the channels' schedule pointers when reading the epg.data file, so that the first WhatsOn menu will come up faster. - If a shutdown is requested, but the shutdown script doesn't actually halt the system, it is now tried again after 5 minutes (suggested by Helmut Auer). - Separated the 'install' target into several individual targets; renamed the 'plugins-install' target to 'install-plugins' (thanks to Helmut Auer).
197 lines
6.8 KiB
C
197 lines
6.8 KiB
C
/*
|
|
* interface.c: Abstract user interface layer
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: interface.c 1.73 2006/01/29 12:35:50 kls Exp $
|
|
*/
|
|
|
|
#include "interface.h"
|
|
#include <ctype.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include "i18n.h"
|
|
#include "status.h"
|
|
|
|
cInterface *Interface = NULL;
|
|
|
|
cInterface::cInterface(int SVDRPport)
|
|
{
|
|
interrupted = false;
|
|
SVDRP = NULL;
|
|
if (SVDRPport)
|
|
SVDRP = new cSVDRP(SVDRPport);
|
|
}
|
|
|
|
cInterface::~cInterface()
|
|
{
|
|
delete SVDRP;
|
|
}
|
|
|
|
eKeys cInterface::GetKey(bool Wait)
|
|
{
|
|
if (!cRemote::HasKeys())
|
|
Skins.Flush();
|
|
if (SVDRP) {
|
|
if (SVDRP->Process())
|
|
Wait = false;
|
|
}
|
|
return cRemote::Get(Wait ? 1000 : 10);
|
|
}
|
|
|
|
eKeys cInterface::Wait(int Seconds, bool KeepChar)
|
|
{
|
|
if (Seconds == 0)
|
|
Seconds = Setup.OSDMessageTime;
|
|
Skins.Flush();
|
|
eKeys Key = kNone;
|
|
time_t timeout = time(NULL) + Seconds;
|
|
for (;;) {
|
|
Key = GetKey();
|
|
if (ISRAWKEY(Key) || time(NULL) > timeout || interrupted)
|
|
break;
|
|
}
|
|
if (KeepChar && ISRAWKEY(Key))
|
|
cRemote::Put(Key);
|
|
interrupted = false;
|
|
return Key;
|
|
}
|
|
|
|
bool cInterface::Confirm(const char *s, int Seconds, bool WaitForTimeout)
|
|
{
|
|
isyslog("confirm: %s", s);
|
|
eKeys k = Skins.Message(mtWarning, s, Seconds);
|
|
bool result = WaitForTimeout ? k == kNone : k == kOk;
|
|
isyslog("%sconfirmed", result ? "" : "not ");
|
|
return result;
|
|
}
|
|
|
|
bool cInterface::QueryKeys(cRemote *Remote, cSkinDisplayMenu *DisplayMenu)
|
|
{
|
|
DisplayMenu->SetItem(tr("Phase 1: Detecting RC code type"), 2, false, false);
|
|
DisplayMenu->SetItem(tr("Press any key on the RC unit"), 4, false, false);
|
|
DisplayMenu->Flush();
|
|
if (Remote->Initialize()) {
|
|
DisplayMenu->SetItem(tr("RC code detected!"), 4, false, false);
|
|
DisplayMenu->SetItem(tr("Do not press any key..."), 5, false, false);
|
|
DisplayMenu->Flush();
|
|
sleep(3);
|
|
DisplayMenu->SetItem("", 4, false, false);
|
|
DisplayMenu->SetItem("", 5, false, false);
|
|
|
|
DisplayMenu->SetItem(tr("Phase 2: Learning specific key codes"), 2, false, false);
|
|
eKeys NewKey = kUp;
|
|
while (NewKey != kNone) {
|
|
char *Prompt;
|
|
char buf[32];
|
|
snprintf(buf, sizeof(buf), "Key$%s", cKey::ToString(NewKey));
|
|
asprintf(&Prompt, tr("Press key for '%s'"), tr(buf));
|
|
DisplayMenu->SetItem(Prompt, 4, false, false);
|
|
free(Prompt);
|
|
cRemote::Clear();
|
|
DisplayMenu->Flush();
|
|
for (eKeys k = NewKey; k == NewKey; ) {
|
|
char *NewCode = NULL;
|
|
eKeys Key = cRemote::Get(100, &NewCode);
|
|
switch (Key) {
|
|
case kUp: if (NewKey > kUp) {
|
|
NewKey = eKeys(NewKey - 1);
|
|
cKey *last = Keys.Last();
|
|
if (last && last->Key() == NewKey)
|
|
Keys.Del(last);
|
|
}
|
|
break;
|
|
case kDown: DisplayMenu->SetItem(tr("Press 'Up' to confirm"), 4, false, false);
|
|
DisplayMenu->SetItem(tr("Press 'Down' to continue"), 5, false, false);
|
|
DisplayMenu->SetItem("", 6, false, false);
|
|
DisplayMenu->SetItem("", 7, false, false);
|
|
DisplayMenu->SetItem("", 8, false, false);
|
|
DisplayMenu->Flush();
|
|
for (;;) {
|
|
Key = cRemote::Get(100);
|
|
if (Key == kUp) {
|
|
DisplayMenu->Clear();
|
|
return true;
|
|
}
|
|
else if (Key == kDown) {
|
|
DisplayMenu->SetItem("", 5, false, false);
|
|
k = kNone; // breaks the outer for() loop
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case kMenu: NewKey = eKeys(NewKey + 1);
|
|
break;
|
|
case kNone: if (NewCode) {
|
|
dsyslog("new %s code: %s = %s", Remote->Name(), NewCode, cKey::ToString(NewKey));
|
|
Keys.Add(new cKey(Remote->Name(), NewCode, NewKey));
|
|
NewKey = eKeys(NewKey + 1);
|
|
free(NewCode);
|
|
}
|
|
break;
|
|
default: break;
|
|
}
|
|
}
|
|
if (NewKey > kUp)
|
|
DisplayMenu->SetItem(tr("(press 'Up' to go back)"), 6, false, false);
|
|
else
|
|
DisplayMenu->SetItem("", 6, false, false);
|
|
if (NewKey > kDown)
|
|
DisplayMenu->SetItem(tr("(press 'Down' to end key definition)"), 7, false, false);
|
|
else
|
|
DisplayMenu->SetItem("", 7, false, false);
|
|
if (NewKey > kMenu)
|
|
DisplayMenu->SetItem(tr("(press 'Menu' to skip this key)"), 8, false, false);
|
|
else
|
|
DisplayMenu->SetItem("", 8, false, false);
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void cInterface::LearnKeys(void)
|
|
{
|
|
for (cRemote *Remote = Remotes.First(); Remote; Remote = Remotes.Next(Remote)) {
|
|
if (!Remote->Ready()) {
|
|
esyslog("ERROR: remote control %s not ready!", Remote->Name());
|
|
continue;
|
|
}
|
|
bool known = Keys.KnowsRemote(Remote->Name());
|
|
dsyslog("remote control %s - %s", Remote->Name(), known ? "keys known" : "learning keys");
|
|
if (!known) {
|
|
cSkinDisplayMenu *DisplayMenu = Skins.Current()->DisplayMenu();
|
|
char Headline[256];
|
|
snprintf(Headline, sizeof(Headline), tr("Learning Remote Control Keys"));
|
|
cRemote::Clear();
|
|
DisplayMenu->SetTitle(Headline);
|
|
DisplayMenu->SetItem(Remote->Name(), 0, false, false);
|
|
cRemote::SetLearning(Remote);
|
|
bool rc = QueryKeys(Remote, DisplayMenu);
|
|
cRemote::SetLearning(NULL);
|
|
DisplayMenu->Clear();
|
|
if (!rc) {
|
|
delete DisplayMenu;
|
|
continue;
|
|
}
|
|
DisplayMenu->SetItem(Remote->Name(), 0, false, false);
|
|
DisplayMenu->SetItem(tr("Phase 3: Saving key codes"), 2, false, false);
|
|
DisplayMenu->SetItem(tr("Press 'Up' to save, 'Down' to cancel"), 4, false, false);
|
|
for (;;) {
|
|
eKeys key = GetKey();
|
|
if (key == kUp) {
|
|
Keys.Save();
|
|
delete DisplayMenu;
|
|
break;
|
|
}
|
|
else if (key == kDown) {
|
|
Keys.Load();
|
|
delete DisplayMenu;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|