/*************************************************************************** * __________ __ ___. * 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 #include #include #include #include #include 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(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 (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(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(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 (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 (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 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)) { QFileInfo wpsfile(*file); QDir base(wpsfile.canonicalPath()); 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 children = root->children(); for(QList::iterator i = children.begin() ; i != children.end(); i++) setChildrenUnselectable(*i); }