1ee2cddf1a
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@27795 a1c6a512-1295-4272-9138-f99709370657
460 lines
13 KiB
C++
460 lines
13 KiB
C++
/***************************************************************************
|
|
* __________ __ ___.
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
* \/ \/ \/ \/ \/
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2010 Robert Bieber
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
* KIND, either express or implied.
|
|
*
|
|
****************************************************************************/
|
|
|
|
#include <QPainter>
|
|
#include <QPainterPath>
|
|
#include <QGraphicsSceneMouseEvent>
|
|
#include <QTransform>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <cmath>
|
|
|
|
#include "rbviewport.h"
|
|
#include "rbscreen.h"
|
|
#include "rbrenderinfo.h"
|
|
#include "parsetreemodel.h"
|
|
#include "tag_table.h"
|
|
#include "skin_parser.h"
|
|
#include "skindocument.h"
|
|
|
|
/* Pixels/second of text scrolling */
|
|
const double RBViewport::scrollRate = 30;
|
|
|
|
RBViewport::RBViewport(skin_element* node, const RBRenderInfo& info,
|
|
ParseTreeNode* pNode)
|
|
: RBMovable(info.screen()), foreground(info.screen()->foreground()),
|
|
background(info.screen()->background()), textOffset(0,0),
|
|
screen(info.screen()), textAlign(Left), showStatusBar(false),
|
|
statusBarTexture(":/render/statusbar.png"),
|
|
leftGraphic(0), centerGraphic(0), rightGraphic(0), scrollTime(0),
|
|
node(pNode), doc(info.document())
|
|
{
|
|
mirrored = info.screen()->isRtlMirrored()
|
|
&& info.device()->data("rtl").toBool();
|
|
|
|
if(!node->tag)
|
|
{
|
|
/* Default viewport takes up the entire screen */
|
|
size = QRectF(0, 0, info.screen()->getWidth(),
|
|
info.screen()->getHeight());
|
|
customUI = false;
|
|
font = screen->getFont(1);
|
|
|
|
screen->setDefault(this);
|
|
|
|
if(screen->getCustomUI())
|
|
{
|
|
RBViewport* cui = screen->getCustomUI();
|
|
size = cui->boundingRect();
|
|
setPos(cui->pos());
|
|
|
|
}
|
|
|
|
/* Making sure the default viewport can't be graphically manipulated */
|
|
setFlag(ItemIsSelectable, false);
|
|
setFlag(ItemIsMovable, false);
|
|
|
|
if(info.model()->rowCount(QModelIndex()) > 1)
|
|
{
|
|
/* If there is more than one viewport in the document */
|
|
textOffset.setX(-1);
|
|
}
|
|
else
|
|
{
|
|
setVisible(true);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
QString ident;
|
|
int x,y,w,h;
|
|
/* Rendering one of the other types of viewport */
|
|
switch(node->tag->name[1])
|
|
{
|
|
case '\0':
|
|
customUI = false;
|
|
baseParam= 0;
|
|
break;
|
|
|
|
case 'l':
|
|
/* A preloaded viewport definition */
|
|
ident = node->params[0].data.text;
|
|
customUI = false;
|
|
if(!screen->viewPortDisplayed(ident))
|
|
hide();
|
|
info.screen()->loadViewport(ident, this);
|
|
baseParam= 1;
|
|
break;
|
|
|
|
case 'i':
|
|
/* Custom UI Viewport */
|
|
customUI = true;
|
|
baseParam= 1;
|
|
if(node->params[0].type == skin_tag_parameter::DEFAULT)
|
|
{
|
|
setVisible(true);
|
|
}
|
|
else
|
|
{
|
|
hide();
|
|
info.screen()->loadViewport(ident, this);
|
|
}
|
|
break;
|
|
}
|
|
/* Now we grab the info common to all viewports */
|
|
int param = baseParam;
|
|
x = node->params[param++].data.number;
|
|
if(x < 0)
|
|
x = info.screen()->boundingRect().right() + x;
|
|
y = node->params[param++].data.number;
|
|
if(y < 0)
|
|
y = info.screen()->boundingRect().bottom() + y;
|
|
|
|
if(node->params[param].type == skin_tag_parameter::DEFAULT)
|
|
w = info.screen()->getWidth() - x;
|
|
else
|
|
w = node->params[param].data.number;
|
|
if(w < 0)
|
|
w = info.screen()->getWidth() + w - x;
|
|
|
|
if(node->params[++param].type == skin_tag_parameter::DEFAULT)
|
|
h = info.screen()->getHeight() - y;
|
|
else
|
|
h = node->params[param].data.number;
|
|
if(h < 0)
|
|
h = info.screen()->getHeight() + h - y;
|
|
|
|
/* Adjusting to screen coordinates if necessary */
|
|
if(screen->parentItem() != 0)
|
|
{
|
|
x -= screen->parentItem()->pos().x();
|
|
y -= screen->parentItem()->pos().y();
|
|
}
|
|
|
|
/* Mirroring if necessary */
|
|
if(mirrored)
|
|
{
|
|
x = parentItem()->boundingRect().width() - w - x;
|
|
}
|
|
|
|
if(node->params[++param].type == skin_tag_parameter::DEFAULT)
|
|
font = screen->getFont(1);
|
|
else
|
|
font = screen->getFont(node->params[param].data.number);
|
|
|
|
setPos(x, y);
|
|
size = QRectF(0, 0, w, h);
|
|
}
|
|
|
|
debug = info.device()->data("showviewports").toBool();
|
|
lineHeight = font->lineHeight();
|
|
if(customUI)
|
|
screen->setCustomUI(this);
|
|
}
|
|
|
|
RBViewport::~RBViewport()
|
|
{
|
|
}
|
|
|
|
QPainterPath RBViewport::shape() const
|
|
{
|
|
QPainterPath retval;
|
|
retval.addRect(size);
|
|
return retval;
|
|
}
|
|
|
|
void RBViewport::paint(QPainter *painter,
|
|
const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
if(!screen->hasBackdrop() && background != screen->background())
|
|
{
|
|
painter->fillRect(size, QBrush(background));
|
|
}
|
|
|
|
painter->setBrush(Qt::NoBrush);
|
|
painter->setPen(customUI ? Qt::blue : Qt::red);
|
|
if(debug)
|
|
painter->drawRect(size);
|
|
|
|
if(showStatusBar)
|
|
painter->fillRect(QRectF(0, 0, size.width(), 8), statusBarTexture);
|
|
|
|
RBMovable::paint(painter, option, widget);
|
|
}
|
|
|
|
void RBViewport::newLine()
|
|
{
|
|
if(textOffset.x() < 0)
|
|
return;
|
|
|
|
if(leftText != "")
|
|
alignLeft();
|
|
|
|
if(centerText != "")
|
|
alignCenter();
|
|
|
|
if(rightText != "")
|
|
alignRight();
|
|
|
|
textOffset.setY(textOffset.y() + lineHeight);
|
|
textOffset.setX(0);
|
|
textAlign = Left;
|
|
|
|
leftText.clear();
|
|
rightText.clear();
|
|
centerText.clear();
|
|
|
|
leftGraphic = 0;
|
|
centerGraphic = 0;
|
|
rightGraphic = 0;
|
|
|
|
scrollTime = 0;
|
|
}
|
|
|
|
void RBViewport::write(QString text)
|
|
{
|
|
if(textOffset.x() < 0)
|
|
return;
|
|
|
|
Alignment align = textAlign;
|
|
if(mirrored && align == Left)
|
|
align = Right;
|
|
else if(mirrored && align == Right)
|
|
align = Left;
|
|
|
|
if(align == Left)
|
|
{
|
|
leftText.append(text);
|
|
}
|
|
else if(align == Center)
|
|
{
|
|
centerText.append(text);
|
|
}
|
|
else if(align == Right)
|
|
{
|
|
rightText.append(text);
|
|
}
|
|
}
|
|
|
|
void RBViewport::showPlaylist(const RBRenderInfo &info, int start,
|
|
skin_element *id3, skin_element *noId3)
|
|
{
|
|
/* Determining whether ID3 info is available */
|
|
skin_element* root = id3;
|
|
|
|
/* The line will be a linked list */
|
|
if(root->children_count > 0)
|
|
root = root->children[0];
|
|
|
|
int song = start + info.device()->data("pp").toInt();
|
|
int numSongs = info.device()->data("pe").toInt();
|
|
int halfWay = (numSongs - song) / 2 + 1 + song;
|
|
|
|
while(song <= numSongs && textOffset.y() + lineHeight < size.height())
|
|
{
|
|
if(song == halfWay)
|
|
{
|
|
root = noId3;
|
|
if(root->children_count > 0)
|
|
root = root->children[0];
|
|
}
|
|
skin_element* current = root;
|
|
while(current)
|
|
{
|
|
|
|
if(current->type == TEXT)
|
|
{
|
|
write(QString((char*)current->data));
|
|
}
|
|
|
|
if(current->type == TAG)
|
|
{
|
|
QString tag(current->tag->name);
|
|
if(tag == "pp")
|
|
{
|
|
write(QString::number(song));
|
|
}
|
|
else if(tag == "pt")
|
|
{
|
|
write(QObject::tr("00:00"));
|
|
}
|
|
else if(tag[0] == 'i' || tag[0] == 'f')
|
|
{
|
|
if(song == info.device()->data("pp").toInt())
|
|
{
|
|
write(info.device()->data(tag).toString());
|
|
}
|
|
else
|
|
{
|
|
/* If we're not on the current track, use the next
|
|
* track info
|
|
*/
|
|
if(tag[0] == 'i')
|
|
tag = QString("I") + tag.right(1);
|
|
else
|
|
tag = QString("F") + tag.right(1);
|
|
write(info.device()->data(tag).toString());
|
|
}
|
|
}
|
|
}
|
|
|
|
current = current->next;
|
|
}
|
|
newLine();
|
|
song++;
|
|
}
|
|
}
|
|
|
|
void RBViewport::makeFullScreen()
|
|
{
|
|
size = screen->boundingRect();
|
|
setPos(screen->pos());
|
|
}
|
|
|
|
void RBViewport::saveGeometry()
|
|
{
|
|
QRectF bounds = boundingRect();
|
|
QPointF origin = pos();
|
|
|
|
node->modParam(static_cast<int>(origin.x()), baseParam);
|
|
node->modParam(static_cast<int>(origin.y()), baseParam + 1);
|
|
node->modParam(static_cast<int>(bounds.width()), baseParam + 2);
|
|
node->modParam(static_cast<int>(bounds.height()), baseParam + 3);
|
|
}
|
|
|
|
void RBViewport::alignLeft()
|
|
{
|
|
int y = textOffset.y();
|
|
|
|
if(leftGraphic)
|
|
delete leftGraphic;
|
|
|
|
leftGraphic = font->renderText(leftText, foreground, size.width(), this);
|
|
leftGraphic->setPos(0, y);
|
|
|
|
/* Setting scroll position if necessary */
|
|
int difference = leftGraphic->realWidth()
|
|
- leftGraphic->boundingRect().width();
|
|
if(difference > 0)
|
|
{
|
|
/* Subtracting out complete cycles */
|
|
double totalTime = 2 * difference / scrollRate;
|
|
scrollTime -= totalTime * std::floor(scrollTime / totalTime);
|
|
|
|
/* Calculating the offset */
|
|
if(scrollTime < static_cast<double>(difference) / scrollRate)
|
|
{
|
|
leftGraphic->setOffset(scrollRate * scrollTime);
|
|
}
|
|
else
|
|
{
|
|
scrollTime -= static_cast<double>(difference) / scrollRate;
|
|
leftGraphic->setOffset(difference - scrollRate * scrollTime);
|
|
}
|
|
}
|
|
}
|
|
|
|
void RBViewport::alignCenter()
|
|
{
|
|
int y = textOffset.y();
|
|
int x = 0;
|
|
|
|
if(centerGraphic)
|
|
delete centerGraphic;
|
|
|
|
centerGraphic = font->renderText(centerText, foreground, size.width(),
|
|
this);
|
|
|
|
if(centerGraphic->boundingRect().width() < size.width())
|
|
{
|
|
x = size.width() - centerGraphic->boundingRect().width();
|
|
x /= 2;
|
|
}
|
|
else
|
|
{
|
|
x = 0;
|
|
}
|
|
|
|
centerGraphic->setPos(x, y);
|
|
|
|
/* Setting scroll position if necessary */
|
|
int difference = centerGraphic->realWidth()
|
|
- centerGraphic->boundingRect().width();
|
|
if(difference > 0)
|
|
{
|
|
/* Subtracting out complete cycles */
|
|
double totalTime = 2 * difference / scrollRate;
|
|
scrollTime -= totalTime * std::floor(scrollTime / totalTime);
|
|
|
|
/* Calculating the offset */
|
|
if(scrollTime < static_cast<double>(difference) / scrollRate)
|
|
{
|
|
centerGraphic->setOffset(scrollRate * scrollTime);
|
|
}
|
|
else
|
|
{
|
|
scrollTime -= static_cast<double>(difference) / scrollRate;
|
|
centerGraphic->setOffset(difference - scrollRate * scrollTime);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
void RBViewport::alignRight()
|
|
{
|
|
int y = textOffset.y();
|
|
int x = 0;
|
|
|
|
if(rightGraphic)
|
|
delete rightGraphic;
|
|
|
|
rightGraphic = font->renderText(rightText, foreground, size.width(), this);
|
|
|
|
if(rightGraphic->boundingRect().width() < size.width())
|
|
x = size.width() - rightGraphic->boundingRect().width();
|
|
else
|
|
x = 0;
|
|
|
|
rightGraphic->setPos(x, y);
|
|
|
|
/* Setting scroll position if necessary */
|
|
int difference = rightGraphic->realWidth()
|
|
- rightGraphic->boundingRect().width();
|
|
if(difference > 0)
|
|
{
|
|
/* Subtracting out complete cycles */
|
|
double totalTime = 2 * difference / scrollRate;
|
|
scrollTime -= totalTime * std::floor(scrollTime / totalTime);
|
|
|
|
/* Calculating the offset */
|
|
if(scrollTime < static_cast<double>(difference) / scrollRate)
|
|
{
|
|
rightGraphic->setOffset(scrollRate * scrollTime);
|
|
}
|
|
else
|
|
{
|
|
scrollTime -= static_cast<double>(difference) / scrollRate;
|
|
rightGraphic->setOffset(difference - scrollRate * scrollTime);
|
|
}
|
|
}
|
|
}
|
|
|