263 lines
7.7 KiB
TypeScript
263 lines
7.7 KiB
TypeScript
import Gdk from 'gi://Gdk?version=4.0';
|
|
import Gtk from 'gi://Gtk?version=4.0';
|
|
|
|
import { property, register } from 'astal';
|
|
import Binding, { type Connectable, type Subscribable } from 'astal/binding';
|
|
import {
|
|
hook,
|
|
noImplicitDestroy,
|
|
setChildren,
|
|
construct,
|
|
} from './_astal';
|
|
|
|
export type BindableChild = Gtk.Widget | Binding<Gtk.Widget>;
|
|
|
|
export const type = Symbol('child type');
|
|
const dummyBuilder = new Gtk.Builder();
|
|
|
|
interface EventController<Self extends Gtk.Widget> {
|
|
onFocusEnter?: (self: Self) => void
|
|
onFocusLeave?: (self: Self) => void
|
|
|
|
onKeyPressed?: (self: Self, keyval: number, keycode: number, state: Gdk.ModifierType) => void
|
|
onKeyReleased?: (self: Self, keyval: number, keycode: number, state: Gdk.ModifierType) => void
|
|
onKeyModifier?: (self: Self, state: Gdk.ModifierType) => void
|
|
|
|
onLegacy?: (self: Self, event: Gdk.Event) => void
|
|
onButtonPressed?: (self: Self, state: Gdk.ButtonEvent) => void
|
|
onButtonReleased?: (self: Self, state: Gdk.ButtonEvent) => void
|
|
|
|
onHoverEnter?: (self: Self, x: number, y: number) => void
|
|
onHoverLeave?: (self: Self) => void
|
|
onMotion?: (self: Self, x: number, y: number) => void
|
|
|
|
onScroll?: (self: Self, dx: number, dy: number) => void
|
|
onScrollDecelerate?: (self: Self, vel_x: number, vel_y: number) => void
|
|
}
|
|
|
|
const setupControllers = <T>(widget: Gtk.Widget, {
|
|
onFocusEnter,
|
|
onFocusLeave,
|
|
onKeyPressed,
|
|
onKeyReleased,
|
|
onKeyModifier,
|
|
onLegacy,
|
|
onButtonPressed,
|
|
onButtonReleased,
|
|
onHoverEnter,
|
|
onHoverLeave,
|
|
onMotion,
|
|
onScroll,
|
|
onScrollDecelerate,
|
|
...props
|
|
}: EventController<Gtk.Widget> & T) => {
|
|
if (onFocusEnter || onFocusLeave) {
|
|
const focus = new Gtk.EventControllerFocus();
|
|
|
|
widget.add_controller(focus);
|
|
|
|
if (onFocusEnter) { focus.connect('focus-enter', () => onFocusEnter(widget)); }
|
|
|
|
if (onFocusLeave) { focus.connect('focus-leave', () => onFocusLeave(widget)); }
|
|
}
|
|
|
|
if (onKeyPressed || onKeyReleased || onKeyModifier) {
|
|
const key = new Gtk.EventControllerKey();
|
|
|
|
widget.add_controller(key);
|
|
|
|
if (onKeyPressed) {
|
|
key.connect('key-pressed', (_, val, code, state) => onKeyPressed(widget, val, code, state));
|
|
}
|
|
|
|
if (onKeyReleased) {
|
|
key.connect('key-released', (_, val, code, state) =>
|
|
onKeyReleased(widget, val, code, state));
|
|
}
|
|
|
|
if (onKeyModifier) { key.connect('modifiers', (_, state) => onKeyModifier(widget, state)); }
|
|
}
|
|
|
|
if (onLegacy || onButtonPressed || onButtonReleased) {
|
|
const legacy = new Gtk.EventControllerLegacy();
|
|
|
|
widget.add_controller(legacy);
|
|
|
|
legacy.connect('event', (_, event) => {
|
|
if (event.get_event_type() === Gdk.EventType.BUTTON_PRESS) {
|
|
onButtonPressed?.(widget, event as Gdk.ButtonEvent);
|
|
}
|
|
|
|
if (event.get_event_type() === Gdk.EventType.BUTTON_RELEASE) {
|
|
onButtonReleased?.(widget, event as Gdk.ButtonEvent);
|
|
}
|
|
|
|
onLegacy?.(widget, event);
|
|
});
|
|
}
|
|
|
|
if (onMotion || onHoverEnter || onHoverLeave) {
|
|
const hover = new Gtk.EventControllerMotion();
|
|
|
|
widget.add_controller(hover);
|
|
|
|
if (onHoverEnter) { hover.connect('enter', (_, x, y) => onHoverEnter(widget, x, y)); }
|
|
|
|
if (onHoverLeave) { hover.connect('leave', () => onHoverLeave(widget)); }
|
|
|
|
if (onMotion) { hover.connect('motion', (_, x, y) => onMotion(widget, x, y)); }
|
|
}
|
|
|
|
if (onScroll || onScrollDecelerate) {
|
|
const scroll = new Gtk.EventControllerScroll();
|
|
|
|
widget.add_controller(scroll);
|
|
|
|
if (onScroll) { scroll.connect('scroll', (_, x, y) => onScroll(widget, x, y)); }
|
|
|
|
if (onScrollDecelerate) {
|
|
scroll.connect('decelerate', (_, x, y) => onScrollDecelerate(widget, x, y));
|
|
}
|
|
}
|
|
|
|
return props;
|
|
};
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
export default <C extends new (...args: any[]) => Gtk.Widget>(
|
|
cls: C,
|
|
clsName = cls.name,
|
|
) => {
|
|
@register({ GTypeName: `RealClass_${clsName}` })
|
|
class Widget extends cls {
|
|
declare private _css: string | undefined;
|
|
declare private _provider: Gtk.CssProvider | undefined;
|
|
|
|
@property(String)
|
|
get css(): string | undefined {
|
|
return this._css;
|
|
}
|
|
|
|
set css(value: string) {
|
|
if (!this._provider) {
|
|
this._provider = new Gtk.CssProvider();
|
|
|
|
this.get_style_context().add_provider(
|
|
this._provider,
|
|
Gtk.STYLE_PROVIDER_PRIORITY_USER,
|
|
);
|
|
}
|
|
|
|
this._css = value;
|
|
this._provider.load_from_string(value);
|
|
}
|
|
|
|
|
|
declare private [type]: string;
|
|
|
|
@property(String)
|
|
get type(): string { return this[type]; }
|
|
|
|
set type(value: string) { this[type] = value; }
|
|
|
|
|
|
get children(): Gtk.Widget[] { return this.getChildren(this); }
|
|
|
|
set children(value: Gtk.Widget[]) { this.setChildren(this, value); }
|
|
|
|
|
|
declare private [noImplicitDestroy]: boolean;
|
|
|
|
@property(String)
|
|
get noImplicitDestroy(): boolean { return this[noImplicitDestroy]; }
|
|
|
|
set noImplicitDestroy(value: boolean) { this[noImplicitDestroy] = value; }
|
|
|
|
|
|
protected getChildren(widget: Gtk.Widget): Gtk.Widget[] {
|
|
if ('get_child' in widget && typeof widget.get_child == 'function') {
|
|
return widget.get_child() ? [widget.get_child()] : [];
|
|
}
|
|
|
|
const children: Gtk.Widget[] = [];
|
|
let ch = widget.get_first_child();
|
|
|
|
while (ch !== null) {
|
|
children.push(ch);
|
|
ch = ch.get_next_sibling();
|
|
}
|
|
|
|
return children;
|
|
}
|
|
|
|
protected setChildren(widget: Gtk.Widget, children: Gtk.Widget[]) {
|
|
for (const child of children) {
|
|
widget.vfunc_add_child(
|
|
dummyBuilder,
|
|
child,
|
|
type in widget ? widget[type] as string : null,
|
|
);
|
|
}
|
|
}
|
|
|
|
[setChildren](children: Gtk.Widget[]) {
|
|
for (const child of (this.getChildren(this))) {
|
|
child?.unparent();
|
|
|
|
if (!children.includes(child) && noImplicitDestroy in this) {
|
|
child.run_dispose();
|
|
}
|
|
}
|
|
|
|
this.setChildren(this, children);
|
|
}
|
|
|
|
|
|
hook(
|
|
object: Connectable,
|
|
signal: string,
|
|
callback: (self: this, ...args: unknown[]) => void,
|
|
): this;
|
|
|
|
hook(
|
|
object: Subscribable,
|
|
callback: (self: this, ...args: unknown[]) => void,
|
|
): this;
|
|
|
|
hook(
|
|
object: Connectable | Subscribable,
|
|
signalOrCallback: string | ((self: this, ...args: unknown[]) => void),
|
|
callback?: (self: this, ...args: unknown[]) => void,
|
|
) {
|
|
hook(this, object, signalOrCallback, callback);
|
|
|
|
return this;
|
|
}
|
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
constructor(...params: any[]) {
|
|
const props = params[0] || {};
|
|
|
|
super('cssName' in props ? { cssName: props.cssName } : {});
|
|
|
|
if ('cssName' in props) {
|
|
delete props.cssName;
|
|
}
|
|
|
|
if (props.noImplicitDestroy) {
|
|
this.noImplicitDestroy = true;
|
|
delete props.noImplicitDestroy;
|
|
}
|
|
|
|
if (props.type) {
|
|
this.type = props.type;
|
|
delete props.type;
|
|
}
|
|
|
|
construct(this, setupControllers(this, props));
|
|
}
|
|
}
|
|
|
|
return Widget;
|
|
};
|