src/hbwidgets/itemviews/hbtreeview.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Wed, 23 Jun 2010 18:33:25 +0300
changeset 6 c3690ec91ef8
parent 5 627c4a0fd0e7
permissions -rw-r--r--
Revision: 201023 Kit: 2010125

/****************************************************************************
**
** 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 HbWidgets 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 "hbtreeview.h"
#include "hbtreeview_p.h"
#include "hbtreeitemcontainer_p.h"
#include "hbtreeviewitem.h"
#include "hbtreeitemselectionmodel_p.h"
#include "hbtreemodeliterator_p.h"

#include <QItemSelection>
#include <QGraphicsSceneMouseEvent>

/*!
    @alpha
    @hbwidgets
    \class HbTreeView
    \brief HbTreeView represents a hierarchical list

    The HbTreeView class provides a default model/view implementation of a tree view.

    A HbTreeView implements a tree representation of items from a model. This class is used to provide standard hierarchical 
    lists.

    HbTreeView implements the interfaces defined by the HbAbstractItemView class to allow it to display data provided 
    by models derived from the QAbstractItemModel class. 

    It is simple to construct a tree view displaying data from a model. In the following example, the contents of a directory 
    are supplied by a QDirModel and displayed as a tree:

    \snippet{ultimatecodesnippet/ultimatecodesnippet.cpp,43}

    The model/view architecture ensures that the contents of the tree view are updated as the model changes.

    Items that have children can be in an expanded (children are visible) or collapsed (children are hidden) state. 

    Each tree view item is represented by an instance of HbTreeViewItem. HbTreeView
    uses HbTreeViewItem prototype to instantiate the tree view items. HbTreeViewItem
    can be subclassed for customization purposes.
    
    By default, HbTreeView uses item recycling. This means that only the
    visible tree view items plus a small buffer of items above and below the visible
    list are instantiated at a time. When the view is scrolled the tree view items are
    recycled so that the buffer size above and below the tree view is kept constant.
*/

/*!
    Constructs a tree view with \a parent.
 */
HbTreeView::HbTreeView(QGraphicsItem *parent)
    : HbAbstractItemView(*new HbTreeViewPrivate(), new HbTreeItemContainer(), new HbTreeModelIterator(), parent)
{
    Q_D( HbTreeView );

    d->q_ptr = this;
    d->init();
}

/*!
    \private
*/
HbTreeView::HbTreeView(HbTreeViewPrivate &dd, HbAbstractItemContainer *container, QGraphicsItem *parent) :
    HbAbstractItemView(dd, container, new HbTreeModelIterator(), parent)
{
    Q_D( HbTreeView );

    d->q_ptr = this;
    d->init();
}


/*!
    Destructs the tree view.
 */
HbTreeView::~HbTreeView()
{
}

/*!
    \reimp
*/
void HbTreeView::scrollTo(const QModelIndex &index, ScrollHint hint)
{
    Q_D(HbTreeView);

    if (    !index.isValid()
        ||  index.model() != d->mModelIterator->model()) {
        return;
    }

    QModelIndex newIndex = index;
    if (!d->mContainer->itemByIndex(index)) {
        QModelIndex parentIndex = index.parent();
        QModelIndex root = rootIndex();
        if (d->isParentValid(parentIndex) || root == index) {
            while (parentIndex.isValid() && parentIndex != root) {
                if (!isExpanded(parentIndex)) {
                    newIndex = parentIndex;
                }
                parentIndex = parentIndex.parent();
            }
        }
    }

    if ( itemRecycling()) {
        if (    !d->mContainer->itemByIndex(newIndex)
            ||  hint != EnsureVisible) {
            if ( hint != PositionAtTop ) {
                // Following two variable applies only for hint EnsureVisible.
                // It is position relative to the view
                bool belowBottom = false;
                bool aboveTop = false;
                if (hint == EnsureVisible) {
                    QModelIndex firstVisibleIndex;
                    QModelIndex lastVisibleIndex;
                    d->mContainer->firstAndLastVisibleModelIndex(firstVisibleIndex, lastVisibleIndex);

                    belowBottom = d->treeModelIterator()->lessThan(lastVisibleIndex, newIndex);
                    aboveTop = d->treeModelIterator()->lessThan(newIndex, firstVisibleIndex);
                }

                if (    hint == PositionAtBottom 
                    ||  hint == PositionAtCenter
                    ||  belowBottom ) {
                    // resolve index to be put to top of the recycle buffer
                    int stepCount = 0;
                    if (hint == PositionAtCenter) {
                        stepCount = d->mContainer->items().count() / 2;
                    } else {
                        stepCount = d->mContainer->items().count()-1;
                    }
                    QModelIndex topIndex = d->searchIndexUp(newIndex, stepCount);
                    d->mContainer->setModelIndexes(topIndex);
                } else if (aboveTop) {
                    d->mContainer->setModelIndexes(newIndex);
                }
                // else - hint is EnsureVisible & item is already visible
            } else {
                d->mContainer->setModelIndexes(newIndex);
            }
        }
    }
    HbAbstractItemView::scrollTo(newIndex, hint);
}

/*!
    \reimp
*/
void HbTreeView::rowsInserted(const QModelIndex &parent, int start, int end)
{
    Q_D(HbTreeView);

    if (d->mModelIterator->model()->columnCount(parent) == 0) {
        return;
    }

    if (d->isParentValid(parent)) {
        if (isExpanded(parent) || parent == d->mModelIterator->rootIndex()) {
            for (int i = start; i <= end; ++i) {
                HbAbstractItemView::rowsInserted(parent, i, i);
                // if there is expanded parent under expanding parent, handle it recursively
                QModelIndex newParent = d->treeModelIterator()->model()->index(i, 0, parent);
                int childCount = d->treeModelIterator()->childCount(newParent);
                if (childCount > 0 && isExpanded(newParent)) {
                    rowsInserted(newParent, 0, childCount - 1);
                } 
            }

            if (d->animationEnabled(true)) {
                if (d->mInSetExpanded) {
                    d->startAppearEffect("treeviewitem", "expand", parent, start, end);
                } else {
                    d->startAppearEffect("viewitem", "appear", parent, start, end);
                }
            }
        }

        HbAbstractViewItem *parentItem = d->mContainer->itemByIndex(parent);
        if (parentItem) {
            parentItem->updateChildItems();
        }
    }
}

/*!
    \reimp
*/
void HbTreeView::rowsRemoved(const QModelIndex &parent, int start, int end)
{
    Q_D(HbTreeView);

    if (d->isParentValid(parent)) {
        if (parent == d->mCurrentIndex.parent() 
            && start <= d->mCurrentIndex.row()
            && d->mCurrentIndex.row() <= end) {
            // new current: 1) next after last deleted (note that
            // start and end index in model prior to deleting)
            // 2) just before first deleted
            QModelIndex newCurrentIndex = d->mModelIterator->nextIndex(d->mCurrentIndex);
            if (!newCurrentIndex.isValid()) {
                newCurrentIndex = d->mModelIterator->previousIndex(d->mCurrentIndex);
            }

            if (d->mSelectionModel) {
                d->mSelectionModel->setCurrentIndex(newCurrentIndex, QItemSelectionModel::NoUpdate);
            } 
        }  

        if (isExpanded(parent) || parent == d->mModelIterator->rootIndex() || d->mInSetExpanded) {
            bool animate = d->animationEnabled(false);

            QList<HbAbstractViewItem *> items = d->mContainer->items();
            for (int i = d->mItemsToBeDeleted.count() - 1; i >= 0; --i) {
                int pos = d->mItemsToBeDeleted.at(i);
                
                HbAbstractViewItem *item = items.at(pos);
                if (item) {
                    if (animate) {
                        HbEffect::cancel(item, "appear");
                        HbEffect::cancel(item, "expand");
                        
                        QString effectType;
                        QString itemType;
                        if (d->mInSetExpanded) {
                            effectType = "collapse";
                            itemType = "treeviewitem";
                        } else {
                            effectType = "disappear";
                            itemType = "viewitem";
                        }

                        item->setFlag(QGraphicsItem::ItemSendsGeometryChanges, true);
                        HbEffect::start(item,
                                itemType,
                                effectType,
                                d->mContainer,
                                "animationFinished");
                        
                        if (HbEffect::effectRunning(item, effectType)) {
                            d->mContainer->removeItem(pos, animate);
                        }
                    } else {
                        d->mContainer->removeItem(pos, animate);
                    }
                }
            }
        }

        HbAbstractViewItem *parentItem = d->mContainer->itemByIndex(parent);
        if (parentItem) {
            parentItem->updateChildItems();
        }
    }
}

/*!
    \reimp
*/
void HbTreeView::rowsAboutToBeRemoved(const QModelIndex &index, int start, int end)
{
    Q_D(HbTreeView);

    if (d->isParentValid(index)) {
        d->mItemsToBeDeleted.clear();
        
        QList <HbAbstractViewItem *> items = d->mContainer->items();
        int itemCount = items.count();

        // Add the view items given as parameters and all their child items to a list for deletion.
        for (int i = start; i <= end; ++i) {
            QModelIndex parent = model()->index(i, 0, index);

            for (int j = 0; j < itemCount; ++j) {
                QModelIndex itemIndex = items.at(j)->modelIndex();
                if (itemIndex == parent || d->isChild(itemIndex, parent)) {
                    d->mItemsToBeDeleted.append(j);
                }
            }
        }
    }
}

/*!
    \reimp
*/
void HbTreeView::columnsInserted(const QModelIndex &parent, int start, int end)
{
    Q_UNUSED(end)
    Q_UNUSED(start)

    Q_D(HbTreeView);

    if (d->isParentValid(parent)) {
        if (d->mModelIterator->model()->columnCount(parent) == 1){
            rowsInserted(parent, 0, d->mModelIterator->indexCount(parent));
        }
    }
}

/*!
    \reimp
*/
void HbTreeView::columnsRemoved(const QModelIndex &parent, int start, int end)
{
    Q_UNUSED(end)
    Q_UNUSED(start)

    Q_D(HbTreeView);

    if (d->isParentValid(parent)) {
        if (d->mModelIterator->model()->columnCount(parent) == 0){
            rowsRemoved(parent, 0, d->mModelIterator->indexCount(parent));
        }
    }

}

/*!
    \reimp
*/
void HbTreeView::dataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight)
{
    Q_D(HbTreeView);
    if (d->isParentValid(topLeft.parent())) {
        HbAbstractItemView::dataChanged(topLeft, bottomRight);
    }
}


/*!
    \reimp
    This function is involded in partial selecting and deselecting parent items.
    Selecting items are implemented by internal selection model class.
*/
void HbTreeView::currentSelectionChanged(const QItemSelection &selected, const QItemSelection &deselected)
{
    Q_D(HbTreeView);
    bool selectionRoot = false;
    if (!d->mSelectionStarted) {
        d->mSelectionStarted = true;
        selectionRoot = true;
    }

    HbAbstractItemView::currentSelectionChanged(selected, deselected);

    if (selectionRoot) {
        QModelIndexList selectedIndexes(selected.indexes());
        QMap<QModelIndex, Qt::CheckState> parentItems;

        int count = selectedIndexes.count();
        for (int i = 0; i < count; ++i) {
            QModelIndex index = selectedIndexes.at(i);

            // Parents are always partially selected when one of the
            // children is selected.
            QModelIndex parentIndex = index.parent();
            if (!parentItems.contains(parentIndex)) {
                while (parentIndex != d->mModelIterator->rootIndex()) {
                    if (!d->mSelectionModel->isSelected(parentIndex)) {
                        parentItems.insert(parentIndex, Qt::PartiallyChecked);
                    } 
                    // else - occurs when node is selected. Node
                    // may come first in the selection list and only after that its children
                    parentIndex = parentIndex.parent();
                }
            }
        }

        QModelIndexList deselectedIndexes(deselected.indexes());
        QItemSelection deselectedChildren;

        count = deselectedIndexes.count();
        for (int i = 0; i < count; ++i) {
            QModelIndex index = deselectedIndexes.at(i);
            bool solveParentCheckState = true;

            // Solves the partial selection/non-selection state for the parents
            // by checking status of siblings. Parent will PartiallyChecked, if
            // sibling is either checked or partially checked.

            // There is one exception, which this slot cannot handle. It is handled by HbTreeItemSelectionModel::select().
            // See its documentation
            QModelIndex parentIndex = index.parent();
            while (parentIndex != d->mModelIterator->rootIndex()) {
                Qt::CheckState parentCheckState = Qt::PartiallyChecked;
                if (solveParentCheckState) {
                    int childCount = model()->rowCount(parentIndex);
                    for (int current = 0; current < childCount; ++current) {
                        QModelIndex childIndex = parentIndex.child(current, parentIndex.column());

                        // Resolves child check state.
                        Qt::CheckState childCheckState = Qt::Unchecked;
                        if (d->mSelectionModel->isSelected(childIndex)) {
                            childCheckState = Qt::PartiallyChecked;
                        } else if (parentItems.contains(childIndex)) {
                            // handles use case: node1, which has child node2, which has child leaf1.
                            // Initially node1 is selected, then leaf1 is selected.
                            // Node2 may have been set Unchecked into map before node1.
                            // When node1 will be handled, node2 is unchecked in the map but not yet in the view item
                            childCheckState = parentItems.value(childIndex);
                        } else {
                            QVariant value = d->mContainer->itemTransientState(childIndex).value("checkState");
                            if (value.isValid()) {
                                childCheckState = (Qt::CheckState)value.toInt();
                            }
                        }
                        if (childCheckState != Qt::Unchecked) {
                            // No need to solve further parents check state as
                            // one child item is selected -> all parents are partially
                            // selected.
                            solveParentCheckState = false;
                            break;
                        }

                        if (current == childCount - 1) {
                            parentCheckState = Qt::Unchecked;
                        }
                    }   
                }
                if (!parentItems.contains(parentIndex)) {
                    // handling use case: parent initially multi selected, then a leaf is multi deselected
                    QItemSelection selection(parentIndex, parentIndex);
                    // Using HbTreeItemSelectionModel must be bypassed, because it would deselect every children, too
                    d->mSelectionModel->QItemSelectionModel::select(selection, QItemSelectionModel::Deselect);
                }
                parentItems.insert(parentIndex, parentCheckState);
                parentIndex = parentIndex.parent();
            }
        }

        QMap<QModelIndex, Qt::CheckState>::iterator iterator = parentItems.begin();
        while (iterator != parentItems.end()) {
            HbAbstractViewItem *item = d->mContainer->itemByIndex(iterator.key());
            if (item) {
                item->setCheckState(iterator.value());
            } 

            d->mContainer->setItemTransientStateValue(iterator.key(), "checkState", iterator.value());
            iterator++;
        }

        d->mSelectionStarted = false;
    }
}

/*!
    Sets the item referred to by \a index to either collapse or expanded, depending on the value of \a expanded.

    \sa isExpanded()
*/
void HbTreeView::setExpanded(const QModelIndex &index, bool expanded)
{
    Q_D(HbTreeView);

    if (isExpanded(index) != expanded) {
        d->mInSetExpanded = true;
        d->treeModelIterator()->itemExpansionChanged(index);

        d->mContainer->setItemTransientStateValue(index, "expanded", expanded);

        int childCount = d->treeModelIterator()->childCount(index) - 1;

        if (expanded) {
            rowsAboutToBeInserted(index, 0, childCount);
            rowsInserted(index, 0, childCount);
        } else {
            rowsAboutToBeRemoved(index, 0, childCount);
            rowsRemoved(index, 0, childCount);
        }

        HbTreeViewItem *item = qobject_cast<HbTreeViewItem *>(itemByIndex(index));
        if (item) {
            item->setExpanded(expanded);
        } 

        d->mInSetExpanded = false;
    }
}

/*!
    Returns true if the model item \a index is expanded; otherwise returns false.

    \sa setExpanded()
*/
bool HbTreeView::isExpanded(const QModelIndex &index) const
{
    Q_D(const HbTreeView);

    QVariant flags = d->mContainer->itemTransientState(index).value("expanded");
    if (flags.isValid() && flags.toBool() == true) {
        return true;
    } else {
        return false;
    }
}
/*!
    Overrides default indentation of tree view items defined in style sheet.
    Items, which are not immediate children of root item are indented by
    indentation * level pixels. 
    Level is count of parent items between item itself and root item.

    If negative value is set, then indentation from style sheet is used.

    \sa indentation()
*/
void HbTreeView::setIndentation(qreal indentation)
{
    Q_D(HbTreeView);
    Q_ASSERT_X(qobject_cast<HbTreeItemContainer*>(d->mContainer), "HbTreeView::setIndentation", "wrong container type");
    qobject_cast<HbTreeItemContainer*>(d->mContainer)->setIndentation(indentation);
}

/*!
    Returns indentation of tree view items. 
    The returned value is either default value or value set by setIndentation().

    Default value is -1. In this case indentation from style sheet is used.

    \sa setIndentation()
*/
qreal HbTreeView::indentation() const
{
    Q_D(const HbTreeView);
    Q_ASSERT_X(qobject_cast<HbTreeItemContainer*>(d->mContainer), "HbTreeView::indentation", "wrong container type");
    return qobject_cast<HbTreeItemContainer*>(d->mContainer)->indentation();
}

/*!
    \reimp
*/
void HbTreeView::reset()
{
    HbAbstractItemView::reset();
}


/*!
    \reimp
*/
void HbTreeView::setRootIndex(const QModelIndex &index)
{
    //Q_D(HbTreeView);
    HbAbstractItemView::setRootIndex(index);
    setExpanded(index, true); 
}

/*!
    \reimp
    Tree view has its internal selection model. Implementation of this virtual function is needed
    to take it into use with new model..
*/
void HbTreeView::setModel(QAbstractItemModel *model, HbAbstractViewItem *prototype)
{
    Q_D(HbTreeView);
    HbAbstractItemView::setModel(model, prototype);
    setSelectionModel(new HbTreeItemSelectionModel(model, d, this));
}

#include "moc_hbtreeview.cpp"