agsdotfilesdotshyprlandhyprland-configricerofirofi-configshell-scriptsswwwwallustwaybarwaybar-modulewaybar-themes
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
423 lines
20 KiB
423 lines
20 KiB
// TODO |
|
// - Make client destroy/create not destroy and recreate the whole thing |
|
// - Active ws hook optimization: only update when moving to next group |
|
// |
|
const { Gdk, Gtk } = imports.gi; |
|
const { Gravity } = imports.gi.Gdk; |
|
import { SCREEN_HEIGHT, SCREEN_WIDTH } from '../../variables.js'; |
|
import App from 'resource:///com/github/Aylur/ags/app.js'; |
|
import Variable from 'resource:///com/github/Aylur/ags/variable.js'; |
|
import Widget from 'resource:///com/github/Aylur/ags/widget.js'; |
|
import * as Utils from 'resource:///com/github/Aylur/ags/utils.js'; |
|
|
|
import Hyprland from 'resource:///com/github/Aylur/ags/service/hyprland.js'; |
|
const { execAsync, exec } = Utils; |
|
import { setupCursorHoverGrab } from '../.widgetutils/cursorhover.js'; |
|
import { dumpToWorkspace, swapWorkspace } from "./actions.js"; |
|
import { substitute } from "../.miscutils/icons.js"; |
|
|
|
const NUM_OF_WORKSPACES_SHOWN = userOptions.overview.numOfCols * userOptions.overview.numOfRows; |
|
const TARGET = [Gtk.TargetEntry.new('text/plain', Gtk.TargetFlags.SAME_APP, 0)]; |
|
const POPUP_CLOSE_TIME = 100; // ms |
|
|
|
const overviewTick = Variable(false); |
|
|
|
export default () => { |
|
const clientMap = new Map(); |
|
let workspaceGroup = 0; |
|
const ContextMenuWorkspaceArray = ({ label, actionFunc, thisWorkspace }) => Widget.MenuItem({ |
|
label: `${label}`, |
|
setup: (menuItem) => { |
|
let submenu = new Gtk.Menu(); |
|
submenu.className = 'menu'; |
|
|
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
const startWorkspace = offset + 1; |
|
const endWorkspace = startWorkspace + NUM_OF_WORKSPACES_SHOWN - 1; |
|
for (let i = startWorkspace; i <= endWorkspace; i++) { |
|
let button = new Gtk.MenuItem({ |
|
label: `Workspace ${i}` |
|
}); |
|
button.connect("activate", () => { |
|
// execAsync([`${onClickBinary}`, `${thisWorkspace}`, `${i}`]).catch(print); |
|
actionFunc(thisWorkspace, i); |
|
overviewTick.setValue(!overviewTick.value); |
|
}); |
|
submenu.append(button); |
|
} |
|
menuItem.set_reserve_indicator(true); |
|
menuItem.set_submenu(submenu); |
|
} |
|
}) |
|
|
|
const Window = ({ address, at: [x, y], size: [w, h], workspace: { id, name }, class: c, title, xwayland }, screenCoords) => { |
|
const revealInfoCondition = (Math.min(w, h) * userOptions.overview.scale > 70); |
|
if (w <= 0 || h <= 0 || (c === '' && title === '') || c.endsWith('-dropterm')) return null; |
|
// Non-primary monitors |
|
if (screenCoords.x != 0) x -= screenCoords.x; |
|
if (screenCoords.y != 0) y -= screenCoords.y; |
|
// Other offscreen adjustments |
|
if (x + w <= 0) x += (Math.floor(x / SCREEN_WIDTH) * SCREEN_WIDTH); |
|
else if (x < 0) { w = x + w; x = 0; } |
|
if (y + h <= 0) x += (Math.floor(y / SCREEN_HEIGHT) * SCREEN_HEIGHT); |
|
else if (y < 0) { h = y + h; y = 0; } |
|
// Truncate if offscreen |
|
if (x + w > SCREEN_WIDTH) w = SCREEN_WIDTH - x; |
|
if (y + h > SCREEN_HEIGHT) h = SCREEN_HEIGHT - y; |
|
|
|
const appIcon = Widget.Icon({ |
|
icon: substitute(c), |
|
size: Math.min(w, h) * userOptions.overview.scale / 2.5, |
|
}); |
|
return Widget.Button({ |
|
attribute: { |
|
address, x, y, w, h, ws: id, |
|
updateIconSize: (self) => { |
|
appIcon.size = Math.min(self.attribute.w, self.attribute.h) * userOptions.overview.scale / 2.5; |
|
}, |
|
}, |
|
className: 'overview-tasks-window', |
|
hpack: 'start', |
|
vpack: 'start', |
|
css: ` |
|
margin-left: ${Math.round(x * userOptions.overview.scale)}px; |
|
margin-top: ${Math.round(y * userOptions.overview.scale)}px; |
|
margin-right: -${Math.round((x + w) * userOptions.overview.scale)}px; |
|
margin-bottom: -${Math.round((y + h) * userOptions.overview.scale)}px; |
|
`, |
|
onClicked: (self) => { |
|
App.closeWindow('overview'); |
|
Utils.timeout(POPUP_CLOSE_TIME, () => Hyprland.messageAsync(`dispatch focuswindow address:${address}`)); |
|
}, |
|
onMiddleClickRelease: () => Hyprland.messageAsync(`dispatch closewindow address:${address}`), |
|
onSecondaryClick: (button) => { |
|
button.toggleClassName('overview-tasks-window-selected', true); |
|
const menu = Widget.Menu({ |
|
className: 'menu', |
|
children: [ |
|
Widget.MenuItem({ |
|
child: Widget.Label({ |
|
xalign: 0, |
|
label: "Close (Middle-click)", |
|
}), |
|
onActivate: () => Hyprland.messageAsync(`dispatch closewindow address:${address}`), |
|
}), |
|
ContextMenuWorkspaceArray({ |
|
label: "Dump windows to workspace", |
|
actionFunc: dumpToWorkspace, |
|
thisWorkspace: Number(id) |
|
}), |
|
ContextMenuWorkspaceArray({ |
|
label: "Swap windows with workspace", |
|
actionFunc: swapWorkspace, |
|
thisWorkspace: Number(id) |
|
}), |
|
], |
|
}); |
|
menu.connect("deactivate", () => { |
|
button.toggleClassName('overview-tasks-window-selected', false); |
|
}) |
|
menu.connect("selection-done", () => { |
|
button.toggleClassName('overview-tasks-window-selected', false); |
|
}) |
|
menu.popup_at_widget(button.get_parent(), Gravity.SOUTH, Gravity.NORTH, null); // Show menu below the button |
|
button.connect("destroy", () => menu.destroy()); |
|
}, |
|
child: Widget.Box({ |
|
homogeneous: true, |
|
child: Widget.Box({ |
|
vertical: true, |
|
vpack: 'center', |
|
className: 'spacing-v-5', |
|
children: [ |
|
appIcon, |
|
// TODO: Add xwayland tag instead of just having italics |
|
Widget.Revealer({ |
|
transition: 'slide_down', |
|
revealChild: revealInfoCondition, |
|
child: Widget.Label({ |
|
maxWidthChars: 10, // Doesn't matter what number |
|
truncate: 'end', |
|
className: `${xwayland ? 'txt txt-italic' : 'txt'}`, |
|
css: ` |
|
font-size: ${Math.min(SCREEN_WIDTH, SCREEN_HEIGHT) * userOptions.overview.scale / 14.6}px; |
|
margin: 0px ${Math.min(SCREEN_WIDTH, SCREEN_HEIGHT) * userOptions.overview.scale / 10}px; |
|
`, |
|
// If the title is too short, include the class |
|
label: (title.length <= 1 ? `${c}: ${title}` : title), |
|
}) |
|
}) |
|
] |
|
}) |
|
}), |
|
tooltipText: `${c}: ${title}`, |
|
setup: (button) => { |
|
setupCursorHoverGrab(button); |
|
|
|
button.drag_source_set(Gdk.ModifierType.BUTTON1_MASK, TARGET, Gdk.DragAction.MOVE); |
|
button.drag_source_set_icon_name(substitute(c)); |
|
// button.drag_source_set_icon_gicon(icon); |
|
|
|
button.connect('drag-begin', (button) => { // On drag start, add the dragging class |
|
button.toggleClassName('overview-tasks-window-dragging', true); |
|
}); |
|
button.connect('drag-data-get', (_w, _c, data) => { // On drag finish, give address |
|
data.set_text(address, address.length); |
|
button.toggleClassName('overview-tasks-window-dragging', false); |
|
}); |
|
}, |
|
}); |
|
} |
|
|
|
const Workspace = (index) => { |
|
// const fixed = Widget.Fixed({ |
|
// attribute: { |
|
// put: (widget, x, y) => { |
|
// fixed.put(widget, x, y); |
|
// }, |
|
// move: (widget, x, y) => { |
|
// fixed.move(widget, x, y); |
|
// }, |
|
// } |
|
// }); |
|
const fixed = Widget.Box({ |
|
attribute: { |
|
put: (widget, x, y) => { |
|
if (!widget.attribute) return; |
|
// Note: x and y are already multiplied by userOptions.overview.scale |
|
const newCss = ` |
|
margin-left: ${Math.round(x)}px; |
|
margin-top: ${Math.round(y)}px; |
|
margin-right: -${Math.round(x + (widget.attribute.w * userOptions.overview.scale))}px; |
|
margin-bottom: -${Math.round(y + (widget.attribute.h * userOptions.overview.scale))}px; |
|
`; |
|
widget.css = newCss; |
|
fixed.pack_start(widget, false, false, 0); |
|
}, |
|
move: (widget, x, y) => { |
|
if (!widget) return; |
|
if (!widget.attribute) return; |
|
// Note: x and y are already multiplied by userOptions.overview.scale |
|
const newCss = ` |
|
margin-left: ${Math.round(x)}px; |
|
margin-top: ${Math.round(y)}px; |
|
margin-right: -${Math.round(x + (widget.attribute.w * userOptions.overview.scale))}px; |
|
margin-bottom: -${Math.round(y + (widget.attribute.h * userOptions.overview.scale))}px; |
|
`; |
|
widget.css = newCss; |
|
}, |
|
} |
|
}) |
|
const WorkspaceNumber = ({ index, ...rest }) => Widget.Label({ |
|
className: 'overview-tasks-workspace-number', |
|
label: `${index}`, |
|
css: ` |
|
margin: ${Math.min(SCREEN_WIDTH, SCREEN_HEIGHT) * userOptions.overview.scale * userOptions.overview.wsNumMarginScale}px; |
|
font-size: ${SCREEN_HEIGHT * userOptions.overview.scale * userOptions.overview.wsNumScale}px; |
|
`, |
|
setup: (self) => self.hook(Hyprland.active.workspace, (self) => { |
|
// Update when going to new ws group |
|
const currentGroup = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN); |
|
self.label = `${currentGroup * NUM_OF_WORKSPACES_SHOWN + index}`; |
|
}), |
|
...rest, |
|
}) |
|
const widget = Widget.Box({ |
|
className: 'overview-tasks-workspace', |
|
vpack: 'center', |
|
css: ` |
|
min-width: ${SCREEN_WIDTH * userOptions.overview.scale}px; |
|
min-height: ${SCREEN_HEIGHT * userOptions.overview.scale}px; |
|
`, |
|
children: [Widget.EventBox({ |
|
hexpand: true, |
|
vexpand: true, |
|
onPrimaryClick: () => { |
|
App.closeWindow('overview'); |
|
Utils.timeout(POPUP_CLOSE_TIME, () => Hyprland.messageAsync(`dispatch workspace ${index}`)); |
|
}, |
|
setup: (eventbox) => { |
|
eventbox.drag_dest_set(Gtk.DestDefaults.ALL, TARGET, Gdk.DragAction.COPY); |
|
eventbox.connect('drag-data-received', (_w, _c, _x, _y, data) => { |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
Hyprland.messageAsync(`dispatch movetoworkspacesilent ${index + offset},address:${data.get_text()}`) |
|
overviewTick.setValue(!overviewTick.value); |
|
}); |
|
}, |
|
child: Widget.Overlay({ |
|
child: Widget.Box({}), |
|
overlays: [ |
|
WorkspaceNumber({ index: index, hpack: 'start', vpack: 'start' }), |
|
fixed |
|
] |
|
}), |
|
})], |
|
}); |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
fixed.attribute.put(WorkspaceNumber(offset + index), 0, 0); |
|
widget.clear = () => { |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
clientMap.forEach((client, address) => { |
|
if (!client) return; |
|
if ((client.attribute.ws <= offset || client.attribute.ws > offset + NUM_OF_WORKSPACES_SHOWN) || |
|
(client.attribute.ws == offset + index)) { |
|
client.destroy(); |
|
client = null; |
|
clientMap.delete(address); |
|
} |
|
}); |
|
} |
|
widget.set = (clientJson, screenCoords) => { |
|
let c = clientMap.get(clientJson.address); |
|
if (c) { |
|
if (c.attribute?.ws !== clientJson.workspace.id) { |
|
c.destroy(); |
|
c = null; |
|
clientMap.delete(clientJson.address); |
|
} |
|
else if (c) { |
|
c.attribute.w = clientJson.size[0]; |
|
c.attribute.h = clientJson.size[1]; |
|
c.attribute.updateIconSize(c); |
|
fixed.attribute.move(c, |
|
Math.max(0, clientJson.at[0] * userOptions.overview.scale), |
|
Math.max(0, clientJson.at[1] * userOptions.overview.scale) |
|
); |
|
return; |
|
} |
|
} |
|
const newWindow = Window(clientJson, screenCoords); |
|
if (newWindow === null) return; |
|
// clientMap.set(clientJson.address, newWindow); |
|
fixed.attribute.put(newWindow, |
|
Math.max(0, newWindow.attribute.x * userOptions.overview.scale), |
|
Math.max(0, newWindow.attribute.y * userOptions.overview.scale) |
|
); |
|
clientMap.set(clientJson.address, newWindow); |
|
}; |
|
widget.unset = (clientAddress) => { |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
let c = clientMap.get(clientAddress); |
|
if (!c) return; |
|
c.destroy(); |
|
c = null; |
|
clientMap.delete(clientAddress); |
|
}; |
|
widget.show = () => { |
|
fixed.show_all(); |
|
} |
|
return widget; |
|
}; |
|
|
|
const arr = (s, n) => { |
|
const array = []; |
|
for (let i = 0; i < n; i++) |
|
array.push(s + i); |
|
|
|
return array; |
|
}; |
|
|
|
const OverviewRow = ({ startWorkspace, workspaces, windowName = 'overview' }) => Widget.Box({ |
|
children: arr(startWorkspace, workspaces).map(Workspace), |
|
attribute: { |
|
monitorMap: [], |
|
getMonitorMap: (box) => { |
|
execAsync('hyprctl -j monitors').then(monitors => { |
|
box.attribute.monitorMap = JSON.parse(monitors).reduce((acc, item) => { |
|
acc[item.id] = { x: item.x, y: item.y }; |
|
return acc; |
|
}, {}); |
|
}); |
|
}, |
|
update: (box) => { |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
if (!App.getWindow(windowName).visible) return; |
|
Hyprland.messageAsync('j/clients').then(clients => { |
|
const allClients = JSON.parse(clients); |
|
const kids = box.get_children(); |
|
kids.forEach(kid => kid.clear()); |
|
for (let i = 0; i < allClients.length; i++) { |
|
const client = allClients[i]; |
|
const childID = client.workspace.id - (offset + startWorkspace); |
|
if (offset + startWorkspace <= client.workspace.id && |
|
client.workspace.id <= offset + startWorkspace + workspaces) { |
|
const screenCoords = box.attribute.monitorMap[client.monitor]; |
|
if (kids[childID]) { |
|
kids[childID].set(client, screenCoords); |
|
} |
|
continue; |
|
} |
|
} |
|
kids.forEach(kid => kid.show()); |
|
}).catch(print); |
|
}, |
|
updateWorkspace: (box, id) => { |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
if (!( // Not in range, ignore |
|
offset + startWorkspace <= id && |
|
id <= offset + startWorkspace + workspaces |
|
)) return; |
|
// if (!App.getWindow(windowName).visible) return; |
|
Hyprland.messageAsync('j/clients').then(clients => { |
|
const allClients = JSON.parse(clients); |
|
const kids = box.get_children(); |
|
for (let i = 0; i < allClients.length; i++) { |
|
const client = allClients[i]; |
|
if (client.workspace.id != id) continue; |
|
const screenCoords = box.attribute.monitorMap[client.monitor]; |
|
kids[id - (offset + startWorkspace)]?.set(client, screenCoords); |
|
} |
|
kids[id - (offset + startWorkspace)]?.show(); |
|
}).catch(print); |
|
}, |
|
}, |
|
setup: (box) => { |
|
box.attribute.getMonitorMap(box); |
|
box |
|
.hook(overviewTick, (box) => box.attribute.update(box)) |
|
.hook(Hyprland, (box, clientAddress) => { |
|
const offset = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN) * NUM_OF_WORKSPACES_SHOWN; |
|
const kids = box.get_children(); |
|
const client = Hyprland.getClient(clientAddress); |
|
if (!client) return; |
|
const id = client.workspace.id; |
|
|
|
box.attribute.updateWorkspace(box, id); |
|
kids[id - (offset + startWorkspace)]?.unset(clientAddress); |
|
}, 'client-removed') |
|
.hook(Hyprland, (box, clientAddress) => { |
|
const client = Hyprland.getClient(clientAddress); |
|
if (!client) return; |
|
box.attribute.updateWorkspace(box, client.workspace.id); |
|
}, 'client-added') |
|
.hook(Hyprland.active.workspace, (box) => { |
|
// Full update when going to new ws group |
|
const previousGroup = box.attribute.workspaceGroup; |
|
const currentGroup = Math.floor((Hyprland.active.workspace.id - 1) / NUM_OF_WORKSPACES_SHOWN); |
|
if (currentGroup !== previousGroup) { |
|
box.attribute.update(box); |
|
box.attribute.workspaceGroup = currentGroup; |
|
} |
|
}) |
|
.hook(App, (box, name, visible) => { // Update on open |
|
if (name == 'overview' && visible) box.attribute.update(box); |
|
}) |
|
}, |
|
}); |
|
|
|
return Widget.Revealer({ |
|
revealChild: true, |
|
transition: 'slide_down', |
|
transitionDuration: userOptions.animations.durationLarge, |
|
child: Widget.Box({ |
|
vertical: true, |
|
className: 'overview-tasks', |
|
children: Array.from({ length: userOptions.overview.numOfRows }, (_, index) => |
|
OverviewRow({ |
|
startWorkspace: 1 + index * userOptions.overview.numOfCols, |
|
workspaces: userOptions.overview.numOfCols, |
|
}) |
|
) |
|
}), |
|
}); |
|
} |