*/
Colour Colour::BLACK(0, 0, 0);
Colour Colour::RED(255, 0, 0);
+Colour Colour::GREEN(0, 255, 0);
Colour Colour::VIDEOBLUE(0, 0, 150);
Colour Colour::VIEWBACKGROUND(0, 0, 100);
Colour Colour::TITLEBARBACKGROUND(0, 0, 200);
// objects deriving from messagequeues, make them derive from
// messagereceiver - then one messagequeue can deliver any message to anywhere
+ // Try to segfault
+ logger->log("Command", Log::DEBUG, "1: %p", m);
+ logger->log("Command", Log::DEBUG, "2: %p", m->to);
+ logger->log("Command", Log::DEBUG, "3: %lu", m->parameter);
// deliver timer
const static ULONG CHILD_CLOSE = 17;
const static ULONG REDRAW_LANG = 18;
const static ULONG TIMER = 19;
+ const static ULONG EPG = 20;
};
#endif
logger->log("Timers", Log::DEBUG, "Timers init start");\r
\r
threadLock(); // lock here, the thread loop will unlock and wait\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 1");\r
+ //logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 1");\r
if (!threadStart())\r
{\r
shutdown();\r
return 1;\r
}\r
\r
-int Timers::setTimer(TimerReceiver* client, int clientReference, time_t requestedTime)\r
+int Timers::setTimer(TimerReceiver* client, int clientReference, long int requestedTime, long int requestedTimeNSEC)\r
{\r
if (!initted) return 0;\r
\r
logger->log("Timers", Log::DEBUG, "Starting set timer 1");\r
\r
+ //logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 2");\r
+ threadLock();\r
+\r
+ // Check that this timer is not already in the list\r
+ TimerList::iterator i;\r
+ Timer* currentTimer = NULL;\r
+ for(i = timerList.begin(); i != timerList.end(); i++)\r
+ {\r
+ currentTimer = *i;\r
+ if ((currentTimer->client == client) && (currentTimer->clientReference == clientReference))\r
+ {\r
+ // Overwrite an existing timer\r
+ currentTimer->requestedTime.tv_sec = requestedTime;\r
+ currentTimer->requestedTime.tv_nsec = requestedTimeNSEC;\r
+ resetThreadFlag = true;\r
+ threadSignalNoLock();\r
+\r
+ //logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 2 (b)");\r
+ threadUnlock();\r
+ return 0;\r
+ }\r
+ }\r
+\r
Timer* t = new Timer();\r
t->client = client;\r
t->clientReference = clientReference;\r
t->requestedTime.tv_sec = requestedTime;\r
- t->requestedTime.tv_nsec = 0;\r
+ t->requestedTime.tv_nsec = requestedTimeNSEC;\r
\r
-logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 2");\r
- threadLock();\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 2");\r
+ //logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 2");\r
timerList.push_back(t);\r
resetThreadFlag = true;\r
threadSignalNoLock();\r
-logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 2");\r
+ //logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 2");\r
threadUnlock();\r
\r
- logger->log("Timers", Log::DEBUG, "1 Have set timer for %p ref %i", client, clientReference);\r
+ logger->log("Timers", Log::DEBUG, "Timer set for %p ref %i", client, clientReference);\r
\r
return 1;\r
}\r
\r
int Timers::setTimer(TimerReceiver* client, int clientReference, struct timespec duration)\r
{\r
- if (!initted) return 0;\r
-\r
- logger->log("Timers", Log::DEBUG, "Starting set timer 2");\r
-\r
- Timer* t = new Timer();\r
- t->client = client;\r
- t->clientReference = clientReference;\r
-\r
struct timespec currentTime;\r
clock_gettime(CLOCK_REALTIME, ¤tTime);\r
\r
- t->requestedTime.tv_sec = currentTime.tv_sec + duration.tv_sec;\r
- t->requestedTime.tv_nsec = currentTime.tv_nsec + duration.tv_nsec;\r
- if (t->requestedTime.tv_nsec > 999999999)\r
+ long int requestedTime;\r
+ long int requestedTimeNSEC;\r
+\r
+ requestedTime = currentTime.tv_sec + duration.tv_sec;\r
+ requestedTimeNSEC = currentTime.tv_nsec + duration.tv_nsec;\r
+ if (requestedTimeNSEC > 999999999)\r
{\r
- ++t->requestedTime.tv_sec;\r
- t->requestedTime.tv_nsec -= 1000000000;\r
+ ++requestedTime;\r
+ requestedTimeNSEC -= 1000000000;\r
logger->log("Timers", Log::DEBUG, "Second rollover - CHECK FIXME");\r
}\r
\r
-logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 3");\r
- threadLock();\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 3");\r
- timerList.push_back(t);\r
- resetThreadFlag = true;\r
- threadSignalNoLock();\r
-logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 3");\r
- threadUnlock();\r
-\r
- logger->log("Timers", Log::DEBUG, "2 Have set timer for %p ref %i", client, clientReference);\r
-\r
- return 1;\r
+ return setTimer(client, clientReference, requestedTime, requestedTimeNSEC);\r
}\r
\r
int Timers::cancelTimer(TimerReceiver* client, int clientReference)\r
\r
logger->log("Timers", Log::DEBUG, "Starting cancel timer %p %i, list size = %i", client, clientReference, timerList.size());\r
\r
-logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 4");\r
+ //logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 4");\r
threadLock();\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 4");\r
+ //logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 4");\r
TimerList::iterator i;\r
Timer* currentTimer = NULL;\r
for(i = timerList.begin(); i != timerList.end(); i++)\r
{\r
currentTimer = *i;\r
- logger->log("Timers", Log::DEBUG, "I: %p %i : %p %i", client, clientReference, currentTimer->client, currentTimer->clientReference);\r
+ //logger->log("Timers", Log::DEBUG, "I: %p %i : %p %i", client, clientReference, currentTimer->client, currentTimer->clientReference);\r
if ((currentTimer->client == client) && (currentTimer->clientReference == clientReference))\r
{\r
timerList.erase(i);\r
{\r
// no timer found\r
logger->log("Timers", Log::DEBUG, "No timer found in cancelTimer %p ref %i", client, clientReference);\r
-logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 4");\r
+ //logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 4");\r
threadUnlock();\r
return 0;\r
}\r
\r
resetThreadFlag = true;\r
threadSignalNoLock();\r
-logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 4");\r
+ //logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX 4");\r
threadUnlock();\r
\r
\r
\r
if (nextTimer)\r
{\r
- logger->log("Timers", Log::DEBUG, "List size: %i. nextTimer: %p. nextTime.tv_sec: %li. nextTime.tv_nsec: %li", timerList.size(), nextTimer, nextTime.tv_sec, nextTime.tv_nsec);\r
+ logger->log("Timers", Log::DEBUG, "List size: %i. nextTimerClient: %p/%i. nextTime.tv_sec: %li. nextTime.tv_nsec: %li", timerList.size(), nextTimer->client, nextTimer->clientReference, nextTime.tv_sec, nextTime.tv_nsec);\r
\r
\r
-logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX (1)");\r
+ //logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX (1)");\r
threadWaitForSignalTimed(&nextTime); // FIXME does this work if the time is in the past?\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 5");\r
+ //logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 5");\r
\r
// unlocks in the wait\r
}\r
else\r
{\r
-logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX (2)");\r
+ //logger->log("Timers", Log::DEBUG, "about to un-LOCK -TIMERS- MUTEX (2)");\r
threadWaitForSignal();\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 6");\r
+ //logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 6");\r
// unlocks in the wait\r
}\r
\r
// now unlock the timers mutex for a fraction of a second\r
// in case the gui thread is waiting on the timers mutex\r
threadUnlock();\r
-logger->log("Timers", Log::DEBUG, "un-LOCKED -TIMERS- MUTEX (3)");\r
- printf("\n\n\n WOOOOO \n\n\n The anti deadlock code is working!!! \n\n\n");\r
- usleep(10000); // 5ms - too long?\r
-logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 7");\r
+ //logger->log("Timers", Log::DEBUG, "un-LOCKED -TIMERS- MUTEX (3)");\r
+ //printf("\n\n\n WOOOOO \n\n\n The anti deadlock code is working!!! \n\n\n");\r
+ usleep(10000); // 10ms - too long?\r
+ //logger->log("Timers", Log::DEBUG, "Waiting for LOCK -TIMERS- MUTEX 7");\r
threadLock();\r
-logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 7");\r
+ //logger->log("Timers", Log::DEBUG, "LOCKED -TIMERS- MUTEX 7");\r
resetThreadFlag = true;\r
}\r
else\r
#include "command.h"\r
#include "timerreceiver.h"\r
\r
+// FIXME - ensure all objects that call settimer call cancel timer if they are being deleted\r
+\r
+/*\r
+\r
+Timers documentation\r
+\r
+Call setTimer to set a timer.... cancelTimer to delete a running timer.\r
+Derive your object from TimerReceiver (timerreceiver.h), implement timercall() in your class\r
+and supply your 'this' pointer to setTimer.\r
+\r
+Once a timer has fired it does not exist anymore, you have to keep creating them if you want\r
+a constant pulse.\r
+\r
+clientReference is any int of your choice. It will be supplied back to you in the timercall()\r
+so you can identify which timer has fired if you have more than one.\r
+\r
+You can reset a timer by calling setTimer again. This will not create 2 timers, it will overwrite the first one.\r
+\r
+You must not allow a timer to fire on an object that has been deleted already, unless you want\r
+segfaulty hell.\r
+\r
+*/\r
+\r
class Timer\r
{\r
public:\r
int init();\r
int shutdown();\r
\r
- // FIXME - ensure all objects that call settimer call cancel timer if they are being deleted\r
-\r
- int setTimer(TimerReceiver* client, int clientReference, time_t requestedTime);\r
+ int setTimer(TimerReceiver* client, int clientReference, long int requestedTime, long int requestedTimeNSEC=0);\r
int setTimer(TimerReceiver* client, int clientReference, timespec duration);\r
int cancelTimer(TimerReceiver* client, int clientReference);\r
\r
if (Video::getInstance()->getFormat() == Video::PAL)\r
{\r
create(632, 520);\r
- setScreenPos(44, 30);\r
+ setScreenPos(60, 30);\r
}\r
else\r
{\r
}
}
-// FIXME - make this take a optional region for smaller updates (not whole views)
void ViewMan::updateView(View* toUpdate, Region* regionToUpdate)
{
// Get the z index of the view
#include "vlivebanner.h"
+VLiveBanner* VLiveBanner::instance = NULL;
+
VLiveBanner::VLiveBanner(View* tparent, Channel* channel, bool bannerTakesCommands)
{
+ instance = this;
eventList = NULL;
parent = tparent;
takeCommands = bannerTakesCommands;
+ clockRegion.x = 440;
+ clockRegion.y = 0;
+ clockRegion.w = 60;
+ clockRegion.h = 30;
+
create(500, 120);
if (Video::getInstance()->getFormat() == Video::PAL)
{
VLiveBanner::~VLiveBanner()
{
+ instance = NULL;
+ Timers::getInstance()->cancelTimer(this, 1);
+ Timers::getInstance()->cancelTimer(this, 2);
delData();
}
+VLiveBanner* VLiveBanner::getInstance()
+{
+ return instance;
+}
+
void VLiveBanner::delData()
{
if (eventList)
}
// Reset the timer as it probably took 1-2 seconds to change the channel
- Timers::getInstance()->cancelTimer(this, 1); // if it exists
Timers::getInstance()->setTimer(this, 1, (struct timespec){4, 0});
}
}
rectangle(7, area.h - 24, 18, 16, Colour::RED);
drawText(tr("info"), 32, area.h - 25, Colour::LIGHTTEXT);
+
+ rectangle(110, area.h - 24, 18, 16, Colour::GREEN);
+ drawText(tr("EPG"), 135, area.h - 25, Colour::LIGHTTEXT);
+
+ drawClock();
}
int VLiveBanner::handleCommand(int command)
sl.draw();
show();
+
+ // Arrows pressed, go to an 8s timer
+ Timers::getInstance()->setTimer(this, 1, (struct timespec){8, 0});
+
return 2;
}
case Remote::DF_DOWN:
sl.draw();
show();
+
+ // Arrows pressed, go to an 8s timer
+ Timers::getInstance()->setTimer(this, 1, (struct timespec){8, 0});
+
return 2;
}
case Remote::CHANNELUP:
}
return 2; // should not get here
}
+ case Remote::GREEN:
+ {
+ // full epg
+ Timers::getInstance()->cancelTimer(this, 1); // if it exists
+ Message* m = new Message();
+ m->message = Message::EPG;
+ m->to = parent;
+ m->from = this;
+ ViewMan::getInstance()->postMessage(m);
+ return 4;
+ }
}
return 1;
void VLiveBanner::timercall(int clientReference)
{
- // delete me!
- Message* m = new Message();
- m->message = Message::CLOSE_ME;
- m->to = ViewMan::getInstance();
- m->from = this;
- ViewMan::getInstance()->postMessage(m);
+ if (clientReference == 1)
+ {
+ // delete me!
+ Message* m = new Message();
+ m->message = Message::CLOSE_ME;
+ m->to = ViewMan::getInstance();
+ m->from = this;
+ ViewMan::getInstance()->postMessage(m);
+ }
+ else if (clientReference == 2)
+ {
+ // redraw clock
+ drawClock();
+ ViewMan::getInstance()->updateView(this, &clockRegion);
+ }
+}
+
+void VLiveBanner::drawClock()
+{
+ // Blank the area first
+ rectangle(area.w - 60, 0, 60, 30, titleBarColour);
+
+ char timeString[20];
+ time_t t;
+ time(&t);
+ struct tm* tms = localtime(&t);
+ strftime(timeString, 19, "%H:%M", tms);
+ drawTextRJ(timeString, 490, 5, Colour::LIGHTTEXT);
+
+ time_t dt = 60 - (t % 60); // seconds to the next minute
+ if (dt == 0) dt = 60; // advance a whole minute if necessary
+ dt += t; // get a time_t value for it rather than using duration
+ // (so it will occur at the actual second and not second and a half)
+
+ Timers::getInstance()->setTimer(this, 2, dt);
}
public:
VLiveBanner(View* parent, Channel* channel, bool bannerTakesCommands);
~VLiveBanner();
+ static VLiveBanner* getInstance();
void delData();
void setChannel(Channel* channel);
void timercall(int clientReference);
private:
+ static VLiveBanner* instance;
View* parent;
WSelectList sl;
Channel* currentChannel;
EventList* eventList;
bool takeCommands;
-
+ void drawClock();
+ Region clockRegion;
};
#endif
setBackgroundColour(Colour::VIEWBACKGROUND);
}
+VMute::~VMute()
+{
+ // Make sure the timer is deleted
+ Timers::getInstance()->cancelTimer(this, 1);
+}
+
void VMute::draw()
{
View::draw();
w.setSurfaceOffset(5, 5);
w.draw();
- Timers::getInstance()->cancelTimer(this, 1); // if it exists
Timers::getInstance()->setTimer(this, 1, (struct timespec){2, 0});
}
{
public:
VMute();
+ ~VMute();
void draw();
int handleCommand(int command);
void timercall(int clientReference);
unavailable = 0;
unavailableView = NULL;
streamType = tstreamType;
- vlb = (VLiveBanner*)1; // Can't be NULL because then that is sent to ViewMan::remove and it takes the top view off .. FIXME!
epgmode=false;
if (streamType == VDR::RADIO) player = new PlayerVideo(Command::getInstance(), 0, 1);
else player = new PlayerVideo(Command::getInstance(), 0, 0);
channelChange(OFFSET, DOWN);
return 2;
}
+ case Remote::PREVCHANNEL:
+ {
+ if (unavailable) showUnavailable(0);
+ else stop();
+ channelChange(PREVIOUS, 0);
+ return 2;
+ }
case Remote::OK:
{
doBanner(true);
channelChange(OFFSET, UP);
if(!epgmode)
{
+ VLiveBanner* vlb = VLiveBanner::getInstance(); // guaranteed to be one
vlb->setChannel((*chanList)[currentChannel]);
vlb->draw();
vlb->show();
channelChange(OFFSET, DOWN);
if(!epgmode)
{
+ VLiveBanner* vlb = VLiveBanner::getInstance(); // guaranteed to be one
vlb->setChannel((*chanList)[currentChannel]);
vlb->draw();
vlb->show();
stop();
play(1);
}
+ else if (m->message == Message::EPG)
+ {
+ Log::getInstance()->log("VVideoLive", Log::DEBUG, "EPG requested from live banner");
+
+ if (!epgmode)
+ {
+ showEPG();
+ epgmode=!epgmode; // shouldn't this be within the braces? // same for above in handleCommand, ask Brian FIXME
+ }
+ }
}
void VVideoLive::doBanner(bool bannerTakesCommands)
{
if(epgmode)
return;
- vlb = new VLiveBanner(this, (*chanList)[currentChannel], bannerTakesCommands);
+
+ if (VLiveBanner::getInstance()) return; // there already is one
+
+ VLiveBanner* vlb = new VLiveBanner(this, (*chanList)[currentChannel], bannerTakesCommands);
Message* m = new Message();
m->from = this;
printf("1\n");
if (unavailable) return;
printf("2\n");
- if (!noRemoveVLB) viewman->removeView(vlb); // if live banner is present, remove it. won't cause damage if its not present
+ if (!noRemoveVLB && VLiveBanner::getInstance()) viewman->removeView(VLiveBanner::getInstance()); // if live banner is present, remove it. won't cause damage if its not present
printf("3\n");
player->stop();
int unavailable;
VInfo* unavailableView;
ULONG streamType;
- VLiveBanner* vlb;
UINT upChannel();
UINT downChannel();
setBackgroundColour(Colour::VIEWBACKGROUND);
}
+VVolume::~VVolume()
+{
+ // Make sure the timer is deleted
+ Timers::getInstance()->cancelTimer(this, 1);
+}
+
void VVolume::draw()
{
View::draw();
w.draw();
}
- Timers::getInstance()->cancelTimer(this, 1); // if it exists
Timers::getInstance()->setTimer(this, 1, (struct timespec){2, 0});
}
{
public:
VVolume();
+ ~VVolume();
void draw();
int handleCommand(int command);
void timercall(int clientReference);
struct tm* tms = localtime(&t);
strftime(timeString, 19, "%H:%M", tms);
drawTextRJ(timeString, 450, 5, Colour::LIGHTTEXT);
- time_t dt = 60 - (t % 60);
- if (dt == 0) dt = 60;
- Timers::getInstance()->setTimer(this, 1, (struct timespec){dt, 0});
+
+ time_t dt = 60 - (t % 60); // seconds to the next minute
+ if (dt == 0) dt = 60; // advance a whole minute if necessary
+ dt += t; // get a time_t value for it rather than using duration
+ // (so it will occur at the actual second and not second and a half)
+
+ Timers::getInstance()->setTimer(this, 1, dt);
}
void VWelcome::timercall(int clientReference)