/****************************************************************************
**
** 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 "hbbadgeicon_p.h"
#include "hbicon.h"
#include "hbbadgeiconinfo_p.h"
#include <QPainter>
/*!
@beta
@hbcore
\internal
\class HbBadgeIcon
\brief HbBadgeIcon represents zero or more badge icons to be drawn
over an HbIcon.
This class uses a QList<HbBadgeIconInfo> in z-order to store the badges
that must be applied when rendering an HbIcon.
This class is to be used only internally within HbCore. Furthermore,
This class is not intended to be a superclass for anything else.
*/
/*!
Default constructor; needs to do nothing.
\internal
*/
HbBadgeIcon::HbBadgeIcon()
{
// This space intentionally left blank
}
/*!
Destructor; no explicit work needs to be done here.
\internal
*/
HbBadgeIcon::~HbBadgeIcon()
{
// This space intentionally left blank
}
/*!
Add an icon to be drawn at the specified alignment in the
given z-plane after scaling a predetermined amount. The
badges are stored in increasing z-order value.
Note that scaling is here for API compatibility later, but this
implementation provides no scaling of the badge icon.
\internal
*/
void HbBadgeIcon::addBadge(Qt::Alignment alignment,
const HbIcon &icon,
int zValue,
const QSizeF &sizeFactor,
Qt::AspectRatioMode aspectRatio)
{
HbBadgeIconInfo info(icon, alignment, zValue, sizeFactor, aspectRatio);
int badgeCount = mBadgeIcons.count();
if (badgeCount == 0) {
mBadgeIcons.append(info);
} else {
bool added = false;
//Find a spot to insert the badgeinfo in the list.
for (int i = badgeCount - 1; i >= 0; i--) {
if (mBadgeIcons[i].zValue() > zValue) {
mBadgeIcons.insert(i + 1, info);
added = true;
break;
}
}
//Not added anywhere so put it at the top.
if (!added) {
mBadgeIcons.prepend(info);
}
}
}
/*!
Removes all badges with a specific \a _iconName as the HbIcon::iconName().
\internal
*/
bool HbBadgeIcon::removeBadge(const HbIcon &badge)
{
bool result = false;
// Could make this a binary search, but complexity isn't worth it for normal use cases.
int count = mBadgeIcons.count();
for (int i = count - 1; i >= 0; --i) {
const HbBadgeIconInfo &info = mBadgeIcons.at(i);
if (info.icon() == badge) {
mBadgeIcons.removeAt(i);
result = true;
}
}
return result;
}
/*!
Indicate whether or not this icon is badged.
\internal
*/
bool HbBadgeIcon::isBadged() const
{
return mBadgeIcons.size() > 0;
}
/*!
Remove all badge icons from this icon
\internal
*/
void HbBadgeIcon::removeAllBadges()
{
mBadgeIcons.clear();
}
/*!
Returns the list of all badge icons
*/
const QList<HbBadgeIconInfo> HbBadgeIcon::badges() const
{
return mBadgeIcons;
}
/*!
\internal
Paint all badges in z-order.
*/
void HbBadgeIcon::paint(QPainter *painter,
const QRectF &rect,
QIcon::Mode mode,
QIcon::State state,
bool mirror)
{
int count = mBadgeIcons.count();
for (int i = count - 1; i >= 0; i--) {
HbBadgeIconInfo badge = mBadgeIcons.at(i);
// Fix the alignment.
Qt::Alignment align = badge.alignment();
Qt::Alignment absAlign = align;
if (mirror) {
absAlign = align & ~(Qt::AlignRight | Qt::AlignLeft);
if (align & Qt::AlignLeft) {
absAlign |= Qt::AlignRight;
}
if (align & Qt::AlignRight) {
absAlign |= Qt::AlignLeft;
}
}
// Update the size.
QSizeF sizeFactor = badge.sizeFactor();
if (!sizeFactor.isNull()) {
QSizeF targetSizeF(sizeFactor.width() * rect.width(), sizeFactor.height() * rect.height());
QSize targetSize = targetSizeF.toSize();
HbIcon icon = badge.icon();
if (targetSize != icon.size().toSize()) {
icon.setSize(targetSize);
badge.setIcon(icon);
mBadgeIcons[i] = badge;
}
}
// And finally draw the badge.
badge.icon().paint(painter,
rect,
badge.aspectRatio(),
absAlign,
mode,
state);
}
}
/*!
\internal
*/
void HbBadgeIcon::externalize(QDataStream &stream)
{
int count = mBadgeIcons.count();
// Write out how many badges we'll save first
stream << count;
// And write each item
for (int i = 0; i < count; i++) {
const HbBadgeIconInfo &badge(mBadgeIcons.at(i));
stream << badge.icon();
stream << (qint32) badge.alignment();
stream << badge.zValue();
stream << badge.sizeFactor();
stream << (qint32) badge.aspectRatio();
}
}
/*!
\Internal
*/
void HbBadgeIcon::internalize(QDataStream &stream)
{
int howManyBadges;
stream >> howManyBadges;
for (int i = 0; i < howManyBadges; ++i) {
qint32 align;
qint32 zValue;
HbIcon icon;
QSizeF sizeFactor;
qint32 aspectRatio;
stream >> icon;
stream >> align;
stream >> zValue;
stream >> sizeFactor;
stream >> aspectRatio;
HbBadgeIconInfo info(icon, (Qt::Alignment) align, zValue,
sizeFactor, (Qt::AspectRatioMode) aspectRatio);
mBadgeIcons.append(info);
}
}