2011-04-27 16:14:04 -03:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2011 by Daker Fernandes Pinheiro <dakerfp@gmail.com>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Library General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* 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 General Public License for more details
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this program; if not, write to the
|
|
|
|
* Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import QtQuick 1.0
|
|
|
|
import org.kde.plasma.core 0.1 as PlasmaCore
|
2011-12-22 15:16:01 +02:00
|
|
|
import "private" as Private
|
2011-04-27 16:14:04 -03:00
|
|
|
|
|
|
|
// TODO: create a value indicator for plasma?
|
2012-12-17 22:16:05 +01:00
|
|
|
/**
|
|
|
|
* An interactive slider component with Plasma look and feel.
|
|
|
|
*/
|
2011-04-28 07:43:18 -03:00
|
|
|
Item {
|
2011-04-27 16:14:04 -03:00
|
|
|
id: slider
|
|
|
|
|
|
|
|
// Common API
|
2012-12-17 22:16:05 +01:00
|
|
|
/**
|
|
|
|
* type:real
|
|
|
|
* How many steps the slider value can be selected within its range value.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property alias stepSize: range.stepSize
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* type:real
|
|
|
|
* Minimum value that the slider's value can assume.
|
|
|
|
*
|
|
|
|
* The default value is 0.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property alias minimumValue: range.minimumValue
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* type:real
|
|
|
|
* Maximum value that the slider's value can assume.
|
|
|
|
*
|
|
|
|
* The default value is 1.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property alias maximumValue: range.maximumValue
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* type:real
|
|
|
|
* This property holds the value selected inside the minimum to maximum
|
|
|
|
* range of value.
|
|
|
|
*
|
|
|
|
* The default value is 0.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property alias value: range.value
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Orientation for this component.
|
|
|
|
*
|
|
|
|
* The orientation can be either Qt.Horizontal or Qt.Vertical.
|
|
|
|
*
|
|
|
|
* The default is Qt.Horizontal.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property int orientation: Qt.Horizontal
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* type:bool
|
|
|
|
*
|
|
|
|
* True if the Slider is being pressed.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property alias pressed: mouseArea.pressed
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This property holds if a value indicator element will be shown while is
|
|
|
|
* dragged or not.
|
|
|
|
*
|
|
|
|
* @warning The value indicator is not implemented in the Plasma Slider.
|
|
|
|
*
|
|
|
|
* The default value is false.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property bool valueIndicatorVisible: false
|
2012-12-17 22:16:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This property holds the text being displayed in the value indicator.
|
|
|
|
*
|
|
|
|
* @warning The value indicator is not implemented in the Plasma Slider.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property string valueIndicatorText: value
|
|
|
|
|
|
|
|
// Plasma API
|
2012-12-17 22:16:05 +01:00
|
|
|
/**
|
|
|
|
* type:bool
|
|
|
|
* This property holds if the slider visualizations has an inverted
|
|
|
|
* direction.
|
|
|
|
*
|
|
|
|
* The default value is false.
|
|
|
|
*/
|
2011-04-27 16:14:04 -03:00
|
|
|
property alias inverted: range.inverted
|
|
|
|
|
2011-11-21 16:24:18 +01:00
|
|
|
width: contents.isVertical ? theme.defaultFont.mSize.height*1.6 : 200
|
|
|
|
height: contents.isVertical ? 200 : theme.defaultFont.mSize.height*1.6
|
2011-07-18 19:09:50 -03:00
|
|
|
// TODO: needs to define if there will be specific graphics for
|
|
|
|
// disabled sliders
|
|
|
|
opacity: enabled ? 1.0 : 0.5
|
2011-04-27 16:14:04 -03:00
|
|
|
|
2011-07-08 13:18:58 -03:00
|
|
|
Keys.onUpPressed: {
|
2011-11-21 16:24:18 +01:00
|
|
|
if (!enabled || !contents.isVertical)
|
2011-07-08 13:18:58 -03:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (inverted)
|
|
|
|
value -= stepSize;
|
|
|
|
else
|
|
|
|
value += stepSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
Keys.onDownPressed: {
|
2011-07-18 19:09:50 -03:00
|
|
|
if (!enabled || !enabled)
|
|
|
|
return;
|
|
|
|
|
2011-11-21 16:24:18 +01:00
|
|
|
if (!contents.isVertical)
|
2011-07-08 13:18:58 -03:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (inverted)
|
|
|
|
value += stepSize;
|
|
|
|
else
|
|
|
|
value -= stepSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
Keys.onLeftPressed: {
|
2011-11-21 16:24:18 +01:00
|
|
|
if (!enabled || contents.isVertical)
|
2011-07-08 13:18:58 -03:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (inverted)
|
|
|
|
value += stepSize;
|
|
|
|
else
|
|
|
|
value -= stepSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
Keys.onRightPressed: {
|
2011-11-21 16:24:18 +01:00
|
|
|
if (!enabled || contents.isVertical)
|
2011-07-08 13:18:58 -03:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (inverted)
|
|
|
|
value -= stepSize;
|
|
|
|
else
|
|
|
|
value += stepSize;
|
|
|
|
}
|
|
|
|
|
2011-04-27 16:14:04 -03:00
|
|
|
Item {
|
|
|
|
id: contents
|
|
|
|
|
2011-11-21 16:24:18 +01:00
|
|
|
// Plasma API
|
|
|
|
property bool animated: true
|
2012-06-16 14:53:30 +02:00
|
|
|
property real handleWidth: contents.isVertical ? grooveSvg.elementSize("vertical-slider-handle").width : grooveSvg.elementSize("horizontal-slider-handle").width
|
|
|
|
property real handleHeight: contents.isVertical ? grooveSvg.elementSize("vertical-slider-handle").height : grooveSvg.elementSize("horizontal-slider-handle").height
|
2011-11-21 16:24:18 +01:00
|
|
|
|
|
|
|
// Convenience API
|
|
|
|
property bool isVertical: orientation == Qt.Vertical
|
|
|
|
|
|
|
|
width: contents.isVertical ? slider.height : slider.width
|
|
|
|
height: contents.isVertical ? slider.width : slider.height
|
|
|
|
rotation: contents.isVertical ? -90 : 0
|
2011-04-27 16:14:04 -03:00
|
|
|
|
|
|
|
anchors.centerIn: parent
|
|
|
|
|
2011-09-13 21:21:29 +02:00
|
|
|
RangeModel {
|
2011-04-27 16:14:04 -03:00
|
|
|
id: range
|
|
|
|
|
|
|
|
minimumValue: 0.0
|
|
|
|
maximumValue: 1.0
|
|
|
|
value: 0
|
|
|
|
stepSize: 0.0
|
|
|
|
inverted: false
|
2011-11-08 15:12:44 +01:00
|
|
|
positionAtMinimum: 0
|
|
|
|
positionAtMaximum: contents.width - handle.width
|
2011-04-27 16:14:04 -03:00
|
|
|
}
|
|
|
|
|
2011-10-11 14:26:51 +02:00
|
|
|
PlasmaCore.Svg {
|
|
|
|
id: grooveSvg
|
2011-10-11 17:35:23 +02:00
|
|
|
imagePath: "widgets/slider"
|
2011-10-11 14:26:51 +02:00
|
|
|
}
|
|
|
|
PlasmaCore.FrameSvgItem {
|
2011-04-27 16:14:04 -03:00
|
|
|
id: groove
|
2011-10-11 17:35:23 +02:00
|
|
|
imagePath: "widgets/slider"
|
|
|
|
prefix: "groove"
|
2011-10-11 14:26:51 +02:00
|
|
|
//FIXME: frameSvg should have a minimumSize attribute, could be added to kdelibs 4.7(maybe just the qml binding is enough)?
|
2011-10-11 17:35:23 +02:00
|
|
|
height: grooveSvg.elementSize("groove-topleft").height + grooveSvg.elementSize("groove-bottomleft").height
|
2011-04-27 16:14:04 -03:00
|
|
|
anchors {
|
|
|
|
left: parent.left
|
|
|
|
right: parent.right
|
|
|
|
verticalCenter: parent.verticalCenter
|
|
|
|
}
|
2011-10-11 14:26:51 +02:00
|
|
|
}
|
|
|
|
PlasmaCore.FrameSvgItem {
|
|
|
|
id: highlight
|
2011-10-11 17:35:23 +02:00
|
|
|
imagePath: "widgets/slider"
|
|
|
|
prefix: "groove-highlight"
|
2011-10-11 14:26:51 +02:00
|
|
|
height: groove.height
|
2011-11-08 15:12:44 +01:00
|
|
|
width: inverted ? groove.width - handle.x : fakeHandle.x + handle.width/2
|
2011-10-11 14:52:39 +02:00
|
|
|
x: inverted ? handle.x : 0
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
|
2012-06-13 22:48:22 +01:00
|
|
|
//use the same animation when resizing a slider as moving the slider this keeps it in line when using key shortcuts
|
|
|
|
Behavior on width {
|
|
|
|
enabled: !mouseArea.drag.active && contents.animated
|
|
|
|
PropertyAnimation {
|
|
|
|
duration: behavior.enabled ? 150 : 0
|
|
|
|
easing.type: Easing.OutSine
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-11 14:26:51 +02:00
|
|
|
visible: range.position > 0 && slider.enabled
|
2011-04-27 16:14:04 -03:00
|
|
|
}
|
|
|
|
|
2011-12-22 17:34:09 +02:00
|
|
|
Private.RoundShadow {
|
2011-11-08 15:12:44 +01:00
|
|
|
id: shadow
|
|
|
|
imagePath: "widgets/slider"
|
2012-06-16 14:53:30 +02:00
|
|
|
focusElement: contents.isVertical ? "vertical-slider-focus" : "horizontal-slider-focus"
|
|
|
|
hoverElement: contents.isVertical ? "vertical-slider-hover" : "horizontal-slider-hover"
|
|
|
|
shadowElement: contents.isVertical ? "vertical-slider-shadow" : "horizontal-slider-shadow"
|
2011-11-10 17:23:51 +01:00
|
|
|
state: slider.activeFocus ? "focus" : (mouseArea.containsMouse ? "hover" : "shadow")
|
2011-11-08 15:12:44 +01:00
|
|
|
anchors.fill: handle
|
2012-06-16 14:53:30 +02:00
|
|
|
//We rotate the handle below, we need to rotate the shadow back as well
|
|
|
|
rotation: contents.isVertical ? 90 : 0
|
2011-04-27 16:14:04 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
PlasmaCore.SvgItem {
|
|
|
|
id: handle
|
|
|
|
|
|
|
|
x: fakeHandle.x
|
|
|
|
anchors {
|
|
|
|
verticalCenter: groove.verticalCenter
|
|
|
|
}
|
2012-04-27 15:14:43 +02:00
|
|
|
width: contents.handleWidth
|
|
|
|
height: contents.handleHeight
|
2012-06-16 14:53:30 +02:00
|
|
|
//Rotate the handle back for vertical slider so all the handles have the same shadow effect
|
|
|
|
rotation: contents.isVertical ? 90 : 0
|
2011-04-27 16:14:04 -03:00
|
|
|
svg: PlasmaCore.Svg { imagePath: "widgets/slider" }
|
2012-06-16 14:53:30 +02:00
|
|
|
elementId: contents.isVertical ? "vertical-slider-handle" : "horizontal-slider-handle"
|
2011-04-27 16:14:04 -03:00
|
|
|
|
|
|
|
Behavior on x {
|
|
|
|
id: behavior
|
2012-04-27 00:36:26 +02:00
|
|
|
enabled: !mouseArea.drag.active && contents.animated
|
2011-04-27 16:14:04 -03:00
|
|
|
|
|
|
|
PropertyAnimation {
|
|
|
|
duration: behavior.enabled ? 150 : 0
|
|
|
|
easing.type: Easing.OutSine
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Item {
|
|
|
|
id: fakeHandle
|
|
|
|
width: handle.width
|
|
|
|
height: handle.height
|
|
|
|
}
|
|
|
|
|
|
|
|
MouseArea {
|
|
|
|
id: mouseArea
|
|
|
|
|
|
|
|
anchors.fill: parent
|
2011-07-18 19:09:50 -03:00
|
|
|
enabled: slider.enabled
|
2011-04-27 16:14:04 -03:00
|
|
|
drag {
|
|
|
|
target: fakeHandle
|
|
|
|
axis: Drag.XAxis
|
|
|
|
minimumX: range.positionAtMinimum
|
|
|
|
maximumX: range.positionAtMaximum
|
|
|
|
}
|
|
|
|
hoverEnabled: true
|
|
|
|
|
|
|
|
onPressed: {
|
|
|
|
// Clamp the value
|
2011-11-08 15:12:44 +01:00
|
|
|
var newX = Math.max(mouse.x, drag.minimumX)
|
|
|
|
newX = Math.min(newX, drag.maximumX)
|
2011-04-27 16:14:04 -03:00
|
|
|
|
|
|
|
// Debounce the press: a press event inside the handler will not
|
|
|
|
// change its position, the user needs to drag it.
|
2011-11-08 15:12:44 +01:00
|
|
|
if (Math.abs(newX - fakeHandle.x) > handle.width / 2) {
|
2012-04-27 15:14:43 +02:00
|
|
|
range.position = newX - handle.width / 2
|
2011-11-08 15:12:44 +01:00
|
|
|
}
|
2011-04-27 16:14:04 -03:00
|
|
|
|
2011-11-08 15:12:44 +01:00
|
|
|
slider.forceActiveFocus()
|
2011-04-27 16:14:04 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Binding {
|
|
|
|
target: range
|
|
|
|
property: "position"
|
|
|
|
value: fakeHandle.x
|
|
|
|
}
|
|
|
|
|
|
|
|
// During the drag, we simply ignore position set from the range, this
|
|
|
|
// means that setting a value while dragging will not "interrupt" the
|
|
|
|
// dragging activity.
|
|
|
|
Binding {
|
|
|
|
when: !mouseArea.drag.active
|
|
|
|
target: fakeHandle
|
|
|
|
property: "x"
|
|
|
|
value: range.position
|
|
|
|
}
|
2012-06-14 19:26:51 -03:00
|
|
|
}
|