177 lines
5.5 KiB
TypeScript
177 lines
5.5 KiB
TypeScript
import { bind, timeout } from 'astal';
|
|
import { App, Astal, Gtk, Widget } from 'astal/gtk3';
|
|
|
|
import AstalWp from 'gi://AstalWp';
|
|
|
|
import { ProgressBar } from '../misc/subclasses';
|
|
import PopupWindow from '../misc/popup-window';
|
|
import Brightness from '../../services/brightness';
|
|
|
|
/* Types */
|
|
declare global {
|
|
function popup_osd(osd: string): void;
|
|
}
|
|
|
|
|
|
const HIDE_DELAY = 2000;
|
|
const transition_duration = 300;
|
|
|
|
export default () => {
|
|
let n_showing = 0;
|
|
let stack: Widget.Stack | undefined;
|
|
|
|
const popup = (osd: string) => {
|
|
if (!stack) {
|
|
return;
|
|
}
|
|
|
|
++n_showing;
|
|
stack.shown = osd;
|
|
|
|
App.get_window('win-osd')?.set_visible(true);
|
|
|
|
timeout(HIDE_DELAY, () => {
|
|
--n_showing;
|
|
|
|
if (n_showing === 0) {
|
|
App.get_window('win-osd')?.set_visible(false);
|
|
}
|
|
});
|
|
};
|
|
|
|
globalThis.popup_osd = popup;
|
|
|
|
const brightness = Brightness.get_default();
|
|
const speaker = AstalWp.get_default()?.get_audio()?.get_default_speaker();
|
|
const microphone = AstalWp.get_default()?.get_audio()?.get_default_microphone();
|
|
|
|
if (!speaker || !microphone) {
|
|
throw new Error('Could not find default audio devices.');
|
|
}
|
|
|
|
return (
|
|
<PopupWindow
|
|
name="osd"
|
|
anchor={Astal.WindowAnchor.BOTTOM}
|
|
exclusivity={Astal.Exclusivity.IGNORE}
|
|
close_on_unfocus="stay"
|
|
transition="slide bottom"
|
|
>
|
|
<stack
|
|
className="osd"
|
|
transitionDuration={transition_duration}
|
|
setup={(self) => {
|
|
timeout(3 * 1000, () => {
|
|
stack = self;
|
|
});
|
|
}}
|
|
>
|
|
|
|
<box
|
|
name="speaker"
|
|
css="margin-bottom: 80px;"
|
|
|
|
setup={(self) => {
|
|
self.hook(speaker, 'notify::mute', () => {
|
|
popup('speaker');
|
|
});
|
|
}}
|
|
>
|
|
<box className="osd-item widget">
|
|
<icon icon={bind(speaker, 'volumeIcon')} />
|
|
|
|
<ProgressBar
|
|
fraction={bind(speaker, 'volume')}
|
|
sensitive={bind(speaker, 'mute').as((v) => !v)}
|
|
valign={Gtk.Align.CENTER}
|
|
/>
|
|
</box>
|
|
</box>
|
|
|
|
<box
|
|
name="microphone"
|
|
css="margin-bottom: 80px;"
|
|
|
|
setup={(self) => {
|
|
self.hook(microphone, 'notify::mute', () => {
|
|
popup('microphone');
|
|
});
|
|
}}
|
|
>
|
|
<box className="osd-item widget">
|
|
<icon icon={bind(microphone, 'volumeIcon')} />
|
|
|
|
<ProgressBar
|
|
fraction={bind(microphone, 'volume')}
|
|
sensitive={bind(microphone, 'mute').as((v) => !v)}
|
|
valign={Gtk.Align.CENTER}
|
|
/>
|
|
</box>
|
|
</box>
|
|
|
|
<box
|
|
name="brightness"
|
|
css="margin-bottom: 80px;"
|
|
|
|
setup={(self) => {
|
|
self.hook(brightness, 'notify::screen-icon', () => {
|
|
popup('brightness');
|
|
});
|
|
}}
|
|
>
|
|
<box className="osd-item widget">
|
|
<icon icon={bind(brightness, 'screenIcon')} />
|
|
|
|
<ProgressBar
|
|
fraction={bind(brightness, 'screen')}
|
|
valign={Gtk.Align.CENTER}
|
|
/>
|
|
</box>
|
|
</box>
|
|
|
|
{
|
|
brightness.hasKbd && (
|
|
<box
|
|
name="keyboard"
|
|
css="margin-bottom: 80px;"
|
|
|
|
setup={(self) => {
|
|
self.hook(brightness, 'notify::kbd-level', () => {
|
|
popup('keyboard');
|
|
});
|
|
}}
|
|
>
|
|
<box className="osd-item widget">
|
|
<icon icon="keyboard-brightness-symbolic" />
|
|
|
|
<ProgressBar
|
|
fraction={bind(brightness, 'kbdLevel').as((v) => (v ?? 0) / 2)}
|
|
sensitive={bind(brightness, 'kbdLevel').as((v) => v !== 0)}
|
|
valign={Gtk.Align.CENTER}
|
|
/>
|
|
</box>
|
|
</box>
|
|
)
|
|
}
|
|
|
|
<box
|
|
name="caps"
|
|
css="margin-bottom: 80px;"
|
|
|
|
setup={(self) => {
|
|
self.hook(brightness, 'notify::caps-icon', () => {
|
|
popup('caps');
|
|
});
|
|
}}
|
|
>
|
|
<box className="osd-item widget">
|
|
<icon icon={bind(brightness, 'capsIcon')} />
|
|
|
|
<label label="Caps Lock" />
|
|
</box>
|
|
</box>
|
|
|
|
</stack>
|
|
</PopupWindow>
|
|
);
|
|
};
|