mirror of
https://github.com/elementary/gala.git
synced 2024-12-21 16:21:37 +03:00
b2bfe28062
Co-authored-by: Leo <lenemter@gmail.com>
577 lines
19 KiB
Vala
577 lines
19 KiB
Vala
/*
|
|
* Copyright 2017 Adam Bieńkowski
|
|
* Copyright 2023 elementary, Inc. <https://elementary.io>
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*/
|
|
|
|
public class Gala.Plugins.PIP.PopupWindow : Clutter.Actor {
|
|
private int button_size;
|
|
private int container_margin;
|
|
private const uint FADE_OUT_TIMEOUT = 200;
|
|
private const float MINIMUM_SCALE = 0.1f;
|
|
private const float MAXIMUM_SCALE = 1.0f;
|
|
private const int SCREEN_MARGIN = 0;
|
|
private const float OFF_SCREEN_PERCENT = 0.5f;
|
|
private const int OFF_SCREEN_VISIBLE_PIXELS = 80;
|
|
|
|
public signal void closed ();
|
|
|
|
public Meta.Display display { get; construct; }
|
|
public Meta.WindowActor window_actor { get; construct; }
|
|
|
|
private Clutter.Clone clone; // clone itself
|
|
private Clutter.Actor clone_container; // clips the clone
|
|
private Clutter.Actor container; // draws the shadow
|
|
private Gala.CloseButton close_button;
|
|
private Clutter.Actor resize_button;
|
|
private DragDropAction move_action;
|
|
|
|
private float begin_resize_width = 0.0f;
|
|
private float begin_resize_height = 0.0f;
|
|
private float resize_start_x = 0.0f;
|
|
private float resize_start_y = 0.0f;
|
|
|
|
private bool resizing = false;
|
|
private bool off_screen = false;
|
|
private Clutter.Grab? grab = null;
|
|
|
|
private static unowned Meta.Window? previous_focus = null;
|
|
|
|
// From https://opensourcehacker.com/2011/12/01/calculate-aspect-ratio-conserving-resize-for-images-in-javascript/
|
|
private static void calculate_aspect_ratio_size_fit (float src_width, float src_height, float max_width, float max_height,
|
|
out float width, out float height) {
|
|
float ratio = float.min (max_width / src_width, max_height / src_height);
|
|
width = src_width * ratio;
|
|
height = src_height * ratio;
|
|
}
|
|
|
|
private static bool get_window_is_normal (Meta.Window window) {
|
|
var window_type = window.get_window_type ();
|
|
return window_type == Meta.WindowType.NORMAL
|
|
|| window_type == Meta.WindowType.DIALOG
|
|
|| window_type == Meta.WindowType.MODAL_DIALOG;
|
|
}
|
|
|
|
public PopupWindow (Meta.Display display, Meta.WindowActor window_actor) {
|
|
Object (display: display, window_actor: window_actor);
|
|
}
|
|
|
|
construct {
|
|
var scale = display.get_monitor_scale (display.get_current_monitor ());
|
|
|
|
button_size = Gala.Utils.scale_to_int (36, scale);
|
|
container_margin = button_size / 2;
|
|
|
|
reactive = true;
|
|
set_pivot_point (0.5f, 0.5f);
|
|
set_easing_mode (Clutter.AnimationMode.EASE_IN_QUAD);
|
|
|
|
clone = new Clutter.Clone (window_actor);
|
|
|
|
move_action = new DragDropAction (DragDropActionType.SOURCE, "pip");
|
|
move_action.drag_begin.connect (on_move_begin);
|
|
move_action.drag_canceled.connect (on_move_end);
|
|
move_action.actor_clicked.connect (activate);
|
|
|
|
clone_container = new Clutter.Actor () {
|
|
scale_x = 0.35f,
|
|
scale_y = 0.35f
|
|
};
|
|
clone_container.add_child (clone);
|
|
|
|
container = new Clutter.Actor () {
|
|
reactive = true
|
|
};
|
|
container.add_child (clone_container);
|
|
container.add_effect (new ShadowEffect ("window"));
|
|
container.add_action (move_action);
|
|
|
|
update_size ();
|
|
|
|
#if HAS_MUTTER45
|
|
Mtk.Rectangle monitor_rect;
|
|
#else
|
|
Meta.Rectangle monitor_rect;
|
|
#endif
|
|
get_current_monitor_rect (out monitor_rect);
|
|
|
|
float x_position, y_position;
|
|
if (Clutter.get_default_text_direction () == Clutter.TextDirection.RTL) {
|
|
x_position = SCREEN_MARGIN + monitor_rect.x;
|
|
} else {
|
|
x_position = monitor_rect.width + monitor_rect.x - SCREEN_MARGIN - width;
|
|
}
|
|
y_position = monitor_rect.height + monitor_rect.y - SCREEN_MARGIN - height;
|
|
|
|
set_position (x_position, y_position);
|
|
|
|
close_button = new Gala.CloseButton (scale) {
|
|
opacity = 0
|
|
};
|
|
// TODO: Check if close button should be on the right
|
|
close_button.add_constraint (new Clutter.AlignConstraint (this, Clutter.AlignAxis.X_AXIS, 0.0f));
|
|
close_button.add_constraint (new Clutter.AlignConstraint (this, Clutter.AlignAxis.Y_AXIS, 0.0f));
|
|
close_button.triggered.connect (on_close_click_clicked);
|
|
|
|
resize_button = Utils.create_resize_button (scale);
|
|
resize_button.opacity = 0;
|
|
resize_button.reactive = true;
|
|
resize_button.add_constraint (new Clutter.AlignConstraint (this, Clutter.AlignAxis.X_AXIS, 1.0f));
|
|
resize_button.add_constraint (new Clutter.AlignConstraint (this, Clutter.AlignAxis.Y_AXIS, 1.0f));
|
|
resize_button.button_press_event.connect (on_resize_button_press);
|
|
|
|
add_child (container);
|
|
add_child (close_button);
|
|
add_child (resize_button);
|
|
|
|
window_actor.notify["allocation"].connect (on_allocation_changed);
|
|
container.set_position (container_margin, container_margin);
|
|
update_clone_clip ();
|
|
|
|
unowned var window = window_actor.get_meta_window ();
|
|
window.unmanaged.connect (on_close_click_clicked);
|
|
window.notify["appears-focused"].connect (update_window_focus);
|
|
|
|
unowned var workspace_manager = display.get_workspace_manager ();
|
|
workspace_manager.active_workspace_changed.connect (update_window_focus);
|
|
}
|
|
|
|
public override void show () {
|
|
base.show ();
|
|
|
|
opacity = 0;
|
|
|
|
save_easing_state ();
|
|
set_easing_duration (AnimationsSettings.get_animation_duration (200));
|
|
opacity = 255;
|
|
restore_easing_state ();
|
|
}
|
|
|
|
public override void hide () {
|
|
opacity = 255;
|
|
|
|
var duration = AnimationsSettings.get_animation_duration (200);
|
|
save_easing_state ();
|
|
set_easing_duration (duration);
|
|
opacity = 0;
|
|
restore_easing_state ();
|
|
|
|
if (duration == 0) {
|
|
base.hide ();
|
|
} else {
|
|
ulong completed_id = 0;
|
|
completed_id = transitions_completed.connect (() => {
|
|
disconnect (completed_id);
|
|
base.hide ();
|
|
});
|
|
}
|
|
}
|
|
|
|
#if HAS_MUTTER45
|
|
public override bool enter_event (Clutter.Event event) {
|
|
#else
|
|
public override bool enter_event (Clutter.CrossingEvent event) {
|
|
#endif
|
|
var duration = AnimationsSettings.get_animation_duration (300);
|
|
|
|
close_button.save_easing_state ();
|
|
close_button.set_easing_duration (duration);
|
|
close_button.opacity = 255;
|
|
close_button.restore_easing_state ();
|
|
|
|
resize_button.save_easing_state ();
|
|
resize_button.set_easing_duration (duration);
|
|
resize_button.opacity = 255;
|
|
resize_button.restore_easing_state ();
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
}
|
|
|
|
#if HAS_MUTTER45
|
|
public override bool leave_event (Clutter.Event event) {
|
|
#else
|
|
public override bool leave_event (Clutter.CrossingEvent event) {
|
|
#endif
|
|
var duration = AnimationsSettings.get_animation_duration (300);
|
|
|
|
close_button.save_easing_state ();
|
|
close_button.set_easing_duration (duration);
|
|
close_button.opacity = 0;
|
|
close_button.restore_easing_state ();
|
|
|
|
resize_button.save_easing_state ();
|
|
resize_button.set_easing_duration (duration);
|
|
resize_button.opacity = 0;
|
|
resize_button.restore_easing_state ();
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
}
|
|
|
|
public void set_container_clip (Graphene.Rect? container_clip) {
|
|
clone_container.clip_rect = container_clip;
|
|
update_clone_container_scale ();
|
|
on_allocation_changed ();
|
|
}
|
|
|
|
private Clutter.Actor on_move_begin () {
|
|
display.set_cursor (Meta.Cursor.DND_IN_DRAG);
|
|
|
|
return this;
|
|
}
|
|
|
|
private void on_move_end () {
|
|
reactive = true;
|
|
update_screen_position ();
|
|
display.set_cursor (Meta.Cursor.DEFAULT);
|
|
}
|
|
|
|
#if HAS_MUTTER45
|
|
private bool on_resize_button_press (Clutter.Event event) {
|
|
#else
|
|
private bool on_resize_button_press (Clutter.ButtonEvent event) {
|
|
#endif
|
|
if (resizing || event.get_button () != Clutter.Button.PRIMARY) {
|
|
return Clutter.EVENT_STOP;
|
|
}
|
|
|
|
resizing = true;
|
|
|
|
event.get_coords (out resize_start_x, out resize_start_y);
|
|
|
|
begin_resize_width = width;
|
|
begin_resize_height = height;
|
|
|
|
grab = resize_button.get_stage ().grab (resize_button);
|
|
resize_button.event.connect (on_resize_event);
|
|
|
|
display.set_cursor (Meta.Cursor.SE_RESIZE);
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
}
|
|
|
|
private bool on_resize_event (Clutter.Event event) {
|
|
if (!resizing) {
|
|
return Clutter.EVENT_STOP;
|
|
}
|
|
|
|
switch (event.get_type ()) {
|
|
case Clutter.EventType.MOTION:
|
|
var mods = event.get_state ();
|
|
if (!(Clutter.ModifierType.BUTTON1_MASK in mods)) {
|
|
stop_resizing ();
|
|
break;
|
|
}
|
|
|
|
float event_x, event_y;
|
|
event.get_coords (out event_x, out event_y);
|
|
float diff_x = event_x - resize_start_x;
|
|
float diff_y = event_y - resize_start_y;
|
|
|
|
width = begin_resize_width + diff_x;
|
|
height = begin_resize_height + diff_y;
|
|
|
|
update_clone_container_scale ();
|
|
update_size ();
|
|
|
|
break;
|
|
case Clutter.EventType.BUTTON_RELEASE:
|
|
if (event.get_button () == Clutter.Button.PRIMARY) {
|
|
stop_resizing ();
|
|
}
|
|
|
|
break;
|
|
case Clutter.EventType.LEAVE:
|
|
case Clutter.EventType.ENTER:
|
|
return Clutter.EVENT_PROPAGATE;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return Clutter.EVENT_STOP;
|
|
}
|
|
|
|
private void stop_resizing () {
|
|
if (!resizing) {
|
|
return;
|
|
}
|
|
|
|
if (grab != null) {
|
|
grab.dismiss ();
|
|
resize_button.event.disconnect (on_resize_event);
|
|
grab = null;
|
|
}
|
|
|
|
resizing = false;
|
|
|
|
update_screen_position ();
|
|
|
|
display.set_cursor (Meta.Cursor.DEFAULT);
|
|
}
|
|
|
|
private void on_allocation_changed () {
|
|
update_clone_clip ();
|
|
update_size ();
|
|
}
|
|
|
|
private void on_close_click_clicked () {
|
|
var duration = AnimationsSettings.get_animation_duration (FADE_OUT_TIMEOUT);
|
|
|
|
save_easing_state ();
|
|
set_easing_duration (duration);
|
|
opacity = 0;
|
|
restore_easing_state ();
|
|
|
|
Clutter.Threads.Timeout.add (duration, () => {
|
|
closed ();
|
|
return Source.REMOVE;
|
|
});
|
|
}
|
|
|
|
private void update_window_focus () {
|
|
unowned Meta.Window focus_window = display.get_focus_window ();
|
|
if ((focus_window != null && !get_window_is_normal (focus_window))
|
|
|| (previous_focus != null && !get_window_is_normal (previous_focus))) {
|
|
previous_focus = focus_window;
|
|
return;
|
|
}
|
|
|
|
unowned var workspace_manager = display.get_workspace_manager ();
|
|
unowned var active_workspace = workspace_manager.get_active_workspace ();
|
|
unowned var window = window_actor.get_meta_window ();
|
|
|
|
if (window.appears_focused && window.located_on_workspace (active_workspace)) {
|
|
hide ();
|
|
} else if (!window_actor.is_destroyed ()) {
|
|
show ();
|
|
}
|
|
|
|
previous_focus = focus_window;
|
|
}
|
|
|
|
private void update_size () {
|
|
int clone_container_width, clone_container_height;
|
|
|
|
if (clone_container.has_clip) {
|
|
float src_width = 0.0f, src_height = 0.0f;
|
|
clone_container.get_clip (null, null, out src_width, out src_height);
|
|
clone_container_width = (int) (src_width * clone_container.scale_x);
|
|
clone_container_height = (int) (src_height * clone_container.scale_y);
|
|
} else {
|
|
clone_container_width = (int) (clone_container.width * clone_container.scale_x);
|
|
clone_container_height = (int) (clone_container.height * clone_container.scale_y);
|
|
}
|
|
|
|
container.width = clone_container_width;
|
|
container.height = clone_container_height;
|
|
|
|
width = clone_container_width + button_size;
|
|
height = clone_container_height + button_size;
|
|
}
|
|
|
|
/*
|
|
* Offsets clone by csd shadow size.
|
|
*/
|
|
private void update_clone_clip () {
|
|
var rect = window_actor.get_meta_window ().get_frame_rect ();
|
|
|
|
float x_offset = rect.x - window_actor.x;
|
|
float y_offset = rect.y - window_actor.y;
|
|
clone.set_clip (x_offset, y_offset, rect.width, rect.height);
|
|
clone.set_position (-x_offset, -y_offset);
|
|
|
|
clone_container.set_size (rect.width, rect.height);
|
|
}
|
|
|
|
private void update_clone_container_scale () {
|
|
float src_width = 1.0f, src_height = 1.0f;
|
|
if (clone_container.has_clip) {
|
|
clone_container.get_clip (null, null, out src_width, out src_height);
|
|
} else {
|
|
src_width = clone_container.width;
|
|
src_height = clone_container.height;
|
|
}
|
|
|
|
float max_width = width - button_size;
|
|
float max_height = height - button_size;
|
|
|
|
float new_width, new_height;
|
|
calculate_aspect_ratio_size_fit (
|
|
src_width, src_height,
|
|
max_width, max_height,
|
|
out new_width, out new_height
|
|
);
|
|
|
|
float window_width = 1.0f, window_height = 1.0f;
|
|
get_target_window_size (out window_width, out window_height);
|
|
|
|
float new_scale_x = new_width / window_width;
|
|
float new_scale_y = new_height / window_height;
|
|
|
|
clone_container.scale_x = new_scale_x.clamp (MINIMUM_SCALE, MAXIMUM_SCALE);
|
|
clone_container.scale_y = new_scale_y.clamp (MINIMUM_SCALE, MAXIMUM_SCALE);
|
|
|
|
update_clone_container_position ();
|
|
}
|
|
|
|
private void update_clone_container_position () {
|
|
if (clone_container.has_clip) {
|
|
float clip_x = 0.0f, clip_y = 0.0f;
|
|
clone_container.get_clip (out clip_x, out clip_y, null, null);
|
|
clone_container.x = (float) (-clip_x * clone_container.scale_x);
|
|
clone_container.y = (float) (-clip_y * clone_container.scale_y);
|
|
}
|
|
}
|
|
|
|
private void update_screen_position () {
|
|
if (!place_window_off_screen ()) {
|
|
place_window_in_screen ();
|
|
}
|
|
}
|
|
|
|
private void place_window_in_screen () {
|
|
off_screen = false;
|
|
|
|
#if HAS_MUTTER45
|
|
Mtk.Rectangle monitor_rect;
|
|
#else
|
|
Meta.Rectangle monitor_rect;
|
|
#endif
|
|
get_current_monitor_rect (out monitor_rect);
|
|
|
|
int monitor_x = monitor_rect.x;
|
|
int monitor_y = monitor_rect.y;
|
|
int monitor_width = monitor_rect.width;
|
|
int monitor_height = monitor_rect.height;
|
|
|
|
var screen_limit_start_x = SCREEN_MARGIN + monitor_x;
|
|
var screen_limit_end_x = monitor_width + monitor_x - SCREEN_MARGIN - width;
|
|
var screen_limit_start_y = SCREEN_MARGIN + monitor_y;
|
|
var screen_limit_end_y = monitor_height + monitor_y - SCREEN_MARGIN - height;
|
|
|
|
var duration = AnimationsSettings.get_animation_duration (300);
|
|
|
|
save_easing_state ();
|
|
set_easing_mode (Clutter.AnimationMode.EASE_OUT_BACK);
|
|
set_easing_duration (duration);
|
|
x = x.clamp (screen_limit_start_x, screen_limit_end_x);
|
|
y = y.clamp (screen_limit_start_y, screen_limit_end_y);
|
|
restore_easing_state ();
|
|
}
|
|
|
|
private bool place_window_off_screen () {
|
|
off_screen = false;
|
|
|
|
var duration = AnimationsSettings.get_animation_duration (300);
|
|
|
|
save_easing_state ();
|
|
set_easing_mode (Clutter.AnimationMode.EASE_OUT_BACK);
|
|
set_easing_duration (duration);
|
|
|
|
#if HAS_MUTTER45
|
|
Mtk.Rectangle monitor_rect;
|
|
#else
|
|
Meta.Rectangle monitor_rect;
|
|
#endif
|
|
get_current_monitor_rect (out monitor_rect);
|
|
|
|
int monitor_x = monitor_rect.x;
|
|
int monitor_y = monitor_rect.y;
|
|
int monitor_width = monitor_rect.width;
|
|
int monitor_height = monitor_rect.height;
|
|
|
|
// X axis off screen
|
|
var off_screen_x_threshold = width * OFF_SCREEN_PERCENT;
|
|
|
|
var off_screen_x = (x - monitor_x) < -off_screen_x_threshold;
|
|
if (off_screen_x
|
|
&& !coord_is_in_other_monitor (x, Clutter.Orientation.HORIZONTAL)) {
|
|
off_screen = true;
|
|
x = monitor_x - width + OFF_SCREEN_VISIBLE_PIXELS;
|
|
}
|
|
|
|
var off_screen_w = (x + width) > (monitor_x + monitor_width + off_screen_x_threshold);
|
|
if (off_screen_w
|
|
&& !coord_is_in_other_monitor (x + width, Clutter.Orientation.HORIZONTAL)) {
|
|
off_screen = true;
|
|
x = monitor_x + monitor_width - OFF_SCREEN_VISIBLE_PIXELS;
|
|
}
|
|
|
|
// Y axis off screen
|
|
var off_screen_y_threshold = height * OFF_SCREEN_PERCENT;
|
|
|
|
var off_screen_y = (y - monitor_y) < -off_screen_y_threshold;
|
|
if (off_screen_y
|
|
&& !coord_is_in_other_monitor (y, Clutter.Orientation.VERTICAL)) {
|
|
off_screen = true;
|
|
y = monitor_y - height + OFF_SCREEN_VISIBLE_PIXELS;
|
|
}
|
|
|
|
var off_screen_h = (y + height) > (monitor_y + monitor_height + off_screen_y_threshold);
|
|
if (off_screen_h
|
|
&& !coord_is_in_other_monitor (y + height, Clutter.Orientation.VERTICAL)) {
|
|
off_screen = true;
|
|
y = monitor_y + monitor_height - OFF_SCREEN_VISIBLE_PIXELS;
|
|
}
|
|
|
|
restore_easing_state ();
|
|
|
|
return off_screen;
|
|
}
|
|
|
|
private bool coord_is_in_other_monitor (float coord, Clutter.Orientation axis) {
|
|
int n_monitors = display.get_n_monitors ();
|
|
|
|
if (n_monitors == 1) {
|
|
return false;
|
|
}
|
|
|
|
int current = display.get_current_monitor ();
|
|
for (int i = 0; i < n_monitors; i++) {
|
|
if (i != current) {
|
|
var monitor_rect = display.get_monitor_geometry (i);
|
|
bool in_monitor = false;
|
|
|
|
if (axis == Clutter.Orientation.HORIZONTAL) {
|
|
in_monitor = (coord >= monitor_rect.x) && (coord <= monitor_rect.x + monitor_rect.width);
|
|
} else {
|
|
in_monitor = (coord >= monitor_rect.y) && (coord <= monitor_rect.y + monitor_rect.height);
|
|
}
|
|
|
|
if (in_monitor) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
#if HAS_MUTTER45
|
|
private void get_current_monitor_rect (out Mtk.Rectangle rect) {
|
|
#else
|
|
private void get_current_monitor_rect (out Meta.Rectangle rect) {
|
|
#endif
|
|
rect = display.get_monitor_geometry (display.get_current_monitor ());
|
|
}
|
|
|
|
private void get_target_window_size (out float width, out float height) {
|
|
if (clone_container.has_clip) {
|
|
clone_container.get_clip (null, null, out width, out height);
|
|
} else if (clone.has_clip) {
|
|
clone.get_clip (null, null, out width, out height);
|
|
} else {
|
|
width = clone.width;
|
|
height = clone.height;
|
|
}
|
|
}
|
|
|
|
private void activate () {
|
|
if (off_screen) {
|
|
place_window_in_screen ();
|
|
} else {
|
|
var window = window_actor.get_meta_window ();
|
|
window.activate (Clutter.get_current_event_time ());
|
|
}
|
|
}
|
|
}
|