5602342613
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@27775 a1c6a512-1295-4272-9138-f99709370657
397 lines
11 KiB
C++
397 lines
11 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 "parsetreemodel.h"
|
|
#include "symbols.h"
|
|
#include "rbscreen.h"
|
|
#include "rbrenderinfo.h"
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <QObject>
|
|
#include <QPixmap>
|
|
#include <QMap>
|
|
#include <QDir>
|
|
|
|
#include <iostream>
|
|
|
|
ParseTreeModel::ParseTreeModel(const char* document, QObject* parent):
|
|
QAbstractItemModel(parent), sbsModel(0)
|
|
{
|
|
this->tree = skin_parse(document);
|
|
|
|
if(tree)
|
|
this->root = new ParseTreeNode(tree, this);
|
|
else
|
|
this->root = 0;
|
|
|
|
scene = new RBScene();
|
|
}
|
|
|
|
|
|
ParseTreeModel::~ParseTreeModel()
|
|
{
|
|
if(root)
|
|
delete root;
|
|
if(tree)
|
|
skin_free_tree(tree);
|
|
if(sbsModel)
|
|
sbsModel->deleteLater();
|
|
}
|
|
|
|
QString ParseTreeModel::genCode()
|
|
{
|
|
if(root)
|
|
return root->genCode();
|
|
else
|
|
return "";
|
|
}
|
|
|
|
QString ParseTreeModel::changeTree(const char *document)
|
|
{
|
|
struct skin_element* test = skin_parse(document);
|
|
|
|
if(!test)
|
|
{
|
|
QString error = tr("Error on line ") +
|
|
QString::number(skin_error_line())
|
|
+ tr(", column ") + QString::number(skin_error_col())
|
|
+ tr(": ") + QString(skin_error_message());
|
|
return error;
|
|
}
|
|
|
|
ParseTreeNode* temp = new ParseTreeNode(test, this);
|
|
|
|
if(root)
|
|
{
|
|
emit beginRemoveRows(QModelIndex(), 0, root->numChildren() - 1);
|
|
delete root;
|
|
emit endRemoveRows();
|
|
}
|
|
|
|
root = temp;
|
|
|
|
emit beginInsertRows(QModelIndex(), 0, temp->numChildren() - 1);
|
|
emit endInsertRows();
|
|
|
|
return tr("Document Parses Successfully");
|
|
|
|
}
|
|
|
|
QModelIndex ParseTreeModel::index(int row, int column,
|
|
const QModelIndex& parent) const
|
|
{
|
|
if(!hasIndex(row, column, parent))
|
|
return QModelIndex();
|
|
|
|
ParseTreeNode* foundParent;
|
|
|
|
if(parent.isValid())
|
|
foundParent = static_cast<ParseTreeNode*>(parent.internalPointer());
|
|
else
|
|
foundParent = root;
|
|
|
|
if(row < foundParent->numChildren() && row >= 0)
|
|
return createIndex(row, column, foundParent->child(row));
|
|
else
|
|
return QModelIndex();
|
|
}
|
|
|
|
QModelIndex ParseTreeModel::parent(const QModelIndex &child) const
|
|
{
|
|
if(!child.isValid())
|
|
return QModelIndex();
|
|
|
|
ParseTreeNode* foundParent = static_cast<ParseTreeNode*>
|
|
(child.internalPointer())->getParent();
|
|
|
|
if(foundParent == root)
|
|
return QModelIndex();
|
|
|
|
return createIndex(foundParent->getRow(), 0, foundParent);
|
|
}
|
|
|
|
int ParseTreeModel::rowCount(const QModelIndex &parent) const
|
|
{
|
|
if(!root)
|
|
return 0;
|
|
|
|
if(!parent.isValid())
|
|
return root->numChildren();
|
|
|
|
if(parent.column() != typeColumn)
|
|
return 0;
|
|
|
|
return static_cast<ParseTreeNode*>(parent.internalPointer())->numChildren();
|
|
}
|
|
|
|
int ParseTreeModel::columnCount(const QModelIndex &parent) const
|
|
{
|
|
if(parent.isValid())
|
|
return numColumns;
|
|
else
|
|
return numColumns;
|
|
}
|
|
|
|
QVariant ParseTreeModel::data(const QModelIndex &index, int role) const
|
|
{
|
|
if(!index.isValid())
|
|
return QVariant();
|
|
|
|
if(role != Qt::DisplayRole)
|
|
return QVariant();
|
|
|
|
return static_cast<ParseTreeNode*>(index.internalPointer())->
|
|
data(index.column());
|
|
}
|
|
|
|
QVariant ParseTreeModel::headerData(int col, Qt::Orientation orientation,
|
|
int role) const
|
|
{
|
|
if(orientation != Qt::Horizontal)
|
|
return QVariant();
|
|
|
|
if(col >= numColumns)
|
|
return QVariant();
|
|
|
|
if(role != Qt::DisplayRole)
|
|
return QVariant();
|
|
|
|
switch(col)
|
|
{
|
|
case typeColumn:
|
|
return QObject::tr("Type");
|
|
|
|
case lineColumn:
|
|
return QObject::tr("Line");
|
|
|
|
case valueColumn:
|
|
return QObject::tr("Value");
|
|
}
|
|
|
|
return QVariant();
|
|
}
|
|
|
|
Qt::ItemFlags ParseTreeModel::flags(const QModelIndex &index) const
|
|
{
|
|
Qt::ItemFlags retval = Qt::ItemIsEnabled | Qt::ItemIsSelectable;
|
|
|
|
ParseTreeNode* element = static_cast<ParseTreeNode*>
|
|
(index.internalPointer());
|
|
|
|
if((element->isParam()
|
|
|| element->getElement()->type == TEXT
|
|
|| element->getElement()->type == COMMENT)
|
|
&& index.column() == valueColumn)
|
|
{
|
|
retval |= Qt::ItemIsEditable;
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
bool ParseTreeModel::setData(const QModelIndex &index, const QVariant &value,
|
|
int role)
|
|
{
|
|
if(role != Qt::EditRole)
|
|
return false;
|
|
|
|
if(index.column() != valueColumn)
|
|
return false;
|
|
|
|
ParseTreeNode* node = static_cast<ParseTreeNode*>
|
|
(index.internalPointer());
|
|
|
|
if(node->isParam())
|
|
{
|
|
struct skin_tag_parameter* param = node->getParam();
|
|
|
|
/* Now that we've established that we do, in fact, have a parameter,
|
|
* set it to its new value if an acceptable one has been entered
|
|
*/
|
|
if(value.toString().trimmed() == QString(QChar(DEFAULTSYM)))
|
|
{
|
|
if(islower(param->type_code))
|
|
param->type = skin_tag_parameter::DEFAULT;
|
|
else
|
|
return false;
|
|
}
|
|
else if(tolower(param->type_code) == 's'
|
|
|| tolower(param->type_code) == 'f')
|
|
{
|
|
if(param->type == skin_tag_parameter::STRING)
|
|
free(param->data.text);
|
|
|
|
param->type = skin_tag_parameter::STRING;
|
|
param->data.text = strdup(value.toString().trimmed().toAscii());
|
|
}
|
|
else if(tolower(param->type_code) == 'i')
|
|
{
|
|
if(!value.canConvert(QVariant::Int))
|
|
return false;
|
|
|
|
param->type = skin_tag_parameter::INTEGER;
|
|
param->data.number = value.toInt();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
struct skin_element* element = node->getElement();
|
|
|
|
if(element->type != COMMENT && element->type != TEXT)
|
|
return false;
|
|
|
|
free(element->data);
|
|
element->data = strdup(value.toString().trimmed().toAscii());
|
|
}
|
|
|
|
emit dataChanged(index, index);
|
|
return true;
|
|
}
|
|
|
|
RBScene* ParseTreeModel::render(ProjectModel* project,
|
|
DeviceState* device,
|
|
SkinDocument* doc, const QString* file)
|
|
{
|
|
scene->clear();
|
|
|
|
/* Setting the background */
|
|
scene->setBackgroundBrush(QBrush(QPixmap(":/render/scenebg.png")));
|
|
|
|
/* Preparing settings */
|
|
QMap<QString, QString> settings;
|
|
if(project)
|
|
settings = project->getSettings();
|
|
|
|
/* Setting themebase if it can't be derived from the project */
|
|
if(settings.value("themebase", "") == "" && file && QFile::exists(*file))
|
|
{
|
|
QDir base(*file);
|
|
base.cdUp();
|
|
settings.insert("themebase", base.canonicalPath());
|
|
}
|
|
|
|
/* Finding imagebase and determining remote/wps status */
|
|
bool remote = false;
|
|
bool wps = false;
|
|
if(file)
|
|
{
|
|
QString skinFile = *file;
|
|
QStringList decomp = skinFile.split("/");
|
|
skinFile = decomp[decomp.count() - 1];
|
|
skinFile.chop(skinFile.length() - skinFile.lastIndexOf("."));
|
|
settings.insert("imagepath", settings.value("themebase","") + "/wps/" +
|
|
skinFile);
|
|
|
|
decomp = file->split(".");
|
|
QString extension = decomp.last();
|
|
if(extension[0] == 'r')
|
|
remote = true;
|
|
if(extension.right(3).toLower() == "wps"
|
|
|| extension.right(3).toLower() == "fms")
|
|
wps = true;
|
|
}
|
|
|
|
/* Rendering SBS, if necessary */
|
|
RBScreen* screen = 0;
|
|
if(wps && device->data("rendersbs").toBool())
|
|
{
|
|
QString sbsFile = settings.value(remote ? "rsbs" : "sbs", "");
|
|
sbsFile.replace("/.rockbox" , settings.value("themebase",""));
|
|
|
|
if(QFile::exists(sbsFile))
|
|
{
|
|
QFile sbs(sbsFile);
|
|
sbs.open(QFile::ReadOnly | QFile::Text);
|
|
|
|
if(sbsModel)
|
|
sbsModel->deleteLater();
|
|
sbsModel = new ParseTreeModel(QString(sbs.readAll()).toAscii());
|
|
|
|
if(sbsModel->root != 0)
|
|
{
|
|
RBRenderInfo sbsInfo(sbsModel, project, doc, &settings, device,
|
|
screen);
|
|
|
|
screen = new RBScreen(sbsInfo, remote);
|
|
scene->addItem(screen);
|
|
|
|
sbsInfo = RBRenderInfo(sbsModel, project, doc, &settings,
|
|
device, screen);
|
|
sbsModel->root->render(sbsInfo);
|
|
screen->endSbsRender();
|
|
|
|
setChildrenUnselectable(screen);
|
|
}
|
|
}
|
|
}
|
|
|
|
RBRenderInfo info(this, project, doc, &settings, device, screen);
|
|
|
|
/* Adding the screen */
|
|
if(!screen)
|
|
{
|
|
screen = new RBScreen(info, remote);
|
|
scene->addItem(screen);
|
|
}
|
|
|
|
scene->setScreenSize(screen->boundingRect());
|
|
|
|
info = RBRenderInfo(this, project, doc, &settings, device, screen);
|
|
|
|
|
|
/* Rendering the tree */
|
|
if(root)
|
|
root->render(info);
|
|
|
|
return scene;
|
|
}
|
|
|
|
void ParseTreeModel::paramChanged(ParseTreeNode *param)
|
|
{
|
|
QModelIndex left = indexFromPointer(param);
|
|
QModelIndex right = createIndex(left.row(), 2, left.internalPointer());
|
|
emit dataChanged(left, right);
|
|
}
|
|
|
|
QModelIndex ParseTreeModel::indexFromPointer(ParseTreeNode *p)
|
|
{
|
|
/* Recursively finding an index for an arbitrary pointer within the tree */
|
|
if(!p->getParent())
|
|
return QModelIndex();
|
|
return index(p->getRow(), 0, indexFromPointer(p->getParent()));
|
|
}
|
|
|
|
void ParseTreeModel::setChildrenUnselectable(QGraphicsItem *root)
|
|
{
|
|
root->setFlag(QGraphicsItem::ItemIsSelectable, false);
|
|
root->setFlag(QGraphicsItem::ItemIsMovable, false);
|
|
|
|
QList<QGraphicsItem*> children = root->children();
|
|
for(QList<QGraphicsItem*>::iterator i = children.begin()
|
|
; i != children.end(); i++)
|
|
setChildrenUnselectable(*i);
|
|
}
|