~elopio/ubuntu-ui-toolkit/toolbar_emulator

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
/*
 * Copyright 2012 Canonical Ltd.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation; version 3.
 *
 * 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 Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */

// FIXME(loicm) An option should be added to render the value text outside of
//     the thumb, on top or bottom of the thumb (or left and right once we got
//     support for vertical orientation). Moreover, having the value text inside
//     the thumb makes it complicated to fit numbers with high values (or high
//     number of decimals) or visually awkward since the text is hidden by the
//     mouse cursor (or the finger) when dragged.

// FIXME(loicm) An animation should be added when the user clicks on an area
//     outside the thumb to smoothly move it under the mouse.

// FIXME(loicm) Maybe add support for vertical orientation (orientation
//     property). Don't forget to adapt the inverted property documentation
//     accordingly. Note that UIKit doesn't have vertical sliders.

// FIXME(loicm) Add support for keyboard shortcuts (basically left/right).

import QtQuick 2.0
import "mathUtils.js" as MathUtils
import "sliderUtils.js" as SliderFuncs
// FIXME: When a module contains QML, C++ and JavaScript elements exported,
// we need to use named imports otherwise namespace collision is reported
// by the QML engine. As workaround, we use Theming named import.
// Bug to watch: https://bugreports.qt-project.org/browse/QTBUG-27645
import "." 0.1 as Theming

/*!
    \qmltype Slider
    \inqmlmodule Ubuntu.Components 0.1
    \ingroup ubuntu
    \brief Slider is a component to select a value from a continuous range of
     values.

    The slider's sensing area is defined by the width and height, therefore
    delegates should take this into account when defining the visuals, and
    alter these values to align the graphics' sizes.

    \l {http://design.ubuntu.com/apps/building-blocks/slider}{See also the Design Guidelines on Sliders}.

    Example:
    \qml
    Item {
        Slider {
            function formatValue(v) { return v.toFixed(2) }
            minimumValue: -3.14
            maximumValue: 3.14
            value: 0.0
            live: true
        }
    }
    \endqml

    \section2 Theming

    The slider's default style class is \b slider and style properties depend on
    the actual delegate defined by the theme, except of one property which defines
    the spacing units between the slider's bar and thumb, called \b thumbSpacing.
    The slider uses one single touch sensing area to position the thumb within the
    bar. Therefore However delegates must define the following properties:
    \list
    \li * \b bar - the slider's bar object
    \li * \b thumb - the slider's thumb object
    \endlist

    Beside these, the library provies functions for delegates to update liveValue and
    normalizedValue in SliderUtils module.

    \b{This component is under heavy development.}
*/
AbstractButton {
    id: slider

    width: units.gu(38)
    height: units.gu(5)

    Theming.ItemStyle.class: "slider"

    // FIXME(loicm) Add Support for the inverted property. There's an ongoing
    //     debate on whether we should use that property (like every other
    //     toolkits) or add new enumerations to the orientation property and
    //     get something like LeftToRight, RightToLeft, TopToBottom and
    //     BottomToTop. The last proposition appears to have the advantage that
    //     it's not culture centric (ie inverted for LeftToRight and RightToLeft
    //     isn't natural for everybody).

    // /*!
    //    \preliminary
    //    Selects whether or not the minimum value starts from the right.
    // */
    // property bool inverted: false

    /*!
       \preliminary
       The minimum value from the continuous range of values. If this value is
       greater than maximumValue, the component will be in an inconsistent
       state.
    */
    property real minimumValue: 0.0

    /*!
       \preliminary
       The maximum value from the continuous range of values. If this value is
       lesser than minimumValue, the component will be in an inconsistent state.
    */
    property real maximumValue: 100.0

    // FIXME(loicm) Add Support for the stepSize property.

    // /*!
    //    \preliminary
    //    The distance between two selectable values in the range defined by
    //    [minimumValue, maximumValue].
    // */
    // property real stepSize: 1.0

    /*!
       \preliminary
       The current value of the slider. This property is not changed while the
       thumb is dragged unless the live property is set to true.
    */
    property real value: 0.0

    /*!
       \preliminary
       Defines whether the value is updated while the thumb is dragged or just
       when the thumb is released.
    */
    property bool live: false


    /*!
       \preliminary
       The signal is emitted when there is a click on the slider. The onThumb parameter provides information if the click, was inside of the thumb element.
    */
    signal touched(bool onThumb)


    /*!
      \preliminary
      This function is used by the value indicator to show the current value.
      Reimplement this function if you want to show different information. By
      default, the value v is rounded to the nearest interger value.

      \b Note: this function will be deprecated, and will be solved with particular
      delegates for the thumb.
     */
    function formatValue(v) {
        return v.toFixed(0)
    }

    // Private symbols.

    /*! \internal */
    onValueChanged: internals.liveValue = slider.value

    /*! \internal */
    Component.onCompleted: internals.updateMouseArea()

    /*! \internal */
    onPressedChanged: internals.mouseAreaPressed()

    /*! \internal */
    property alias __internals: internals
    QtObject {
        id: internals

        property real thumbSpacing: slider.Theming.ItemStyle.delegate ? slider.Theming.ItemStyle.delegate.thumbSpacing : 0
        property Item bar: slider.Theming.ItemStyle.delegate ? slider.Theming.ItemStyle.delegate.bar : null
        property Item thumb: slider.Theming.ItemStyle.delegate ? slider.Theming.ItemStyle.delegate.thumb :  null

        property real liveValue: 0.0
        property real normalizedValue: MathUtils.clamp((liveValue - slider.minimumValue) /
                                                         (slider.maximumValue - slider.minimumValue),
                                                         0.0, 1.0)

        property real dragInitMouseX: 0.0
        property real dragInitNormalizedValue: 0.0
        property real thumbWidth: thumb ? thumb.width - thumbSpacing : 0.0
        property real thumbSpace: bar ? bar.width - (2.0 * thumbSpacing + thumbWidth) : 0.0

        function updateMouseArea() {
            slider.__mouseArea.positionChanged.connect(internals.mouseAreaPositionchanged);
        }

        function mouseAreaPressed() {
            if (!thumb || !bar)
                return;
            if (slider.__mouseArea.pressedButtons == Qt.LeftButton) {
                // Left button pressed.
                var mouseX = slider.__mouseArea.mouseX;
                var mouseY = slider.__mouseArea.mouseY;
                if (mouseY >= thumb.y && mouseY <= thumb.y + thumb.height) {
                    if (mouseX >= thumb.x && mouseX <= thumb.x + thumb.width) {
                        // Button pressed inside the thumb.
                        dragInitMouseX = mouseX;
                        dragInitNormalizedValue = normalizedValue;
                        slider.touched(true);
                    } else if (mouseX > thumbSpacing &&
                               mouseX < bar.width - thumbSpacing) {
                        // Button pressed outside the thumb.
                        var normalizedPosition = (slider.__mouseArea.mouseX - thumbSpacing - thumbWidth * 0.5) / thumbSpace;
                        normalizedPosition = MathUtils.clamp(normalizedPosition, 0.0, 1.0);
                        liveValue = MathUtils.lerp(normalizedPosition, slider.minimumValue, slider.maximumValue);
                        dragInitMouseX = mouseX;
                        dragInitNormalizedValue = normalizedValue;
                        if (slider.live) {
                            slider.value = liveValue;
                        }
                        slider.touched(false);
                    }
                }
            } else {
                // Button released.
                if (!slider.live) {
                    slider.value = liveValue;
                }
            }
        }

        function mouseAreaPositionchanged() {
            // Left button dragging.
            if (slider.pressed) {
                var normalizedOffsetX = (slider.__mouseArea.mouseX - dragInitMouseX) / thumbSpace;
                var v = MathUtils.clamp(dragInitNormalizedValue + normalizedOffsetX, 0.0, 1.0);
                liveValue = MathUtils.lerp(v, slider.minimumValue, slider.maximumValue);
                if (slider.live) {
                    slider.value = liveValue;
                }
            }
        }
    }
}