2011-07-12 16:12:28 -04:00
|
|
|
/* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */
|
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
import Atk from 'gi://Atk';
|
|
|
|
import Clutter from 'gi://Clutter';
|
|
|
|
import GObject from 'gi://GObject';
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
import * as BarLevel from './barLevel.js';
|
2018-02-08 19:04:42 +01:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
const SLIDER_SCROLL_STEP = 0.02; /* Slider scrolling step in % */
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
export const Slider = GObject.registerClass({
|
2019-07-25 18:53:00 +02:00
|
|
|
Signals: {
|
|
|
|
'drag-begin': {},
|
2019-08-20 23:43:54 +02:00
|
|
|
'drag-end': {},
|
|
|
|
},
|
2019-07-25 18:53:00 +02:00
|
|
|
}, class Slider extends BarLevel.BarLevel {
|
|
|
|
_init(value) {
|
|
|
|
super._init({
|
|
|
|
value,
|
|
|
|
style_class: 'slider',
|
|
|
|
can_focus: true,
|
2018-02-08 19:04:42 +01:00
|
|
|
reactive: true,
|
2019-10-21 20:44:00 +02:00
|
|
|
accessible_role: Atk.Role.SLIDER,
|
|
|
|
x_expand: true,
|
2019-07-25 18:53:00 +02:00
|
|
|
});
|
2018-02-08 19:04:42 +01:00
|
|
|
|
2019-09-10 07:42:48 +02:00
|
|
|
this._releaseId = 0;
|
2011-07-12 16:12:28 -04:00
|
|
|
this._dragging = false;
|
2013-08-22 23:15:35 +02:00
|
|
|
|
2023-10-26 20:09:23 +02:00
|
|
|
this._handleRadius = 0;
|
|
|
|
this._handleBorderWidth = 0;
|
|
|
|
this._handleBorderColor = null;
|
|
|
|
|
2017-12-02 01:27:35 +01:00
|
|
|
this._customAccessible.connect('get-minimum-increment', this._getMinimumIncrement.bind(this));
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-10-26 20:09:23 +02:00
|
|
|
vfunc_style_changed() {
|
|
|
|
super.vfunc_style_changed();
|
|
|
|
|
|
|
|
const themeNode = this.get_theme_node();
|
|
|
|
this._handleRadius = themeNode.get_length('-slider-handle-radius');
|
|
|
|
this._handleBorderWidth =
|
|
|
|
themeNode.get_length('-slider-handle-border-width');
|
|
|
|
const [hasHandleColor, handleBorderColor] =
|
|
|
|
themeNode.lookup_color('-slider-handle-border-color', false);
|
|
|
|
this._handleBorderColor = hasHandleColor ? handleBorderColor : null;
|
|
|
|
}
|
|
|
|
|
2019-07-25 18:53:00 +02:00
|
|
|
vfunc_repaint() {
|
|
|
|
super.vfunc_repaint();
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2018-02-08 19:04:42 +01:00
|
|
|
// Add handle
|
2019-07-25 18:53:00 +02:00
|
|
|
let cr = this.get_context();
|
|
|
|
let themeNode = this.get_theme_node();
|
|
|
|
let [width, height] = this.get_surface_size();
|
2023-09-06 16:30:19 +03:00
|
|
|
const rtl = this.get_text_direction() === Clutter.TextDirection.RTL;
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-10-26 20:09:23 +02:00
|
|
|
const ceiledHandleRadius = Math.ceil(this._handleRadius + this._handleBorderWidth);
|
2020-03-01 14:01:14 +01:00
|
|
|
const handleY = height / 2;
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-09-06 16:30:19 +03:00
|
|
|
let handleX = ceiledHandleRadius +
|
|
|
|
(width - 2 * ceiledHandleRadius) * this._value / this._maxValue;
|
|
|
|
if (rtl)
|
|
|
|
handleX = width - handleX;
|
|
|
|
|
2011-07-12 16:12:28 -04:00
|
|
|
let color = themeNode.get_foreground_color();
|
2023-07-18 15:50:48 +02:00
|
|
|
cr.setSourceColor(color);
|
2023-10-26 20:09:23 +02:00
|
|
|
cr.arc(handleX, handleY, this._handleRadius, 0, 2 * Math.PI);
|
2011-07-12 16:12:28 -04:00
|
|
|
cr.fillPreserve();
|
2023-10-26 20:09:23 +02:00
|
|
|
if (this._handleBorderColor && this._handleBorderWidth) {
|
|
|
|
cr.setSourceColor(this._handleBorderColor);
|
|
|
|
cr.setLineWidth(this._handleBorderWidth);
|
2013-06-21 14:24:06 -04:00
|
|
|
cr.stroke();
|
2011-07-12 16:12:28 -04:00
|
|
|
}
|
|
|
|
cr.$dispose();
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-10-26 13:39:38 +02:00
|
|
|
_getPreferredHeight() {
|
|
|
|
const barHeight = super._getPreferredHeight();
|
|
|
|
const handleHeight = 2 * this._handleRadius + this._handleBorderWidth;
|
|
|
|
return Math.max(barHeight, handleHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
_getPreferredWidth() {
|
|
|
|
const barWidth = super._getPreferredWidth();
|
|
|
|
const handleWidth = 2 * this._handleRadius + this._handleBorderWidth;
|
|
|
|
return Math.max(barWidth, handleWidth);
|
|
|
|
}
|
|
|
|
|
2023-08-08 18:22:11 +02:00
|
|
|
vfunc_button_press_event(event) {
|
|
|
|
return this.startDragging(event);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2013-06-11 23:39:56 -04:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
startDragging(event) {
|
2013-06-11 23:43:04 -04:00
|
|
|
if (this._dragging)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-07-12 16:12:28 -04:00
|
|
|
|
|
|
|
this._dragging = true;
|
|
|
|
|
2013-06-11 23:43:04 -04:00
|
|
|
let device = event.get_device();
|
2014-07-22 12:37:06 +02:00
|
|
|
let sequence = event.get_event_sequence();
|
|
|
|
|
2021-11-18 00:00:52 +01:00
|
|
|
this._grab = global.stage.grab(this);
|
2014-07-22 12:37:06 +02:00
|
|
|
|
2013-06-11 23:43:04 -04:00
|
|
|
this._grabbedDevice = device;
|
2014-07-22 12:37:06 +02:00
|
|
|
this._grabbedSequence = sequence;
|
|
|
|
|
2016-01-27 22:35:11 +00:00
|
|
|
// We need to emit 'drag-begin' before moving the handle to make
|
2019-07-25 18:53:00 +02:00
|
|
|
// sure that no 'notify::value' signal is emitted before this one.
|
2016-01-27 22:35:11 +00:00
|
|
|
this.emit('drag-begin');
|
|
|
|
|
2011-07-12 16:12:28 -04:00
|
|
|
let absX, absY;
|
|
|
|
[absX, absY] = event.get_coords();
|
|
|
|
this._moveHandle(absX, absY);
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_endDragging() {
|
2011-07-12 16:12:28 -04:00
|
|
|
if (this._dragging) {
|
2019-09-22 13:31:02 +02:00
|
|
|
if (this._releaseId) {
|
2019-07-25 18:53:00 +02:00
|
|
|
this.disconnect(this._releaseId);
|
2019-09-22 13:31:02 +02:00
|
|
|
this._releaseId = 0;
|
|
|
|
}
|
|
|
|
|
2021-11-18 00:00:52 +01:00
|
|
|
if (this._grab) {
|
|
|
|
this._grab.dismiss();
|
|
|
|
this._grab = null;
|
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2014-07-22 12:37:06 +02:00
|
|
|
this._grabbedSequence = null;
|
2013-06-11 23:43:04 -04:00
|
|
|
this._grabbedDevice = null;
|
2011-07-12 16:12:28 -04:00
|
|
|
this._dragging = false;
|
|
|
|
|
|
|
|
this.emit('drag-end');
|
|
|
|
}
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2019-09-10 07:42:48 +02:00
|
|
|
vfunc_button_release_event() {
|
|
|
|
if (this._dragging && !this._grabbedSequence)
|
|
|
|
return this._endDragging();
|
|
|
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
}
|
|
|
|
|
2023-08-08 18:22:11 +02:00
|
|
|
vfunc_touch_event(event) {
|
2014-07-22 12:37:06 +02:00
|
|
|
let sequence = event.get_event_sequence();
|
|
|
|
|
|
|
|
if (!this._dragging &&
|
2023-08-07 02:51:19 +02:00
|
|
|
event.type() === Clutter.EventType.TOUCH_BEGIN) {
|
2014-07-22 12:37:06 +02:00
|
|
|
this.startDragging(event);
|
|
|
|
return Clutter.EVENT_STOP;
|
2021-11-18 00:00:52 +01:00
|
|
|
} else if (this._grabbedSequence &&
|
|
|
|
sequence.get_slot() === this._grabbedSequence.get_slot()) {
|
2023-08-07 02:51:19 +02:00
|
|
|
if (event.type() === Clutter.EventType.TOUCH_UPDATE)
|
2019-09-10 07:42:48 +02:00
|
|
|
return this._motionEvent(this, event);
|
2023-08-07 02:51:19 +02:00
|
|
|
else if (event.type() === Clutter.EventType.TOUCH_END)
|
2014-07-22 12:37:06 +02:00
|
|
|
return this._endDragging();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2014-07-22 12:37:06 +02:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
scroll(event) {
|
2011-07-12 16:12:28 -04:00
|
|
|
let direction = event.get_scroll_direction();
|
2023-09-22 13:21:32 +02:00
|
|
|
let delta = 0;
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-09-22 13:16:59 +02:00
|
|
|
if (event.get_flags() & Clutter.EventFlags.FLAG_POINTER_EMULATED)
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-08-07 02:51:19 +02:00
|
|
|
if (direction === Clutter.ScrollDirection.DOWN) {
|
2011-07-12 16:12:28 -04:00
|
|
|
delta = -SLIDER_SCROLL_STEP;
|
2023-08-07 02:51:19 +02:00
|
|
|
} else if (direction === Clutter.ScrollDirection.UP) {
|
2019-01-30 00:01:21 +01:00
|
|
|
delta = SLIDER_SCROLL_STEP;
|
2023-08-07 02:51:19 +02:00
|
|
|
} else if (direction === Clutter.ScrollDirection.SMOOTH) {
|
2019-02-01 14:41:55 +01:00
|
|
|
let [, dy] = event.get_scroll_delta();
|
2011-07-12 16:12:28 -04:00
|
|
|
// Even though the slider is horizontal, use dy to match
|
|
|
|
// the UP/DOWN above.
|
2015-01-10 13:12:48 +01:00
|
|
|
delta = -dy * SLIDER_SCROLL_STEP;
|
2011-07-12 16:12:28 -04:00
|
|
|
}
|
|
|
|
|
2019-08-12 17:40:54 +02:00
|
|
|
this.value = Math.min(Math.max(0, this._value + delta), this._maxValue);
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-08-08 18:22:11 +02:00
|
|
|
vfunc_scroll_event(event) {
|
|
|
|
return this.scroll(event);
|
2019-09-10 07:42:48 +02:00
|
|
|
}
|
|
|
|
|
2023-08-08 18:22:11 +02:00
|
|
|
vfunc_motion_event(event) {
|
2019-09-10 07:42:48 +02:00
|
|
|
if (this._dragging && !this._grabbedSequence)
|
2023-08-08 18:22:11 +02:00
|
|
|
return this._motionEvent(this, event);
|
2019-09-10 07:42:48 +02:00
|
|
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
_motionEvent(actor, event) {
|
2011-07-12 16:12:28 -04:00
|
|
|
let absX, absY;
|
|
|
|
[absX, absY] = event.get_coords();
|
|
|
|
this._moveHandle(absX, absY);
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2023-08-08 18:14:04 +02:00
|
|
|
vfunc_key_press_event(event) {
|
|
|
|
let key = event.get_key_symbol();
|
2023-08-07 02:51:19 +02:00
|
|
|
if (key === Clutter.KEY_Right || key === Clutter.KEY_Left) {
|
|
|
|
let delta = key === Clutter.KEY_Right ? 0.1 : -0.1;
|
2019-08-12 17:40:54 +02:00
|
|
|
this.value = Math.max(0, Math.min(this._value + delta, this._maxValue));
|
2013-11-29 18:17:34 +00:00
|
|
|
return Clutter.EVENT_STOP;
|
2011-07-12 16:12:28 -04:00
|
|
|
}
|
2023-08-08 18:14:04 +02:00
|
|
|
return super.vfunc_key_press_event(event);
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2019-01-31 15:08:10 +01:00
|
|
|
_moveHandle(absX, _absY) {
|
2019-01-31 15:04:56 +01:00
|
|
|
let relX, sliderX;
|
2019-07-25 18:53:00 +02:00
|
|
|
[sliderX] = this.get_transformed_position();
|
2023-09-06 16:30:19 +03:00
|
|
|
const rtl = this.get_text_direction() === Clutter.TextDirection.RTL;
|
|
|
|
let width = this._barLevelWidth;
|
|
|
|
|
2011-07-12 16:12:28 -04:00
|
|
|
relX = absX - sliderX;
|
2023-09-06 16:30:19 +03:00
|
|
|
if (rtl)
|
|
|
|
relX = width - relX;
|
2011-07-12 16:12:28 -04:00
|
|
|
|
|
|
|
let newvalue;
|
2023-10-26 20:09:23 +02:00
|
|
|
if (relX < this._handleRadius)
|
2011-07-12 16:12:28 -04:00
|
|
|
newvalue = 0;
|
2023-10-26 20:09:23 +02:00
|
|
|
else if (relX > width - this._handleRadius)
|
2011-07-12 16:12:28 -04:00
|
|
|
newvalue = 1;
|
|
|
|
else
|
2023-10-26 20:09:23 +02:00
|
|
|
newvalue = (relX - this._handleRadius) / (width - 2 * this._handleRadius);
|
2019-08-12 17:40:54 +02:00
|
|
|
this.value = newvalue * this._maxValue;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2011-07-12 16:12:28 -04:00
|
|
|
|
2019-02-04 12:30:53 +01:00
|
|
|
_getMinimumIncrement() {
|
2013-08-22 23:15:35 +02:00
|
|
|
return 0.1;
|
2017-10-31 02:19:44 +01:00
|
|
|
}
|
2019-07-25 18:53:00 +02:00
|
|
|
});
|