Add working directory selector to config view, suppress GTK baseline warning

- Config view gains a CWD subsection with entry, browse, and apply
- Suppress spurious GtkDropDown baseline warning via custom log writer
- Move extract_path before first use, simplify path status indicators
This commit is contained in:
Chris Punches
2026-03-17 01:58:42 -04:00
parent e55da16193
commit 36164c01a6
4 changed files with 149 additions and 12 deletions

View File

@@ -19,10 +19,23 @@
#include <gtk/gtk.h>
#include <iostream>
#include <string>
#include <cstring>
#include "models/project.h"
#include "models/grex_config.h"
#include "views/main_window.h"
// Suppress known GTK4 bug: GtkDropDown's internal arrow GtkImage reports
// INT_MIN baselines instead of -1, triggering a spurious warning.
static GLogWriterOutput grex_log_writer(GLogLevelFlags level, const GLogField* fields,
gsize n_fields, gpointer) {
for (gsize i = 0; i < n_fields; i++) {
if (std::strcmp(fields[i].key, "MESSAGE") == 0 &&
std::strstr(static_cast<const char*>(fields[i].value), "reported baselines") != nullptr)
return G_LOG_WRITER_HANDLED;
}
return g_log_writer_default(level, fields, n_fields, nullptr);
}
static grex::Project* g_project = nullptr;
static grex::GrexConfig* g_grex_config = nullptr;
static grex::MainWindow* g_main_window = nullptr;
@@ -65,6 +78,7 @@ static void on_activate(GtkApplication* app, gpointer) {
}
int main(int argc, char* argv[]) {
g_log_set_writer_func(grex_log_writer, nullptr, nullptr);
auto* app = gtk_application_new("org.darkhorselinux.grex", G_APPLICATION_HANDLES_COMMAND_LINE);
GOptionEntry entries[] = {

View File

@@ -96,6 +96,11 @@ struct DialogState {
static void update_sensitivity(DialogState* s);
static void validate_fields(DialogState* s);
static std::string extract_path(const std::string& s) {
auto pos = s.find(' ');
return pos == std::string::npos ? s : s.substr(0, pos);
}
static gboolean dlg_switch_state_set_cb(GtkSwitch* sw, gboolean new_state, gpointer data) {
auto* b = static_cast<DlgSwitchBinding*>(data);
gtk_switch_set_state(sw, new_state);
@@ -277,7 +282,7 @@ static GtkWidget* make_file_row(DialogState* s, GtkWidget* grid, int row, const
return;
}
namespace fs = std::filesystem;
fs::path p(raw);
fs::path p(extract_path(raw));
if (p.is_relative()) {
auto root = fbd->state->project->resolved_project_root();
if (!root.empty())
@@ -365,11 +370,6 @@ static void update_sensitivity(DialogState* s) {
show(active && s->working_copy.supply_environment, {s->label_environment, s->box_environment});
}
static std::string extract_path(const std::string& s) {
auto pos = s.find(' ');
return pos == std::string::npos ? s : s.substr(0, pos);
}
static void validate_fields(DialogState* s) {
if (s->loading) return;

View File

@@ -19,8 +19,12 @@
#include "views/config_view.h"
#include "util/unsaved_dialog.h"
#include "util/json_helpers.h"
#include <cerrno>
#include <climits>
#include <cstdlib>
#include <cstring>
#include <filesystem>
#include <unistd.h>
namespace grex {
@@ -123,6 +127,124 @@ ConfigView::ConfigView(Project& project) : project_(project) {
gtk_grid_set_column_spacing(GTK_GRID(vars_grid_), 16);
gtk_box_append(GTK_BOX(vars_box), vars_grid_);
// --- Current Working Directory subsection ---
gtk_box_append(GTK_BOX(vars_box), gtk_separator_new(GTK_ORIENTATION_HORIZONTAL));
auto* cwd_label = gtk_label_new(nullptr);
gtk_label_set_markup(GTK_LABEL(cwd_label), "<b>Current Working Directory</b>");
gtk_label_set_xalign(GTK_LABEL(cwd_label), 0.0f);
gtk_box_append(GTK_BOX(vars_box), cwd_label);
auto* cwd_desc = gtk_label_new("Set the working directory for this session. Relative paths in unit validation resolve against this.");
gtk_label_set_xalign(GTK_LABEL(cwd_desc), 0.0f);
gtk_label_set_wrap(GTK_LABEL(cwd_desc), TRUE);
gtk_widget_add_css_class(cwd_desc, "dim-label");
gtk_box_append(GTK_BOX(vars_box), cwd_desc);
auto* cwd_grid = gtk_grid_new();
gtk_grid_set_row_spacing(GTK_GRID(cwd_grid), 8);
gtk_grid_set_column_spacing(GTK_GRID(cwd_grid), 12);
auto* cwd_key = gtk_label_new("Directory");
gtk_label_set_xalign(GTK_LABEL(cwd_key), 1.0f);
gtk_widget_set_size_request(cwd_key, 140, -1);
gtk_widget_add_css_class(cwd_key, "dim-label");
gtk_grid_attach(GTK_GRID(cwd_grid), cwd_key, 0, 0, 1, 1);
auto* cwd_hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 4);
gtk_widget_set_hexpand(cwd_hbox, TRUE);
char cwd_buf[PATH_MAX];
const char* initial_cwd = getcwd(cwd_buf, sizeof(cwd_buf));
cwd_entry_ = gtk_entry_new();
gtk_widget_set_hexpand(cwd_entry_, TRUE);
gtk_editable_set_text(GTK_EDITABLE(cwd_entry_), initial_cwd ? initial_cwd : "");
gtk_box_append(GTK_BOX(cwd_hbox), cwd_entry_);
auto* cwd_browse = gtk_button_new_with_label("Browse...");
gtk_box_append(GTK_BOX(cwd_hbox), cwd_browse);
auto* cwd_apply = gtk_button_new_with_label("Apply");
gtk_box_append(GTK_BOX(cwd_hbox), cwd_apply);
gtk_grid_attach(GTK_GRID(cwd_grid), cwd_hbox, 1, 0, 1, 1);
cwd_status_ = gtk_label_new(nullptr);
gtk_label_set_xalign(GTK_LABEL(cwd_status_), 0.0f);
gtk_widget_add_css_class(cwd_status_, "dim-label");
gtk_grid_attach(GTK_GRID(cwd_grid), cwd_status_, 1, 1, 1, 1);
gtk_box_append(GTK_BOX(vars_box), cwd_grid);
// Apply button — chdir and report
g_signal_connect(cwd_apply, "clicked", G_CALLBACK(+[](GtkButton*, gpointer d) {
auto* self = static_cast<ConfigView*>(d);
auto dir = std::string(gtk_editable_get_text(GTK_EDITABLE(self->cwd_entry_)));
if (dir.empty()) {
self->project_.report_status("Error: working directory is empty");
return;
}
if (chdir(dir.c_str()) == 0) {
char buf[PATH_MAX];
const char* actual = getcwd(buf, sizeof(buf));
gtk_editable_set_text(GTK_EDITABLE(self->cwd_entry_), actual ? actual : dir.c_str());
gtk_label_set_text(GTK_LABEL(self->cwd_status_), "");
self->project_.report_status("Working directory set to: " + std::string(actual ? actual : dir.c_str()));
self->update_resolved_labels();
} else {
auto msg = "Error: cannot chdir to '" + dir + "': " + std::string(strerror(errno));
self->project_.report_status(msg);
gtk_label_set_markup(GTK_LABEL(self->cwd_status_),
("<span foreground=\"#cc0000\">" + msg + "</span>").c_str());
}
}), this);
// Browse button — folder chooser
g_signal_connect(cwd_browse, "clicked", G_CALLBACK(+[](GtkButton*, gpointer d) {
auto* self = static_cast<ConfigView*>(d);
auto* window = GTK_WINDOW(gtk_widget_get_ancestor(self->root_, GTK_TYPE_WINDOW));
auto* dialog = gtk_file_dialog_new();
gtk_file_dialog_set_title(dialog, "Select Working Directory");
struct CwdBrowseCtx { ConfigView* view; };
auto* ctx = new CwdBrowseCtx{self};
gtk_file_dialog_select_folder(dialog, window, nullptr,
+[](GObject* source, GAsyncResult* res, gpointer data) {
auto* ctx = static_cast<CwdBrowseCtx*>(data);
GError* error = nullptr;
auto* file = gtk_file_dialog_select_folder_finish(GTK_FILE_DIALOG(source), res, &error);
if (file) {
auto* path = g_file_get_path(file);
gtk_editable_set_text(GTK_EDITABLE(ctx->view->cwd_entry_), path);
g_free(path);
g_object_unref(file);
} else if (error) {
g_error_free(error);
}
delete ctx;
}, ctx);
}), this);
// Enter in entry triggers apply
g_signal_connect(cwd_entry_, "activate", G_CALLBACK(+[](GtkEntry*, gpointer d) {
auto* self = static_cast<ConfigView*>(d);
auto dir = std::string(gtk_editable_get_text(GTK_EDITABLE(self->cwd_entry_)));
if (dir.empty()) return;
if (chdir(dir.c_str()) == 0) {
char buf[PATH_MAX];
const char* actual = getcwd(buf, sizeof(buf));
gtk_editable_set_text(GTK_EDITABLE(self->cwd_entry_), actual ? actual : dir.c_str());
gtk_label_set_text(GTK_LABEL(self->cwd_status_), "");
self->project_.report_status("Working directory set to: " + std::string(actual ? actual : dir.c_str()));
self->update_resolved_labels();
} else {
auto msg = "Error: cannot chdir to '" + dir + "': " + std::string(strerror(errno));
self->project_.report_status(msg);
gtk_label_set_markup(GTK_LABEL(self->cwd_status_),
("<span foreground=\"#cc0000\">" + msg + "</span>").c_str());
}
}), this);
gtk_frame_set_child(GTK_FRAME(vars_frame), vars_box);
gtk_box_append(GTK_BOX(config_content_), vars_frame);
@@ -297,14 +419,12 @@ void ConfigView::update_resolved_labels() {
// Status indicator
bool path_ok = (display != "(unresolved)") &&
(std::filesystem::exists(display) || std::filesystem::is_directory(display));
auto* indicator = gtk_label_new(nullptr);
auto* indicator = gtk_label_new(path_ok ? "OK" : "ERR");
gtk_label_set_xalign(GTK_LABEL(indicator), 0.5f);
if (display == "(unresolved)")
gtk_label_set_markup(GTK_LABEL(indicator), "<span foreground=\"#cc0000\">\u2718</span>");
else if (path_ok)
gtk_label_set_markup(GTK_LABEL(indicator), "<span foreground=\"#4e9a06\">\u2714</span>");
if (path_ok)
gtk_widget_add_css_class(indicator, "success");
else
gtk_label_set_markup(GTK_LABEL(indicator), "<span foreground=\"#cc0000\">\u2718</span>");
gtk_widget_add_css_class(indicator, "error");
gtk_grid_attach(GTK_GRID(resolved_grid_), indicator, 1, row, 1, 1);
auto* val_label = gtk_label_new(nullptr);

View File

@@ -19,6 +19,7 @@
#pragma once
#include <gtk/gtk.h>
#include <vector>
#include <string>
#include "models/project.h"
namespace grex {
@@ -58,6 +59,8 @@ private:
GtkWidget* btn_close_;
GtkWidget* btn_save_;
GtkWidget* config_content_;
GtkWidget* cwd_entry_;
GtkWidget* cwd_status_;
void build_config_fields();
void build_variables_section();