/********************************************************************
 KWin - the KDE window manager
 This file is part of the KDE project.

Copyright (C) 2008 Lubos Lunak <l.lunak@suse.cz>
Copyright (C) 2009 Lucas Murray <lmurray@undefinedfire.com>

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 program 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 this program.  If not, see <http://www.gnu.org/licenses/>.
*********************************************************************/

#include "monitor.h"

#include <kdebug.h>
#include <kstandarddirs.h>
#include <qgraphicsitem.h>
#include <qgraphicsview.h>
#include <qgraphicsscene.h>
#include <qgraphicssceneevent.h>
#include <qmenu.h>
#include <qdesktopwidget.h>
#include <qapplication.h>
#include <plasma/framesvg.h>

namespace KWin
{

Monitor::Monitor( QWidget* parent )
    : ScreenPreviewWidget( parent )
    {
    QDesktopWidget *desktop = QApplication::desktop();
    QRect avail = desktop->availableGeometry(desktop->screenNumber(this));
    setRatio((qreal)avail.width()/(qreal)avail.height());
    for( int i = 0;
         i < 8;
         ++i )
        popups[ i ] = new QMenu( this );
    scene = new QGraphicsScene( this );
    view = new QGraphicsView( scene, this );
    view->setBackgroundBrush( Qt::black );
    view->setHorizontalScrollBarPolicy( Qt::ScrollBarAlwaysOff );
    view->setVerticalScrollBarPolicy( Qt::ScrollBarAlwaysOff );
    view->setFocusPolicy( Qt::NoFocus );
    view->setFrameShape( QFrame::NoFrame );
    for( int i = 0;
         i < 8;
         ++i )
        {
        items[ i ] = new Corner( this );
        scene->addItem( items[ i ] );
        hidden[ i ] = false;
        grp[ i ] = new QActionGroup( this );
        }
    checkSize();
    }

void Monitor::clear()
    {
    for( int i = 0;
         i < 8;
         ++i )
        {
        popups[ i ]->clear();
        setEdge( i, false );
        setEdgeHidden( i, false );
        delete grp[ i ];
        grp[ i ] = new QActionGroup( this );
        }
    }

void Monitor::resizeEvent( QResizeEvent* e )
    {
    ScreenPreviewWidget::resizeEvent( e );
    checkSize();
    }
    
void Monitor::checkSize()
    {
    QRect contentsRect = previewRect();
    int w = 151;
    int h = 115;
    view->setGeometry(contentsRect);
    scene->setSceneRect(QRect(QPoint(0,0), contentsRect.size()));
    int x2 = ( contentsRect.width() - 20 ) / 2;
    int x3 = contentsRect.width() - 20;
    int y2 = ( contentsRect.height() - 20 ) / 2;
    int y3 = contentsRect.height() - 20;
    items[ 0 ]->setRect( 0, y2, 20, 20 );
    items[ 1 ]->setRect( x3, y2, 20, 20 );
    items[ 2 ]->setRect( x2, 0, 20, 20 );
    items[ 3 ]->setRect( x2, y3, 20, 20 );
    items[ 4 ]->setRect( 0, 0, 20, 20 );
    items[ 5 ]->setRect( x3, 0, 20, 20 );
    items[ 6 ]->setRect( 0, y3, 20, 20 );
    items[ 7 ]->setRect( x3, y3, 20, 20 );
    }

void Monitor::setEdge( int edge, bool set )
    {
    items[ edge ]->setActive( set );
    }

bool Monitor::edge( int edge ) const
    {
    return items[ edge ]->brush() == Qt::green;
    }

void Monitor::setEdgeHidden( int edge, bool set )
    {
    hidden[ edge ] = set;
    if( set )
        items[ edge ]->hide();
    else
        items[ edge ]->show();
    }

bool Monitor::edgeHidden( int edge ) const
    {
    return hidden[ edge ];
    }

void Monitor::addEdgeItem( int edge, const QString& item )
    {
    QAction* act = popups[ edge ]->addAction( item );
    act->setCheckable( true );
    popup_actions[ edge ].append( act );
    grp[ edge ]->addAction( act );
    if( popup_actions[ edge ].count() == 1 )
        {
        act->setChecked( true );
        items[ edge ]->setToolTip( item );
        }
    setEdge( edge, !popup_actions[ edge ][ 0 ]->isChecked());
    }

void Monitor::setEdgeItemEnabled( int edge, int index, bool enabled )
    {
    popup_actions[ edge ][ index ]->setEnabled( enabled );
    }

bool Monitor::edgeItemEnabled( int edge, int index ) const
    {
    return popup_actions[ edge ][ index ]->isEnabled();
    }

void Monitor::selectEdgeItem( int edge, int index )
    {
    popup_actions[ edge ][ index ]->setChecked( true );
    setEdge( edge, !popup_actions[ edge ][ 0 ]->isChecked());
    items[ edge ]->setToolTip( popup_actions[ edge ][ index ]->text() );
    }

int Monitor::selectedEdgeItem( int edge ) const
    {
    foreach( QAction* act, popup_actions[ edge ] )
        if( act->isChecked())
            return popup_actions[ edge ].indexOf( act );
    abort();
    }

void Monitor::popup( Corner* c, QPoint pos )
    {
    for( int i = 0;
         i < 8;
         ++i )
        {
        if( items[ i ] == c )
            {
            if( popup_actions[ i ].count() == 0 )
                return;
            if( QAction* a = popups[ i ]->exec( pos ))
                {
                selectEdgeItem( i, popup_actions[ i ].indexOf( a ));
                emit changed();
                emit edgeSelectionChanged( i, popup_actions[ i ].indexOf( a ));
                c->setToolTip( a->text() );
                }
            return;
            }
        }
    abort();
    }

void Monitor::flip( Corner* c, QPoint pos )
    {
    for( int i = 0;
         i < 8;
         ++i )
        {
        if( items[ i ] == c )
            {
            if( popup_actions[ i ].count() == 0 )
                setEdge( i, !edge( i ));
            else
                popup( c, pos );
            return;
            }
        }
    abort();
    }

Monitor::Corner::Corner( Monitor* m )
    : monitor( m ),
      m_active( false ),
      m_hover( false )
    {
    button = new Plasma::FrameSvg();
    button->setImagePath("widgets/button");
    setAcceptHoverEvents(true);
    }

Monitor::Corner::~Corner( )
    {
    delete button;
    }

void Monitor::Corner::contextMenuEvent( QGraphicsSceneContextMenuEvent* e )
    {
    monitor->popup( this, e->screenPos());
    }

void Monitor::Corner::mousePressEvent( QGraphicsSceneMouseEvent* e )
    {
    monitor->flip( this, e->screenPos());
    }

void Monitor::Corner::paint( QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget )
    {
    Q_UNUSED(option)
    Q_UNUSED(widget)

    if (m_hover)
        {
        button->setElementPrefix("normal");

        qreal left, top, right, bottom;
        button->getMargins(left, top, right, bottom);

        button->setElementPrefix("active");
        qreal activeLeft, activeTop, activeRight, activeBottom;
        button->getMargins(activeLeft, activeTop, activeRight, activeBottom);

        QRectF activeRect = QRectF(QPointF(0, 0), rect().size());
        activeRect.adjust(left - activeLeft, top - activeTop,
                      -(right - activeRight), -(bottom - activeBottom));
        button->setElementPrefix("active");
        button->resizeFrame(activeRect.size());
        button->paintFrame(painter, rect().topLeft()+activeRect.topLeft());
        }
        else
        {
        button->setElementPrefix(m_active?"pressed":"normal");
        button->resizeFrame(rect().size());
        button->paintFrame(painter, rect().topLeft());
        }

    if (m_active)
        {
        QPainterPath roundedRect;
        painter->setRenderHint(QPainter::Antialiasing);
        roundedRect.addRoundedRect(rect().adjusted(5,5,-5,-5), 2, 2);
        painter->fillPath(roundedRect, QApplication::palette().text());
        }
    }

void Monitor::Corner::hoverEnterEvent( QGraphicsSceneHoverEvent * e)
{
    m_hover = true;
    update();
}

void Monitor::Corner::hoverLeaveEvent( QGraphicsSceneHoverEvent * e)
{
    m_hover = false;
    update();
}

void Monitor::Corner::setActive( bool active )
    {
    m_active = active;
    update();
    }

bool Monitor::Corner::active() const
    {
    return m_active;
    }
} // namespace

#include "monitor.moc"