2019-02-21 08:44:34 +01:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
|
|
|
|
2023-07-10 02:53:00 -07:00
|
|
|
import Clutter from 'gi://Clutter';
|
|
|
|
import St from 'gi://St';
|
2019-02-21 08:44:34 +01:00
|
|
|
|
|
|
|
// Shamelessly copied from the layout "hotcorner" ripples implementation
|
2023-07-10 02:53:00 -07:00
|
|
|
export class Ripples {
|
2019-02-21 08:44:34 +01:00
|
|
|
constructor(px, py, styleClass) {
|
|
|
|
this._x = 0;
|
|
|
|
this._y = 0;
|
|
|
|
|
|
|
|
this._px = px;
|
|
|
|
this._py = py;
|
|
|
|
|
2020-03-29 23:51:13 +02:00
|
|
|
this._ripple1 = new St.BoxLayout({
|
|
|
|
style_class: styleClass,
|
|
|
|
opacity: 0,
|
|
|
|
can_focus: false,
|
|
|
|
reactive: false,
|
|
|
|
visible: false,
|
|
|
|
});
|
2019-02-21 08:44:34 +01:00
|
|
|
this._ripple1.set_pivot_point(px, py);
|
|
|
|
|
2020-03-29 23:51:13 +02:00
|
|
|
this._ripple2 = new St.BoxLayout({
|
|
|
|
style_class: styleClass,
|
|
|
|
opacity: 0,
|
|
|
|
can_focus: false,
|
|
|
|
reactive: false,
|
|
|
|
visible: false,
|
|
|
|
});
|
2019-02-21 08:44:34 +01:00
|
|
|
this._ripple2.set_pivot_point(px, py);
|
|
|
|
|
2020-03-29 23:51:13 +02:00
|
|
|
this._ripple3 = new St.BoxLayout({
|
|
|
|
style_class: styleClass,
|
|
|
|
opacity: 0,
|
|
|
|
can_focus: false,
|
|
|
|
reactive: false,
|
|
|
|
visible: false,
|
|
|
|
});
|
2019-02-21 08:44:34 +01:00
|
|
|
this._ripple3.set_pivot_point(px, py);
|
|
|
|
}
|
|
|
|
|
2019-08-31 01:40:53 +02:00
|
|
|
destroy() {
|
|
|
|
this._ripple1.destroy();
|
|
|
|
this._ripple2.destroy();
|
|
|
|
this._ripple3.destroy();
|
|
|
|
}
|
|
|
|
|
2018-07-20 21:46:19 +02:00
|
|
|
_animRipple(ripple, delay, duration, startScale, startOpacity, finalScale) {
|
2019-02-21 08:44:34 +01:00
|
|
|
// We draw a ripple by using a source image and animating it scaling
|
|
|
|
// outwards and fading away. We want the ripples to move linearly
|
|
|
|
// or it looks unrealistic, but if the opacity of the ripple goes
|
2017-06-10 05:55:10 +02:00
|
|
|
// linearly to zero it fades away too quickly, so we use a separate
|
|
|
|
// tween to give a non-linear curve to the fade-away and make
|
2019-02-21 08:44:34 +01:00
|
|
|
// it more visible in the middle section.
|
|
|
|
|
|
|
|
ripple.x = this._x;
|
|
|
|
ripple.y = this._y;
|
|
|
|
ripple.visible = true;
|
|
|
|
ripple.opacity = 255 * Math.sqrt(startOpacity);
|
|
|
|
ripple.scale_x = ripple.scale_y = startScale;
|
2019-08-19 19:55:49 +02:00
|
|
|
ripple.set_translation(-this._px * ripple.width, -this._py * ripple.height, 0.0);
|
2019-02-21 08:44:34 +01:00
|
|
|
|
2018-07-20 21:46:19 +02:00
|
|
|
ripple.ease({
|
|
|
|
opacity: 0,
|
|
|
|
delay,
|
|
|
|
duration,
|
2019-08-20 23:43:54 +02:00
|
|
|
mode: Clutter.AnimationMode.EASE_IN_QUAD,
|
2018-07-20 21:46:19 +02:00
|
|
|
});
|
|
|
|
ripple.ease({
|
|
|
|
scale_x: finalScale,
|
|
|
|
scale_y: finalScale,
|
|
|
|
delay,
|
|
|
|
duration,
|
|
|
|
mode: Clutter.AnimationMode.LINEAR,
|
2019-08-20 23:43:54 +02:00
|
|
|
onComplete: () => (ripple.visible = false),
|
2018-07-20 21:46:19 +02:00
|
|
|
});
|
2019-02-21 08:44:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
addTo(stage) {
|
2019-06-29 14:36:13 +02:00
|
|
|
if (this._stage !== undefined)
|
2019-02-21 08:44:34 +01:00
|
|
|
throw new Error('Ripples already added');
|
|
|
|
|
|
|
|
this._stage = stage;
|
2023-11-07 10:47:14 +00:00
|
|
|
this._stage.add_child(this._ripple1);
|
|
|
|
this._stage.add_child(this._ripple2);
|
|
|
|
this._stage.add_child(this._ripple3);
|
2019-02-21 08:44:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
playAnimation(x, y) {
|
2019-06-29 14:36:13 +02:00
|
|
|
if (this._stage === undefined)
|
2019-02-21 08:44:34 +01:00
|
|
|
throw new Error('Ripples not added');
|
|
|
|
|
|
|
|
this._x = x;
|
|
|
|
this._y = y;
|
|
|
|
|
|
|
|
this._stage.set_child_above_sibling(this._ripple1, null);
|
|
|
|
this._stage.set_child_above_sibling(this._ripple2, this._ripple1);
|
|
|
|
this._stage.set_child_above_sibling(this._ripple3, this._ripple2);
|
|
|
|
|
|
|
|
// Show three concentric ripples expanding outwards; the exact
|
|
|
|
// parameters were found by trial and error, so don't look
|
|
|
|
// for them to make perfect sense mathematically
|
|
|
|
|
2018-07-20 21:46:19 +02:00
|
|
|
// delay time scale opacity => scale
|
|
|
|
this._animRipple(this._ripple1, 0, 830, 0.25, 1.0, 1.5);
|
|
|
|
this._animRipple(this._ripple2, 50, 1000, 0.0, 0.7, 1.25);
|
|
|
|
this._animRipple(this._ripple3, 350, 1000, 0.0, 0.3, 1);
|
2019-02-21 08:44:34 +01:00
|
|
|
}
|
2023-07-10 02:53:00 -07:00
|
|
|
}
|