import { App, Astal, Gtk, Widget } from 'astal/gtk3'; import { property, register } from 'astal/gobject'; import { Binding, idle } from 'astal'; import { get_hyprland_monitor, hyprMessage } from '../../lib'; /* Types */ type CloseType = 'none' | 'stay' | 'released' | 'clicked'; type HyprTransition = 'slide' | 'slide top' | 'slide bottom' | 'slide left' | 'slide right' | 'popin' | 'fade'; type PopupCallback = (self?: Widget.Window) => void; export type PopupWindowProps = Widget.WindowProps & { transition?: HyprTransition | Binding close_on_unfocus?: CloseType | Binding on_open?: PopupCallback on_close?: PopupCallback }; @register() export class PopupWindow extends Widget.Window { @property(String) declare transition: HyprTransition | Binding; @property(String) declare close_on_unfocus: CloseType | Binding; on_open: PopupCallback; on_close: PopupCallback; constructor({ transition = 'slide top', close_on_unfocus = 'released', on_open = () => { /**/ }, on_close = () => { /**/ }, name, visible = false, layer = Astal.Layer.OVERLAY, ...rest }: PopupWindowProps) { super({ ...rest, name: `win-${name}`, namespace: `win-${name}`, visible: false, layer, setup: () => idle(() => { // Add way to make window open on startup if (visible) { this.visible = true; } }), }); App.add_window(this); const setTransition = (_: PopupWindow, t: HyprTransition | Binding) => { hyprMessage(`keyword layerrule animation ${t}, ${this.name}`).catch(console.log); }; this.connect('notify::transition', setTransition); this.close_on_unfocus = close_on_unfocus; this.transition = transition; this.on_open = on_open; this.on_close = on_close; this.connect('notify::visible', () => { // Make sure we have the right animation setTransition(this, this.transition); if (this.visible) { this.on_open(this); } else { this.on_close(this); } }); }; async set_x_pos( alloc: Gtk.Allocation, side = 'right' as 'left' | 'right', ) { const monitor = this.gdkmonitor ?? this.get_display().get_monitor_at_point(alloc.x, alloc.y); const transform = get_hyprland_monitor(monitor)?.get_transform(); let width: number; if (transform && (transform === 1 || transform === 3)) { width = monitor.get_geometry().height; } else { width = monitor.get_geometry().width; } this.margin_right = side === 'right' ? (width - alloc.x - alloc.width) : this.margin_right; this.margin_left = side === 'right' ? this.margin_left : (alloc.x - alloc.width); } } export default PopupWindow;