};
static struct gopt_frame_widget gopt_g_widgets[__FIO_OPT_G_NR];
+static GNode *gopt_dep_tree;
+
static GtkWidget *gopt_get_group_frame(GtkWidget *box, unsigned int groupmask)
{
unsigned int mask, group;
static void gopt_set_children_visible(struct fio_option *parent,
gboolean visible)
{
- struct fio_option *o;
- int i;
+ GNode *child, *node;
- /*
- * This isn't super fast, but it should not be an issue. If it is, we
- * can speed it up by caching the lookup at least. Or we can do it
- * once, at init time.
- */
- for (i = 0; fio_options[i].name; i++) {
- o = &fio_options[i];
- if (!o->parent || !o->hide)
- continue;
+ if (parent->hide_on_set)
+ visible = !visible;
+
+ node = g_node_find(gopt_dep_tree, G_IN_ORDER, G_TRAVERSE_ALL, parent);
+ child = g_node_first_child(node);
+ while (child) {
+ struct fio_option *o = child->data;
+ struct gopt *g = o->gui_data;
- if (strcmp(parent->name, o->parent))
- continue;
+ /*
+ * Recurse into child, if it also has children
+ */
+ if (g_node_n_children(child))
+ gopt_set_children_visible(o, visible);
- if (gopt_widgets[i])
- gtk_widget_set_sensitive(gopt_widgets[i], visible);
+ if (gopt_widgets[g->opt_index])
+ gtk_widget_set_sensitive(gopt_widgets[g->opt_index], visible);
+
+ child = g_node_next_sibling(child);
}
}
if (text)
gtk_entry_set_text(GTK_ENTRY(s->entry), text);
gtk_entry_set_editable(GTK_ENTRY(s->entry), 1);
- s->gopt.sig_handler = g_signal_connect(GTK_OBJECT(s->entry), "changed", G_CALLBACK(gopt_str_changed), s);
- g_signal_connect(GTK_OBJECT(s->entry), "destroy", G_CALLBACK(gopt_str_destroy), s);
if (o->def)
gtk_entry_set_text(GTK_ENTRY(s->entry), o->def);
+ s->gopt.sig_handler = g_signal_connect(GTK_OBJECT(s->entry), "changed", G_CALLBACK(gopt_str_changed), s);
+ g_signal_connect(GTK_OBJECT(s->entry), "destroy", G_CALLBACK(gopt_str_destroy), s);
+
gtk_box_pack_start(GTK_BOX(s->gopt.box), s->entry, FALSE, FALSE, 0);
gtk_box_pack_start(GTK_BOX(s->gopt.box), label, FALSE, FALSE, 0);
- o->gui_data = s;
return &s->gopt;
}
{
struct gopt_combo *c = (struct gopt_combo *) data;
struct fio_option *o = &fio_options[c->gopt.opt_index];
+ unsigned int index;
- printf("combo %s changed\n", o->name);
+ index = gtk_combo_box_get_active(GTK_COMBO_BOX(c->combo));
+ gopt_set_children_visible(o, index);
}
static void gopt_combo_destroy(GtkWidget *w, gpointer data)
c->combo = gtk_combo_box_new_text();
gopt_mark_index(&c->gopt, idx);
- c->gopt.sig_handler = g_signal_connect(GTK_OBJECT(c->combo), "changed", G_CALLBACK(gopt_combo_changed), c);
g_signal_connect(GTK_OBJECT(c->combo), "destroy", G_CALLBACK(gopt_combo_destroy), c);
gtk_box_pack_start(GTK_BOX(c->gopt.box), c->combo, FALSE, FALSE, 0);
gtk_box_pack_start(GTK_BOX(c->gopt.box), label, FALSE, FALSE, 0);
- o->gui_data = c;
return c;
}
static struct gopt *gopt_new_combo_str(struct fio_option *o, const char *text,
unsigned int idx)
{
- struct gopt_combo *combo;
+ struct gopt_combo *c;
struct value_pair *vp;
int i, active = 0;
- combo = __gopt_new_combo(o, idx);
+ c = __gopt_new_combo(o, idx);
i = 0;
vp = &o->posval[0];
while (vp->ival) {
- gtk_combo_box_append_text(GTK_COMBO_BOX(combo->combo), vp->ival);
+ gtk_combo_box_append_text(GTK_COMBO_BOX(c->combo), vp->ival);
if (o->def && !strcmp(vp->ival, o->def))
active = i;
if (text && !strcmp(vp->ival, text))
i++;
}
- gtk_combo_box_set_active(GTK_COMBO_BOX(combo->combo), active);
- return &combo->gopt;
+ gtk_combo_box_set_active(GTK_COMBO_BOX(c->combo), active);
+ c->gopt.sig_handler = g_signal_connect(GTK_OBJECT(c->combo), "changed", G_CALLBACK(gopt_combo_changed), c);
+ return &c->gopt;
}
static struct gopt *gopt_new_combo_int(struct fio_option *o, unsigned int *ip,
unsigned int idx)
{
- struct gopt_combo *combo;
+ struct gopt_combo *c;
struct value_pair *vp;
int i, active = 0;
- combo = __gopt_new_combo(o, idx);
+ c = __gopt_new_combo(o, idx);
i = 0;
vp = &o->posval[0];
while (vp->ival) {
- gtk_combo_box_append_text(GTK_COMBO_BOX(combo->combo), vp->ival);
+ gtk_combo_box_append_text(GTK_COMBO_BOX(c->combo), vp->ival);
if (ip && vp->oval == *ip)
active = i;
vp++;
i++;
}
- gtk_combo_box_set_active(GTK_COMBO_BOX(combo->combo), active);
- return &combo->gopt;
+ gtk_combo_box_set_active(GTK_COMBO_BOX(c->combo), active);
+ c->gopt.sig_handler = g_signal_connect(GTK_OBJECT(c->combo), "changed", G_CALLBACK(gopt_combo_changed), c);
+ return &c->gopt;
}
static struct gopt *gopt_new_str_multi(struct fio_option *o, unsigned int idx)
i->lastval = value;
if (o->inv_opt) {
- struct gopt_int *i_inv = o->inv_opt->gui_data;
+ struct gopt *b_inv = o->inv_opt->gui_data;
+ struct gopt_int *i_inv = container_of(b_inv, struct gopt_int, gopt);
int cur_val;
assert(o->type == o->inv_opt->type);
gtk_box_pack_start(GTK_BOX(i->gopt.box), i->spin, FALSE, FALSE, 0);
gtk_box_pack_start(GTK_BOX(i->gopt.box), label, FALSE, FALSE, 0);
- o->gui_data = i;
return i;
}
set = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(b->check));
if (o->inv_opt) {
- struct gopt_bool *b_inv = o->inv_opt->gui_data;
+ struct gopt *g_inv = o->inv_opt->gui_data;
+ struct gopt_bool *b_inv = container_of(g_inv, struct gopt_bool, gopt);
assert(o->type == o->inv_opt->type);
gtk_box_pack_start(GTK_BOX(b->gopt.box), b->check, FALSE, FALSE, 0);
gtk_box_pack_start(GTK_BOX(b->gopt.box), label, FALSE, FALSE, 0);
- o->gui_data = b;
return &b->gopt;
}
gtk_box_pack_start(GTK_BOX(r->gopt.box), label, FALSE, FALSE, 0);
g_signal_connect(G_OBJECT(r->gopt.box), "destroy", G_CALLBACK(gopt_range_destroy), r);
- o->gui_data = r;
return &r->gopt;
}
label = gtk_label_new(o->name);
else
label = gtk_label_new(o->lname);
+ gopt_mark_index(&g->gopt, idx);
g->spin = gtk_spin_button_new_with_range(0.0, 1023.0, 1.0);
gtk_spin_button_set_update_policy(GTK_SPIN_BUTTON(g->spin), GTK_UPDATE_IF_VALID);
gtk_box_pack_start(GTK_BOX(g->gopt.box), label, FALSE, FALSE, 3);
g_signal_connect(G_OBJECT(g->gopt.box), "destroy", G_CALLBACK(gopt_str_val_destroy), g);
- o->gui_data = g;
return &g->gopt;
}
gtk_widget_set_tooltip_text(go->box, o->help);
go->opt_type = o->type;
+ o->gui_data = go;
dest = gopt_get_group_frame(hbox, o->group);
if (!dest)
GtkWidget *hbox = NULL;
int i;
+ gopt_dep_tree = g_node_new(NULL);
+
/*
* First add all options
*/
struct fio_option *o = &fio_options[i];
unsigned int mask = o->category;
struct opt_group *og;
+ GNode *node, *nparent;
+
+ /*
+ * Insert node with either the root parent, or an
+ * option parent.
+ */
+ node = g_node_new(o);
+ nparent = gopt_dep_tree;
+ if (o->parent) {
+ struct fio_option *parent;
+
+ parent = fio_option_find(o->parent);
+ nparent = g_node_find(gopt_dep_tree, G_IN_ORDER, G_TRAVERSE_ALL, parent);
+ if (!nparent) {
+ log_err("fio: did not find parent %s for opt %s\n", o->name, o->parent);
+ nparent = gopt_dep_tree;
+ }
+ }
+
+ g_node_insert(nparent, -1, node);
while ((og = opt_group_from_mask(&mask)) != NULL) {
GtkWidget *vbox = vboxes[ffz(~og->mask)];
gtk_dialog_run(GTK_DIALOG(dialog));
gtk_widget_destroy(dialog);
+
+ g_node_destroy(gopt_dep_tree);
+ gopt_dep_tree = NULL;
memset(gopt_widgets, 0, sizeof(gopt_widgets));
memset(gopt_g_widgets, 0, sizeof(gopt_g_widgets));
}