mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Fixed setting the audio language codes in 'Transfer-Mode' (reported by Rolf Ahrenberg). The actual problem was the call to the Transferring() function in cDevice::AttachPlayer() before assigning the player. - Fixed removing the '-' when entering a channel number where there is no other one that fits the input (thanks to Joachim Wilke). - Fixed the 'libsi' function CharArray::checkSize(), which made a previous workaround in libsi/descriptor.c obsolete (thanks to Marcel Wiesweg). - The "Ok" key in the "Jump" mode of the replay progress display now confirms the jump instead of closing the display (thanks to Christoph Haubrich). - The 'summary' field of a timer definition has been renamed to 'aux', and is now only used for external applications to store auxiliary information with a timer, which has no meaning whatsoever to VDR itself. The contents of the 'aux' field of a timer is copied into the recording's 'info.vdr' file, using the tag character '@'. - The description of a recording is now taken exclusively from its related EPG data. If an application wants to use a different description it needs to set it with SVDRP/PUTE and use table ID 0x00, so that it won't be overwritten (as a side effect, however, this also disables VPS for such an event). - There is no more "Summary" menu when pressing "Ok" in the "Timers" menu. The "Ok" key now always opens the "Edit timer" menu. - The upper 16 bit of a timer's "flags" are no longer treated specially when a timer is modified in the "Edit timer" menu. If an external application needs to know if a timer was modified, it has to keep a copy of the timer's data and compare that to the actual data. - The new function cRecordingInfo::ChannelID() can be used to retrieve the ID of the channel a recording was made from. - The 'info.vdr' file of a recording now also contains the 'E' and 'V' records of the EPG event used when creating it. - The option "Setup/OSD/Sort timers" has been removed. Timers are always sorted by their start time and priority. - The "Blue" key in the "Timers" menu now displays the EPG info of the event the selected timer will record (if available). The "On/Off" function has been shifted to the "Red" button. Editing a timer is done by pressing "Ok". - When determining which event a timer is going to record, all available events in the future are now taken into account (no more limit to 4 hours in the future). This has been done so that the event info is available in the "Timers" menu when pressing the "Blue" button. In order to avoid unnecessary work, each timer now has its own timestamp to control whether its schedule has changed since the last time its event has been set. - Fixed setting events to timers in case a non-VPS event has expired. - There is now a log message "timer ... set to event ..." when defining a timer from the EPG menu. - Lines tagged with '#' in the 'info.vdr' file of a recording are now silently ignored when reading that file (suggested by Peter Bieringer). Such lines can be used by external tools to store arbitrary information. - The 'event id' in EPG data has been extended to 32 bit, so that external tools can generate ids that don't collide with those from the DVB data stream (suggested by Matthias Schniedermeyer). - The DrawBitmap() function now has a new parameter 'Overlay' that allows a bitmap to be drawn with a transparent background (thanks to Alexander Hans). - Fixed cSchedule::GetFollowingEvent() in case there is currently no present event running (thanks to Pekka Mauno).
191 lines
7.1 KiB
C++
191 lines
7.1 KiB
C++
/*
|
|
* epg.h: Electronic Program Guide
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* Original version (as used in VDR before 1.3.0) written by
|
|
* Robert Schneider <Robert.Schneider@web.de> and Rolf Hakenes <hakenes@hippomi.de>.
|
|
*
|
|
* $Id: epg.h 1.33 2006/02/26 13:58:57 kls Exp $
|
|
*/
|
|
|
|
#ifndef __EPG_H
|
|
#define __EPG_H
|
|
|
|
#include "channels.h"
|
|
#include "thread.h"
|
|
#include "tools.h"
|
|
|
|
#define MAXEPGBUGFIXLEVEL 3
|
|
|
|
enum eDumpMode { dmAll, dmPresent, dmFollowing, dmAtTime };
|
|
|
|
struct tComponent {
|
|
uchar stream;
|
|
uchar type;
|
|
char language[MAXLANGCODE2];
|
|
char *description;
|
|
cString ToString(void);
|
|
bool FromString(const char *s);
|
|
};
|
|
|
|
class cComponents {
|
|
private:
|
|
int numComponents;
|
|
tComponent *components;
|
|
void Realloc(int Index);
|
|
public:
|
|
cComponents(void);
|
|
~cComponents(void);
|
|
int NumComponents(void) const { return numComponents; }
|
|
void SetComponent(int Index, const char *s);
|
|
void SetComponent(int Index, uchar Stream, uchar Type, const char *Language, const char *Description);
|
|
tComponent *Component(int Index) const { return (Index < numComponents) ? &components[Index] : NULL; }
|
|
tComponent *GetComponent(int Index, uchar Stream, uchar Type); // Gets the Index'th component of Stream and Type, skipping other components
|
|
};
|
|
|
|
class cSchedule;
|
|
|
|
typedef u_int32_t tEventID;
|
|
|
|
class cEvent : public cListObject {
|
|
friend class cSchedule;
|
|
private:
|
|
cSchedule *schedule; // The Schedule this event belongs to
|
|
tEventID eventID; // Event ID of this event
|
|
uchar tableID; // Table ID this event came from
|
|
uchar version; // Version number of section this event came from
|
|
int runningStatus; // 0=undefined, 1=not running, 2=starts in a few seconds, 3=pausing, 4=running
|
|
char *title; // Title of this event
|
|
char *shortText; // Short description of this event (typically the episode name in case of a series)
|
|
char *description; // Description of this event
|
|
cComponents *components; // The stream components of this event
|
|
time_t startTime; // Start time of this event
|
|
int duration; // Duration of this event in seconds
|
|
time_t vps; // Video Programming Service timestamp (VPS, aka "Programme Identification Label", PIL)
|
|
time_t seen; // When this event was last seen in the data stream
|
|
public:
|
|
cEvent(tEventID EventID);
|
|
~cEvent();
|
|
virtual int Compare(const cListObject &ListObject) const;
|
|
tChannelID ChannelID(void) const;
|
|
const cSchedule *Schedule(void) const { return schedule; }
|
|
tEventID EventID(void) const { return eventID; }
|
|
uchar TableID(void) const { return tableID; }
|
|
uchar Version(void) const { return version; }
|
|
int RunningStatus(void) const { return runningStatus; }
|
|
const char *Title(void) const { return title; }
|
|
const char *ShortText(void) const { return shortText; }
|
|
const char *Description(void) const { return description; }
|
|
const cComponents *Components(void) const { return components; }
|
|
time_t StartTime(void) const { return startTime; }
|
|
time_t EndTime(void) const { return startTime + duration; }
|
|
int Duration(void) const { return duration; }
|
|
time_t Vps(void) const { return vps; }
|
|
time_t Seen(void) const { return seen; }
|
|
bool SeenWithin(int Seconds) const { return time(NULL) - seen < Seconds; }
|
|
bool HasTimer(void) const;
|
|
bool IsRunning(bool OrAboutToStart = false) const;
|
|
cString GetDateString(void) const;
|
|
cString GetTimeString(void) const;
|
|
cString GetEndTimeString(void) const;
|
|
cString GetVpsString(void) const;
|
|
void SetEventID(tEventID EventID);
|
|
void SetTableID(uchar TableID);
|
|
void SetVersion(uchar Version);
|
|
void SetRunningStatus(int RunningStatus, cChannel *Channel = NULL);
|
|
void SetTitle(const char *Title);
|
|
void SetShortText(const char *ShortText);
|
|
void SetDescription(const char *Description);
|
|
void SetComponents(cComponents *Components); // Will take ownership of Components!
|
|
void SetStartTime(time_t StartTime);
|
|
void SetDuration(int Duration);
|
|
void SetVps(time_t Vps);
|
|
void SetSeen(void);
|
|
void Dump(FILE *f, const char *Prefix = "", bool InfoOnly = false) const;
|
|
bool Parse(char *s);
|
|
static bool Read(FILE *f, cSchedule *Schedule);
|
|
void FixEpgBugs(void);
|
|
};
|
|
|
|
class cSchedules;
|
|
|
|
class cSchedule : public cListObject {
|
|
private:
|
|
tChannelID channelID;
|
|
cList<cEvent> events;
|
|
cHash<cEvent> eventsHashID;
|
|
cHash<cEvent> eventsHashStartTime;
|
|
bool hasRunning;
|
|
time_t modified;
|
|
time_t presentSeen;
|
|
public:
|
|
cSchedule(tChannelID ChannelID);
|
|
tChannelID ChannelID(void) const { return channelID; }
|
|
time_t Modified(void) const { return modified; }
|
|
time_t PresentSeen(void) const { return presentSeen; }
|
|
bool PresentSeenWithin(int Seconds) const { return time(NULL) - presentSeen < Seconds; }
|
|
void SetModified(void) { modified = time(NULL); }
|
|
void SetPresentSeen(void) { presentSeen = time(NULL); }
|
|
void SetRunningStatus(cEvent *Event, int RunningStatus, cChannel *Channel = NULL);
|
|
void ClrRunningStatus(cChannel *Channel = NULL);
|
|
void ResetVersions(void);
|
|
void Sort(void);
|
|
void DropOutdated(time_t SegmentStart, time_t SegmentEnd, uchar TableID, uchar Version);
|
|
void Cleanup(time_t Time);
|
|
void Cleanup(void);
|
|
cEvent *AddEvent(cEvent *Event);
|
|
void DelEvent(cEvent *Event);
|
|
void HashEvent(cEvent *Event);
|
|
void UnhashEvent(cEvent *Event);
|
|
const cList<cEvent> *Events(void) const { return &events; }
|
|
const cEvent *GetPresentEvent(void) const;
|
|
const cEvent *GetFollowingEvent(void) const;
|
|
const cEvent *GetEvent(tEventID EventID, time_t StartTime = 0) const;
|
|
const cEvent *GetEventAround(time_t Time) const;
|
|
void Dump(FILE *f, const char *Prefix = "", eDumpMode DumpMode = dmAll, time_t AtTime = 0) const;
|
|
static bool Read(FILE *f, cSchedules *Schedules);
|
|
};
|
|
|
|
class cSchedulesLock {
|
|
private:
|
|
bool locked;
|
|
public:
|
|
cSchedulesLock(bool WriteLock = false, int TimeoutMs = 0);
|
|
~cSchedulesLock();
|
|
bool Locked(void) { return locked; }
|
|
};
|
|
|
|
class cSchedules : public cList<cSchedule> {
|
|
friend class cSchedule;
|
|
friend class cSchedulesLock;
|
|
private:
|
|
cRwLock rwlock;
|
|
static cSchedules schedules;
|
|
static const char *epgDataFileName;
|
|
static time_t lastCleanup;
|
|
static time_t lastDump;
|
|
static time_t modified;
|
|
public:
|
|
static void SetEpgDataFileName(const char *FileName);
|
|
static const cSchedules *Schedules(cSchedulesLock &SchedulesLock);
|
|
///< Caller must provide a cSchedulesLock which has to survive the entire
|
|
///< time the returned cSchedules is accessed. Once the cSchedules is no
|
|
///< longer used, the cSchedulesLock must be destroyed.
|
|
static time_t Modified(void) { return modified; }
|
|
static void SetModified(cSchedule *Schedule);
|
|
static void Cleanup(bool Force = false);
|
|
static void ResetVersions(void);
|
|
static bool ClearAll(void);
|
|
static bool Dump(FILE *f, const char *Prefix = "", eDumpMode DumpMode = dmAll, time_t AtTime = 0);
|
|
static bool Read(FILE *f = NULL);
|
|
cSchedule *AddSchedule(tChannelID ChannelID);
|
|
const cSchedule *GetSchedule(tChannelID ChannelID) const;
|
|
const cSchedule *GetSchedule(const cChannel *Channel, bool AddIfMissing = false) const;
|
|
};
|
|
|
|
void ReportEpgBugFixStats(bool Reset = false);
|
|
|
|
#endif //__EPG_H
|