mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Fixed two errors in 'newplugin' (thanks to Alexander Rieger). - Fixed converting arbitrarily formatted summary.vdr files (thanks to Thomas Günther). - Fixed handling color buttons in cMenuEditStrItem (thanks to Alexander Rieger). - Added cChannel::LinkChannels() and cChannel::RefChannel() (suggested by Helmut Auer). Note that VDR itself doesn't actually use the linked channels, yet, so there is no guarantee that this really works under all circumstances. - Added a missing include statement to the 'sky' plugin (thanks to Alfred Zastrow for reporting this one). - Fixed handling key macros with keys after @plugin (thanks to Rolf Ahrenberg for reporting this one). - Fixed error handling in cCiTransportConnection::RecvTPDU() (thanks to Georg Acher for reporting this one). - Removed obsolete 'shift' code in device.[hc]. - The SVDRP command DELR no longer triggers a complete reload of the global Recordings list, but rather deletes that particular entry. - The list of recordings is now read in a separate thread, resulting in a faster startup if there are a great many of recordings, or the disk(s) have to spin up. If the Recordings menu is opened while the list of recordings is still being read, the menu will be updated accordingly. Plugins that access the global Recordings variable should lock the thread by putting something like cThreadLock RecordingsLock(&Recordings); into the respective code block. Thanks to Carsten Koch for his help in testing and debugging this. - The 'new' indicator in the Recordings menu is now kept up-to-date (thanks to Thomas Günther). - Updated the Romanian OSD texts (thanks to Lucian Muresan). - Updated the Russian OSD texts (thanks to Oleg Roitburd). - The '.update' file in the video directory is now touched when a recording is added or deleted, so that other VDR instances can update their lists (thanks to Alexander Rieger). - Made the function ExchangeChars() public (suggested by Lucian Muresan).
258 lines
7.3 KiB
C
258 lines
7.3 KiB
C
/*
|
|
* keys.c: Remote control Key handling
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: keys.c 1.9 2005/09/17 11:27:40 kls Exp $
|
|
*/
|
|
|
|
#include "keys.h"
|
|
#include "plugin.h"
|
|
|
|
static tKey keyTable[] = { // "Up" and "Down" must be the first two keys!
|
|
{ kUp, "Up" },
|
|
{ kDown, "Down" },
|
|
{ kMenu, "Menu" },
|
|
{ kOk, "Ok" },
|
|
{ kBack, "Back" },
|
|
{ kLeft, "Left" },
|
|
{ kRight, "Right" },
|
|
{ kRed, "Red" },
|
|
{ kGreen, "Green" },
|
|
{ kYellow, "Yellow" },
|
|
{ kBlue, "Blue" },
|
|
{ k0, "0" },
|
|
{ k1, "1" },
|
|
{ k2, "2" },
|
|
{ k3, "3" },
|
|
{ k4, "4" },
|
|
{ k5, "5" },
|
|
{ k6, "6" },
|
|
{ k7, "7" },
|
|
{ k8, "8" },
|
|
{ k9, "9" },
|
|
{ kPlay, "Play" },
|
|
{ kPause, "Pause" },
|
|
{ kStop, "Stop" },
|
|
{ kRecord, "Record" },
|
|
{ kFastFwd, "FastFwd" },
|
|
{ kFastRew, "FastRew" },
|
|
{ kPower, "Power" },
|
|
{ kChanUp, "Channel+" },
|
|
{ kChanDn, "Channel-" },
|
|
{ kVolUp, "Volume+" },
|
|
{ kVolDn, "Volume-" },
|
|
{ kMute, "Mute" },
|
|
{ kAudio, "Audio" },
|
|
{ kSchedule, "Schedule" },
|
|
{ kChannels, "Channels" },
|
|
{ kTimers, "Timers" },
|
|
{ kRecordings, "Recordings" },
|
|
{ kSetup, "Setup" },
|
|
{ kCommands, "Commands" },
|
|
{ kUser1, "User1" },
|
|
{ kUser2, "User2" },
|
|
{ kUser3, "User3" },
|
|
{ kUser4, "User4" },
|
|
{ kUser5, "User5" },
|
|
{ kUser6, "User6" },
|
|
{ kUser7, "User7" },
|
|
{ kUser8, "User8" },
|
|
{ kUser9, "User9" },
|
|
{ kNone, "" },
|
|
{ k_Setup, "_Setup" },
|
|
{ kNone, NULL },
|
|
};
|
|
|
|
// -- cKey -------------------------------------------------------------------
|
|
|
|
cKey::cKey(void)
|
|
{
|
|
remote = code = NULL;
|
|
key = kNone;
|
|
}
|
|
|
|
cKey::cKey(const char *Remote, const char *Code, eKeys Key)
|
|
{
|
|
remote = strdup(Remote);
|
|
code = strdup(Code);
|
|
key = Key;
|
|
}
|
|
|
|
cKey::~cKey()
|
|
{
|
|
free(remote);
|
|
free(code);
|
|
}
|
|
|
|
bool cKey::Parse(char *s)
|
|
{
|
|
char *p = strchr(s, '.');
|
|
if (p) {
|
|
*p++ = 0;
|
|
remote = strdup(s);
|
|
char *q = strpbrk(p, " \t");
|
|
if (q) {
|
|
*q++ = 0;
|
|
key = FromString(p);
|
|
if (key != kNone) {
|
|
q = skipspace(q);
|
|
if (*q) {
|
|
code = strdup(q);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool cKey::Save(FILE *f)
|
|
{
|
|
return fprintf(f, "%s.%-10s %s\n", remote, ToString(key), code) > 0;
|
|
}
|
|
|
|
eKeys cKey::FromString(const char *Name)
|
|
{
|
|
if (Name) {
|
|
for (tKey *k = keyTable; k->name; k++) {
|
|
if (strcasecmp(k->name, Name) == 0)
|
|
return k->type;
|
|
}
|
|
}
|
|
return kNone;
|
|
}
|
|
|
|
const char *cKey::ToString(eKeys Key)
|
|
{
|
|
for (tKey *k = keyTable; k->name; k++) {
|
|
if (k->type == Key)
|
|
return k->name;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// -- cKeys ------------------------------------------------------------------
|
|
|
|
cKeys Keys;
|
|
|
|
bool cKeys::KnowsRemote(const char *Remote)
|
|
{
|
|
if (Remote) {
|
|
for (cKey *k = First(); k; k = Next(k)) {
|
|
if (strcmp(Remote, k->Remote()) == 0)
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
eKeys cKeys::Get(const char *Remote, const char *Code)
|
|
{
|
|
if (Remote && Code) {
|
|
for (cKey *k = First(); k; k = Next(k)) {
|
|
if (strcmp(Remote, k->Remote()) == 0 && strcmp(Code, k->Code()) == 0)
|
|
return k->Key();
|
|
}
|
|
}
|
|
return kNone;
|
|
}
|
|
|
|
const char *cKeys::GetSetup(const char *Remote)
|
|
{
|
|
if (Remote) {
|
|
for (cKey *k = First(); k; k = Next(k)) {
|
|
if (strcmp(Remote, k->Remote()) == 0 && k->Key() == k_Setup)
|
|
return k->Code();
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void cKeys::PutSetup(const char *Remote, const char *Setup)
|
|
{
|
|
if (!GetSetup(Remote))
|
|
Add(new cKey(Remote, Setup, k_Setup));
|
|
else
|
|
esyslog("ERROR: called PutSetup() for %s, but setup has already been defined!", Remote);
|
|
}
|
|
|
|
// -- cKeyMacro --------------------------------------------------------------
|
|
|
|
cKeyMacro::cKeyMacro(void)
|
|
{
|
|
for (int i = 0; i < MAXKEYSINMACRO; i++)
|
|
macro[i] = kNone;
|
|
plugin = NULL;
|
|
}
|
|
|
|
cKeyMacro::~cKeyMacro()
|
|
{
|
|
free(plugin);
|
|
}
|
|
|
|
bool cKeyMacro::Parse(char *s)
|
|
{
|
|
int n = 0;
|
|
char *p;
|
|
char *strtok_next;
|
|
while ((p = strtok_r(s, " \t", &strtok_next)) != NULL) {
|
|
if (n < MAXKEYSINMACRO) {
|
|
if (*p == '@') {
|
|
if (plugin) {
|
|
esyslog("ERROR: only one @plugin allowed per macro");
|
|
return false;
|
|
}
|
|
if (!n) {
|
|
esyslog("ERROR: @plugin can't be first in macro");
|
|
return false;
|
|
}
|
|
macro[n] = k_Plugin;
|
|
if (n < MAXKEYSINMACRO) {
|
|
plugin = strdup(p + 1);
|
|
if (!cPluginManager::GetPlugin(plugin)) {
|
|
esyslog("ERROR: unknown plugin '%s'", plugin);
|
|
// this is not a fatal error - plugins may or may not be loaded
|
|
macro[--n] = kNone; // makes sure the key doesn't cause any side effects
|
|
}
|
|
}
|
|
else {
|
|
esyslog("ERROR: key macro too long");
|
|
return false;
|
|
}
|
|
}
|
|
else {
|
|
macro[n] = cKey::FromString(p);
|
|
if (macro[n] == kNone) {
|
|
esyslog("ERROR: unknown key '%s'", p);
|
|
return false;
|
|
}
|
|
}
|
|
n++;
|
|
s = NULL;
|
|
}
|
|
else {
|
|
esyslog("ERROR: key macro too long");
|
|
return false;
|
|
}
|
|
}
|
|
if (n < 2) {
|
|
esyslog("ERROR: empty key macro");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// -- cKeyMacros -------------------------------------------------------------
|
|
|
|
cKeyMacros KeyMacros;
|
|
|
|
const cKeyMacro *cKeyMacros::Get(eKeys Key)
|
|
{
|
|
for (cKeyMacro *k = First(); k; k = Next(k)) {
|
|
if (*k->Macro() == Key)
|
|
return k;
|
|
}
|
|
return NULL;
|
|
}
|