ganeswidgets/src/HgScrollBufferManager.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 14 May 2010 16:57:01 +0300
changeset 2 49c70dcc3f17
parent 1 e48454f237ca
child 3 c863538fcbb6
permissions -rw-r--r--
Revision: 201017 Kit: 201019

/*
* Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of "Eclipse Public License v1.0"
* which accompanies this distribution, and is available
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
*
* Initial Contributors:
* Nokia Corporation - initial contribution.
*
* Contributors:
*
* Description:
*
*/


#include <QTimer>

#include "HgScrollBufferManager.h"
#include "trace.h"

// -----------------------------------------------------------------------------
// HgScrollBufferManager::HgScrollBufferManager()
// -----------------------------------------------------------------------------
//
HgScrollBufferManager::HgScrollBufferManager(
        int bufferSize,
        int bufferTreshold,
        int initialPosition,
        int totalCount )
:   mBufferSize( bufferSize ),
    mBufferTreshold( bufferTreshold ),
    mBufferPosition( initialPosition ),
    mDiff(0),
    mTotalCount( totalCount ),
    mResetOrdered(false),
    mRequestStart(0),
    mRequestCount(0),
    mReleaseStart(0),
    mReleaseCount(0),
    mFirstTime(true)
    {
    init();
    }

// -----------------------------------------------------------------------------
// HgScrollBufferManager::init()
// -----------------------------------------------------------------------------
//
void HgScrollBufferManager::init()
    {
    mResetOrdered = ETrue;
    mTimer.setSingleShot(true);
    connect(&mTimer, SIGNAL(timeout()), this, SLOT(timeout()));
    }

// -----------------------------------------------------------------------------
// HgScrollBufferManager::~HgScrollBufferManager()
// -----------------------------------------------------------------------------
//
HgScrollBufferManager::~HgScrollBufferManager()
    {
    mTimer.stop();
    }

// -----------------------------------------------------------------------------
// HgScrollBufferManager::resetBuffer()
// -----------------------------------------------------------------------------
//
void HgScrollBufferManager::resetBuffer( int aPosition, int totalCount )
    {
    if( !mResetOrdered )
        {
        // release Old buffer
        mReleaseStart = mBufferPosition;
        mReleaseCount = mBufferSize;
        }

    // set position and count
    mBufferPosition = aPosition - (mBufferSize / 2);
    mTotalCount = totalCount;
    mDiff = 0;

    if( mBufferPosition + mBufferSize > mTotalCount - 1 )
        {
        mBufferPosition = mTotalCount - mBufferSize;
        }

    if(mBufferPosition < 0 )
        {
        mBufferPosition = 0;
        }

    //request new Buffer
    mRequestStart = mBufferPosition;
    mRequestCount = mBufferSize;
    mResetOrdered = ETrue;
    asyncUpdate();
    }

void HgScrollBufferManager::scrollPositionChanged( int newPosition )
    {
    // If all the items fit in the buffer no need to move the buffer.
    if( mTotalCount <= mBufferSize ) return;

    bool forceUpdate = EFalse;
    newPosition -= mBufferSize / 2; // normalize index to Buffer start

    if(newPosition < 0)
        {
        newPosition = 0;
        forceUpdate = ETrue;
        }
    else if( newPosition > mTotalCount - mBufferSize )
        {
        newPosition = mTotalCount - mBufferSize;
        forceUpdate = ETrue;
        }

    mDiff = mBufferPosition - newPosition;

    // Too large change reset whole buffer
    if( mDiff >= mBufferSize || -mDiff >= mBufferSize || mResetOrdered )
        {
        resetBuffer(newPosition + (mBufferSize/2), mTotalCount );
        }
    // Move Up
    else if( mDiff >= mBufferTreshold )
        {
        mRequestCount = mDiff;
        mReleaseCount = mDiff;
        asyncUpdate();
        }
    // Move Down
    else if( -mDiff >= mBufferTreshold )
        {
        mRequestCount = -mDiff;
        mReleaseCount = -mDiff;
        asyncUpdate();
        }
    // Top or bottom has been reached
    else if( forceUpdate && mDiff )
        {
        int diff = mDiff < 0 ? -mDiff : mDiff;
        mRequestCount = diff;
        mReleaseCount = diff;
        asyncUpdate();
        }
    }

void HgScrollBufferManager::timeout()
{
    if(mResetOrdered)
        {
        mResetOrdered = EFalse;
        }
    else
        {
        if(mDiff < 0)
            {
            mReleaseStart = mBufferPosition;
            mRequestStart = mBufferPosition + mBufferSize;
            }
        else if( mDiff > 0)
            {
            mReleaseStart = mBufferPosition + mBufferSize - mDiff;
            mRequestStart = mBufferPosition - mDiff;
            }
        }

    // Release
    int end = mReleaseStart + mReleaseCount < mTotalCount ?
        mReleaseStart + mReleaseCount: mTotalCount;
    end--;
    if(end >= mReleaseStart )
        {
        emit releaseItems( mReleaseStart, end );
        }

    mReleaseCount = 0;

    // Request
    end = mRequestStart + mRequestCount < mTotalCount ?
        mRequestStart + mRequestCount : mTotalCount;

    end--;
    if(end >= mRequestStart )
        {
        emit requestItems( mRequestStart, end );
        }

    mRequestCount = 0;

    // Move Buffer
    mBufferPosition -= mDiff;
    // Reset Diff
    mDiff = 0;
}

bool HgScrollBufferManager::positionInsideBuffer( int position )
{
    return position >= mBufferPosition && position <= (mBufferPosition+mBufferSize);
}

void HgScrollBufferManager::asyncUpdate()
{
    if (!mTimer.isActive() && mFirstTime) {
        mFirstTime = false;
        mTimer.start(0);
    } else {
        timeout();
    }
}

void HgScrollBufferManager::currentBuffer(int& bufferStart, int& bufferEnd)
{
    bufferStart = mBufferPosition;
    bufferEnd = mBufferPosition+mBufferSize > mTotalCount-1 ?
        mTotalCount-1 : mBufferPosition+mBufferSize;
}

void HgScrollBufferManager::addItems(int start, int end)
{
    FUNC_LOG;

    mTotalCount += (end-start+1);
    int lastBufferItem = mBufferPosition+mBufferSize-1;

    if (start < mBufferPosition) {
        simpleAddItems(start, end);
        // New items push the buffer forward, items inside the buffer do not change
    }
    // Check buffer higher limit
    else if (start <= lastBufferItem && end > lastBufferItem) {
        simpleAddItems(start, lastBufferItem);
        // Items added after the buffer are ignored
    }
    else {
        simpleAddItems(start, end);
    }
}

void HgScrollBufferManager::removeItems(int start, int end)
{
    FUNC_LOG;

    int lastBufferItem = mBufferPosition+mBufferSize-1;
    int removedItemCount = end-start+1;

    if (mTotalCount < mBufferSize) {
        // Do nothing
    }
    else if (start > lastBufferItem) {
        // Do nothing
    }
    else if (end < mBufferPosition) {
        mTotalCount = mTotalCount-removedItemCount;
        simpleRemoveItems(start, end);
    }
    else if (start < mBufferPosition && end > lastBufferItem) {
        mTotalCount = mTotalCount-removedItemCount;
        mBufferPosition = qBound(0, mBufferPosition, mTotalCount-mBufferSize);
        resetBuffer(mBufferPosition, mTotalCount);
    }
    // Check buffer higher limit
    else if (start <= lastBufferItem && end > lastBufferItem) {
        mTotalCount = mTotalCount-(end-lastBufferItem);
        simpleRemoveItems(lastBufferItem+1, end);
        mTotalCount = mTotalCount-(lastBufferItem-start+1);
        simpleRemoveItems(start, lastBufferItem);
        // Order does matter
        mTotalCount = mTotalCount-(end-lastBufferItem);
        simpleRemoveItems(lastBufferItem+1, end);
        mTotalCount = mTotalCount-(lastBufferItem-start+1);
        simpleRemoveItems(start, lastBufferItem);
    }
    // Check buffer lower limit
    else if (start < mBufferPosition && end >= mBufferPosition) {
        // Order does matter
        mTotalCount = mTotalCount-(end-mBufferPosition+1);
        simpleRemoveItems(mBufferPosition, end);
        mTotalCount = mTotalCount-(mBufferPosition-start);
        simpleRemoveItems(start, mBufferPosition-1);
    }
    else {
        mTotalCount = mTotalCount-removedItemCount;
        simpleRemoveItems(start, end);
    }
}

void HgScrollBufferManager::moveItems(int start, int end, int target)
{
    int lastBufferItem = mBufferPosition+mBufferSize-1;

    INFO("Move" << start << "-" << end << "to" << target << ",buffer:" << mBufferPosition << "-" << lastBufferItem << "total count:" << mTotalCount);

    if (mTotalCount < mBufferSize) {
        // Do nothing
    }
    else if (start < mBufferPosition && end > lastBufferItem) {
        resetBuffer(mBufferPosition, mTotalCount);
    }
    else if (start > lastBufferItem && target > lastBufferItem) {
        // Do nothing
    }
    else if (start > lastBufferItem && target < mBufferPosition) {
        simpleAddItems(start, end);
    }
    else if (end < mBufferPosition && target < mBufferPosition) {
        // Do nothing
    }
    else if (end < mBufferPosition && target > lastBufferItem) {
        simpleRemoveItems(start, end);
    }
    else if (start >= mBufferPosition && end <= lastBufferItem &&
             target >= mBufferPosition && target <= lastBufferItem) {
        // Do nothing
    }
    else {
        // Rare and complicated use cases: reset the whole buffer
        resetBuffer(mBufferPosition, mTotalCount);
    }
}

void HgScrollBufferManager::flushRequestBuffers()
{
    FUNC_LOG;

    qSort(mReleaseBuffer);
    int releaseCount = mReleaseBuffer.count();
    int lastReleased = -1;
    for (int i = 0; i < releaseCount; i++) {
        UpdatePair update = mReleaseBuffer.at(i);
        emit releaseItems(qMax(lastReleased+1, update.start()), update.end());
        lastReleased = update.end();
    }
    mReleaseBuffer.clear();

    qSort(mRequestBuffer);
    int requestCount = mRequestBuffer.count();
    int lastRequested = -1;
    for (int i = 0; i < requestCount; i++) {
        UpdatePair update = mRequestBuffer.at(i);
        emit requestItems(qMax(lastRequested+1, update.start()), update.end());
        lastRequested = update.end();
    }
    mRequestBuffer.clear();
}

int HgScrollBufferManager::changeBufferPosition(int newPos)
{
    FUNC_LOG;
    INFO("Change buffer position to" << newPos << "total count:" << mTotalCount);
    HANDLE_ERROR_BOOL((newPos >= 0));
    HANDLE_ERROR_BOOL((newPos+mBufferSize <= mTotalCount));

    int bufferShift = newPos-mBufferPosition;
    if (bufferShift > 0) {
        mRequestBuffer.shiftRight(mBufferPosition, bufferShift);
        mReleaseBuffer.shiftRight(mBufferPosition, bufferShift);
    }
    else if (bufferShift < 0) {
        mRequestBuffer.shiftLeft(mBufferPosition, -bufferShift);
        mReleaseBuffer.shiftLeft(mBufferPosition, -bufferShift);
    }
    mBufferPosition = newPos;
    return bufferShift;
}

/**
    This function manages only simple item additions: all items are either
    outside the buffer or inside it.
    Firs call prepare, then update model, then call fecth.
*/
void HgScrollBufferManager::simpleAddItems(int start, int end)
{
    FUNC_LOG;

    int lastBufferItem = mBufferPosition+mBufferSize-1;
    int numAddedItems = end-start+1; // [start, end] inclusive

    if (mTotalCount < mBufferSize) {
        appendRequestBuffer(start, numAddedItems);
    }
    else if (start > lastBufferItem) {
        // Do nothing
    }
    else if (start <= mBufferPosition) {
        changeBufferPosition(mBufferPosition+numAddedItems);
        // No need to fetch items, the indexes just change
    }
    else {
        // free from end
        appendReleaseBuffer(lastBufferItem+1-numAddedItems, numAddedItems);
        mReleaseBuffer.shiftRight(start, numAddedItems);
        mRequestBuffer.shiftRight(start, numAddedItems);
        appendRequestBuffer(start, numAddedItems);
    }
}

/**
    This function manages only simple item removals: all items are either
    outside the buffer or inside it.
    Firs call prepare, then update model, then call fecth.
*/
void HgScrollBufferManager::simpleRemoveItems(int start, int end)
{
    FUNC_LOG;

    int lastBufferItem = mBufferPosition+mBufferSize-1;
    int numRemovedItems = end-start+1; // [start, end] inclusive

    if (start > lastBufferItem) {
        // Do nothing
    }
    else if (end < mBufferPosition) {
        changeBufferPosition(qMax(0, mBufferPosition-numRemovedItems));
        // No need to fetch items, the indexes just change
    }
    else {
        if (mTotalCount < mBufferPosition+mBufferSize) {
            // Buffer is at the end of items
            int bufferShift = changeBufferPosition(qMax(0, mTotalCount-mBufferSize));
            // Fetch from beginning
            // Releasing removed items has been done outside this class
            appendRequestBuffer(mBufferPosition, qAbs(bufferShift));
        }
        else {
            // Fetch from end
            appendRequestBuffer(lastBufferItem+1-numRemovedItems, numRemovedItems);
        }
    }
}

void HgScrollBufferManager::appendRequestBuffer(int start, int count)
{
    FUNC_LOG;
    INFO("Request items" << start << ":" << count)

    mRequestBuffer.add(start, count);
    mReleaseBuffer.remove(start, count);
}

void HgScrollBufferManager::appendReleaseBuffer(int start, int count)
{
    FUNC_LOG;
    INFO("Release items" << start << ":" << count)

    mReleaseBuffer.add(start, count);
    mRequestBuffer.remove(start, count);
}

UpdatePair::UpdatePair(int start, int count) : mStart(start), mCount(count)
{
    HANDLE_ERROR_BOOL(mCount > 0);
}

int UpdatePair::start() const
{
    return mStart;
}

int UpdatePair::end() const
{
    return mStart+mCount-1;
}

bool UpdatePair::adjacent(int start, int count) const
{
    if (start+count < mStart) return false;
    if (start > mStart+mCount) return false;
    return true;
}

bool UpdatePair::contains(const UpdatePair &other) const
{
    if (other.mStart+other.mCount-1 < mStart) return false;
    if (other.mStart > mStart+mCount-1) return false;
    return true;
}

void UpdatePair::extend(int start, int count)
{
    int end = qMax(mStart+mCount, start+count);
    mStart = qMin(mStart, start);
    mCount = end-mStart;
    INFO("Pair extended to:" << mStart << ":" << mCount);
}

void UpdatePair::subtract(int start, int count)
{
    int end = qMin(mStart+mCount, start+count);
    mStart = qMax(mStart, start);
    mCount = end-mStart;
    INFO("Pair reduced to:" << mStart << ":" << mCount);
}

void UpdatePair::shiftRight(int count)
{
    mStart += count;
    INFO("Pair shifted to:" << mStart << ":" << mCount);
}

void UpdatePair::shiftLeft(int count)
{
    mStart -= count;
    HANDLE_ERROR_BOOL((mStart >= 0));
    INFO("Pair shifted to:" << mStart << ":" << mCount);
}

bool UpdatePair::operator== (const UpdatePair &other) const
{
    return mStart == other.mStart && mCount == other.mCount;
}

bool UpdatePair::operator< (const UpdatePair &other) const
{
    return (mStart < other.mStart || mStart == other.mStart && mCount < other.mCount);
}

UpdateBuffer::UpdateBuffer()
{
    FUNC_LOG;
}

void UpdateBuffer::add(int start, int count)
{
    FUNC_LOG;

    int itemCount = this->count();
    for (int i = 0; i < itemCount; i++) {
        if (at(i).contains(UpdatePair(start, count))) {
            // It is already there
            return;
        }
        if (at(i).adjacent(start, count)) {
            (*this)[i].extend(start, count);
            return;
        }
    }
    append(UpdatePair(start, count));
}

void UpdateBuffer::remove(int start, int count)
{
    FUNC_LOG;

    int itemCount = this->count();
    for (int i = itemCount-1; i >= 0; i--) {
        UpdatePair pair = at(i);
        UpdatePair comp = UpdatePair(start, count);
        if (comp.contains(pair)) {
            INFO("Removing pair" << pair.start() << "-" << pair.end());
            removeAt(i);
        }
        else if (pair.contains(comp)) {
            // Subtraction from middle is not applicable in mediawall use cases
            (*this)[i].subtract(start, count);
        }
        // Item may be present in multiple pairs.
    }
}

void UpdateBuffer::shiftRight(int startingFrom, int amount)
{
    FUNC_LOG;

    int itemCount = this->count();
    for (int i = 0; i < itemCount; i++) {
        if (at(i).start() >= startingFrom) {
            (*this)[i].shiftRight(amount);
        }
    }
}

void UpdateBuffer::shiftLeft(int startingFrom, int amount)
{
    FUNC_LOG;

    int itemCount = this->count();
    for (int i = 0; i < itemCount; i++) {
        if (at(i).start() >= startingFrom) {
            (*this)[i].shiftLeft(amount);
        }
    }
}