mirror of
https://gitlab.gnome.org/julianschacher/top-bar-organizer.git
synced 2025-10-27 15:19:09 +00:00
147 lines
5.2 KiB
JavaScript
147 lines
5.2 KiB
JavaScript
"use strict";
|
|
/* exported PrefsBoxOrderItemRow */
|
|
|
|
const Gtk = imports.gi.Gtk;
|
|
const Gdk = imports.gi.Gdk;
|
|
const Gio = imports.gi.Gio;
|
|
const GObject = imports.gi.GObject;
|
|
const Adw = imports.gi.Adw;
|
|
|
|
const ExtensionUtils = imports.misc.extensionUtils;
|
|
const Me = ExtensionUtils.getCurrentExtension();
|
|
|
|
var PrefsBoxOrderItemRow = GObject.registerClass({
|
|
GTypeName: "PrefsBoxOrderItemRow",
|
|
Template: Me.dir.get_child("ui").get_child("prefs-box-order-item-row.ui").get_uri(),
|
|
InternalChildren: [
|
|
"item-name-display-label"
|
|
]
|
|
}, class PrefsBoxOrderItemRow extends Adw.ActionRow {
|
|
#drag_starting_point_x;
|
|
#drag_starting_point_y;
|
|
|
|
constructor(params = {}, item) {
|
|
super(params);
|
|
|
|
this.#associateItem(item);
|
|
this.#setupActions();
|
|
}
|
|
|
|
/**
|
|
* Associate `this` with an item.
|
|
* @param {String} item
|
|
*/
|
|
#associateItem(item) {
|
|
this.item = item;
|
|
|
|
if (item.startsWith("appindicator-kstatusnotifieritem-")) {
|
|
// Set `this._item_name_display_label` to something nicer, if the
|
|
// associated item is an AppIndicator/KStatusNotifierItem item.
|
|
this._item_name_display_label.set_label(item.replace("appindicator-kstatusnotifieritem-", ""));
|
|
} else {
|
|
// Otherwise just set it to `item`.
|
|
this._item_name_display_label.set_label(item);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Setup actions.
|
|
*/
|
|
#setupActions() {
|
|
const actionGroup = new Gio.SimpleActionGroup();
|
|
|
|
const forgetAction = new Gio.SimpleAction({
|
|
name: "forget"
|
|
});
|
|
forgetAction.connect("activate", (_action, _params) => {
|
|
const parentListBox = this.get_parent();
|
|
parentListBox.remove(this);
|
|
parentListBox.saveBoxOrderToSettings();
|
|
});
|
|
actionGroup.add_action(forgetAction);
|
|
|
|
this.insert_action_group("options", actionGroup);
|
|
}
|
|
|
|
onDragPrepare(_source, x, y) {
|
|
const value = new GObject.Value();
|
|
value.init(PrefsBoxOrderItemRow);
|
|
value.set_object(this);
|
|
|
|
this.#drag_starting_point_x = x;
|
|
this.#drag_starting_point_y = y;
|
|
return Gdk.ContentProvider.new_for_value(value);
|
|
}
|
|
|
|
onDragBegin(_source, drag) {
|
|
let dragWidget = new Gtk.ListBox();
|
|
let allocation = this.get_allocation();
|
|
dragWidget.set_size_request(allocation.width, allocation.height);
|
|
|
|
let dragPrefsBoxOrderItemRow = new PrefsBoxOrderItemRow({}, this.item);
|
|
dragWidget.append(dragPrefsBoxOrderItemRow);
|
|
dragWidget.drag_highlight_row(dragPrefsBoxOrderItemRow);
|
|
|
|
let currentDragIcon = Gtk.DragIcon.get_for_drag(drag);
|
|
currentDragIcon.set_child(dragWidget);
|
|
drag.set_hotspot(this.#drag_starting_point_x, this.#drag_starting_point_y);
|
|
}
|
|
|
|
// Handle a new drop on `this` properly.
|
|
// `value` is the thing getting dropped.
|
|
onDrop(_target, value, _x, _y) {
|
|
// If `this` got dropped onto itself, do nothing.
|
|
if (value === this) {
|
|
return;
|
|
}
|
|
|
|
// Get the GtkListBoxes of `this` and the drop value.
|
|
const ownListBox = this.get_parent();
|
|
const valueListBox = value.get_parent();
|
|
|
|
// Get the position of `this` and the drop value.
|
|
const ownPosition = this.get_index();
|
|
const valuePosition = value.get_index();
|
|
|
|
// Remove the drop value from its list box.
|
|
valueListBox.remove(value);
|
|
|
|
// Since an element got potentially removed from the list of `this`,
|
|
// get the position of `this` again.
|
|
const updatedOwnPosition = this.get_index();
|
|
|
|
if (ownListBox !== valueListBox) {
|
|
// First handle the case where `this` and the drop value are in
|
|
// different list boxes.
|
|
if ((ownListBox.boxOrder === "right-box-order" && valueListBox.boxOrder === "left-box-order")
|
|
|| (ownListBox.boxOrder === "right-box-order" && valueListBox.boxOrder === "center-box-order")
|
|
|| (ownListBox.boxOrder === "center-box-order" && valueListBox.boxOrder === "left-box-order")) {
|
|
// If the list box of the drop value comes before the list
|
|
// box of `this`, add the drop value after `this`.
|
|
ownListBox.insert(value, updatedOwnPosition + 1);
|
|
} else {
|
|
// Otherwise, add the drop value where `this` currently is.
|
|
ownListBox.insert(value, updatedOwnPosition);
|
|
}
|
|
} else {
|
|
if (valuePosition < ownPosition) {
|
|
// If the drop value was before `this`, add the drop value
|
|
// after `this`.
|
|
ownListBox.insert(value, updatedOwnPosition + 1);
|
|
} else {
|
|
// Otherwise, add the drop value where `this` currently is.
|
|
ownListBox.insert(value, updatedOwnPosition);
|
|
}
|
|
}
|
|
|
|
/// Finally save the box order(/s) to settings.
|
|
ownListBox.saveBoxOrderToSettings();
|
|
// If the list boxes of `this` and the drop value were different,
|
|
// save an updated box order for the list were the drop value was in
|
|
// as well.
|
|
if (ownListBox !== valueListBox) {
|
|
valueListBox.saveBoxOrderToSettings();
|
|
}
|
|
}
|
|
});
|