src/hbcore/image/hbiconanimationparser.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 14 May 2010 16:09:54 +0300
changeset 2 06ff229162e9
parent 0 16d8024aca5e
child 5 627c4a0fd0e7
permissions -rw-r--r--
Revision: 201017 Kit: 201019

/****************************************************************************
**
** Copyright (C) 2008-2010 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (developer.feedback@nokia.com)
**
** This file is part of the HbCore module of the UI Extensions for Mobile.
**
** GNU Lesser General Public License Usage
** This file may be used under the terms of the GNU Lesser General Public
** License version 2.1 as published by the Free Software Foundation and
** appearing in the file LICENSE.LGPL included in the packaging of this file.
** Please review the following information to ensure the GNU Lesser General
** Public License version 2.1 requirements will be met:
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights.  These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at developer.feedback@nokia.com.
**
****************************************************************************/

#include "hbiconanimationparser_p.h"
#include "hbiconanimationmanager_p.h"
#include "hbimagetraces_p.h"
#include "hbtheme_p.h"
#include "hbthemeclient_p.h"
#include "hbinstance.h"
#include <QDebug>
#include <QFile>
#include <QBuffer>
#include <QByteArray>
#include <QPair>
#include <QStack>

// Private class
HbIconAnimationParser::HbIconAnimationParser() :
    QXmlStreamReader()
{
}

HbIconAnimationParser::~HbIconAnimationParser()
{
}

bool HbIconAnimationParser::parseDefinitionFileShared(
    const QString &fileName, AnimHash &animations, const QString &realFileName)
{
#ifdef HB_ICON_TRACES
    qDebug() << "HbIconAnimationParser: Trying to get" << realFileName << "via themeserver";
#endif
    QByteArray arr = HbThemeClient::global()->getSharedBlob(realFileName);

    if (!arr.isEmpty()) {
#ifdef HB_ICON_TRACES
        qDebug() << "HbIconAnimationParser: Got data from themeserver for" << realFileName
                 << "size=" << arr.size();
#endif
        QBuffer buf(&arr);
        setDevice(&buf);
        return doParseFile(fileName, animations);
    }
#ifdef HB_ICON_TRACES
    qDebug() << "HbIconAnimationParser: Getting" << realFileName << "via themeserver failed";
#endif
    return false;
}

bool HbIconAnimationParser::parseDefinitionFile(
    const QString &fileName, AnimHash &animations, const QString &realFileName)
{
    QFile file(realFileName);

    if (!file.exists()) {
#ifdef HB_ICON_TRACES
        qDebug() << "HbIconAnimationParser: File does not exist:" << realFileName;
#endif
        return false;
    }

    if (!file.open(QFile::ReadOnly | QFile::Text)) {
#ifdef HB_ICON_TRACES
        qDebug() << "HbIconAnimationParser: File opening failed:" << realFileName;
#endif
        return false;
    }
    
    setDevice(&file);
    bool ret = doParseFile(fileName, animations);
    file.close();

    return ret;
}

bool HbIconAnimationParser::doParseFile(
    const QString &fileName, AnimHash &animations)
{
    bool ret = false;

    while (!atEnd()) {
        readNext();

        if (isStartElement()) {
            if (name() == "animations") {
                ret = readAnimations(fileName, animations);
            }
        }        
    }

#ifdef HB_ICON_TRACES
    qDebug() << "Parsing of" << *fileName << "finished with success code" << ret;
#endif

    return ret;
}

/*!
* Reads animation elements.
* Returns true if at least one animation definition was added in the hash.
*/
bool HbIconAnimationParser::readAnimations(
    const QString &fileName, AnimHash &animations)
{
    Q_ASSERT(isStartElement() && name() == "animations");

    bool ret = false;

    while (!atEnd()) {
        readNext();

        if (isEndElement()) {
            break;
        }

        if (isStartElement()) {
            if (name() == "icon") {
                HbIconAnimationData anim;
                anim.definitionFileName = fileName;

                // Read icon element
                QString iconName = readIconData(anim);

                if (!anim.def.isNull() && !iconName.isEmpty()) {
                    // Remove old definition to avoid duplicates
                    animations.remove(iconName);
                    // Insert the new animation definition in the hash
                    animations.insert(iconName, anim);
                    ret = true;
                }
            } else {
                readUnknownElement();
            }
        }
    }

    return ret;
}

/*!
 * Parses the icon data.
 */
QString HbIconAnimationParser::readIconData(HbIconAnimationData &data)
{
    Q_ASSERT(isStartElement() && name() == "icon");
    
    QString iconName;
    int defaultDuration = 0;
    QStack< QPair<int, int> > repStk;

    QXmlStreamAttributes attrs = attributes();

    // Get old frame list from definition. It should be empty at this point.
    QList<HbIconAnimationDefinition::AnimationFrame> frames = data.def.frameList();

    // Read icon attributes
    foreach (const QXmlStreamAttribute &attr, attrs) {
        // "name" = ...
        if (attr.name().toString() == "name") {
            iconName = attr.value().toString();
        } else if (attr.name().toString() == "playmode") {
            // Defaults to "playonce"
            if (attr.value().toString() == "loop") {
                data.def.setPlayMode(HbIconAnimationDefinition::Loop);
            }
        } else if (attr.name().toString() == "frame_duration") {
            // E.g. frame_duration="500". This means that 500 is used as the duration for
            // all frames for which the duration is not explicitly given.
            defaultDuration = attr.value().toString().toInt();
        }
    }

    // Read frame elements inside icon element.
    while (!atEnd()) {
        readNext();

        if (isEndElement()) {
            // Check if reached </icon>
            if (name() == "icon") {
                break;
            }
            // Check if reached </loop>
            if (name() == "loop") {
                QPair<int, int> repData = repStk.pop();
                // Assign these values to the last frame of the loop that has just ended.
                if (!frames.isEmpty()) {
                    // Instead of a single index/count pair there is a list because one
                    // frame may contain multiple jumps (when multiple </loop> elements
                    // come directly after each other).
                    HbIconAnimationDefinition::AnimationFrame::Jump jump;
                    jump.targetFrameIndex = repData.first;
                    jump.repeatCount = repData.second;
                    frames.last().jumps.append(jump);
                }
            }
        }

        if (isStartElement()) {

            // Frame element inside icon or loop element.
            if (name() == "frame") {
                HbIconAnimationDefinition::AnimationFrame frame;
                frame.duration = defaultDuration;

                QXmlStreamAttributes attrs = attributes();
                // Read attributes
                foreach (const QXmlStreamAttribute &attr, attrs) {
                    // duration attribute
                    if (attr.name().toString() == "duration") {
                        frame.duration = attr.value().toString().toInt();
                        if (frame.duration < 0) {
                            frame.duration = 0;
                        }
                    }
                }

                frame.iconName = readElementText();
                frames.append(frame);

            } else if (name() == "loop") {
                // <loop count="2">
                //     <frame src=":/graphicsitemdemo/frame2.svg"/>
                //     ...
                // </loop>
                int repeatCount = 0;
                QXmlStreamAttributes attrs = attributes();
                foreach (const QXmlStreamAttribute &attr, attrs) {
                    if (attr.name().toString() == "count") {
                        repeatCount = attr.value().toString().toInt();
                    }
                }
                // If count attribute is not given or is <= 0 then the loop element is ignored.
                if (repeatCount > 0) {
                    // Target frame index for this loop's last frame will be the index of
                    // the frame that comes next.
                    repStk.push(QPair<int, int>(frames.count(), repeatCount)); 
                }

            } else {
                readUnknownElement();
            }
        }
    }

    // Set framelist in the definition.
    data.def.setFrameList(frames);

    return iconName;
}

/*!
    Handles unknown elements.
*/
void HbIconAnimationParser::readUnknownElement()
{
    Q_ASSERT(isStartElement());

    while (!atEnd()) {
        readNext();

        if (isEndElement()) {
            break;
        }

        if (isStartElement()) {
            readUnknownElement();
        }
    }
}

// End of File