2024-11-20 00:54:02 -05:00
|
|
|
import { idle, readFile, Process } from 'astal';
|
2024-11-14 11:13:57 -05:00
|
|
|
import { App, Astal, Gtk, Widget } from 'astal/gtk3';
|
2024-11-11 20:33:44 -05:00
|
|
|
|
|
|
|
import AstalGreet from 'gi://AstalGreet';
|
|
|
|
|
2024-11-20 00:54:02 -05:00
|
|
|
import { centerCursor } from '../../lib';
|
2024-11-11 20:33:44 -05:00
|
|
|
|
2024-11-20 00:54:02 -05:00
|
|
|
|
|
|
|
export default (hyprpaper: InstanceType<typeof Process>) => {
|
2024-11-19 23:55:17 -05:00
|
|
|
const DEFAULT_NAME = 'matt';
|
|
|
|
const PARSED_INDEX = {
|
|
|
|
name: 0,
|
|
|
|
uid: 2,
|
|
|
|
gid: 3,
|
|
|
|
desc: 4,
|
|
|
|
home: 5,
|
|
|
|
shell: 6,
|
|
|
|
};
|
|
|
|
|
|
|
|
const parsePasswd = (fileContent: string) => {
|
|
|
|
const splitUsers = fileContent.split('\n');
|
|
|
|
const parsedUsers = splitUsers.map((u) => {
|
|
|
|
const user = u.split(':');
|
|
|
|
|
|
|
|
return {
|
|
|
|
name: user[PARSED_INDEX.name],
|
|
|
|
uid: Number(user[PARSED_INDEX.uid]),
|
|
|
|
gid: Number(user[PARSED_INDEX.gid]),
|
|
|
|
desc: user[PARSED_INDEX.desc],
|
|
|
|
home: user[PARSED_INDEX.home],
|
|
|
|
shell: user[PARSED_INDEX.shell],
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
// Filter out system users, nixbld users and nobody
|
|
|
|
return parsedUsers.filter((u) => {
|
|
|
|
return u.uid >= 1000 &&
|
|
|
|
!u.name.includes('nixbld') &&
|
|
|
|
u.name !== 'nobody';
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const users = parsePasswd(readFile('/etc/passwd'));
|
|
|
|
|
|
|
|
const dropdown = new Gtk.ComboBoxText();
|
|
|
|
|
|
|
|
dropdown.show_all();
|
|
|
|
|
|
|
|
users.forEach((u) => {
|
|
|
|
dropdown.append(null, u.name);
|
2024-11-11 20:33:44 -05:00
|
|
|
});
|
|
|
|
|
2024-11-19 23:55:17 -05:00
|
|
|
const response = <label /> as Widget.Label;
|
|
|
|
|
|
|
|
const password = (
|
|
|
|
<entry
|
|
|
|
placeholderText="Password"
|
|
|
|
visibility={false}
|
|
|
|
|
|
|
|
setup={(self) => idle(() => {
|
|
|
|
self.grab_focus();
|
|
|
|
})}
|
|
|
|
|
|
|
|
onActivate={(self) => {
|
|
|
|
AstalGreet.login(
|
|
|
|
dropdown.get_active_text() ?? '',
|
2024-12-29 17:22:55 -05:00
|
|
|
self.get_text() || '',
|
2024-11-19 23:55:17 -05:00
|
|
|
'Hyprland',
|
|
|
|
(_, res) => {
|
|
|
|
try {
|
|
|
|
AstalGreet.login_finish(res);
|
2024-11-20 00:54:02 -05:00
|
|
|
App.get_window('greeter')?.set_visible(false);
|
|
|
|
hyprpaper.kill();
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
App.quit();
|
|
|
|
}, 500);
|
2024-11-19 23:55:17 -05:00
|
|
|
}
|
|
|
|
catch (error) {
|
2024-12-29 17:22:55 -05:00
|
|
|
response.set_label(JSON.stringify(error));
|
2024-11-19 23:55:17 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<window
|
|
|
|
name="greeter"
|
2024-11-20 00:54:02 -05:00
|
|
|
application={App}
|
2024-11-19 23:55:17 -05:00
|
|
|
keymode={Astal.Keymode.ON_DEMAND}
|
2024-11-20 00:54:02 -05:00
|
|
|
visible={false}
|
|
|
|
setup={(self) => {
|
|
|
|
centerCursor();
|
|
|
|
setTimeout(() => {
|
2024-12-29 17:22:55 -05:00
|
|
|
self.set_visible(true);
|
2024-11-20 00:54:02 -05:00
|
|
|
password.grab_focus();
|
|
|
|
}, 1000);
|
|
|
|
}}
|
2024-11-11 20:33:44 -05:00
|
|
|
>
|
|
|
|
<box
|
|
|
|
vertical
|
|
|
|
halign={Gtk.Align.CENTER}
|
|
|
|
valign={Gtk.Align.CENTER}
|
|
|
|
hexpand
|
|
|
|
vexpand
|
2024-11-19 23:55:17 -05:00
|
|
|
className="base"
|
2024-11-11 20:33:44 -05:00
|
|
|
>
|
2024-11-19 23:55:17 -05:00
|
|
|
<box
|
|
|
|
vertical
|
|
|
|
halign={Gtk.Align.CENTER}
|
|
|
|
valign={Gtk.Align.CENTER}
|
|
|
|
hexpand
|
|
|
|
vexpand
|
|
|
|
className="linked"
|
|
|
|
|
|
|
|
setup={() => {
|
|
|
|
idle(() => {
|
|
|
|
const usernames = users.map((u) => u.name);
|
|
|
|
|
|
|
|
if (usernames.includes(DEFAULT_NAME)) {
|
|
|
|
dropdown.set_active(usernames.indexOf(DEFAULT_NAME));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{dropdown}
|
|
|
|
{password}
|
|
|
|
</box>
|
|
|
|
|
|
|
|
{response}
|
2024-11-11 20:33:44 -05:00
|
|
|
</box>
|
2024-11-19 23:55:17 -05:00
|
|
|
</window>
|
|
|
|
);
|
|
|
|
};
|