/*
* Copyright (C) 2010 Ixonos Plc.
*
* This file is part of fosdem-schedule.
*
* fosdem-schedule 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.
*
* fosdem-schedule is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along with
* fosdem-schedule. If not, see .
*/
#include "delegate.h"
#include "eventmodel.h"
#include
#include
#include
const int RADIUS = 10;
const int SPACER = 10;
const double scaleFactor1 = 0.4;
const double scaleFactor2 = 0.8;
Delegate::Delegate(QTreeView *aParent)
: QItemDelegate(aParent)
, mViewPtr(aParent)
{
mControls.clear();
defineControls();
}
Delegate::~Delegate()
{
QListIterator i(mControls.keys());
while (i.hasNext())
{
delete mControls[i.next()]->image();
}
}
void Delegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
if(!mViewPtr)
return;
painter->save();
QColor bkgrColor = Qt::cyan;
//QColor bkgrColor = QColor(0xAA,0xAA,0xAA);
QColor conflictColor = Qt::yellow;
QPen borderPen(bkgrColor.darker());
if(hasParent(index))
{
int aux = option.rect.height() - mControls[FavouriteControlOn]->drawPoint().y() - mControls[FavouriteControlOn]->image()->height();
// font SMALL
QFont fontSmall = option.font;
fontSmall.setBold(false);
fontSmall.setPixelSize(aux*0.2);
QFontMetrics fmSmall(fontSmall);
// font SMALL bold
QFont fontSmallB = fontSmall;
fontSmallB.setBold(true);
QFontMetrics fmSmallB(fontSmallB);
// font BIG
QFont fontBig = option.font;
fontBig.setBold(false);
fontBig.setPixelSize(aux*0.33);
QFontMetrics fmBig(fontBig);
// font BIG bold
QFont fontBigB = fontBig;
fontBigB.setBold(true);
QFontMetrics fmBigB(fontBigB);
//int spacer = (fmSmall.boundingRect("999").width() < SPACER) ? SPACER : fmSmall.boundingRect("999").width();
//Time conflicts are colored differently
if(static_cast(index.internalPointer())->hasTimeConflict())
bkgrColor = conflictColor;
QLinearGradient itemGradient(option.rect.topLeft(), option.rect.bottomLeft());
itemGradient.setColorAt(0.0, Qt::white);
itemGradient.setColorAt(0.25, bkgrColor);
itemGradient.setColorAt(0.5, bkgrColor);
itemGradient.setColorAt(0.75, bkgrColor);
itemGradient.setColorAt(1.0, Qt::white);
if(isLast(index))
{
QPainterPath endPath;
endPath.moveTo(option.rect.topLeft());
endPath.lineTo(option.rect.bottomLeft()-QPoint(0, RADIUS));
endPath.arcTo(option.rect.left(), option.rect.bottom()-2*RADIUS, 2*RADIUS, 2*RADIUS, 180, 90);
endPath.lineTo(option.rect.bottomRight()-QPoint(RADIUS, 0));
endPath.arcTo(option.rect.right()-2*RADIUS, option.rect.bottom()-2*RADIUS, 2*RADIUS, 2*RADIUS, 270, 90);
endPath.lineTo(option.rect.topRight());
//painter->setBrush( bkgrColor );
painter->setBrush(itemGradient);
painter->setPen(borderPen);
painter->drawPath(endPath);
painter->setFont(option.font);
}
else // middle elements
{
//painter->setBrush( bkgrColor );
painter->setBrush(itemGradient);
painter->setPen(Qt::NoPen);
painter->drawRect(option.rect);
painter->setPen(borderPen);
// vertical lines
painter->drawLine(option.rect.topLeft(), option.rect.bottomLeft());
painter->drawLine(option.rect.topRight(), option.rect.bottomRight());
// horizontal lines
painter->drawLine(option.rect.bottomLeft(), option.rect.bottomRight());
painter->setFont(option.font);
}
// draw Controls
// favourite
if(static_cast(index.internalPointer())->isFavourite())
painter->drawImage(mControls[FavouriteControlOn]->drawPoint(option.rect),*mControls[FavouriteControlOn]->image());
else
painter->drawImage(mControls[FavouriteControlOff]->drawPoint(option.rect),*mControls[FavouriteControlOff]->image());
#ifdef MAEMO
// alarm
if(static_cast(index.internalPointer())->hasAlarm())
painter->drawImage(mControls[AlarmControlOn]->drawPoint(option.rect),*mControls[AlarmControlOn]->image());
else
painter->drawImage(mControls[AlarmControlOff]->drawPoint(option.rect),*mControls[AlarmControlOff]->image());
#endif
// map
painter->drawImage(mControls[MapControl]->drawPoint(option.rect),*mControls[MapControl]->image());
// Time conflict
if(static_cast(index.internalPointer())->hasTimeConflict())
painter->drawImage(mControls[WarningControl]->drawPoint(option.rect),*mControls[WarningControl]->image());
// draw texts
Event *event = static_cast(index.internalPointer());
QPointF titlePointF(mControls[FavouriteControlOn]->drawPoint(option.rect));
titlePointF.setX(option.rect.x()+SPACER);
titlePointF.setY(titlePointF.y()+mControls[FavouriteControlOn]->image()->height());
QTime start = event->start().time();
painter->setFont(fontBig);
painter->drawText(titlePointF,start.toString("hh:mm") + "-" + start.addSecs(event->duration()).toString("hh:mm") + ", " + event->room());
// title
titlePointF.setY(titlePointF.y()+fmBig.height()-fmBig.descent());
painter->setFont(fontBigB);
QString title = event->title();
if(fmBigB.boundingRect(title).width() > (option.rect.width()-2*SPACER)) // the title won't fit the screen
{
// chop words from the end
while( (fmBigB.boundingRect(title + "...").width() > (option.rect.width()-2*SPACER)) && !title.isEmpty())
{
title.chop(1);
// chop characters one-by-one from the end
while( (!title.at(title.length()-1).isSpace()) && !title.isEmpty())
{
title.chop(1);
}
}
title += "...";
}
painter->drawText(titlePointF,title);
// persons
titlePointF.setY(titlePointF.y()+fmSmall.height()-fmSmall.descent());
painter->setFont(fontSmall);
painter->drawText(titlePointF,"Presenter(s): " + event->persons().join(" and "));
// track
titlePointF.setY(titlePointF.y()+fmSmall.height()-fmSmall.descent());
painter->drawText(titlePointF,"Track: " + Track::retrieveTrackName(event->trackId()));
}
else // doesn't have parent - time-groups' elements (top items)
{
QFont fontSmall = option.font;
fontSmall.setBold(true);
fontSmall.setPixelSize(option.rect.height()*scaleFactor1);
QFontMetrics fmSmall(fontSmall);
QFont fontBig = option.font;
fontBig.setBold(true);
fontBig.setPixelSize(option.rect.height()*scaleFactor2);
QFontMetrics fmBig(fontBig);
int spacer = (fmSmall.boundingRect("999").width() < SPACER) ? SPACER : fmSmall.boundingRect("999").width();
QLinearGradient titleGradient(option.rect.topLeft(), option.rect.topRight());
//titleGradient.setColorAt(0.0, Qt::white);
titleGradient.setColorAt(0.0, bkgrColor);
titleGradient.setColorAt(0.5, Qt::white);
titleGradient.setColorAt(1.0, bkgrColor);
QPainterPath titlePath;
if(isExpanded(index))
{
titlePath.moveTo(option.rect.bottomLeft());
titlePath.lineTo(option.rect.topLeft()+QPoint(0, RADIUS));
titlePath.arcTo(option.rect.left(), option.rect.top(), 2*RADIUS, 2*RADIUS, 180, -90);
titlePath.lineTo(option.rect.topRight()-QPoint(RADIUS, 0));
titlePath.arcTo(option.rect.right()-2*RADIUS, option.rect.top(), 2*RADIUS, 2*RADIUS, 90, -90);
titlePath.lineTo(option.rect.bottomRight());
titlePath.closeSubpath();
}
else
{
titlePath.lineTo(option.rect.topLeft()+QPoint(0, RADIUS));
titlePath.arcTo(option.rect.left(), option.rect.top(), 2*RADIUS, 2*RADIUS, 180, -90);
titlePath.lineTo(option.rect.topRight()-QPoint(RADIUS, 0));
titlePath.arcTo(option.rect.right()-2*RADIUS, option.rect.top(), 2*RADIUS, 2*RADIUS, 90, -90);
titlePath.lineTo(option.rect.bottomRight()-QPoint(0, RADIUS));
titlePath.arcTo(option.rect.right()-2*RADIUS, option.rect.bottom()-2*RADIUS, 2*RADIUS, 2*RADIUS, 0, -90);
titlePath.lineTo(option.rect.bottomLeft()+QPoint(RADIUS, 0));
titlePath.arcTo(option.rect.left(), option.rect.bottom()-2*RADIUS, 2*RADIUS, 2*RADIUS, 270, -90);
titlePath.closeSubpath();
}
painter->setBrush(titleGradient);
painter->setPen(borderPen);
painter->drawPath(titlePath);
// draw icons
painter->setFont(fontSmall);
QImage *image = mControls[FavouriteControlOn]->image();
QPoint drawPoint =
option.rect.topRight()
- QPoint(
spacer + image->width(),
- option.rect.height()/2 + image->height()/2);
painter->drawImage(drawPoint,*image);
painter->drawText(drawPoint+QPoint(image->width()+2, image->height() - 2),
QString::number(numberOfFavourities(index)));
#ifdef MAEMO
drawPoint.setX(drawPoint.x() - spacer - image->width());
painter->drawImage(drawPoint,*mControls[AlarmControlOn]->image());
painter->drawText(drawPoint+QPoint(image->width()+2, image->height() - 2),
QString::number(numberOfAlarms(index)));
#endif
// draw texts
QString numEvents = QString::number(index.model()->rowCount(index)).append("/");
drawPoint.setX(drawPoint.x() - spacer - fmSmall.boundingRect(numEvents).width());
drawPoint.setY(drawPoint.y()+image->height() - 2);
painter->drawText(drawPoint,numEvents);
QPointF titlePointF = QPoint(
option.rect.x()+SPACER,
option.rect.y()+option.rect.height()-fmBig.descent());
painter->setFont(fontBig);
painter->drawText(titlePointF,qVariantValue(index.data()));
}
//// HIGHLIGHTING SELECTED ITEM
//if (option.state & QStyle::State_Selected)
//painter->fillRect(option.rect, option.palette.highlight());
painter->restore();
}
QSize Delegate::sizeHint(const QStyleOptionViewItem &option, const QModelIndex &index) const
{
Q_UNUSED(option)
if (index.internalId() == 0) // time group
{
return QSize(40,40);
}
else // event
{
return QSize(100,100);
}
}
bool Delegate::hasParent( const QModelIndex &index ) const
{
if( index.parent().isValid() )
return true;
else
return false;
}
bool Delegate::isLast( const QModelIndex &index ) const
{
if(!hasParent(index))
return false; // what should be returned here?
if(index.row() >= (index.model()->rowCount(index.parent())-1))
return true;
else
return false;
}
bool Delegate::isExpanded( const QModelIndex &index ) const
{
if( !mViewPtr )
return false;
else
return mViewPtr->isExpanded( index );
}
Delegate::ControlId Delegate::whichControlClicked(const QModelIndex &aIndex, const QPoint &aPoint) const
{
if(!hasParent(aIndex)) // time-group item (root item)
return ControlNone;
QListIterator i(mControls.keys());
while (i.hasNext())
{
ControlId id = i.next();
if(mControls[id]->drawRect(static_cast(parent())->visualRect(aIndex)).contains(aPoint))
{
if(id == WarningControl)
{
if(static_cast(aIndex.internalPointer())->hasTimeConflict())
return id;
}
else
return id;
}
}
return ControlNone;
}
void Delegate::defineControls()
{
Control *control;
QPoint p(0,0);
// FAVOURITE ICONs
// on
control = new Control(FavouriteControlOn,QString(":icons/favourite-onBig.png"));
p = QPoint(0,SPACER);
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(FavouriteControlOn,control);
// off
control = new Control(FavouriteControlOff,QString(":icons/favourite-offBig.png"));
p = QPoint(0,SPACER);
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(FavouriteControlOff,control);
#ifdef MAEMO
// ALARM ICONs
// on
control = new Control(AlarmControlOn,QString(":icons/alarm-onBig.png"));
p = mControls[FavouriteControlOn]->drawPoint();
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(AlarmControlOn,control);
// off
control = new Control(AlarmControlOff,QString(":icons/alarm-offBig.png"));
p = mControls[FavouriteControlOff]->drawPoint();
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(AlarmControlOff,control);
// MAP ICON
control = new Control(MapControl,QString(":icons/compassBig.png"));
p = mControls[AlarmControlOn]->drawPoint();
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(MapControl,control);
#else
// MAP ICON
control = new Control(MapControl,QString(":icons/compassBig.png"));
p = mControls[FavouriteControlOn]->drawPoint();
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(MapControl,control);
#endif
// WARNING ICON
control = new Control(WarningControl,QString(":icons/exclamation.png"));
p = mControls[MapControl]->drawPoint();
p.setX(p.x()-control->image()->width()-SPACER);
control->setDrawPoint(p);
mControls.insert(WarningControl,control);
}
bool Delegate::isPointFromRect(const QPoint &aPoint, const QRect &aRect) const
{
if( (aPoint.x()>=aRect.left() && aPoint.x()<=aRect.right()) && (aPoint.y()>=aRect.top() && aPoint.y()<=aRect.bottom()) )
return true;
return false;
}
int Delegate::numberOfFavourities(const QModelIndex &index) const
{
if(index.parent().isValid()) // it's event, not time-group
return 0;
int nrofFavs = 0;
for(int i=0; irowCount(index); i++)
if(static_cast(index.child(i,0).internalPointer())->isFavourite())
nrofFavs++;
return nrofFavs;
}
int Delegate::numberOfAlarms(const QModelIndex &index) const
{
if(index.parent().isValid()) // it's event, not time-group
return 0;
int nrofAlarms = 0;
for(int i=0; irowCount(index); i++)
if(static_cast(index.child(i,0).internalPointer())->hasAlarm())
nrofAlarms++;
return nrofAlarms;
}