2025-01-12 20:12:46 -05:00
|
|
|
import { property, register } from 'astal';
|
2025-01-14 10:12:20 -05:00
|
|
|
import { Gtk, hook } from 'astal/gtk4';
|
|
|
|
import { type Connectable, type Subscribable } from 'astal/binding';
|
|
|
|
|
|
|
|
import construct from './construct';
|
|
|
|
import setupControllers from './controller';
|
2025-01-14 00:09:11 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
import {
|
2025-01-14 10:12:20 -05:00
|
|
|
type BindableProps,
|
2025-01-14 23:39:57 -05:00
|
|
|
childType,
|
|
|
|
type Cursor,
|
2025-01-14 10:12:20 -05:00
|
|
|
dummyBuilder,
|
|
|
|
type MixinParams,
|
2025-01-01 02:55:15 -05:00
|
|
|
noImplicitDestroy,
|
|
|
|
setChildren,
|
2025-01-14 10:12:20 -05:00
|
|
|
} from './generics';
|
2025-01-01 02:55:15 -05:00
|
|
|
|
|
|
|
|
2025-01-14 00:09:11 -05:00
|
|
|
export default <
|
2025-01-14 10:12:20 -05:00
|
|
|
C extends new (...props: MixinParams) => Gtk.Widget,
|
2025-01-14 00:09:11 -05:00
|
|
|
ConstructorProps,
|
|
|
|
>(
|
2025-01-12 20:12:46 -05:00
|
|
|
cls: C,
|
|
|
|
clsName = cls.name,
|
|
|
|
) => {
|
|
|
|
@register({ GTypeName: `RealClass_${clsName}` })
|
2025-01-01 02:55:15 -05:00
|
|
|
class Widget extends cls {
|
|
|
|
declare private _css: string | undefined;
|
|
|
|
declare private _provider: Gtk.CssProvider | undefined;
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
@property(String)
|
2025-01-01 02:55:15 -05:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-14 10:12:20 -05:00
|
|
|
declare private [childType]: string;
|
2025-01-12 20:12:46 -05:00
|
|
|
|
|
|
|
@property(String)
|
2025-01-14 10:12:20 -05:00
|
|
|
get type(): string { return this[childType]; }
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-14 10:12:20 -05:00
|
|
|
set type(value: string) { this[childType] = value; }
|
2025-01-01 02:55:15 -05:00
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-13 10:51:02 -05:00
|
|
|
@property(Object)
|
2025-01-01 02:55:15 -05:00
|
|
|
get children(): Gtk.Widget[] { return this.getChildren(this); }
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
set children(value: Gtk.Widget[]) { this.setChildren(this, value); }
|
|
|
|
|
|
|
|
|
|
|
|
declare private [noImplicitDestroy]: boolean;
|
2025-01-12 20:12:46 -05:00
|
|
|
|
|
|
|
@property(String)
|
2025-01-01 02:55:15 -05:00
|
|
|
get noImplicitDestroy(): boolean { return this[noImplicitDestroy]; }
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
set noImplicitDestroy(value: boolean) { this[noImplicitDestroy] = value; }
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
protected setChildren(widget: Gtk.Widget, children: Gtk.Widget[]) {
|
2025-01-01 02:55:15 -05:00
|
|
|
for (const child of children) {
|
|
|
|
widget.vfunc_add_child(
|
|
|
|
dummyBuilder,
|
|
|
|
child,
|
2025-01-14 10:12:20 -05:00
|
|
|
childType in widget ? widget[childType] as string : null,
|
2025-01-01 02:55:15 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
[setChildren](children: Gtk.Widget[]) {
|
|
|
|
for (const child of (this.getChildren(this))) {
|
|
|
|
child?.unparent();
|
2025-01-01 02:55:15 -05:00
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
if (!children.includes(child) && noImplicitDestroy in this) {
|
|
|
|
child.run_dispose();
|
2025-01-01 02:55:15 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
this.setChildren(this, children);
|
2025-01-01 02:55:15 -05:00
|
|
|
}
|
|
|
|
|
2025-01-14 23:39:57 -05:00
|
|
|
private _cursorName: Cursor = 'default';
|
|
|
|
|
|
|
|
@property(String)
|
|
|
|
get cursorName(): Cursor {
|
|
|
|
return this._cursorName;
|
|
|
|
}
|
|
|
|
|
|
|
|
set cursorName(val: Cursor) {
|
|
|
|
this._cursorName = val;
|
|
|
|
this.set_cursor_from_name(val);
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
hook(
|
|
|
|
object: Connectable,
|
|
|
|
signal: string,
|
2025-01-12 20:12:46 -05:00
|
|
|
callback: (self: this, ...args: unknown[]) => void,
|
2025-01-01 02:55:15 -05:00
|
|
|
): this;
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
hook(
|
|
|
|
object: Subscribable,
|
2025-01-12 20:12:46 -05:00
|
|
|
callback: (self: this, ...args: unknown[]) => void,
|
2025-01-01 02:55:15 -05:00
|
|
|
): this;
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-01 02:55:15 -05:00
|
|
|
hook(
|
|
|
|
object: Connectable | Subscribable,
|
2025-01-12 20:12:46 -05:00
|
|
|
signalOrCallback: string | ((self: this, ...args: unknown[]) => void),
|
|
|
|
callback?: (self: this, ...args: unknown[]) => void,
|
2025-01-01 02:55:15 -05:00
|
|
|
) {
|
|
|
|
hook(this, object, signalOrCallback, callback);
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
|
2025-01-14 10:12:20 -05:00
|
|
|
constructor(...params: MixinParams) {
|
2025-01-01 02:55:15 -05:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2025-01-12 20:12:46 -05:00
|
|
|
construct(this, setupControllers(this, props));
|
2025-01-01 02:55:15 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-01-14 00:09:11 -05:00
|
|
|
type Constructor<Instance, Props> = new (...args: Props[]) => Instance;
|
|
|
|
|
|
|
|
type WidgetClass = Constructor<
|
|
|
|
Widget & Gtk.Widget & InstanceType<C>,
|
|
|
|
Partial<BindableProps<ConstructorProps>>
|
|
|
|
>;
|
|
|
|
|
2025-01-14 10:12:20 -05:00
|
|
|
// override the parameters of the `super` constructor
|
2025-01-14 00:09:11 -05:00
|
|
|
return Widget as unknown as WidgetClass;
|
2025-01-01 02:55:15 -05:00
|
|
|
};
|