Compare commits

..

No commits in common. "master" and "v1.0" have entirely different histories.
master ... v1.0

9 changed files with 126 additions and 501 deletions

2
.gitignore vendored
View File

@ -1,3 +1 @@
ssdd ssdd
resources.c
resources.h

View File

@ -1,55 +0,0 @@
# Compiler
CC = gcc
# Compiler and optimization flags
CFLAGS ?= -Wall -O2
CFLAGS += $(shell pkg-config --cflags gtk4)
CFLAGS += -DGDK_VERSION_MAX_ALLOWED=GDK_VERSION_4_0 -DGDK_VERSION_MIN_REQUIRED=GDK_VERSION_4_0
LDFLAGS ?=
LDFLAGS += $(shell pkg-config --libs gtk4)
# Source files
SRC = ssdd.c resources.c
# Output executable
TARGET = ssdd
# Resource files
RESOURCE_XML = resources.gresource.xml
RESOURCE_C = resources.c
RESOURCE_H = resources.h
# Installation directories
PREFIX ?= /usr/local
BINDIR = $(PREFIX)/bin
DATADIR = $(PREFIX)/share/ssdd
# Default target
all: $(TARGET)
# Build the target
$(TARGET): $(RESOURCE_C) $(SRC)
$(CC) $(CFLAGS) -o $(TARGET) $(SRC) $(LDFLAGS)
# Compile resources
$(RESOURCE_C) $(RESOURCE_H): $(RESOURCE_XML)
glib-compile-resources $(RESOURCE_XML) --generate-source --target=$(RESOURCE_C)
glib-compile-resources $(RESOURCE_XML) --generate-header --target=$(RESOURCE_H)
# Install target
install: $(TARGET)
install -d $(DESTDIR)$(BINDIR)
install -m 755 $(TARGET) $(DESTDIR)$(BINDIR)
install -d $(DESTDIR)$(DATADIR)
install -m 644 $(RESOURCE_XML) $(DESTDIR)$(DATADIR)
# Uninstall target
uninstall:
rm -f $(DESTDIR)$(BINDIR)/$(TARGET)
rm -rf $(DESTDIR)$(DATADIR)
# Clean target
clean:
rm -f $(TARGET) $(RESOURCE_C) $(RESOURCE_H)
.PHONY: all clean install uninstall

View File

@ -1,86 +1,39 @@
# ssdd: Simple Shutdown Dialog for Openbox # ssdd
A simple Shutdown Dialog for Openbox written in C using GTK 4 A simple Shutdown Dialog for Openbox written in C using GTK
![Project Screenshot](ssdd.png) ![Project Screenshot](ssdd.png)
![Settings screenshot](ssdd-settings.png) ## Why?
![Settings screenshot](ssdd-about.png) I just bought a new laptop and on my workstation I was using [ssd from Sawfish](https://github.com/SawfishWM/ssd) which I loved. I didn't want to go through all the steps of installing the necessary libraries and dependencies to get it to work, so I decided to create my own.
**Simple Shutdown Dialog (ssdd)** is a simple yet stylish shutdown dialog for Openbox, crafted in C using GTK 4. ## Dependencies and compilation
## Why ssdd? This app requires GTK+ 3.0 development libraries and gcc or clang.
As a long-time Openbox enthusiast, I've always found the default exit dialog a bit lackluster. Modern systems deserve a more refined shutdown experience. While there are other options out there, I figured one more wouldn't hurt, right? I am using this command to compile the program:
Inspired by the elegant `ssd` from Sawfish, I decided to create my own tailored solution for Openbox. This way, you can avoid the hassle of installing extra dependencies and enjoy a sleek shutdown dialog that complements your Openbox setup.
## Features
- **Clean and Intuitive Interface:** ssdd presents clear options for Logout, Reboot, Shutdown, Switch User, Suspend, Hibernate, Settings, and Exit.
- **Configurable Commands:** Easily customize the commands executed for each action via the settings dialog.
- **Lightweight and Efficient:** Designed to be fast and resource-friendly, perfectly suited for Openbox's minimalist philosophy.
- **Modern GTK 4 Interface:** Built with GTK 4 for a modern look and feel.
## Dependencies and Compilation
ssdd requires:
* GTK 4 development libraries
* Glib 2 development libraries
* gcc or clang
### Easy Compilation
Edit the `Makefile` or use the following commands:
Using GCC:
```shell ```shell
% make all # Compile % gcc ssdd.c -o ssdd `pkg-config --cflags --libs gtk+-3.0`
% sudo make install # Install to /usr/local
% sudo make install PREFIX=/usr # Install to /usr
``` ```
### Manual compilation Using Clang:
```shell
First generate the resources. % clang ssdd.c -o ssdd `pkg-config --cflags --libs gtk+-3.0`
```bash
% glib-compile-resources resources.gresource.xml --generate-source --target=resources.c
% glib-compile-resources resources.gresource.xml --generate-header --target=resources.h
``` ```
```bash This produces the binary `ssdd` which you can place in your $PATH.
# Using GCC:
% gcc ssdd.c resources.c -o ssdd `pkg-config --cflags --libs gtk4`
# Using Clang: ## Configure Openbox to use it.
% clang ssdd.c resources.c -o ssdd `pkg-config --cflags --libs gtk4`
```
Place the `ssdd` binary in your `$PATH` (e.g., `~/bin`). `% sudo nvim /etc/xdg/openbox/menu.xml`
### Integrate with Openbox Find the line with the standard Openbox Exit option and change it to
1. Edit your Openbox menu: `<item label="Log Out"><action name="Execute"><execute>ssdd</execute></item>`
```bash The reconfigure Openbox to use the new setting.
% sudo nvim /etc/xdg/openbox/menu.xml
```
2. Replace the default Exit entry with:
```xml
<item label="Log Out"><action name="Execute"><execute>ssdd</execute></action></item>
```
3. Reconfigure Openbox:
```bash
% openbox --reconfigure
```
### Contributing
Contributions are welcome! Feel free to open issues or submit pull requests.
`% openbox --reconfigure`

View File

@ -1,7 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<gresources>
<gresource prefix="/org/gtk/ssdd">
<file>ssdd-icon.png</file>
</gresource>
</gresources>

Binary file not shown.

Before

Width:  |  Height:  |  Size: 74 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 57 KiB

After

Width:  |  Height:  |  Size: 176 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 35 KiB

468
ssdd.c
View File

@ -1,382 +1,118 @@
#include <gtk/gtk.h> #include <gtk/gtk.h>
#include <glib/gstdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h>
#include "resources.h"
// Function declarations static void execute_command(const gchar *command) {
static void execute_command(const gchar *command, GtkWindow *parent); int ret = system(command);
static void show_settings_dialog(GtkWindow *parent); if (ret != 0) {
static void show_about_tab(GtkWidget *box); GtkWidget *dialog;
static void show_settings_tab(GtkWidget *box); dialog = gtk_message_dialog_new(NULL,
static void button_clicked(GtkWidget *widget, gpointer data); GTK_DIALOG_DESTROY_WITH_PARENT,
static gboolean on_key_pressed(GtkEventControllerKey *controller, guint keyval, guint keycode, GdkModifierType state, gpointer user_data);
static void show_confirmation_dialog(GtkWindow *parent_window, const gchar *label, const gchar *command);
static void create_button(GtkWidget *grid, GtkApplication *app, const gchar *label_text, const gchar *icon_name, const gchar *command, int pos);
static void save_configuration(const gchar *commands[]);
static void load_configuration(gchar *commands[]);
static gchar *get_config_path(void);
static gchar *get_config_dir(void);
static void on_save_button_clicked(GtkButton *button, gpointer user_data);
static void on_confirmation_response(GtkDialog *dialog, gint response_id, gpointer user_data);
static gchar *get_config_dir(void) {
return g_build_filename(g_get_user_config_dir(), "ssdd", NULL);
}
static gchar *get_config_path(void) {
return g_build_filename(g_get_user_config_dir(), "ssdd", "config", NULL);
}
static void execute_command(const gchar *command, GtkWindow *parent) {
GError *error = NULL;
gboolean ret = g_spawn_command_line_async(command, &error);
if (!ret) {
gchar *error_message = g_strdup_printf("Error executing command '%s': %s", command, error->message);
GtkWidget *dialog = gtk_message_dialog_new(parent,
GTK_DIALOG_MODAL,
GTK_MESSAGE_ERROR, GTK_MESSAGE_ERROR,
GTK_BUTTONS_CLOSE, GTK_BUTTONS_CLOSE,
"%s", error_message); "Error executing command: %s",
g_signal_connect(dialog, "response", G_CALLBACK(gtk_window_destroy), NULL); command);
gtk_window_present(GTK_WINDOW(dialog)); gtk_dialog_run(GTK_DIALOG(dialog));
gtk_widget_destroy(dialog);
g_free(error_message);
g_error_free(error);
} }
} }
static void show_settings_dialog(GtkWindow *parent) { static void show_confirmation_dialog(GtkWidget *widget, const gchar *label, const gchar *command) {
GtkWidget *dialog;
gint response;
gchar *message = g_strdup_printf("Are you sure you want to %s?", label);
dialog = gtk_message_dialog_new(NULL,
GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_QUESTION,
GTK_BUTTONS_NONE,
"%s",
message);
gtk_dialog_add_button(GTK_DIALOG(dialog), "Yes", GTK_RESPONSE_YES);
gtk_dialog_add_button(GTK_DIALOG(dialog), "No", GTK_RESPONSE_NO);
response = gtk_dialog_run(GTK_DIALOG(dialog));
gtk_widget_destroy(dialog);
g_free(message);
if (response == GTK_RESPONSE_YES) {
execute_command(command);
}
}
static void show_about_dialog(GtkWidget *widget) {
GtkWidget *dialog; GtkWidget *dialog;
GtkWidget *content_area; GtkWidget *content_area;
GtkWidget *notebook;
GtkWidget *settings_tab;
GtkWidget *about_tab;
dialog = gtk_dialog_new();
gtk_window_set_title(GTK_WINDOW(dialog), "Settings");
gtk_window_set_transient_for(GTK_WINDOW(dialog), parent);
gtk_window_set_modal(GTK_WINDOW(dialog), TRUE);
gtk_window_set_destroy_with_parent(GTK_WINDOW(dialog), TRUE);
content_area = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
gtk_window_set_child(GTK_WINDOW(dialog), content_area);
notebook = gtk_notebook_new();
gtk_box_append(GTK_BOX(content_area), notebook);
settings_tab = gtk_box_new(GTK_ORIENTATION_VERTICAL, 10);
gtk_widget_set_margin_top(settings_tab, 10);
gtk_widget_set_margin_bottom(settings_tab, 10);
gtk_widget_set_margin_start(settings_tab, 10);
gtk_widget_set_margin_end(settings_tab, 10);
gtk_notebook_append_page(GTK_NOTEBOOK(notebook), settings_tab, gtk_label_new("Settings"));
about_tab = gtk_box_new(GTK_ORIENTATION_VERTICAL, 10);
gtk_widget_set_margin_top(about_tab, 10);
gtk_widget_set_margin_bottom(about_tab, 10);
gtk_widget_set_margin_start(about_tab, 10);
gtk_widget_set_margin_end(about_tab, 10);
gtk_notebook_append_page(GTK_NOTEBOOK(notebook), about_tab, gtk_label_new("About"));
show_settings_tab(settings_tab);
show_about_tab(about_tab);
GtkWidget *close_button = gtk_button_new_with_label("Close");
gtk_box_append(GTK_BOX(content_area), close_button);
g_signal_connect_swapped(close_button, "clicked", G_CALLBACK(gtk_window_destroy), dialog);
gtk_window_present(GTK_WINDOW(dialog));
}
static void show_settings_tab(GtkWidget *box) {
const gchar *labels[] = {
"Logout Command",
"Reboot Command",
"Shutdown Command",
"Switch User Command",
"Suspend Command",
"Hibernate Command"
};
gchar *commands[6];
load_configuration(commands);
GtkWidget *grid = gtk_grid_new();
gtk_grid_set_row_spacing(GTK_GRID(grid), 10);
gtk_grid_set_column_spacing(GTK_GRID(grid), 10);
gtk_box_append(GTK_BOX(box), grid);
GtkWidget **entries = g_new(GtkWidget*, 6);
for (int i = 0; i < 6; i++) {
GtkWidget *label = gtk_label_new(labels[i]);
GtkWidget *entry = gtk_entry_new();
gtk_editable_set_text(GTK_EDITABLE(entry), commands[i]);
gtk_widget_set_hexpand(entry, TRUE);
gtk_widget_set_halign(label, GTK_ALIGN_END);
gtk_grid_attach(GTK_GRID(grid), label, 0, i, 1, 1);
gtk_grid_attach(GTK_GRID(grid), entry, 1, i, 1, 1);
entries[i] = entry;
}
GtkWidget *save_button = gtk_button_new_with_label("Save");
g_object_set_data_full(G_OBJECT(save_button), "entries", entries, (GDestroyNotify)g_free);
g_signal_connect(save_button, "clicked", G_CALLBACK(on_save_button_clicked), NULL);
gtk_box_append(GTK_BOX(box), save_button);
for (int i = 0; i < 6; i++) {
g_free(commands[i]);
}
}
static void on_save_button_clicked(GtkButton *button, gpointer user_data) {
GtkWidget **entries = g_object_get_data(G_OBJECT(button), "entries");
const gchar *commands[6];
for (int i = 0; i < 6; i++) {
commands[i] = gtk_editable_get_text(GTK_EDITABLE(entries[i]));
}
save_configuration(commands);
}
static void show_about_tab(GtkWidget *box) {
GtkWidget *label; GtkWidget *label;
GtkWidget *image; GtkWidget *image;
GtkWidget *box;
const gchar *about_text = const gchar *about_text =
"\n<b>About Simple ShutDown Dialog</b>\n\n" "About Stig's ShutDown Dialog\n\n"
"<b>Version:</b> 2.0\n" "<b>Version:</b> 1.0\n"
"<b>Author:</b> kekePower\n" "<b>Author:</b> kekePower\n"
"<b>URL:</b> <a href=\"https://git.kekepower.com/kekePower/ssdd\">https://git.kekepower.com/kekePower/ssdd</a>\n" "<b>URL:</b> <a href=\"https://git.kekepower.com/kekePower/ssdd\">https://git.kekepower.com/kekePower/ssdd</a>\n"
"<b>Description:</b> A Simple ShutDown Dialog for Openbox.\n"; "<b>Description:</b> This is a simple Shutdown Dialog for Openbox.";
image = gtk_image_new_from_resource("/org/gtk/ssdd/ssdd-icon.png"); dialog = gtk_dialog_new_with_buttons("About Stig's ShutDown Dialog",
gtk_image_set_pixel_size(GTK_IMAGE(image), 250); NULL,
gtk_box_append(GTK_BOX(box), image); GTK_DIALOG_DESTROY_WITH_PARENT,
"_Close",
GTK_RESPONSE_CLOSE,
NULL);
content_area = gtk_dialog_get_content_area(GTK_DIALOG(dialog));
box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 5);
gtk_container_add(GTK_CONTAINER(content_area), box);
image = gtk_image_new_from_file("ssdd-icon.png");
gtk_image_set_pixel_size(GTK_IMAGE(image), 250); // Assuming original size is 500x500
gtk_box_pack_start(GTK_BOX(box), image, FALSE, FALSE, 0);
label = gtk_label_new(NULL); label = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(label), about_text); gtk_label_set_markup(GTK_LABEL(label), about_text);
gtk_label_set_selectable(GTK_LABEL(label), TRUE); gtk_label_set_selectable(GTK_LABEL(label), TRUE);
gtk_widget_set_halign(label, GTK_ALIGN_START); gtk_widget_set_halign(label, GTK_ALIGN_START);
gtk_widget_set_valign(label, GTK_ALIGN_START); gtk_widget_set_valign(label, GTK_ALIGN_START);
gtk_box_append(GTK_BOX(box), label); gtk_box_pack_start(GTK_BOX(box), label, TRUE, TRUE, 0);
gtk_widget_show_all(dialog);
gtk_dialog_run(GTK_DIALOG(dialog));
gtk_widget_destroy(dialog);
} }
static void button_clicked(GtkWidget *widget, gpointer data) { static void button_clicked(GtkWidget *widget, gpointer data) {
const gchar *command = (const gchar *) data; const gchar *command = (const gchar *) data;
const gchar *label = g_object_get_data(G_OBJECT(widget), "label"); const gchar *label = gtk_button_get_label(GTK_BUTTON(widget));
GtkApplication *app = g_object_get_data(G_OBJECT(widget), "app");
GtkWindow *parent_window = GTK_WINDOW(gtk_widget_get_root(widget));
if (g_strcmp0(command, "exit") == 0) { if (g_strcmp0(command, "exit") == 0) {
g_application_quit(G_APPLICATION(app)); g_application_quit(G_APPLICATION(g_object_get_data(G_OBJECT(widget), "app")));
return; return;
} }
if (g_strcmp0(command, "settings") == 0) { if (g_strcmp0(command, "about") == 0) {
show_settings_dialog(parent_window); show_about_dialog(widget);
} else { } else {
show_confirmation_dialog(parent_window, label, command); show_confirmation_dialog(widget, label, command);
} }
} }
static gboolean on_key_pressed(GtkEventControllerKey *controller, guint keyval, guint keycode, GdkModifierType state, gpointer user_data) {
if (keyval == GDK_KEY_Escape) {
GtkApplication *app = GTK_APPLICATION(user_data);
g_application_quit(G_APPLICATION(app));
return TRUE;
}
return FALSE;
}
static void show_confirmation_dialog(GtkWindow *parent_window, const gchar *label, const gchar *command) {
GtkWidget *dialog;
gchar *message = g_strdup_printf("Are you sure you want to %s?", label);
dialog = gtk_message_dialog_new(parent_window,
GTK_DIALOG_MODAL,
GTK_MESSAGE_QUESTION,
GTK_BUTTONS_YES_NO,
"%s", message);
// Pass the command via g_object_set_data
g_object_set_data_full(G_OBJECT(dialog), "command", g_strdup(command), g_free);
g_signal_connect(dialog, "response", G_CALLBACK(on_confirmation_response), NULL);
gtk_window_present(GTK_WINDOW(dialog));
g_free(message);
}
static void on_confirmation_response(GtkDialog *dialog, gint response_id, gpointer user_data) {
if (response_id == GTK_RESPONSE_YES) {
const gchar *command = g_object_get_data(G_OBJECT(dialog), "command");
GtkWindow *parent = GTK_WINDOW(gtk_window_get_transient_for(GTK_WINDOW(dialog)));
execute_command(command, parent);
}
gtk_window_destroy(GTK_WINDOW(dialog));
}
static void create_button(GtkWidget *grid, GtkApplication *app, const gchar *label_text,
const gchar *icon_name, const gchar *command, int pos) {
GtkWidget *button;
GtkWidget *box;
GtkWidget *image;
GtkWidget *label;
button = gtk_button_new();
gtk_widget_set_hexpand(button, TRUE); // Allow button to expand horizontally
gtk_widget_set_vexpand(button, TRUE); // Allow button to expand vertically
box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0); // Set spacing to 0 for tighter layout
gtk_widget_set_valign(box, GTK_ALIGN_CENTER); // Center the content vertically
gtk_widget_set_halign(box, GTK_ALIGN_CENTER); // Center the content horizontally
gtk_button_set_child(GTK_BUTTON(button), box);
image = gtk_image_new_from_icon_name(icon_name);
gtk_box_append(GTK_BOX(box), image);
label = gtk_label_new(label_text);
gtk_box_append(GTK_BOX(box), label);
// Reduce margins around the button content
gtk_widget_set_margin_top(box, 5);
gtk_widget_set_margin_bottom(box, 5);
gtk_widget_set_margin_start(box, 5);
gtk_widget_set_margin_end(box, 5);
g_object_set_data(G_OBJECT(button), "app", app);
g_object_set_data(G_OBJECT(button), "label", (gpointer)label_text);
g_signal_connect(button, "clicked", G_CALLBACK(button_clicked), (gpointer)command);
gtk_grid_attach(GTK_GRID(grid), button, pos % 4, pos / 4, 1, 1);
}
static void save_configuration(const gchar *commands[]) {
GError *error = NULL;
gchar *config_dir = get_config_dir();
gchar *config_path = get_config_path();
g_mkdir_with_parents(config_dir, 0755);
GString *config_data = g_string_new(NULL);
g_string_append_printf(config_data, "LOGOUT_COMMAND=%s\n", commands[0]);
g_string_append_printf(config_data, "REBOOT_COMMAND=%s\n", commands[1]);
g_string_append_printf(config_data, "SHUTDOWN_COMMAND=%s\n", commands[2]);
g_string_append_printf(config_data, "SWITCH_USER_COMMAND=%s\n", commands[3]);
g_string_append_printf(config_data, "SUSPEND_COMMAND=%s\n", commands[4]);
g_string_append_printf(config_data, "HIBERNATE_COMMAND=%s\n", commands[5]);
g_file_set_contents(config_path, config_data->str, -1, &error);
if (error) {
g_warning("Failed to save configuration: %s", error->message);
g_error_free(error);
}
g_string_free(config_data, TRUE);
g_free(config_dir);
g_free(config_path);
}
static void load_configuration(gchar *commands[]) {
GError *error = NULL;
gchar *config_data = NULL;
gchar *config_dir = get_config_dir();
gchar *config_path = get_config_path();
g_mkdir_with_parents(config_dir, 0755);
if (!g_file_test(config_path, G_FILE_TEST_EXISTS)) {
g_warning("Configuration file not found. Generating a default configuration.");
const gchar *default_commands[] = {
"openbox --exit",
"systemctl reboot",
"systemctl poweroff",
"dm-tool switch-to-greeter",
"systemctl suspend",
"systemctl hibernate"
};
save_configuration(default_commands);
}
if (!g_file_get_contents(config_path, &config_data, NULL, &error)) {
g_warning("Failed to load configuration: %s", error->message);
g_error_free(error);
g_free(config_dir);
g_free(config_path);
return;
}
gchar **lines = g_strsplit(config_data, "\n", -1);
for (int i = 0; i < 6; i++) {
gchar **key_value = g_strsplit(lines[i], "=", 2);
if (!key_value[0] || !key_value[1]) {
g_warning("Invalid entry in configuration file at line %d. Using default command.", i + 1);
const gchar *default_commands[] = {
"openbox --exit",
"systemctl reboot",
"systemctl poweroff",
"dm-tool switch-to-greeter",
"systemctl suspend",
"systemctl hibernate"
};
commands[i] = g_strdup(default_commands[i]);
g_strfreev(key_value);
continue;
}
if (g_strcmp0(key_value[0], "LOGOUT_COMMAND") == 0) {
commands[0] = g_strdup(key_value[1]);
} else if (g_strcmp0(key_value[0], "REBOOT_COMMAND") == 0) {
commands[1] = g_strdup(key_value[1]);
} else if (g_strcmp0(key_value[0], "SHUTDOWN_COMMAND") == 0) {
commands[2] = g_strdup(key_value[1]);
} else if (g_strcmp0(key_value[0], "SWITCH_USER_COMMAND") == 0) {
commands[3] = g_strdup(key_value[1]);
} else if (g_strcmp0(key_value[0], "SUSPEND_COMMAND") == 0) {
commands[4] = g_strdup(key_value[1]);
} else if (g_strcmp0(key_value[0], "HIBERNATE_COMMAND") == 0) {
commands[5] = g_strdup(key_value[1]);
} else {
g_warning("Unknown key in configuration: %s", key_value[0]);
}
g_strfreev(key_value);
}
for (int i = 0; i < 6; i++) {
if (commands[i] == NULL || g_strcmp0(commands[i], "") == 0) {
g_warning("Command at index %d is invalid. Assigning default value.", i);
const gchar *default_commands[] = {
"openbox --exit",
"systemctl reboot",
"systemctl poweroff",
"dm-tool switch-to-greeter",
"systemctl suspend",
"systemctl hibernate"
};
commands[i] = g_strdup(default_commands[i]);
}
}
g_strfreev(lines);
g_free(config_data);
g_free(config_dir);
g_free(config_path);
}
static void activate(GtkApplication *app, gpointer user_data) { static void activate(GtkApplication *app, gpointer user_data) {
GtkWidget *window; GtkWidget *window;
GtkWidget *grid; GtkWidget *grid;
gchar *commands[6]; GtkWidget *button;
load_configuration(commands); GtkWidget *image;
GtkWidget *box;
GtkWidget *label;
const gchar *buttons[] = {
"openbox --exit",
"dbus-send --system --print-reply --dest=org.freedesktop.login1 /org/freedesktop/login1 org.freedesktop.login1.Manager.Reboot boolean:true",
"dbus-send --system --print-reply --dest=org.freedesktop.login1 /org/freedesktop/login1 org.freedesktop.login1.Manager.PowerOff boolean:true",
"dm-tool switch-to-greeter",
"dbus-send --system --print-reply --dest=org.freedesktop.login1 /org/freedesktop/login1 org.freedesktop.login1.Manager.Suspend boolean:true",
"dbus-send --system --print-reply --dest=org.freedesktop.login1 /org/freedesktop/login1 org.freedesktop.login1.Manager.Hibernate boolean:true",
"about",
"exit"
};
const gchar *icons[] = { const gchar *icons[] = {
"system-log-out", "system-log-out",
"view-refresh", "view-refresh",
@ -384,7 +120,7 @@ static void activate(GtkApplication *app, gpointer user_data) {
"system-users", "system-users",
"media-playback-pause", "media-playback-pause",
"media-playback-stop", "media-playback-stop",
"preferences-system", "help-about",
"application-exit" "application-exit"
}; };
const gchar *labels[] = { const gchar *labels[] = {
@ -394,45 +130,45 @@ static void activate(GtkApplication *app, gpointer user_data) {
"Switch User", "Switch User",
"Suspend", "Suspend",
"Hibernate", "Hibernate",
"Settings", "About",
"Exit" "Exit"
}; };
window = gtk_application_window_new(app); window = gtk_application_window_new(app);
gtk_window_set_title(GTK_WINDOW(window), "Simple ShutDown Dialog"); gtk_window_set_title(GTK_WINDOW(window), "Exit Openbox");
gtk_window_set_resizable(GTK_WINDOW(window), FALSE); gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
gtk_window_set_icon_from_file(GTK_WINDOW(window), "ssdd-icon.png", NULL);
grid = gtk_grid_new(); grid = gtk_grid_new();
gtk_grid_set_row_spacing(GTK_GRID(grid), 2); // Reduce vertical spacing between grid rows gtk_container_add(GTK_CONTAINER(window), grid);
gtk_grid_set_column_spacing(GTK_GRID(grid), 2); // Reduce horizontal spacing between grid columns
gtk_widget_set_margin_top(grid, 10);
gtk_widget_set_margin_bottom(grid, 10);
gtk_widget_set_margin_start(grid, 10);
gtk_widget_set_margin_end(grid, 10);
gtk_window_set_child(GTK_WINDOW(window), grid);
for (int i = 0; i < 6; i++) { for (int i = 0; i < 8; i++) {
create_button(grid, app, labels[i], icons[i], commands[i], i); button = gtk_button_new();
g_free(commands[i]); box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 5);
gtk_container_add(GTK_CONTAINER(button), box);
image = gtk_image_new_from_icon_name(icons[i], GTK_ICON_SIZE_BUTTON);
gtk_box_pack_start(GTK_BOX(box), image, TRUE, TRUE, 0);
label = gtk_label_new(labels[i]);
gtk_box_pack_start(GTK_BOX(box), label, TRUE, TRUE, 0);
gtk_button_set_label(GTK_BUTTON(button), labels[i]);
g_object_set_data(G_OBJECT(button), "app", app);
g_signal_connect(button, "clicked", G_CALLBACK(button_clicked), (gpointer) buttons[i]);
gtk_grid_attach(GTK_GRID(grid), button, i % 4, i / 4, 1, 1);
} }
create_button(grid, app, labels[6], icons[6], "settings", 6);
create_button(grid, app, labels[7], icons[7], "exit", 7);
// Key event handling gtk_widget_show_all(window);
GtkEventController *key_controller = gtk_event_controller_key_new();
gtk_widget_add_controller(window, key_controller);
g_signal_connect(key_controller, "key-pressed", G_CALLBACK(on_key_pressed), app);
gtk_window_present(GTK_WINDOW(window));
} }
int main(int argc, char **argv) { int main(int argc, char **argv) {
GtkApplication *app; GtkApplication *app;
int status; int status;
g_resources_register(resources_get_resource()); app = gtk_application_new("org.gtk.example", G_APPLICATION_DEFAULT_FLAGS);
app = gtk_application_new("org.gtk.ssdd", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect(app, "activate", G_CALLBACK(activate), NULL); g_signal_connect(app, "activate", G_CALLBACK(activate), NULL);
status = g_application_run(G_APPLICATION(app), argc, argv); status = g_application_run(G_APPLICATION(app), argc, argv);
g_object_unref(app); g_object_unref(app);

BIN
ssdd.png

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

After

Width:  |  Height:  |  Size: 15 KiB