src/hbcore/utils/hbsmileytheme.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 17 Sep 2010 08:32:10 +0300
changeset 28 b7da29130b0e
parent 21 4633027730f5
permissions -rw-r--r--
Revision: 201035 Kit: 201037

/****************************************************************************
**
** 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 "hbsmileytheme.h"
#include "hbsmileythemeparser_p.h"
#include <QVariant>
#include <QSharedData>
#include <QHash>
#include <QSet>


class HbSmileyThemePrivate : public QSharedData
{
public:

    HbSmileyThemePrivate();
    HbSmileyThemePrivate(const HbSmileyThemePrivate &other);
    ~HbSmileyThemePrivate();

public: // data

    QHash<QString,QSet<QString> > smileyToPatterns;
    QHash <QString, QString> patternToSmiley;
    mutable QRegExp regexp;
    mutable bool dirty;
};

HbSmileyThemePrivate::HbSmileyThemePrivate()
{
}

HbSmileyThemePrivate::HbSmileyThemePrivate(const HbSmileyThemePrivate &other) : QSharedData(other),
        smileyToPatterns(other.smileyToPatterns),
        patternToSmiley(other.patternToSmiley),
        dirty(true)
{
}

HbSmileyThemePrivate::~HbSmileyThemePrivate()
{
}


/*!
    @beta
    @hbcore
    \class HbSmileyTheme
    \brief HbSmileyTheme is container class for defining a smiley theme.

    Smiley is a sequence of keyboard characters used to represent a person's mood or emotion, especially :) or :-)
    or other smiling depiction. Smileys are also represented by graphics so called emoticons.
    In HbSmileyTheme \a smiley refers to emoticons and \a pattern refers to character sequence representation of a smiley.
    In a smiley theme each smiley is associated with one or multiple patterns. Each smiley is identified by a string that
    is the file name of the graphics file representing the corresponding emoticon.

    This class implements copy-on-write semantics, so copy constructor and assignment operator
    only perform shallow copy and are fast. A deep copy is done if the theme properties are changed.

    Brief example of how to create a smiley themes and use it.

    Example usage:
    \dontinclude ultimatecodesnippet/ultimatecodesnippet.cpp
    \skip Start of snippet 46
    \until End of snippet 46

    Smiley themes can be loaded from smiley theme (.sml) files.

    Example usage:
    \dontinclude ultimatecodesnippet/ultimatecodesnippet.cpp
    \skip Start of snippet 47
    \until End of snippet 47

    Here is an example of a smiley theme file describing the file syntax.

    \include unittest_hbsmileytheme/my_smileys_theme.sml


    \sa HbAbstractEdit
*/


/*!
* Default constructor. Constructs a null smiley theme.
*
*  \sa isNull()
*/
HbSmileyTheme::HbSmileyTheme()
{
    d = new HbSmileyThemePrivate();
}

/*!
* Constructs a smiley theme from a smiley theme definition file.
* If the file does not exist or is of an unknown format, the smiley theme becomes a null theme.
*
*  \sa isNull() load()
*/
HbSmileyTheme::HbSmileyTheme(const QString &fileName)
{
    d = new HbSmileyThemePrivate();
    load(fileName);
}


/*!
*   Constructs a smiley theme that is a copy of the given smiley theme \a other.
*/
HbSmileyTheme::HbSmileyTheme(const HbSmileyTheme &other) : d(other.d)
{
}

/*!
*   Assigns the given smiley theme \a other to this smiley theme and returns a reference
*   to this smiley theme.
*
*   \sa HbSmileyTheme()
*/
HbSmileyTheme &HbSmileyTheme::operator=(const HbSmileyTheme &other)
{
    if (&other != this) {
        d = other.d;
    }
    return *this;
}

/*!
* Destroys the smiley theme.
*/
HbSmileyTheme::~HbSmileyTheme()
{
}

/*!
* Loads a smiley theme from a smiley theme definition file
* Returns true if the smiley theme was successfully loaded; otherwise returns false.
*/
bool HbSmileyTheme::load (const QString &fileName)
{
    HbSmileyTheme tmpTheme;
    HbSmileyThemeParser parser;

    bool ret = parser.parseDefinitionFile(fileName,tmpTheme);
    if (ret) {
        *this = tmpTheme;
    }

    d->dirty = true;

    return ret;
}


/*!
* Returns true if the smiley theme doesn't contain any smileys.
* This is the case when the smiley theme is default constructed.
*/
bool HbSmileyTheme::isNull() const
{
    return d->smileyToPatterns.isEmpty();
}

/*!
* Inserts a smiley to the smiley theme with a given pattern.
* \a smiley identifies smiley and represents the icon name of the smiley.
* It can be any name that is valid for an icon name for HbIcon.
*
* \a pattern is a textual representation of the smiley.
* If the smiley already exists with \a smiley in the theme then \a pattern will be
* an additional textual representation for the smiley.
*
* \sa remove() replace()
*/
void HbSmileyTheme::insert(const QString &smiley, const QString &pattern)
{
    if (!smiley.isEmpty() && !pattern.isEmpty()) {
        d.detach();
        QSet<QString> patterns = d->smileyToPatterns.value(smiley);
        patterns << pattern;
        d->smileyToPatterns[smiley] = patterns;
        d->patternToSmiley[pattern] = smiley;
        d->dirty = true;
    }    
}


/*!
* Inserts a smiley to the smiley theme with a given pattern list.
* \a smiley identifies smiley and represents the icon name of the smiley.
* It can be any name that is valid for an icon name for HbIcon.
*
*
* \a patterns is a list of textual representation of the smiley.
* If the smiley already exists with \a smiley in the theme then \a pattern list will be
* additional textual representations for the smiley.
*
* \sa remove() replace()
*/
void HbSmileyTheme::insert(const QString &smiley, const QStringList &patterns)
{
    foreach(QString pattern, patterns) {
        insert(smiley,pattern);
    }
}

/*!
* Replaces the smiley patterns for \a smiley.
* This is equivalent to calling remove(smiley) and subsequently insert(smiley,patterns).
*
* \sa remove() insert()
*/
void HbSmileyTheme::replace(const QString &smiley, const QStringList &patterns)
{
    remove(smiley);
    insert(smiley,patterns);
}

/*!
* Replaces the smiley patterns for \a smiley with a single \a pattern.
* This is equivalent to calling remove(smiley) and subsequently insert(smiley,pattern).
*
*
* \sa remove() insert()
*/
void HbSmileyTheme::replace(const QString &smiley, const QString &pattern)
{
    remove(smiley);
    insert(smiley,pattern);
}

/*!
* Removes \a smiley from the smiley theme along with its associated patterns.
*
* \sa insert() replace() removePattern()
*/
void HbSmileyTheme::remove(const QString &smiley)
{
    QSet<QString> tmpPatterns = d->smileyToPatterns.value(smiley);
    if (!tmpPatterns.isEmpty()) {
        d.detach();
        d->smileyToPatterns.remove(smiley);
        foreach(QString pattern, tmpPatterns) {
            d->patternToSmiley.remove(pattern);
        }
        d->dirty = true;
    }
}

/*!
* Removes \a pattern from the theme.
*
*
* \sa insert() replace() remove()
*/
void HbSmileyTheme::removePattern(const QString &pattern)
{
    QString smiley = d->patternToSmiley.value(pattern);
    if (!smiley.isEmpty()) {
        QSet<QString> tmpPatterns = d->smileyToPatterns.value(smiley);
        if (!tmpPatterns.isEmpty()) {
            tmpPatterns.remove(pattern);
            d.detach();
            if (tmpPatterns.isEmpty()) {
                d->smileyToPatterns.remove(smiley);
            } else {
                d->smileyToPatterns[smiley] = tmpPatterns;
            }
            d->patternToSmiley.remove(pattern);
            d->dirty = true;
        }
    }
}

/*!
*
* Remove all smileys from the theme and sets the smiley theme to the same state as
* when default constructed.
*
* \sa isNull()
*/
void HbSmileyTheme::clear()
{
    if (!isNull()) {
        d.detach();
        d->smileyToPatterns.clear();
        d->patternToSmiley.clear();
        d->dirty = true;
    }
}

/*!
* Returns an existing smiley from the smiley theme for a given \a pattern
* If the smiley doesn't exist it returns a null string.
*
* \sa smileys()
*/
QString HbSmileyTheme::smiley(const QString &pattern) const
{
    return d->patternToSmiley.value(pattern);
}


/*!
* Returns the existing smileys from the smiley theme.
*
* \sa patterns()
*/
QStringList HbSmileyTheme::smileys() const
{
    return d->smileyToPatterns.keys();
}


/*!
* Returns the currently associated textual patterns for \a smiley.
*
* \sa smileys()
*/
QStringList HbSmileyTheme::patterns(const QString &smiley) const
{
    QSet<QString> tmpPatterns = d->smileyToPatterns.value(smiley);
    return tmpPatterns.toList();
}

/*!
* Convenience method.
* Returns the textual patterns for all the smileys in the theme.
*
* \sa smileys()
*/
QStringList HbSmileyTheme::patterns() const
{
    return d->patternToSmiley.keys();
}

/*!
* Returns the regular expression for matching the smiley patterns from the theme.
*
* \sa patterns()
*/
const QRegExp& HbSmileyTheme::regExp() const
{
    if (d->dirty) {
        QString regexpStr;
        foreach (QString pattern, patterns()) {
            regexpStr += QRegExp::escape(pattern) + "|";
        }
        regexpStr.remove(regexpStr.count()-1, 1);

        d->regexp.setPattern(regexpStr);
        d->dirty = false;
    }

    return d->regexp;
}

/*!
* Returns the smiley theme as a QVariant.
*/
HbSmileyTheme::operator QVariant() const
{
    return QVariant::fromValue(*this);
}