radioapp/radiouiengine/src/radiotimerpool.cpp
author hgs
Fri, 15 Oct 2010 16:26:27 +0300
changeset 57 21be958eb3ce
permissions -rw-r--r--
201041
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
57
hgs
parents:
diff changeset
     1
/*
hgs
parents:
diff changeset
     2
* Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
hgs
parents:
diff changeset
     3
* All rights reserved.
hgs
parents:
diff changeset
     4
* This component and the accompanying materials are made available
hgs
parents:
diff changeset
     5
* under the terms of "Eclipse Public License v1.0"
hgs
parents:
diff changeset
     6
* which accompanies this distribution, and is available
hgs
parents:
diff changeset
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
hgs
parents:
diff changeset
     8
*
hgs
parents:
diff changeset
     9
* Initial Contributors:
hgs
parents:
diff changeset
    10
* Nokia Corporation - initial contribution.
hgs
parents:
diff changeset
    11
*
hgs
parents:
diff changeset
    12
* Contributors:
hgs
parents:
diff changeset
    13
*
hgs
parents:
diff changeset
    14
* Description:
hgs
parents:
diff changeset
    15
*
hgs
parents:
diff changeset
    16
*/
hgs
parents:
diff changeset
    17
hgs
parents:
diff changeset
    18
// System includes
hgs
parents:
diff changeset
    19
#include <QTimer>
hgs
parents:
diff changeset
    20
#include <QMetaMethod>
hgs
parents:
diff changeset
    21
hgs
parents:
diff changeset
    22
// User includes
hgs
parents:
diff changeset
    23
#include "radiotimerpool.h"
hgs
parents:
diff changeset
    24
#include "radiologger.h"
hgs
parents:
diff changeset
    25
hgs
parents:
diff changeset
    26
const int TIMER_BUFFER_SIZE = 2;
hgs
parents:
diff changeset
    27
hgs
parents:
diff changeset
    28
/*!
hgs
parents:
diff changeset
    29
 *
hgs
parents:
diff changeset
    30
 */
hgs
parents:
diff changeset
    31
RadioTimerPool::RadioTimerItem::RadioTimerItem( RadioTimerPool* parent ) :
hgs
parents:
diff changeset
    32
    mTimer( new QTimer() )
hgs
parents:
diff changeset
    33
{
hgs
parents:
diff changeset
    34
    Radio::connect( mTimer.data(),  SIGNAL(timeout()),
hgs
parents:
diff changeset
    35
                    parent,         SLOT(timerFired()) );
hgs
parents:
diff changeset
    36
    mTimer->setSingleShot( true );
hgs
parents:
diff changeset
    37
}
hgs
parents:
diff changeset
    38
hgs
parents:
diff changeset
    39
/*!
hgs
parents:
diff changeset
    40
 *
hgs
parents:
diff changeset
    41
 */
hgs
parents:
diff changeset
    42
RadioTimerPool::RadioTimerItem::~RadioTimerItem()
hgs
parents:
diff changeset
    43
{
hgs
parents:
diff changeset
    44
}
hgs
parents:
diff changeset
    45
hgs
parents:
diff changeset
    46
/*!
hgs
parents:
diff changeset
    47
 *
hgs
parents:
diff changeset
    48
 */
hgs
parents:
diff changeset
    49
void RadioTimerPool::RadioTimerItem::reset()
hgs
parents:
diff changeset
    50
{
hgs
parents:
diff changeset
    51
    mTimer->stop();
hgs
parents:
diff changeset
    52
    mId = -1;
hgs
parents:
diff changeset
    53
    mParam = QVariant();
hgs
parents:
diff changeset
    54
    mReceiver = NULL;
hgs
parents:
diff changeset
    55
    mReceiverMember.clear();
hgs
parents:
diff changeset
    56
}
hgs
parents:
diff changeset
    57
hgs
parents:
diff changeset
    58
/*!
hgs
parents:
diff changeset
    59
 *
hgs
parents:
diff changeset
    60
 */
hgs
parents:
diff changeset
    61
RadioTimerPool::RadioTimerPool( QObject* parent ) :
hgs
parents:
diff changeset
    62
    QObject( parent )
hgs
parents:
diff changeset
    63
{
hgs
parents:
diff changeset
    64
    createItem();
hgs
parents:
diff changeset
    65
}
hgs
parents:
diff changeset
    66
hgs
parents:
diff changeset
    67
/*!
hgs
parents:
diff changeset
    68
 *
hgs
parents:
diff changeset
    69
 */
hgs
parents:
diff changeset
    70
RadioTimerPool::~RadioTimerPool()
hgs
parents:
diff changeset
    71
{
hgs
parents:
diff changeset
    72
    qDeleteAll( mTimerMap.values() );
hgs
parents:
diff changeset
    73
}
hgs
parents:
diff changeset
    74
hgs
parents:
diff changeset
    75
/*!
hgs
parents:
diff changeset
    76
 *
hgs
parents:
diff changeset
    77
 */
hgs
parents:
diff changeset
    78
void RadioTimerPool::startTimer( int msec, int id, QObject* receiver, const char* member, QVariant param )
hgs
parents:
diff changeset
    79
{
hgs
parents:
diff changeset
    80
    LOG_METHOD;
hgs
parents:
diff changeset
    81
    RadioTimerItem* availableItem = NULL;
hgs
parents:
diff changeset
    82
    foreach ( RadioTimerItem* item, mTimerMap.values() ) {
hgs
parents:
diff changeset
    83
        if ( !item->mTimer->isActive() ) {
hgs
parents:
diff changeset
    84
            availableItem = item;
hgs
parents:
diff changeset
    85
        } else {
hgs
parents:
diff changeset
    86
            if ( item->mReceiver == receiver && item->mId == id ) {
hgs
parents:
diff changeset
    87
                LOG_FORMAT( "RadioTimerPool::startTimer. Timer %d already running for class %s", id, receiver->metaObject()->className() );
hgs
parents:
diff changeset
    88
            }
hgs
parents:
diff changeset
    89
        }
hgs
parents:
diff changeset
    90
    }
hgs
parents:
diff changeset
    91
hgs
parents:
diff changeset
    92
    if ( !availableItem ) {
hgs
parents:
diff changeset
    93
        availableItem = createItem();
hgs
parents:
diff changeset
    94
        LOG( "RadioTimerPool::startTimer. Creating new timer" );
hgs
parents:
diff changeset
    95
    }
hgs
parents:
diff changeset
    96
    LOG_FORMAT( "Amount of timers: %d", mTimerMap.count() );
hgs
parents:
diff changeset
    97
hgs
parents:
diff changeset
    98
    availableItem->mId = id;
hgs
parents:
diff changeset
    99
    availableItem->mParam = param;
hgs
parents:
diff changeset
   100
    availableItem->mReceiver = receiver;
hgs
parents:
diff changeset
   101
hgs
parents:
diff changeset
   102
    // SIGNAL() and SLOT() macros add a number in the beginning of member function name for some internal housekeeping
hgs
parents:
diff changeset
   103
    // That causes the method to not be found from the meta object so we need to trim it off
hgs
parents:
diff changeset
   104
    availableItem->mReceiverMember = member;
hgs
parents:
diff changeset
   105
    availableItem->mReceiverMember = availableItem->mReceiverMember.right( availableItem->mReceiverMember.length() - 1 );
hgs
parents:
diff changeset
   106
hgs
parents:
diff changeset
   107
    availableItem->mTimer->start( msec );
hgs
parents:
diff changeset
   108
}
hgs
parents:
diff changeset
   109
hgs
parents:
diff changeset
   110
/*!
hgs
parents:
diff changeset
   111
 *
hgs
parents:
diff changeset
   112
 */
hgs
parents:
diff changeset
   113
void RadioTimerPool::cancelTimer( int id, const QObject* receiver )
hgs
parents:
diff changeset
   114
{
hgs
parents:
diff changeset
   115
    foreach ( RadioTimerItem* item, mTimerMap.values() ) {
hgs
parents:
diff changeset
   116
        if ( item->mId == id && item->mReceiver == receiver ) {
hgs
parents:
diff changeset
   117
            item->reset();
hgs
parents:
diff changeset
   118
            break;
hgs
parents:
diff changeset
   119
        }
hgs
parents:
diff changeset
   120
    }
hgs
parents:
diff changeset
   121
    freeUnusedTimers();
hgs
parents:
diff changeset
   122
}
hgs
parents:
diff changeset
   123
hgs
parents:
diff changeset
   124
/*!
hgs
parents:
diff changeset
   125
 *
hgs
parents:
diff changeset
   126
 */
hgs
parents:
diff changeset
   127
bool RadioTimerPool::isTimerActive( int id, const QObject* receiver ) const
hgs
parents:
diff changeset
   128
{
hgs
parents:
diff changeset
   129
    foreach ( const RadioTimerItem* item, mTimerMap.values() ) {
hgs
parents:
diff changeset
   130
        if ( item->mId == id &&
hgs
parents:
diff changeset
   131
             item->mReceiver == receiver &&
hgs
parents:
diff changeset
   132
             item->mTimer->isActive() ) {
hgs
parents:
diff changeset
   133
            return true;
hgs
parents:
diff changeset
   134
        }
hgs
parents:
diff changeset
   135
    }
hgs
parents:
diff changeset
   136
    return false;
hgs
parents:
diff changeset
   137
}
hgs
parents:
diff changeset
   138
hgs
parents:
diff changeset
   139
/*!
hgs
parents:
diff changeset
   140
 *
hgs
parents:
diff changeset
   141
 */
hgs
parents:
diff changeset
   142
void RadioTimerPool::freeUnusedTimers()
hgs
parents:
diff changeset
   143
{
hgs
parents:
diff changeset
   144
    QList<RadioTimerItem*> unusedItems;
hgs
parents:
diff changeset
   145
    foreach ( RadioTimerItem* item, mTimerMap.values() ) {
hgs
parents:
diff changeset
   146
        if ( !item->mTimer->isActive() ) {
hgs
parents:
diff changeset
   147
            unusedItems.append( item );
hgs
parents:
diff changeset
   148
        }
hgs
parents:
diff changeset
   149
    }
hgs
parents:
diff changeset
   150
hgs
parents:
diff changeset
   151
    if ( unusedItems.count() > TIMER_BUFFER_SIZE ) {
hgs
parents:
diff changeset
   152
        foreach( const RadioTimerItem* item, unusedItems ) {
hgs
parents:
diff changeset
   153
            mTimerMap.remove( item->mTimer.data() );
hgs
parents:
diff changeset
   154
            delete item;
hgs
parents:
diff changeset
   155
        }
hgs
parents:
diff changeset
   156
    }
hgs
parents:
diff changeset
   157
}
hgs
parents:
diff changeset
   158
hgs
parents:
diff changeset
   159
/*!
hgs
parents:
diff changeset
   160
 * Private slot
hgs
parents:
diff changeset
   161
 */
hgs
parents:
diff changeset
   162
void RadioTimerPool::timerFired()
hgs
parents:
diff changeset
   163
{
hgs
parents:
diff changeset
   164
    QTimer* timer = static_cast<QTimer*>( sender() );
hgs
parents:
diff changeset
   165
    if ( mTimerMap.contains( timer ) ) {
hgs
parents:
diff changeset
   166
        RadioTimerItem* item = mTimerMap.value( timer );
hgs
parents:
diff changeset
   167
        if ( item->mReceiver ) {
hgs
parents:
diff changeset
   168
            QByteArray normalizedSignature = QMetaObject::normalizedSignature( item->mReceiverMember.toAscii().constData() );
hgs
parents:
diff changeset
   169
            int methodIndex = item->mReceiver->metaObject()->indexOfMethod( normalizedSignature );
hgs
parents:
diff changeset
   170
            if ( methodIndex != -1 ) {
hgs
parents:
diff changeset
   171
                QMetaMethod method = item->mReceiver->metaObject()->method( methodIndex );
hgs
parents:
diff changeset
   172
                const int paramCount = method.parameterTypes().count();
hgs
parents:
diff changeset
   173
                if ( paramCount == 0 ) {
hgs
parents:
diff changeset
   174
                    method.invoke( item->mReceiver, Qt::QueuedConnection );
hgs
parents:
diff changeset
   175
                } else if ( paramCount == 1 ) {
hgs
parents:
diff changeset
   176
                    method.invoke( item->mReceiver, Qt::QueuedConnection, Q_ARG( int, item->mId ) );
hgs
parents:
diff changeset
   177
                } else if ( paramCount == 2 ) {
hgs
parents:
diff changeset
   178
                    method.invoke( item->mReceiver, Qt::QueuedConnection,
hgs
parents:
diff changeset
   179
                                   Q_ARG( int, item->mId ), Q_ARG( QVariant, item->mParam ) );
hgs
parents:
diff changeset
   180
                }
hgs
parents:
diff changeset
   181
            }
hgs
parents:
diff changeset
   182
hgs
parents:
diff changeset
   183
            item->reset();
hgs
parents:
diff changeset
   184
        }
hgs
parents:
diff changeset
   185
    }
hgs
parents:
diff changeset
   186
}
hgs
parents:
diff changeset
   187
hgs
parents:
diff changeset
   188
/*!
hgs
parents:
diff changeset
   189
 *
hgs
parents:
diff changeset
   190
 */
hgs
parents:
diff changeset
   191
RadioTimerPool::RadioTimerItem* RadioTimerPool::createItem()
hgs
parents:
diff changeset
   192
{
hgs
parents:
diff changeset
   193
    RadioTimerItem* item = new RadioTimerItem( this );
hgs
parents:
diff changeset
   194
    mTimerMap.insert( item->mTimer.data(), item );
hgs
parents:
diff changeset
   195
    return item;
hgs
parents:
diff changeset
   196
}