Back to Darktable

imageop_gui.c/h - GUI Helper Functions for IOP Modules

dev-doc/imageop_gui.md

0.410.2 KB
Original Source

imageop_gui.c/h - GUI Helper Functions for IOP Modules

This document covers the helper functions in src/develop/imageop_gui.c and imageop_gui.h that simplify creating GUI widgets for darktable's image operation (IOP) modules.

Overview

These functions work with darktable's introspection system to automatically:

  • Configure widget ranges, defaults, and labels from struct definitions
  • Set up callbacks that sync widget values to module parameters
  • Register widgets with the shortcut/action system

Primary Widget Creation Functions

dt_bauhaus_slider_from_params()

c
GtkWidget *dt_bauhaus_slider_from_params(dt_iop_module_t *self, const char *param);

Creates a slider widget automatically configured from the module's parameter struct definition.

Parameters:

  • self: The module instance
  • param: Name of the field in params_t struct (or use N_("field") to mark for translation)

What it does:

  1. Reads $MIN, $MAX, $DEFAULT comments from the struct definition
  2. Sets up the slider range and default value
  3. Uses $DESCRIPTION as the widget label
  4. Creates an automatic callback that syncs slider → self->params->field
  5. Packs the widget into self->widget (the current container)
  6. Registers with the action/shortcut system

Array Indexing: For array parameters, use bracket notation:

c
// For: float Dmin[3]; in params_t
g->Dmin_R = dt_bauhaus_slider_from_params(self, "Dmin[0]");  // Red
g->Dmin_G = dt_bauhaus_slider_from_params(self, "Dmin[1]");  // Green
g->Dmin_B = dt_bauhaus_slider_from_params(self, "Dmin[2]");  // Blue

Example (from exposure.c):

c
g->exposure = dt_bauhaus_slider_from_params(self, N_("exposure"));
dt_bauhaus_slider_set_digits(g->exposure, 3);
dt_bauhaus_slider_set_format(g->exposure, _(" EV"));
dt_bauhaus_slider_set_soft_range(g->exposure, -3.0, 4.0);

dt_bauhaus_combobox_from_params()

c
GtkWidget *dt_bauhaus_combobox_from_params(dt_iop_module_t *self, const char *param);

Creates a combobox automatically populated from an enum definition.

Parameters:

  • self: The module instance
  • param: Name of the enum field in params_t

What it does:

  1. Reads enum values and their $DESCRIPTION comments
  2. Populates the combobox with all enum entries
  3. Sets up callback to store the enum value (not index) in self->params
  4. Packs into self->widget and registers with action system

Example (from filmicrgb.c):

c
// In params_t:
// typedef enum { PRESERVE_NONE, PRESERVE_LUMINANCE, ... } dt_preserve_color_t;
// dt_preserve_color_t preserve_color; // $DESCRIPTION: "preserve chrominance"

g->preserve_color = dt_bauhaus_combobox_from_params(self, "preserve_color");

dt_bauhaus_toggle_from_params()

c
GtkWidget *dt_bauhaus_toggle_from_params(dt_iop_module_t *self, const char *param);

Creates a toggle button (checkbox) linked to a boolean parameter.

Parameters:

  • self: The module instance
  • param: Name of the gboolean field in params_t

Returns: A GtkToggleButton widget.

Important: Unlike sliders/comboboxes, toggle buttons require manual update in gui_update():

c
void gui_update(dt_iop_module_t *self)
{
  dt_iop_mymodule_gui_data_t *g = self->gui_data;
  dt_iop_mymodule_params_t *p = self->params;

  // Sliders/comboboxes update automatically, but toggles need this:
  gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(g->my_toggle), p->my_bool);
}

Example (from vignette.c):

c
g->autoratio = dt_bauhaus_toggle_from_params(self, "autoratio");

Button Creation Functions

dt_iop_togglebutton_new()

c
GtkWidget *dt_iop_togglebutton_new(
    dt_iop_module_t *self,
    const char *section,        // Section name for shortcuts (or NULL)
    const gchar *label,         // Label for action system (N_("label"))
    const gchar *ctrl_label,    // Ctrl+click action label (or NULL)
    GCallback callback,         // Function called on toggle
    gboolean local,             // TRUE = local shortcut, FALSE = global
    guint accel_key,            // Accelerator key (0 for none)
    GdkModifierType mods,       // Modifier keys
    DTGTKCairoPaintIconFunc paint,  // Icon paint function
    GtkWidget *box              // Container to pack into (or NULL)
);

Creates a toggle button with an icon, typically used for mode switches or mask display toggles.

Icon Paint Functions (from dtgtk/paint.h):

  • dtgtk_cairo_paint_showmask - Mask display icon
  • dtgtk_cairo_paint_eye / dtgtk_cairo_paint_eye_toggle - Eye icon
  • dtgtk_cairo_paint_colorpicker - Color picker pipette
  • dtgtk_cairo_paint_masks_brush - Brush tool
  • dtgtk_cairo_paint_masks_circle - Circle mask
  • dtgtk_cairo_paint_masks_ellipse - Ellipse mask
  • dtgtk_cairo_paint_masks_path - Path/bezier mask

Example (from toneequal.c):

c
g->show_luminance_mask = dt_iop_togglebutton_new(
    self,                           // module
    N_("display"),                  // section for shortcuts
    N_("show luminance mask"),      // action label
    NULL,                           // no ctrl+click action
    G_CALLBACK(show_luminance_mask_callback),
    FALSE,                          // global shortcut
    0, 0,                           // no accelerator
    dtgtk_cairo_paint_showmask,     // mask icon
    self->widget                    // pack into module widget
);

dt_iop_button_new()

c
GtkWidget *dt_iop_button_new(
    dt_iop_module_t *self,
    const gchar *label,         // Label for action system
    GCallback callback,         // Function called on click
    gboolean local,             // TRUE = local shortcut
    guint accel_key,            // Accelerator key
    GdkModifierType mods,       // Modifier keys
    DTGTKCairoPaintIconFunc paint,  // Icon paint function
    gint paintflags,            // Paint flags (CPF_DIRECTION_*, etc.)
    GtkWidget *box              // Container to pack into
);

Creates a regular (non-toggle) button with an icon.

Example (from flip.c):

c
dt_iop_button_new(self, N_("rotate 90 degrees CCW"),
                  G_CALLBACK(rotate_ccw), FALSE, 0, 0,
                  dtgtk_cairo_paint_refresh,
                  CPF_DIRECTION_UP,  // Icon direction flag
                  self->widget);

Section Organization

DT_IOP_SECTION_FOR_PARAMS Macro

c
dt_iop_module_t *DT_IOP_SECTION_FOR_PARAMS(dt_iop_module_t *self, const char *name);
// Or with explicit container:
dt_iop_module_t *DT_IOP_SECTION_FOR_PARAMS(dt_iop_module_t *self, const char *name, GtkWidget *box);

Creates a logical section for organizing shortcuts hierarchically. Widgets created with this "section module" appear under module/section/widget in the shortcut system.

Important: This does NOT create visual sections - use dt_ui_section_label_new() for that.

Example (from colorbalancergb.c):

c
// Create section for "chroma" controls
dt_iop_module_t *sect = DT_IOP_SECTION_FOR_PARAMS(self, N_("chroma"));

// These sliders will be registered as "colorbalancergb/chroma/global chroma"
dt_bauhaus_slider_from_params(sect, "chroma_global");
dt_bauhaus_slider_from_params(sect, "chroma_highlights");
dt_bauhaus_slider_from_params(sect, "chroma_midtones");
dt_bauhaus_slider_from_params(sect, "chroma_shadows");

Collapsible Sections

dt_gui_new_collapsible_section()

c
#include "gui/gtk.h"

dt_gui_collapsible_section_t cs;
dt_gui_new_collapsible_section(&cs,
    "plugins/darkroom/mymodule/expand_advanced",  // conf key for persisting state
    _("advanced options"),                          // header label
    GTK_BOX(self->widget),                         // parent container
    DT_ACTION(self));                              // module for shortcuts

After creation, pack widgets into cs.container:

c
// Temporarily redirect widget packing into the collapsible section
self->widget = GTK_WIDGET(cs.container);

g->detail_slider = dt_bauhaus_slider_from_params(self, "detail");
g->method_combo = dt_bauhaus_combobox_from_params(self, "method");

self->widget = main_box;  // restore original packing target

The section state (expanded/collapsed) is automatically persisted via the configuration key. Store the dt_gui_collapsible_section_t in your gui_data_t if you need to reference it later.

Related functions:

  • dt_gui_update_collapsible_section(&cs) — sync section state from config
  • dt_gui_hide_collapsible_section(&cs) — programmatically collapse

Helper Function

dt_mask_scroll_increases()

c
gboolean dt_mask_scroll_increases(int up);

Returns TRUE if scrolling "up" should increase a value, respecting user preferences. Use this when implementing custom scroll behavior.


Interpolation Combobox

dt_bauhaus_combobox_new_interpolation()

c
GtkWidget *dt_bauhaus_combobox_new_interpolation(dt_iop_module_t *self);

Creates a standard combobox pre-populated with interpolation methods (bilinear, bicubic, lanczos, etc.). Used by modules that need to offer interpolation choices.


Key Points to Remember

  1. _from_params functions auto-pack widgets into self->widget. Set self->widget to the correct container before calling them.

  2. Toggles need manual gui_update() - unlike sliders/comboboxes which auto-sync.

  3. Slider format order matters - Set factor before format before digits to avoid rounding issues. See sliders.md for all configuration options and recipes.

  4. Sections are for shortcuts, not visual organization. Use dt_ui_section_label_new() for visual headers.

  5. Color pickers wrap sliders - The picker becomes the widget to store/pack, not the original slider. See sliders.md for attachment details.

  6. Don't set widget values in gui_init() - Only create and configure widgets. Setting values belongs in gui_update(), which is called when params are available.

  7. Always call gui_changed() from gui_update() - End gui_update() with gui_changed(self, NULL, NULL) to ensure all UI state adjustments (visibility, sensitivity) are applied consistently. See GUI.md for the full event flow.

  8. Use dt_ui_label_new() for labels - Never use gtk_label_new() directly. The wrapper ensures long text is ellipsized and doesn't stretch the panel width.