diff --git a/declarativeimports/plasmacomponents/qml/SectionScroller.js b/declarativeimports/plasmacomponents/qml/SectionScroller.js new file mode 100644 index 000000000..7b987fede --- /dev/null +++ b/declarativeimports/plasmacomponents/qml/SectionScroller.js @@ -0,0 +1,71 @@ +/**************************************************************************** +** +** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the Qt Components project. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +var _sectionData = []; +var _sections = []; + +function initSectionData(list) { + if (!list || !list.model) return; + _sectionData = []; + _sections = []; + var current = ""; + var prop = list.section.property; + + for (var i = 0, count = list.model.count; i < count; i++) { + var item = list.model.get(i); + if (item[prop] !== current) { + current = item[prop]; + _sections.push(current); + _sectionData.push({ index: i, header: current }); + } + } +} + +function closestSection(pos) { + var tmp = (_sections.length) * pos; + var val = Math.ceil(tmp) // TODO: better algorithm + val = val < 2 ? 1 : val; + return _sections[val-1]; +} + +function indexOf(sectionName) { + var val = _sectionData[_sections.indexOf(sectionName)].index; + return val === 0 || val > 0 ? val : -1; +} diff --git a/declarativeimports/plasmacomponents/qml/SectionScroller.qml b/declarativeimports/plasmacomponents/qml/SectionScroller.qml new file mode 100644 index 000000000..d93aa2aa3 --- /dev/null +++ b/declarativeimports/plasmacomponents/qml/SectionScroller.qml @@ -0,0 +1,218 @@ +/**************************************************************************** +** +** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). +** All rights reserved. +** Contact: Nokia Corporation (qt-info@nokia.com) +** +** This file is part of the Qt Components project. +** +** $QT_BEGIN_LICENSE:BSD$ +** You may use this file under the terms of the BSD license as follows: +** +** "Redistribution and use in source and binary forms, with or without +** modification, are permitted provided that the following conditions are +** met: +** * Redistributions of source code must retain the above copyright +** notice, this list of conditions and the following disclaimer. +** * Redistributions in binary form must reproduce the above copyright +** notice, this list of conditions and the following disclaimer in +** the documentation and/or other materials provided with the +** distribution. +** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor +** the names of its contributors may be used to endorse or promote +** products derived from this software without specific prior written +** permission. +** +** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE." +** $QT_END_LICENSE$ +** +****************************************************************************/ + +import QtQuick 1.1 +import "SectionScroller.js" as Sections +import org.kde.plasma.core 0.1 as PlasmaCore + +/** + * It's similar to a ScrollBar or a ScrollDecorator. + * It's interactive and works on ListViews that have section.property set, + * so its contents are categorized. + * An indicator will say to what category the user scrolled to. + * + * Useful for things like address books or things sorted by date. + * Don't use with models too big (thousands of items) because implies + * loading all the items to memory, as well loses precision. + */ +Item { + id: root + + /** + * The listview this scroll indicator will work on + */ + property ListView listView + + onListViewChanged: { + if (listView && listView.model) + internal.initDirtyObserver(); + } + + Connections { + target: listView + onModelChanged: { + if (listView && listView.model) + internal.initDirtyObserver(); + } + } + + width: 48 + + anchors { + right: listView.right + top: listView.top + bottom: listView.bottom + } + + RangeModel { + id: range + + minimumValue: 0 + maximumValue: Math.max(0, listView.contentHeight - listView.height) + stepSize: 0 + //inverted: true + positionAtMinimum: handle.height / 2 + positionAtMaximum: root.height - handle.height - handle.height / 2 + value: listView.contentY + onValueChanged: { + if (listView.moving) { + return + } else { + listView.contentY = value + } + } + //position: handle.y + onPositionChanged: { + if (!dragArea.pressed) { + handle.y = position + } + } + } + + Rectangle { + anchors.fill: parent + color: Qt.rgba(0,0,0,0.3) + } + + PlasmaCore.Theme { + id: theme + } + + Rectangle { + id: handle + width: 6 + height: 6 + color: theme.textColor + opacity: 0.7 + anchors.horizontalCenter: parent.horizontalCenter + border { + width: 1 + color: theme.backgroundColor + } + onYChanged: { + if (dragArea.pressed) { + range.position = y + } + sectionLabel.text = Sections.closestSection(y/listView.height) + } + Behavior on y { + NumberAnimation { + duration: 150 + } + } + } + PlasmaCore.FrameSvgItem { + imagePath: "widgets/tooltip" + width: childrenRect.width + margins.left + margins.right + height: childrenRect.height + margins.top + margins.bottom + Label { + id: sectionLabel + font.pointSize: theme.defaultFont.pointSize*3 + x: parent.margins.left + y: parent.margins.top + } + y: Math.min(root.height-height, Math.max(0, handle.y - height/2)) + anchors { + //verticalCenter: handle.verticalCenter + right: parent.left + } + opacity: dragArea.pressed?1:0 + Behavior on opacity { + NumberAnimation { + duration: 250 + } + } + } + /*Repeater { + id: sectionsRepeater + delegate: Label { + anchors.horizontalCenter: parent.horizontalCenter + text: Sections._sections[modelData] + y: Sections._sectionData[modelData].index*(listView.height/listView.model.count) + } + }*/ + MouseArea { + id: dragArea + anchors.fill: parent + enabled: scrollbar.enabled + drag { + target: handle + axis: Drag.YAxis + minimumY: range.positionAtMinimum + maximumY: range.positionAtMaximum + } + onPressed: { + mouse.accepted = true + handle.y = mouse.y + } + + } + + QtObject { + id: internal + + function initDirtyObserver() { + Sections.initSectionData(listView); + function dirtyObserver() { + if (!internal.modelDirty) { + internal.modelDirty = true; + dirtyTimer.running = true; + } + } + + if (listView.model.countChanged) + listView.model.countChanged.connect(dirtyObserver); + + if (listView.model.itemsChanged) + listView.model.itemsChanged.connect(dirtyObserver); + + if (listView.model.itemsInserted) + listView.model.itemsInserted.connect(dirtyObserver); + + if (listView.model.itemsMoved) + listView.model.itemsMoved.connect(dirtyObserver); + + if (listView.model.itemsRemoved) + listView.model.itemsRemoved.connect(dirtyObserver); + + sectionsRepeater.model = Sections._sections.length + } + } +} diff --git a/declarativeimports/plasmacomponents/qml/qmldir b/declarativeimports/plasmacomponents/qml/qmldir index 3a296af05..0a0efb355 100644 --- a/declarativeimports/plasmacomponents/qml/qmldir +++ b/declarativeimports/plasmacomponents/qml/qmldir @@ -24,7 +24,9 @@ ToolBar 0.1 ToolBar.qml ToolButton 0.1 ToolButton.qml ListItem 0.1 ToolButton.qml + TabBar 0.1 TabBar.qml TabButton 0.1 TabButton.qml TabGroup 0.1 TabGroup.qml +SectionScroller 0.1 SectionScroller.qml