mirror of
https://projects.vdr-developer.org/git/vdr-plugin-skindesigner.git
synced 2023-10-19 17:58:31 +02:00
1526 lines
52 KiB
C
1526 lines
52 KiB
C
#include "view.h"
|
|
#include "../config.h"
|
|
#include "../libcore/helpers.h"
|
|
#include "../libcore/imageloader.h"
|
|
|
|
using namespace std;
|
|
|
|
cView::cView(cTemplateView *tmplView) : cPixmapContainer(tmplView->DrawGebugGrid() ? tmplView->GetNumPixmaps() + 1 : tmplView->GetNumPixmaps()) {
|
|
this->tmplView = tmplView;
|
|
SetFadeTime(tmplView->GetNumericParameter(ptFadeTime));
|
|
SetShiftTime(tmplView->GetNumericParameter(ptShiftTime));
|
|
SetStartPos(tmplView->GetNumericParameter(ptStartX), tmplView->GetNumericParameter(ptStartY));
|
|
SetShiftType(tmplView->GetNumericParameter(ptShiftType));
|
|
SetShiftMode(tmplView->GetNumericParameter(ptShiftMode));
|
|
tvScaled = tmplView->GetScalingWindow(scalingWindow);
|
|
if (tvScaled) {
|
|
cDevice::PrimaryDevice()->ScaleVideo(scalingWindow);
|
|
}
|
|
tmplViewElement = NULL;
|
|
tmplTab = NULL;
|
|
Init();
|
|
}
|
|
|
|
cView::cView(cTemplateViewElement *tmplViewElement) : cPixmapContainer(tmplViewElement ? tmplViewElement->GetNumPixmaps() : 0) {
|
|
this->tmplViewElement = tmplViewElement;
|
|
tmplView = NULL;
|
|
tmplTab = NULL;
|
|
tvScaled = false;
|
|
Init();
|
|
}
|
|
|
|
cView::cView(cTemplateViewTab *tmplTab) : cPixmapContainer(1) {
|
|
this->tmplTab = tmplTab;
|
|
tmplView = NULL;
|
|
tmplViewElement = NULL;
|
|
tvScaled = false;
|
|
Init();
|
|
}
|
|
|
|
cView::~cView() {
|
|
CancelSave();
|
|
|
|
if (tvScaled) {
|
|
cDevice::PrimaryDevice()->ScaleVideo(cRect::Null);
|
|
}
|
|
//clear detached views
|
|
Lock();
|
|
for (map<eViewElement,cViewElement*>::iterator dVeIt = detachedViewElements.begin(); dVeIt != detachedViewElements.end(); dVeIt++) {
|
|
cViewElement *ve = dVeIt->second;
|
|
delete ve;
|
|
}
|
|
Unlock();
|
|
//clear animations
|
|
Lock();
|
|
for (multimap<int, cAnimation*>::iterator animIt = animations.begin(); animIt != animations.end(); animIt++) {
|
|
cAnimation *anim = animIt->second;
|
|
anim->Stop();
|
|
delete anim;
|
|
}
|
|
Unlock();
|
|
//shift or fade out
|
|
if (fadeOut) {
|
|
if (IsAnimated())
|
|
ShiftOut();
|
|
else {
|
|
FadeOut();
|
|
}
|
|
}
|
|
}
|
|
|
|
void cView::DrawDebugGrid(void) {
|
|
if (tmplView && tmplView->DrawGebugGrid()) {
|
|
DoDrawDebugGrid();
|
|
}
|
|
}
|
|
|
|
void cView::Init(void) {
|
|
fadeOut = true;
|
|
viewInit = true;
|
|
scrolling = false;
|
|
veScroll = veUndefined;
|
|
scrollingPix = -1;
|
|
scrollOrientation = orHorizontal;
|
|
scrollDelay = 0;
|
|
scrollMode = smNone;
|
|
scrollSpeed = ssMedium;
|
|
animCat = 0;
|
|
currentlyScrolling = false;
|
|
}
|
|
|
|
void cView::Action(void) {
|
|
SetInitFinished();
|
|
if (IsAnimated()) {
|
|
ShiftIn();
|
|
} else {
|
|
FadeIn();
|
|
}
|
|
DoFlush();
|
|
if (scrolling) {
|
|
DoSleep(scrollDelay);
|
|
if (!Running())
|
|
return;
|
|
if (scrollOrientation == orHorizontal) {
|
|
ActivateScrolling();
|
|
ScrollHorizontal(scrollingPix, scrollDelay, scrollSpeed, scrollMode);
|
|
} else {
|
|
ScrollVertical(scrollingPix, scrollDelay, scrollSpeed);
|
|
}
|
|
}
|
|
}
|
|
|
|
void cView::Stop(void) {
|
|
CancelSave();
|
|
}
|
|
|
|
/********************************************************************************
|
|
* Protected Functions
|
|
********************************************************************************/
|
|
|
|
void cView::DrawViewElement(eViewElement ve, map <string,string> *stringTokens, map <string,int> *intTokens, map < string, vector< map< string, string > > > *loopTokens) {
|
|
//setting correct ViewElement, depending which constructor was used
|
|
cTemplateViewElement *viewElement = NULL;
|
|
if (tmplViewElement) {
|
|
viewElement = tmplViewElement;
|
|
animCat = 0;
|
|
} else if (tmplView) {
|
|
viewElement = tmplView->GetViewElement(ve);
|
|
animCat = ve;
|
|
}
|
|
if (!viewElement)
|
|
return;
|
|
|
|
if (viewElement->DebugTokens()) {
|
|
DebugTokens(tmplView ? (tmplView->GetViewElementName(ve)) : "current view", stringTokens, intTokens, loopTokens);
|
|
}
|
|
//iterate through pixmaps of viewelement
|
|
int pixCurrent = viewElement->GetPixOffset();
|
|
if (pixCurrent < 0)
|
|
return;
|
|
viewElement->InitPixmapNodeIterator();
|
|
cTemplatePixmapNode *pixNode = NULL;
|
|
while(pixNode = viewElement->GetNextPixmapNode()) {
|
|
cTemplatePixmap *pix = dynamic_cast<cTemplatePixmap*>(pixNode);
|
|
if (pix) {
|
|
bool draw = PreparePixmap(ve, pixCurrent, pix, stringTokens, intTokens, loopTokens);
|
|
if (draw)
|
|
DrawPixmap(pixCurrent, pix, loopTokens);
|
|
pixCurrent++;
|
|
} else {
|
|
cTemplatePixmapContainer *pixContainer = dynamic_cast<cTemplatePixmapContainer*>(pixNode);
|
|
pixContainer->ParseDynamicParameters(stringTokens, intTokens);
|
|
if (pixContainer->DoDebug()) {
|
|
pixContainer->Debug();
|
|
}
|
|
if (!pixContainer->DoExecute()) {
|
|
pixCurrent += pixContainer->NumPixmaps();
|
|
continue;
|
|
}
|
|
pixContainer->InitIterator();
|
|
cTemplatePixmap *pix = NULL;
|
|
while (pix = pixContainer->GetNextPixmap()) {
|
|
bool draw = PreparePixmap(ve, pixCurrent, pix, stringTokens, intTokens, loopTokens);
|
|
if (draw)
|
|
DrawPixmap(pixCurrent, pix, loopTokens);
|
|
pixCurrent++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool cView::PreparePixmap(eViewElement ve, int num, cTemplatePixmap *pix, map <string,string> *stringTokens, map <string,int> *intTokens, map < string, vector< map< string, string > > > *loopTokens) {
|
|
//check if already drawn background area, this can be skipped
|
|
if (PixmapExists(num) && pix->BackgroundArea()) {
|
|
return false;
|
|
}
|
|
//reset Template
|
|
pix->ClearDynamicParameters();
|
|
//create Pixmap if already fully parsed
|
|
if (!PixmapExists(num) && pix->Ready() && pix->DoExecute() && !pix->Scrolling()) {
|
|
CreateViewPixmap(num, pix);
|
|
}
|
|
//check if pixmap needs dynamic parameters
|
|
if ((!pix->Ready() || !pix->DoExecute()) && !pix->Scrolling()) {
|
|
//parse dynamic parameters and initiate functions
|
|
pix->ParseDynamicParameters(stringTokens, intTokens, true);
|
|
if (pix->Ready() && pix->DoExecute()) {
|
|
CreateViewPixmap(num, pix);
|
|
}
|
|
} else {
|
|
//parse dynamic parameters but not initiate functions
|
|
pix->ParseDynamicParameters(stringTokens, intTokens, false);
|
|
}
|
|
//if pixmap still not valid, skip
|
|
if (!pix->Ready() && !pix->Scrolling()) {
|
|
return false;
|
|
}
|
|
//if condition for pixmap set, check if cond is true
|
|
if (!pix->DoExecute()) {
|
|
return false;
|
|
}
|
|
//parse dynamic tokens of pixmap functions
|
|
pix->ClearDynamicFunctionParameters();
|
|
pix->ParseDynamicFunctionParameters(stringTokens, intTokens, loopTokens);
|
|
|
|
if (!PixmapExists(num) && pix->Scrolling()) {
|
|
cSize drawportSize;
|
|
scrolling = pix->CalculateDrawPortSize(drawportSize, loopTokens);
|
|
if (scrolling) {
|
|
CreateScrollingPixmap(num, pix, drawportSize);
|
|
pix->SetScrollingTextWidth();
|
|
veScroll = ve;
|
|
scrollingPix = num;
|
|
scrollOrientation = pix->GetNumericParameter(ptOrientation);
|
|
scrollMode = pix->GetNumericParameter(ptScrollMode);
|
|
scrollDelay = pix->GetNumericParameter(ptDelay);
|
|
scrollSpeed = pix->GetNumericParameter(ptScrollSpeed);
|
|
} else {
|
|
CreateViewPixmap(num, pix);
|
|
}
|
|
}
|
|
if (pix->DoDebug()) {
|
|
pix->Debug();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void cView::ClearViewElement(eViewElement ve) {
|
|
cViewElement *detachedVE = GetViewElement(ve);
|
|
if (detachedVE) {
|
|
detachedVE->Clear();
|
|
return;
|
|
} else {
|
|
if (DetachViewElement(ve))
|
|
return;
|
|
}
|
|
cTemplateViewElement *viewElement = NULL;
|
|
int currentAnimCat = ve;
|
|
if (tmplViewElement) {
|
|
viewElement = tmplViewElement;
|
|
currentAnimCat = 0;
|
|
} else if (tmplView) {
|
|
viewElement = tmplView->GetViewElement(ve);
|
|
}
|
|
if (!viewElement)
|
|
return;
|
|
int pixCurrent = viewElement->GetPixOffset();
|
|
if (pixCurrent < 0)
|
|
return;
|
|
cTemplatePixmap *pix = NULL;
|
|
viewElement->InitPixmapIterator();
|
|
while(pix = viewElement->GetNextPixmap()) {
|
|
if (!pix->BackgroundArea()) {
|
|
Fill(pixCurrent, clrTransparent);
|
|
}
|
|
pixCurrent++;
|
|
}
|
|
ClearAnimations(currentAnimCat);
|
|
}
|
|
|
|
void cView::DestroyViewElement(eViewElement ve) {
|
|
if (!tmplView)
|
|
return;
|
|
cTemplateViewElement *viewElement = tmplView->GetViewElement(ve);
|
|
if (!viewElement)
|
|
return;
|
|
int pixCurrent = viewElement->GetPixOffset();
|
|
if (pixCurrent < 0)
|
|
return;
|
|
cTemplatePixmap *pix = NULL;
|
|
viewElement->InitPixmapIterator();
|
|
while(pix = viewElement->GetNextPixmap()) {
|
|
DestroyPixmap(pixCurrent);
|
|
pixCurrent++;
|
|
}
|
|
ClearAnimations(ve);
|
|
}
|
|
|
|
void cView::DestroyDetachedViewElement(eViewElement ve) {
|
|
map < eViewElement, cViewElement* >::iterator hit = detachedViewElements.find(ve);
|
|
if (hit == detachedViewElements.end())
|
|
return;
|
|
cViewElement *viewElement = hit->second;
|
|
delete viewElement;
|
|
detachedViewElements.erase(hit);
|
|
}
|
|
|
|
|
|
void cView::ClearAnimations(int cat) {
|
|
//stop and delete all animated elements from this viewelement
|
|
if (animations.size() == 0)
|
|
return;
|
|
pair<multimap<int,cAnimation*>::iterator, multimap<int,cAnimation*>::iterator> rangeAnims;
|
|
rangeAnims = animations.equal_range(cat);
|
|
for (multimap<int,cAnimation*>::iterator it = rangeAnims.first; it!=rangeAnims.second; ++it) {
|
|
cAnimation *anim = it->second;
|
|
anim->Stop();
|
|
delete anim;
|
|
}
|
|
animations.erase(cat);
|
|
}
|
|
|
|
void cView::HideAnimations(void) {
|
|
for (multimap<int,cAnimation*>::iterator it = animations.begin(); it!=animations.end(); ++it) {
|
|
cAnimation *anim = it->second;
|
|
anim->HidePixmaps();
|
|
}
|
|
}
|
|
|
|
void cView::ShowAnimations(void) {
|
|
for (multimap<int,cAnimation*>::iterator it = animations.begin(); it!=animations.end(); ++it) {
|
|
cAnimation *anim = it->second;
|
|
anim->ShowPixmaps();
|
|
}
|
|
}
|
|
|
|
void cView::ActivateScrolling(void) {
|
|
if (veScroll == veUndefined)
|
|
return;
|
|
cTemplateViewElement *scrollViewElement = NULL;
|
|
if (tmplView) {
|
|
scrollViewElement = tmplView->GetViewElement(veScroll);
|
|
}
|
|
if (!scrollViewElement)
|
|
return;
|
|
|
|
ClearViewElement(veScroll);
|
|
currentlyScrolling = true;
|
|
|
|
int pixCurrent = scrollViewElement->GetPixOffset();
|
|
if (pixCurrent < 0)
|
|
return;
|
|
scrollViewElement->InitPixmapIterator();
|
|
cTemplatePixmap *pix = NULL;
|
|
while(pix = scrollViewElement->GetNextPixmap()) {
|
|
DrawPixmap(pixCurrent, pix);
|
|
pixCurrent++;
|
|
}
|
|
}
|
|
|
|
bool cView::ExecuteViewElement(eViewElement ve) {
|
|
if (!tmplView)
|
|
return false;
|
|
bool doExecute = tmplView->ExecuteView(ve);
|
|
if (!doExecute)
|
|
return false;
|
|
return tmplView->GetNumPixmapsViewElement(ve);
|
|
}
|
|
|
|
bool cView::DetachViewElement(eViewElement ve) {
|
|
if (!tmplView)
|
|
return false;
|
|
return tmplView->DetachViewElement(ve);
|
|
}
|
|
|
|
bool cView::ViewElementScrolls(eViewElement ve) {
|
|
if (scrollingPix < 0)
|
|
return false;
|
|
if (!tmplView)
|
|
return false;
|
|
cTemplateViewElement *viewElement = tmplView->GetViewElement(ve);
|
|
if (!viewElement)
|
|
return false;
|
|
int pixStart = viewElement->GetPixOffset();
|
|
int numPixmaps = viewElement->GetNumPixmaps();
|
|
if ( (scrollingPix >= pixStart) && (scrollingPix < (pixStart + numPixmaps)) )
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
cViewElement *cView::GetViewElement(eViewElement ve) {
|
|
map < eViewElement, cViewElement* >::iterator hit = detachedViewElements.find(ve);
|
|
if (hit == detachedViewElements.end())
|
|
return NULL;
|
|
cViewElement *viewElement = hit->second;
|
|
return viewElement;
|
|
}
|
|
|
|
void cView::AddViewElement(eViewElement ve, cViewElement *viewElement) {
|
|
detachedViewElements.insert(pair< eViewElement, cViewElement* >(ve, viewElement));
|
|
}
|
|
|
|
void cView::CreateViewPixmap(int num, cTemplatePixmap *pix, cRect *size) {
|
|
cRect pixSize;
|
|
if (size) {
|
|
pixSize = *size;
|
|
} else {
|
|
pixSize = pix->GetPixmapSize();
|
|
}
|
|
int layer = pix->GetNumericParameter(ptLayer);
|
|
int transparency = pix->GetNumericParameter(ptTransparency);
|
|
SetTransparency(num, transparency);
|
|
CreatePixmap(num, layer, pixSize);
|
|
}
|
|
|
|
void cView::CreateScrollingPixmap(int num, cTemplatePixmap *pix, cSize &drawportSize) {
|
|
cRect pixViewPort = pix->GetPixmapSize();
|
|
cRect drawPort;
|
|
drawPort.SetX(0);
|
|
drawPort.SetY(0);
|
|
drawPort.SetWidth(drawportSize.Width());
|
|
drawPort.SetHeight(drawportSize.Height());
|
|
int layer = pix->GetNumericParameter(ptLayer);
|
|
int transparency = pix->GetNumericParameter(ptTransparency);
|
|
SetTransparency(num, transparency);
|
|
CreatePixmap(num, layer, pixViewPort, drawPort);
|
|
}
|
|
|
|
void cView::DrawPixmap(int num, cTemplatePixmap *pix, map < string, vector< map< string, string > > > *loopTokens, bool flushPerLoop) {
|
|
pix->InitFunctionIterator();
|
|
cTemplateFunction *func = NULL;
|
|
while(func = pix->GetNextFunction()) {
|
|
eFuncType type = func->GetType();
|
|
if (func->DoDebug()) {
|
|
func->Debug();
|
|
}
|
|
if (!func->DoExecute()) {
|
|
continue;
|
|
}
|
|
switch (type) {
|
|
case ftFill:
|
|
DoFill(num, func);
|
|
break;
|
|
case ftDrawText:
|
|
DoDrawText(num, func);
|
|
break;
|
|
case ftDrawTextVertical:
|
|
DoDrawTextVertical(num, func);
|
|
break;
|
|
case ftDrawTextBox: {
|
|
int floating = func->GetNumericParameter(ptFloat);
|
|
if (floating > flNone) {
|
|
DoDrawFloatingTextBox(num, func);
|
|
} else {
|
|
DoDrawTextBox(num, func);
|
|
}
|
|
break; }
|
|
case ftDrawRectangle:
|
|
DoDrawRectangle(num, func);
|
|
break;
|
|
case ftDrawEllipse:
|
|
DoDrawEllipse(num, func);
|
|
break;
|
|
case ftDrawSlope:
|
|
DoDrawSlope(num, func);
|
|
break;
|
|
case ftDrawImage:
|
|
DoDrawImage(num, func);
|
|
break;
|
|
case ftLoop:
|
|
if (loopTokens)
|
|
DrawLoop(num, func, loopTokens);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (flushPerLoop) {
|
|
DoFlush();
|
|
}
|
|
}
|
|
}
|
|
|
|
void cView::DrawLoop(int numPixmap, cTemplateFunction *func, map < string, vector< map< string, string > > > *loopTokens) {
|
|
cTemplateLoopFunction *loopFunc = dynamic_cast<cTemplateLoopFunction*>(func);
|
|
if (!loopFunc)
|
|
return;
|
|
|
|
int loopX0 = loopFunc->GetNumericParameter(ptX);
|
|
if (loopX0 < 0) loopX0 = 0;
|
|
int loopY0 = loopFunc->GetNumericParameter(ptY);
|
|
if (loopY0 < 0) loopY0 = 0;
|
|
int orientation = loopFunc->GetNumericParameter(ptOrientation);
|
|
int loopWidth = loopFunc->GetNumericParameter(ptWidth);
|
|
if (loopWidth <= 0)
|
|
loopWidth = loopFunc->GetContainerWidth();
|
|
int loopHeight = loopFunc->GetNumericParameter(ptHeight);
|
|
if (loopHeight <= 0)
|
|
loopHeight = loopFunc->GetContainerHeight();
|
|
int columnWidth = loopFunc->GetNumericParameter(ptColumnWidth);
|
|
int rowHeight = loopFunc->GetNumericParameter(ptRowHeight);
|
|
int overflow = loopFunc->GetNumericParameter(ptOverflow);
|
|
|
|
int x0 = loopX0;
|
|
int y0 = loopY0;
|
|
|
|
string loopTokenName = loopFunc->GetParameter(ptName);
|
|
|
|
map < string, vector< map< string, string > > >::iterator hit = loopTokens->find(loopTokenName);
|
|
if (hit == loopTokens->end())
|
|
return;
|
|
vector< map<string,string> > loopToken = hit->second;
|
|
int lineNumber=0;
|
|
for (vector< map<string,string> >::iterator line = loopToken.begin(); line != loopToken.end(); line++) {
|
|
//check overflow behaviour
|
|
if (overflow == otCut) {
|
|
if (orientation == orHorizontal) {
|
|
if (lineNumber * columnWidth > loopWidth) {
|
|
return;
|
|
}
|
|
} else if (orientation == orVertical) {
|
|
if (lineNumber * rowHeight > loopHeight) {
|
|
return;
|
|
}
|
|
}
|
|
} else if (overflow == otWrap && orientation == orHorizontal) {
|
|
if (x0 + columnWidth > loopWidth) {
|
|
x0 = loopX0;
|
|
if (rowHeight > 0) {
|
|
y0 += rowHeight;
|
|
} else {
|
|
y0 += loopFunc->GetLoopElementsHeight();
|
|
}
|
|
}
|
|
}
|
|
map<string,string> tokens = *line;
|
|
loopFunc->ClearDynamicParameters();
|
|
loopFunc->ParseDynamicParameters(&tokens);
|
|
loopFunc->InitIterator();
|
|
cTemplateFunction *func = NULL;
|
|
while(func = loopFunc->GetNextFunction()) {
|
|
//do debug?
|
|
if (func->DoDebug())
|
|
func->Debug();
|
|
//check if set condition is true
|
|
if (!func->DoExecute()) {
|
|
continue;
|
|
}
|
|
//execute
|
|
eFuncType type = func->GetType();
|
|
switch (type) {
|
|
case ftDrawText:
|
|
DoDrawText(numPixmap, func, x0, y0);
|
|
break;
|
|
case ftDrawTextBox:
|
|
DoDrawTextBox(numPixmap, func, x0, y0);
|
|
break;
|
|
case ftDrawRectangle:
|
|
DoDrawRectangle(numPixmap, func, x0, y0);
|
|
break;
|
|
case ftDrawEllipse:
|
|
DoDrawEllipse(numPixmap, func, x0, y0);
|
|
break;
|
|
case ftDrawSlope:
|
|
DoDrawSlope(numPixmap, func, x0, y0);
|
|
break;
|
|
case ftDrawImage:
|
|
DoDrawImage(numPixmap, func, x0, y0);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
//calculate position of next loop element
|
|
if (orientation == orHorizontal) {
|
|
if (columnWidth > 0) {
|
|
x0 += columnWidth;
|
|
} else {
|
|
x0 += loopFunc->GetLoopElementsWidth();
|
|
}
|
|
} else if (orientation == orVertical) {
|
|
if (rowHeight > 0) {
|
|
y0 += rowHeight;
|
|
} else {
|
|
y0 += loopFunc->GetLoopElementsHeight();
|
|
}
|
|
}
|
|
lineNumber++;
|
|
//DoFlush();
|
|
}
|
|
}
|
|
|
|
void cView::DebugTokens(string viewElement, map<string,string> *stringTokens, map<string,int> *intTokens, map < string, vector< map< string, string > > > *loopTokens) {
|
|
esyslog("skindesigner: ------------------------------ Tokens for %s:", viewElement.c_str());
|
|
if (stringTokens) {
|
|
for (map<string,string>::iterator st = stringTokens->begin(); st != stringTokens->end(); st++) {
|
|
esyslog("skindesigner: string var \"%s\" = \"%s\"", (st->first).c_str(), (st->second).c_str());
|
|
}
|
|
}
|
|
if (intTokens) {
|
|
for (map<string,int>::iterator it = intTokens->begin(); it != intTokens->end(); it++) {
|
|
esyslog("skindesigner: int var \"%s\" = %d", (it->first).c_str(), it->second);
|
|
}
|
|
}
|
|
if (loopTokens) {
|
|
for(map < string, vector< map< string, string > > >::iterator it1 = loopTokens->begin(); it1 != loopTokens->end(); it1++) {
|
|
int line = 0;
|
|
string tokenName = it1->first;
|
|
vector< map<string,string> > tokens = it1->second;
|
|
esyslog("skindesigner: loop token %s", tokenName.c_str());
|
|
for (vector< map<string,string> >::iterator it2 = tokens.begin(); it2 != tokens.end(); it2++) {
|
|
esyslog("skindesigner: loop tokens line %d:", line++);
|
|
map<string,string> element = *it2;
|
|
for (map<string,string>::iterator el = element.begin(); el != element.end(); el++) {
|
|
esyslog("skindesigner: name: %s, value: %s", (el->first).c_str(), (el->second).c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*****************************************************************
|
|
* Private Functions
|
|
*****************************************************************/
|
|
|
|
void cView::DoDrawDebugGrid(void) {
|
|
int stepsX = tmplView->DebugGridX();
|
|
int stepsY = tmplView->DebugGridY();
|
|
cRect osdSize = tmplView->GetOsdSize();
|
|
tColor col = tmplView->DebugGridColor();
|
|
tColor colFont = tmplView->DebugGridFontColor();
|
|
|
|
cRect size(0, 0, osdSize.Width(), osdSize.Height());
|
|
//use last pixmap as grid pixmap
|
|
int numGridPixmap = NumPixmaps() - 1;
|
|
CreatePixmap(numGridPixmap, 7, size);
|
|
|
|
int width = size.Width();
|
|
int height = size.Height();
|
|
float stepWidthX = (double)width / (double)stepsX;
|
|
float stepWidthY = (double)height / (double)stepsY;
|
|
int fontSize = height / stepsY / 5;
|
|
|
|
for (int i = 0; i <= stepsX; i++) {
|
|
int x = (float)i * stepWidthX;
|
|
if (i==stepsX)
|
|
x = x-1;
|
|
cRect line(x, 0, 1, height);
|
|
DrawRectangle(numGridPixmap, line, col);
|
|
if (i==stepsX)
|
|
continue;
|
|
float percent = (float)i / (float)stepsX * 100.0;
|
|
cPoint textPosPerc(x+2, 2);
|
|
cPoint textPosAbs(x+2, fontSize + 4);
|
|
DrawText(numGridPixmap, textPosPerc, *cString::sprintf("%.1f%%", percent), colFont, 0x00000000, "vdrOsd", fontSize);
|
|
DrawText(numGridPixmap, textPosAbs, *cString::sprintf("%dpx", x), colFont, 0x00000000, "vdrOsd", fontSize);
|
|
}
|
|
for (int i = 0; i <= stepsY; i++) {
|
|
int y = (float)i * stepWidthY;
|
|
if (i==stepsY)
|
|
y = y-1;
|
|
cRect line(0, y, width, 1);
|
|
DrawRectangle(numGridPixmap, line, col);
|
|
if (i==0 || i==stepsY)
|
|
continue;
|
|
float percent = (float)i / (float)stepsY * 100.0;
|
|
cPoint textPosPerc(2, y + 2);
|
|
cPoint textPosAbs(2, y + fontSize + 4);
|
|
DrawText(numGridPixmap, textPosPerc, *cString::sprintf("%.1f%%", percent), colFont, 0x00000000, "vdrOsd", fontSize);
|
|
DrawText(numGridPixmap, textPosAbs, *cString::sprintf("%dpx", y), colFont, 0x00000000, "vdrOsd", fontSize);
|
|
}
|
|
}
|
|
|
|
void cView::DoFill(int num, cTemplateFunction *func) {
|
|
tColor col = func->GetColorParameter(ptColor);
|
|
Fill(num, col);
|
|
}
|
|
|
|
void cView::DoDrawText(int num, cTemplateFunction *func, int x0, int y0) {
|
|
int x = func->GetNumericParameter(ptX);
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (y < 0) y = 0;
|
|
y += y0;
|
|
cPoint pos(x,y);
|
|
string fontName = func->GetFontName();
|
|
int fontSize = func->GetNumericParameter(ptFontSize);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
tColor clrBack = clrTransparent;
|
|
string text = "";
|
|
if (!currentlyScrolling) {
|
|
text = func->GetText(true);
|
|
} else {
|
|
text = func->GetText(false);
|
|
}
|
|
if (func->IsAnimated()) {
|
|
DrawAnimatedText(num, func, pos, text, clr, fontName, fontSize);
|
|
} else {
|
|
DrawText(num, pos, text.c_str(), clr, clrBack, fontName, fontSize);
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawTextVertical(int num, cTemplateFunction *func, int x0, int y0) {
|
|
string fontName = func->GetFontName();
|
|
int fontSize = func->GetNumericParameter(ptFontSize);
|
|
int direction = func->GetNumericParameter(ptDirection);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
string text = func->GetText(false);
|
|
cImage *textVertical = imgCache->GetVerticalText(text, clr, fontName, fontSize, direction);
|
|
if (!textVertical)
|
|
return;
|
|
|
|
//align has to be set here because here we know the image size
|
|
int x = 0;
|
|
int y = 0;
|
|
int align = func->GetNumericParameter(ptAlign);
|
|
if (align == alCenter) {
|
|
int containerWidth = func->GetContainerWidth();
|
|
x = (containerWidth - textVertical->Width()) / 2;
|
|
} else if (align == alLeft) {
|
|
x = 0;
|
|
} else if (align == alRight) {
|
|
int containerWidth = func->GetContainerWidth();
|
|
x = (containerWidth - textVertical->Width());
|
|
} else {
|
|
x = func->GetNumericParameter(ptX);
|
|
}
|
|
|
|
int valign = func->GetNumericParameter(ptValign);
|
|
if (valign == alCenter) {
|
|
int containerHeight = func->GetContainerHeight();
|
|
y = (containerHeight - textVertical->Height()) / 2;
|
|
} else if (align == alTop) {
|
|
y = 0;
|
|
} else if (align == alBottom) {
|
|
int containerHeight = func->GetContainerHeight();
|
|
y = (containerHeight - textVertical->Height());
|
|
} else {
|
|
y = func->GetNumericParameter(ptY);
|
|
}
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
if (y < 0) y = func->GetContainerHeight() - textVertical->Height() - 5;
|
|
y += y0;
|
|
cPoint pos(x,y);
|
|
|
|
if (func->IsAnimated()) {
|
|
cRect posAnim(x, y, textVertical->Width(), textVertical->Height());
|
|
DrawAnimatedImage(num, func, posAnim, textVertical);
|
|
} else {
|
|
DrawImage(num, pos, *textVertical);
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawTextBox(int num, cTemplateFunction *func, int x0, int y0) {
|
|
string text = func->GetText(false);
|
|
if (text.size() < 3)
|
|
return;
|
|
int x = func->GetNumericParameter(ptX);
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
if (y < 0) y = 0;
|
|
y += y0;
|
|
int width = func->GetNumericParameter(ptWidth);
|
|
int height = func->GetNumericParameter(ptHeight);
|
|
string fontName = func->GetFontName();
|
|
int fontSize = func->GetNumericParameter(ptFontSize);
|
|
int align = func->GetNumericParameter(ptAlign);
|
|
int maxLines = func->GetNumericParameter(ptMaxLines);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
tColor clrBack = clrTransparent;
|
|
const cFont *font = fontManager->Font(fontName, fontSize);
|
|
if (!font)
|
|
return;
|
|
fontManager->Lock();
|
|
cTextWrapper wrapper;
|
|
wrapper.Set(text.c_str(), font, width);
|
|
fontManager->Unlock();
|
|
int fontHeight = fontManager->Height(fontName, fontSize);
|
|
int lines = wrapper.Lines();
|
|
int yLine = y;
|
|
for (int line=0; line < lines; line++) {
|
|
int xLine = x;
|
|
if (align == alCenter) {
|
|
int textWidth = font->Width(wrapper.GetLine(line));
|
|
xLine += (width - textWidth)/2;
|
|
} else if (align == alRight) {
|
|
int textWidth = font->Width(wrapper.GetLine(line));
|
|
xLine += (width - textWidth);
|
|
}
|
|
cPoint pos(xLine, yLine);
|
|
if (maxLines > 0 && line == maxLines-1) {
|
|
string lastLine = wrapper.GetLine(line);
|
|
if (lines > maxLines) {
|
|
lastLine += "...";
|
|
}
|
|
DrawText(num, pos, lastLine.c_str(), clr, clrBack, fontName, fontSize);
|
|
break;
|
|
} else if (height > 0 && yLine - y + 2*fontHeight > height) {
|
|
DrawText(num, pos, "...", clr, clrBack, fontName, fontSize);
|
|
break;
|
|
}
|
|
DrawText(num, pos, wrapper.GetLine(line), clr, clrBack, fontName, fontSize);
|
|
yLine += fontHeight;
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawFloatingTextBox(int num, cTemplateFunction *func) {
|
|
string text = func->GetText(false);
|
|
if (text.size() < 3)
|
|
return;
|
|
int x = func->GetNumericParameter(ptX);
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (x < 0) x = 0;
|
|
if (y < 0) y = 0;
|
|
int width = func->GetNumericParameter(ptWidth);
|
|
int height = func->GetNumericParameter(ptHeight);
|
|
string fontName = func->GetFontName();
|
|
int fontSize = func->GetNumericParameter(ptFontSize);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
tColor clrBack = clrTransparent;
|
|
const cFont *font = fontManager->Font(fontName, fontSize);
|
|
if (!font)
|
|
return;
|
|
int floatType = func->GetNumericParameter(ptFloat);
|
|
int floatWidth = func->GetNumericParameter(ptFloatWidth);
|
|
int floatHeight = func->GetNumericParameter(ptFloatHeight);
|
|
|
|
cTextWrapper wTextTall;
|
|
cTextWrapper wTextFull;
|
|
|
|
int fontHeight = fontManager->Height(fontName, fontSize);
|
|
int linesNarrow = floatHeight / fontHeight;
|
|
int widthNarrow = width - floatWidth;
|
|
int linesDrawn = 0;
|
|
int curY = 0;
|
|
bool drawNarrow = true;
|
|
|
|
splitstring s(text.c_str());
|
|
std::vector<std::string> flds = s.split('\n', 1);
|
|
|
|
if (flds.size() < 1)
|
|
return;
|
|
|
|
std::stringstream sstrTextTall;
|
|
std::stringstream sstrTextFull;
|
|
|
|
for (int i=0; i < (int)flds.size(); i++) {
|
|
if (!flds[i].size()) {
|
|
//empty line
|
|
linesDrawn++;
|
|
curY += fontHeight;
|
|
if (drawNarrow)
|
|
sstrTextTall << "\n";
|
|
else
|
|
sstrTextFull << "\n";
|
|
} else {
|
|
cTextWrapper wrapper;
|
|
if (drawNarrow) {
|
|
fontManager->Lock();
|
|
wrapper.Set((flds[i].c_str()), font, widthNarrow);
|
|
fontManager->Unlock();
|
|
int newLines = wrapper.Lines();
|
|
//check if wrapper fits completely into narrow area
|
|
if (linesDrawn + newLines < linesNarrow) {
|
|
for (int line = 0; line < wrapper.Lines(); line++) {
|
|
sstrTextTall << wrapper.GetLine(line) << " ";
|
|
}
|
|
sstrTextTall << "\n";
|
|
linesDrawn += newLines;
|
|
} else {
|
|
//this wrapper has to be splitted
|
|
for (int line = 0; line < wrapper.Lines(); line++) {
|
|
if (line + linesDrawn < linesNarrow) {
|
|
sstrTextTall << wrapper.GetLine(line) << " ";
|
|
} else {
|
|
sstrTextFull << wrapper.GetLine(line) << " ";
|
|
}
|
|
}
|
|
sstrTextFull << "\n";
|
|
drawNarrow = false;
|
|
}
|
|
} else {
|
|
fontManager->Lock();
|
|
wrapper.Set((flds[i].c_str()), font, width);
|
|
fontManager->Unlock();
|
|
for (int line = 0; line < wrapper.Lines(); line++) {
|
|
sstrTextFull << wrapper.GetLine(line) << " ";
|
|
}
|
|
sstrTextFull << "\n";
|
|
}
|
|
}
|
|
}
|
|
//VDRs textwrapper swallows linebreaks at the end, so we have to fix that manually
|
|
string textTall = sstrTextTall.str();
|
|
size_t posLastCarriageReturn = textTall.find_last_not_of("\n");
|
|
|
|
int numLinesToAddAtTall = 0;
|
|
if (posLastCarriageReturn != string::npos && (posLastCarriageReturn < textTall.size() - 1)) {
|
|
numLinesToAddAtTall = textTall.size() - posLastCarriageReturn - 2;
|
|
}
|
|
fontManager->Lock();
|
|
wTextTall.Set(textTall.c_str(), font, widthNarrow);
|
|
fontManager->Unlock();
|
|
fontManager->Lock();
|
|
wTextFull.Set(sstrTextFull.str().c_str(), font, width);
|
|
fontManager->Unlock();
|
|
|
|
int textLinesTall = wTextTall.Lines();
|
|
int textLinesFull = wTextFull.Lines();
|
|
|
|
int textXTall = x;
|
|
if (floatType == flTopLeft)
|
|
textXTall = x + floatWidth;
|
|
|
|
int yLine = y;
|
|
for (int line=0; line < textLinesTall; line++) {
|
|
cPoint pos(textXTall, yLine);
|
|
DrawText(num, pos, wTextTall.GetLine(line), clr, clrBack, fontName, fontSize);
|
|
yLine += fontHeight;
|
|
}
|
|
|
|
if (numLinesToAddAtTall) {
|
|
yLine += numLinesToAddAtTall * fontHeight;
|
|
}
|
|
|
|
for (int line=0; line < textLinesFull; line++) {
|
|
cPoint pos(x, yLine);
|
|
if (height > 0 && yLine - y + 2*fontHeight > height) {
|
|
DrawText(num, pos, "...", clr, clrBack, fontName, fontSize);
|
|
break;
|
|
}
|
|
DrawText(num, pos, wTextFull.GetLine(line), clr, clrBack, fontName, fontSize);
|
|
yLine += fontHeight;
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawRectangle(int num, cTemplateFunction *func, int x0, int y0) {
|
|
int x = func->GetNumericParameter(ptX);
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
if (y < 0) y = 0;
|
|
y += y0;
|
|
int w = func->GetNumericParameter(ptWidth);
|
|
int h = func->GetNumericParameter(ptHeight);
|
|
cRect size(x, y, w, h);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
|
|
if (func->IsAnimated()) {
|
|
DrawAnimatedOsdObject(num, func, size, clr, 0);
|
|
} else {
|
|
DrawRectangle(num, size, clr);
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawEllipse(int num, cTemplateFunction *func, int x0, int y0) {
|
|
int x = func->GetNumericParameter(ptX);
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
if (y < 0) y = 0;
|
|
y += y0;
|
|
int w = func->GetNumericParameter(ptWidth);
|
|
int h = func->GetNumericParameter(ptHeight);
|
|
cRect size(x, y, w, h);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
int quadrant = func->GetNumericParameter(ptQuadrant);
|
|
if (quadrant < -4 || quadrant > 8) {
|
|
esyslog("skindesigner: wrong quadrant %d for drawellipse, allowed values are from -4 to 8", quadrant);
|
|
quadrant = 0;
|
|
}
|
|
|
|
if (func->IsAnimated()) {
|
|
DrawAnimatedOsdObject(num, func, size, clr, quadrant);
|
|
} else {
|
|
DrawEllipse(num, size, clr, quadrant);
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawSlope(int num, cTemplateFunction *func, int x0, int y0) {
|
|
int x = func->GetNumericParameter(ptX);
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
if (y < 0) y = 0;
|
|
y += y0;
|
|
int w = func->GetNumericParameter(ptWidth);
|
|
int h = func->GetNumericParameter(ptHeight);
|
|
cRect size(x, y, w, h);
|
|
tColor clr = func->GetColorParameter(ptColor);
|
|
int type = func->GetNumericParameter(ptType);
|
|
if (type < 0 || type > 7) {
|
|
esyslog("skindesigner: wrong type %d for drawslope, allowed values are from 0 to 7", type);
|
|
type = 0;
|
|
}
|
|
if (func->IsAnimated()) {
|
|
DrawAnimatedOsdObject(num, func, size, clr, type);
|
|
} else {
|
|
DrawSlope(num, size, clr, type);
|
|
}
|
|
}
|
|
|
|
void cView::DoDrawImage(int num, cTemplateFunction *func, int x0, int y0) {
|
|
int x = func->GetNumericParameter(ptX);
|
|
int y = func->GetNumericParameter(ptY);
|
|
if (x < 0) x = 0;
|
|
x += x0;
|
|
if (y < 0) y = 0;
|
|
y += y0;
|
|
cPoint pos(x,y);
|
|
int width = func->GetNumericParameter(ptWidth);
|
|
int height = func->GetNumericParameter(ptHeight);
|
|
string path = func->GetImagePath();
|
|
eImageType type = (eImageType)func->GetNumericParameter(ptImageType);
|
|
switch (type) {
|
|
case itChannelLogo: {
|
|
cImage *logo = imgCache->GetLogo(path, width, height);
|
|
if (logo) {
|
|
if (func->IsAnimated()) {
|
|
cRect posAnim(x, y, width, height);
|
|
DrawAnimatedImage(num, func, posAnim, logo);
|
|
} else {
|
|
DrawImage(num, pos, *logo);
|
|
}
|
|
}
|
|
break; }
|
|
case itSepLogo: {
|
|
cImage *sepLogo = imgCache->GetSeparatorLogo(path, width, height);
|
|
if (sepLogo) {
|
|
if (func->IsAnimated()) {
|
|
cRect posAnim(x, y, width, height);
|
|
DrawAnimatedImage(num, func, posAnim, sepLogo);
|
|
} else {
|
|
DrawImage(num, pos, *sepLogo);
|
|
}
|
|
}
|
|
break; }
|
|
case itSkinPart: {
|
|
cImage *skinpart = imgCache->GetSkinpart(path, width, height);
|
|
if (skinpart) {
|
|
if (func->IsAnimated()) {
|
|
cRect posAnim(x, y, width, height);
|
|
DrawAnimatedImage(num, func, posAnim, skinpart);
|
|
} else {
|
|
DrawImage(num, pos, *skinpart);
|
|
}
|
|
}
|
|
break; }
|
|
case itIcon: {
|
|
cImage *icon = imgCache->GetIcon(type, path, width, height);
|
|
if (icon) {
|
|
if (func->IsAnimated()) {
|
|
cRect posAnim(x, y, width, height);
|
|
DrawAnimatedImage(num, func, posAnim, icon);
|
|
} else {
|
|
DrawImage(num, pos, *icon);
|
|
}
|
|
}
|
|
break; }
|
|
case itMenuIcon: {
|
|
cImage *icon = imgCache->GetIcon(type, path, width, height);
|
|
if (icon) {
|
|
if (func->IsAnimated()) {
|
|
cRect posAnim(x, y, width, height);
|
|
DrawAnimatedImage(num, func, posAnim, icon);
|
|
} else {
|
|
DrawImage(num, pos, *icon);
|
|
}
|
|
}
|
|
break; }
|
|
case itImage: {
|
|
cImageLoader imgLoader;
|
|
if (imgLoader.LoadImage(path.c_str())) {
|
|
cImage *image = imgLoader.CreateImage(width, height);
|
|
DrawImage(num, pos, *image);
|
|
delete(image);
|
|
}
|
|
break; }
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void cView::DrawAnimatedImage(int numPix, cTemplateFunction *func, cRect &pos, cImage *image) {
|
|
int layer = Layer(numPix);
|
|
cRect posAnim = CalculateAnimationClip(numPix, pos);
|
|
eAnimType animType = (eAnimType)func->GetNumericParameter(ptAnimType);
|
|
int animFreq = func->GetNumericParameter(ptAnimFreq);
|
|
|
|
cAnimatedImage *anim = new cAnimatedImage(animType, animFreq, posAnim, layer);
|
|
animations.insert(pair<int, cAnimation*>(animCat, anim));
|
|
anim->SetDelay(AnimationDelay());
|
|
anim->SetImage(image);
|
|
anim->Start();
|
|
}
|
|
|
|
void cView::DrawAnimatedText(int numPix, cTemplateFunction *func, cPoint &pos, string text, tColor col, string fontName, int fontSize) {
|
|
int layer = Layer(numPix);
|
|
int textWidth = fontManager->Width(fontName, fontSize, text.c_str());
|
|
int textHeight = fontManager->Height(fontName, fontSize);
|
|
cRect posOrig(pos.X(), pos.Y(), textWidth, textHeight);
|
|
cRect posAnim = CalculateAnimationClip(numPix, posOrig);
|
|
eAnimType animType = (eAnimType)func->GetNumericParameter(ptAnimType);
|
|
int animFreq = func->GetNumericParameter(ptAnimFreq);
|
|
|
|
cAnimatedText *anim = new cAnimatedText(animType, animFreq, posAnim, layer);
|
|
animations.insert(pair<int, cAnimation*>(animCat, anim));
|
|
anim->SetDelay(AnimationDelay());
|
|
anim->SetText(text);
|
|
anim->SetFont(fontName);
|
|
anim->SetFontSize(fontSize);
|
|
anim->SetFontColor(col);
|
|
anim->Start();
|
|
}
|
|
|
|
void cView::DrawAnimatedOsdObject(int numPix, cTemplateFunction *func, cRect &pos, tColor col, int quadrant) {
|
|
int layer = Layer(numPix);
|
|
cRect posAnim = CalculateAnimationClip(numPix, pos);
|
|
eFuncType funcType = func->GetType();
|
|
eAnimType animType = (eAnimType)func->GetNumericParameter(ptAnimType);
|
|
int animFreq = func->GetNumericParameter(ptAnimFreq);
|
|
|
|
cAnimatedOsdObject *anim = new cAnimatedOsdObject(funcType, animType, animFreq, posAnim, layer);
|
|
animations.insert(pair<int, cAnimation*>(animCat, anim));
|
|
anim->SetDelay(AnimationDelay());
|
|
anim->SetColor(col);
|
|
anim->SetQuadrant(quadrant);
|
|
anim->Start();
|
|
}
|
|
|
|
cRect cView::CalculateAnimationClip(int numPix, cRect &pos) {
|
|
cPoint posPix;
|
|
Pos(numPix, posPix);
|
|
cRect posAnim;
|
|
posAnim.SetX(posPix.X() + pos.X());
|
|
posAnim.SetY(posPix.Y() + pos.Y());
|
|
posAnim.SetWidth(pos.Width());
|
|
posAnim.SetHeight(pos.Height());
|
|
return posAnim;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* cViewElement
|
|
************************************************************************/
|
|
|
|
cViewElement::cViewElement(cTemplateViewElement *tmplViewElement) : cView(tmplViewElement) {
|
|
init = true;
|
|
fadeOut = false;
|
|
ve = veUndefined;
|
|
helper = NULL;
|
|
SetTokens = NULL;
|
|
tmplViewElement->SetPixOffset(0);
|
|
delay = tmplViewElement->GetNumericParameter(ptDelay);
|
|
SetFadeTime(tmplViewElement->GetNumericParameter(ptFadeTime));
|
|
SetShiftTime(tmplViewElement->GetNumericParameter(ptShiftTime));
|
|
SetStartPos(tmplViewElement->GetNumericParameter(ptStartX), tmplViewElement->GetNumericParameter(ptStartY));
|
|
SetShiftType(tmplViewElement->GetNumericParameter(ptShiftType));
|
|
SetShiftMode(tmplViewElement->GetNumericParameter(ptShiftMode));
|
|
}
|
|
|
|
cViewElement::cViewElement(cTemplateViewElement *tmplViewElement, cViewHelpers *helper) : cView(tmplViewElement) {
|
|
init = true;
|
|
fadeOut = false;
|
|
ve = veUndefined;
|
|
this->helper = helper;
|
|
SetTokens = NULL;
|
|
tmplViewElement->SetPixOffset(0);
|
|
delay = tmplViewElement->GetNumericParameter(ptDelay);
|
|
SetFadeTime(tmplViewElement->GetNumericParameter(ptFadeTime));
|
|
SetShiftTime(tmplViewElement->GetNumericParameter(ptShiftTime));
|
|
SetStartPos(tmplViewElement->GetNumericParameter(ptStartX), tmplViewElement->GetNumericParameter(ptStartY));
|
|
SetShiftType(tmplViewElement->GetNumericParameter(ptShiftType));
|
|
SetShiftMode(tmplViewElement->GetNumericParameter(ptShiftMode));
|
|
}
|
|
|
|
cViewElement::~cViewElement() {
|
|
CancelSave();
|
|
}
|
|
|
|
bool cViewElement::Render(void) {
|
|
if (!helper || !SetTokens) {
|
|
return false;
|
|
}
|
|
ClearTokens();
|
|
bool done = (helper->*SetTokens)(init, stringTokens, intTokens);
|
|
if (!done) {
|
|
return false;
|
|
}
|
|
init = false;
|
|
ClearViewElement(ve);
|
|
DrawViewElement(ve, &stringTokens, &intTokens);
|
|
return true;
|
|
}
|
|
|
|
void cViewElement::Clear(void) {
|
|
ClearViewElement(ve);
|
|
}
|
|
|
|
void cViewElement::Action(void) {
|
|
DoSleep(delay);
|
|
if (!Running())
|
|
return;
|
|
Render();
|
|
SetInitFinished();
|
|
if (IsAnimated()) {
|
|
ShiftIn();
|
|
} else {
|
|
FadeIn();
|
|
}
|
|
DoFlush();
|
|
if (scrolling) {
|
|
DoSleep(scrollDelay);
|
|
if (!Running())
|
|
return;
|
|
if (scrollOrientation == orHorizontal) {
|
|
ScrollHorizontal(scrollingPix, scrollDelay, scrollSpeed, scrollMode);
|
|
} else {
|
|
ScrollVertical(scrollingPix, scrollDelay, scrollSpeed);
|
|
}
|
|
}
|
|
}
|
|
|
|
void cViewElement::ClearTokens(void) {
|
|
stringTokens.clear();
|
|
intTokens.clear();
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* cViewListItem
|
|
************************************************************************/
|
|
|
|
cViewListItem::cViewListItem(cTemplateViewElement *tmplItem) : cView(tmplItem) {
|
|
fadeOut = false;
|
|
pos = -1;
|
|
numTotal = 0;
|
|
align = alLeft;
|
|
listOrientation = orVertical;
|
|
}
|
|
|
|
cViewListItem::~cViewListItem() {
|
|
|
|
}
|
|
|
|
cRect cViewListItem::DrawListItem(map <string,string> *stringTokens, map <string,int> *intTokens) {
|
|
cRect posItem;
|
|
if (!tmplViewElement)
|
|
return posItem;
|
|
|
|
if (tmplViewElement->DebugTokens()) {
|
|
DebugTokens("ListItem", stringTokens, intTokens);
|
|
}
|
|
|
|
tmplViewElement->InitPixmapIterator();
|
|
int pixCurrent = 0;
|
|
cTemplatePixmapNode *pixNode = NULL;
|
|
while(pixNode = tmplViewElement->GetNextPixmapNode()) {
|
|
cTemplatePixmap *pix = dynamic_cast<cTemplatePixmap*>(pixNode);
|
|
if (pix) {
|
|
SetListElementPosition(pix);
|
|
if (pixCurrent == 0) {
|
|
posItem = pix->GetPixmapSize();
|
|
}
|
|
bool draw = PrepareListItemPixmap(pixCurrent, pix, stringTokens, intTokens);
|
|
if (draw) {
|
|
DrawPixmap(pixCurrent, pix);
|
|
}
|
|
pixCurrent++;
|
|
} else {
|
|
cTemplatePixmapContainer *pixContainer = dynamic_cast<cTemplatePixmapContainer*>(pixNode);
|
|
pixContainer->ParseDynamicParameters(stringTokens, intTokens);
|
|
if (pixContainer->DoDebug()) {
|
|
pixContainer->Debug();
|
|
}
|
|
if (!pixContainer->DoExecute()) {
|
|
pixCurrent += pixContainer->NumPixmaps();
|
|
continue;
|
|
}
|
|
pixContainer->InitIterator();
|
|
cTemplatePixmap *pix = NULL;
|
|
while (pix = pixContainer->GetNextPixmap()) {
|
|
SetListElementPosition(pix);
|
|
bool draw = PrepareListItemPixmap(pixCurrent, pix, stringTokens, intTokens);
|
|
if (draw) {
|
|
DrawPixmap(pixCurrent, pix);
|
|
}
|
|
pixCurrent++;
|
|
}
|
|
}
|
|
}
|
|
return posItem;
|
|
}
|
|
|
|
bool cViewListItem::PrepareListItemPixmap(int num, cTemplatePixmap *pix, map <string,string> *stringTokens, map <string,int> *intTokens) {
|
|
if (!PixmapExists(num)) {
|
|
pix->ParseDynamicParameters(stringTokens, intTokens, true);
|
|
} else {
|
|
pix->ParseDynamicParameters(stringTokens, intTokens, false);
|
|
}
|
|
if (!PixmapExists(num) && pix->Ready() && pix->DoExecute() && !pix->Scrolling()) {
|
|
CreateViewPixmap(num, pix);
|
|
}
|
|
//if pixmap still not valid, skip
|
|
if (!pix->Ready() && !pix->Scrolling()) {
|
|
return false;
|
|
}
|
|
//if condition for pixmap set, check if cond is true
|
|
if (!pix->DoExecute()) {
|
|
return false;
|
|
}
|
|
|
|
pix->ClearDynamicFunctionParameters();
|
|
pix->ParseDynamicFunctionParameters(stringTokens, intTokens, NULL);
|
|
|
|
if (!PixmapExists(num) && pix->Scrolling()) {
|
|
cSize drawportSize;
|
|
scrolling = pix->CalculateDrawPortSize(drawportSize);
|
|
pix->SetScrollingTextWidth();
|
|
if (scrolling) {
|
|
CreateScrollingPixmap(num, pix, drawportSize);
|
|
scrollingPix = num;
|
|
scrollOrientation = pix->GetNumericParameter(ptOrientation);
|
|
scrollMode = pix->GetNumericParameter(ptScrollMode);
|
|
scrollDelay = pix->GetNumericParameter(ptDelay);
|
|
scrollSpeed = pix->GetNumericParameter(ptScrollSpeed);
|
|
} else {
|
|
CreateViewPixmap(num, pix);
|
|
}
|
|
}
|
|
if (pix->DoDebug()) {
|
|
pix->Debug();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void cViewListItem::ClearListItem(void) {
|
|
int pixMax = NumPixmaps();
|
|
for (int pixCurrent = 0; pixCurrent < pixMax; pixCurrent++) {
|
|
Fill(pixCurrent, clrTransparent);
|
|
}
|
|
ClearAnimations(0);
|
|
}
|
|
|
|
void cViewListItem::SetListElementPosition(cTemplatePixmap *pix) {
|
|
int itemWidth = pix->GetNumericParameter(ptWidth);
|
|
int itemHeight = pix->GetNumericParameter(ptHeight);
|
|
int x = 0;
|
|
int y = 0;
|
|
if (listOrientation == orHorizontal) {
|
|
x = container.X();
|
|
int totalWidth = numTotal * itemWidth;
|
|
if (align == alCenter) {
|
|
y += (container.Width() - totalWidth) / 2;
|
|
} else if (align == alBottom) {
|
|
y += (container.Width() - totalWidth);
|
|
}
|
|
x += pos * itemWidth;
|
|
y = pix->GetNumericParameter(ptY);
|
|
} else if (listOrientation == orVertical) {
|
|
y = container.Y();
|
|
int totalHeight = numTotal * itemHeight;
|
|
if (align == alCenter) {
|
|
y += (container.Height() - totalHeight) / 2;
|
|
} else if (align == alBottom) {
|
|
y += (container.Height() - totalHeight);
|
|
}
|
|
y += pos * itemHeight;
|
|
x = pix->GetNumericParameter(ptX);
|
|
}
|
|
pix->SetX(x);
|
|
pix->SetY(y);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* cGrid
|
|
************************************************************************/
|
|
|
|
cGrid::cGrid(cTemplateViewElement *tmplGrid) : cView(tmplGrid) {
|
|
fadeOut = false;
|
|
dirty = true;
|
|
moved = true;
|
|
resized = true;
|
|
current = false;
|
|
x = 0.0;
|
|
y = 0.0;
|
|
width = 0.0;
|
|
height = 0.0;
|
|
}
|
|
|
|
cGrid::~cGrid() {
|
|
}
|
|
|
|
void cGrid::Set(double x, double y, double width, double height,
|
|
map <string,int> *intTokens, map <string,string> *stringTokens) {
|
|
|
|
if ((width != this->width) || (height != this->height)) {
|
|
this->width = width;
|
|
this->height = height;
|
|
resized = true;
|
|
dirty = false;
|
|
} else {
|
|
resized = false;
|
|
}
|
|
if (this->x != x || this->y != y) {
|
|
this->x = x;
|
|
this->y = y;
|
|
moved = true;
|
|
} else {
|
|
moved = false;
|
|
}
|
|
if (intTokens) {
|
|
this->intTokens = *intTokens;
|
|
SetCurrent(current);
|
|
dirty = true;
|
|
}
|
|
if (stringTokens) {
|
|
this->stringTokens = *stringTokens;
|
|
dirty = true;
|
|
}
|
|
}
|
|
|
|
void cGrid::SetCurrent(bool current) {
|
|
this->current = current;
|
|
if (!resized)
|
|
dirty = true;
|
|
intTokens.erase("current");
|
|
intTokens.insert(pair<string,int>("current", current));
|
|
}
|
|
|
|
void cGrid::Move(void) {
|
|
if (!tmplViewElement)
|
|
return;
|
|
tmplViewElement->InitPixmapIterator();
|
|
cTemplatePixmap *pix = NULL;
|
|
int pixCurrent = 0;
|
|
|
|
while(pix = tmplViewElement->GetNextPixmap()) {
|
|
PositionPixmap(pix);
|
|
cRect pixViewPort = pix->GetPixmapSize();
|
|
SetViewPort(pixCurrent, pixViewPort);
|
|
pixCurrent++;
|
|
}
|
|
dirty = false;
|
|
resized = false;
|
|
moved = false;
|
|
}
|
|
|
|
void cGrid::Draw(void) {
|
|
if (!tmplViewElement)
|
|
return;
|
|
if (tmplViewElement->DebugTokens()) {
|
|
DebugTokens("Grid", &stringTokens, &intTokens);
|
|
}
|
|
|
|
tmplViewElement->InitPixmapIterator();
|
|
int pixCurrent = 0;
|
|
cTemplatePixmapNode *pixNode = NULL;
|
|
while(pixNode = tmplViewElement->GetNextPixmapNode()) {
|
|
cTemplatePixmap *pix = dynamic_cast<cTemplatePixmap*>(pixNode);
|
|
if (pix) {
|
|
PositionPixmap(pix);
|
|
bool draw = PrepareGridPixmap(pixCurrent, pix);
|
|
if (pix->DoDebug())
|
|
pix->Debug();
|
|
if (draw) {
|
|
DrawPixmap(pixCurrent, pix);
|
|
}
|
|
pixCurrent++;
|
|
} else {
|
|
cTemplatePixmapContainer *pixContainer = dynamic_cast<cTemplatePixmapContainer*>(pixNode);
|
|
pixContainer->ParseDynamicParameters(&stringTokens, &intTokens);
|
|
if (pixContainer->DoDebug()) {
|
|
pixContainer->Debug();
|
|
}
|
|
if (!pixContainer->DoExecute()) {
|
|
pixCurrent += pixContainer->NumPixmaps();
|
|
continue;
|
|
}
|
|
pixContainer->InitIterator();
|
|
cTemplatePixmap *pix = NULL;
|
|
while (pix = pixContainer->GetNextPixmap()) {
|
|
PositionPixmap(pix);
|
|
bool draw = PrepareGridPixmap(pixCurrent, pix);
|
|
if (pix->DoDebug())
|
|
pix->Debug();
|
|
if (draw) {
|
|
DrawPixmap(pixCurrent, pix);
|
|
}
|
|
pixCurrent++;
|
|
}
|
|
|
|
|
|
}
|
|
}
|
|
dirty = false;
|
|
resized = false;
|
|
moved = false;
|
|
}
|
|
|
|
bool cGrid::PrepareGridPixmap(int num, cTemplatePixmap *pix) {
|
|
if (!PixmapExists(num)) {
|
|
pix->ParseDynamicParameters(&stringTokens, &intTokens, true);
|
|
} else {
|
|
pix->ParseDynamicParameters(&stringTokens, &intTokens, false);
|
|
}
|
|
if (!PixmapExists(num) && pix->Ready() && pix->DoExecute() && !pix->Scrolling()) {
|
|
CreateViewPixmap(num, pix);
|
|
}
|
|
//if pixmap still not valid, skip
|
|
if (!pix->Ready() && !pix->Scrolling()) {
|
|
return false;
|
|
}
|
|
//if condition for pixmap set, check if cond is true
|
|
if (!pix->DoExecute()) {
|
|
return false;
|
|
}
|
|
pix->ClearDynamicFunctionParameters();
|
|
pix->ParseDynamicFunctionParameters(&stringTokens, &intTokens, NULL);
|
|
return true;
|
|
}
|
|
|
|
void cGrid::Clear(void) {
|
|
int pixMax = NumPixmaps();
|
|
for (int pixCurrent = 0; pixCurrent < pixMax; pixCurrent++) {
|
|
Fill(pixCurrent, clrTransparent);
|
|
}
|
|
ClearAnimations(0);
|
|
}
|
|
|
|
void cGrid::DeletePixmaps(void) {
|
|
int pixMax = NumPixmaps();
|
|
for (int pixCurrent = 0; pixCurrent < pixMax; pixCurrent++) {
|
|
DestroyPixmap(pixCurrent);
|
|
}
|
|
ClearAnimations(0);
|
|
}
|
|
|
|
void cGrid::PositionPixmap(cTemplatePixmap *pix) {
|
|
pix->SetXPercent(x);
|
|
pix->SetYPercent(y);
|
|
pix->SetWidthPercent(width);
|
|
pix->SetHeightPercent(height);
|
|
pix->CalculateParameters();
|
|
}
|