1112 lines
28 KiB
C
1112 lines
28 KiB
C
// SPDX-FileCopyrightText: 2021 Stiliyan Varbanov <https://www.fiverr.com/stilvar>
|
|
// SPDX-FileCopyrightText: 2022-2024 Geoffrey D. Bennett <g@b4.vu>
|
|
// SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
/*
|
|
* A Dial widget for GTK-4 similar to GtkScale.
|
|
* 2021 Stiliyan Varbanov www.fiverr.com/stilvar
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <glib-2.0/glib.h>
|
|
#include <glib-object.h>
|
|
#include <cairo/cairo.h>
|
|
#include <stdio.h>
|
|
#include <graphene-1.0/graphene.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include "gtkdial.h"
|
|
|
|
#define DIAL_MIN_WIDTH 60
|
|
#define DIAL_MAX_WIDTH 70
|
|
|
|
static void set_value(GtkDial *dial, double newval);
|
|
|
|
static void gtk_dial_set_property(
|
|
GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec
|
|
);
|
|
|
|
static void gtk_dial_get_property(
|
|
GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec
|
|
);
|
|
|
|
static void gtk_dial_move_slider(GtkDial *dial, GtkScrollType scroll);
|
|
|
|
static void gtk_dial_drag_gesture_begin(
|
|
GtkGestureDrag *gesture,
|
|
double offset_x,
|
|
double offset_y,
|
|
GtkDial *dial
|
|
);
|
|
|
|
static void gtk_dial_drag_gesture_update(
|
|
GtkGestureDrag *gesture,
|
|
double offset_x,
|
|
double offset_y,
|
|
GtkDial *dial
|
|
);
|
|
|
|
static void gtk_dial_drag_gesture_end(
|
|
GtkGestureDrag *gesture,
|
|
double offset_x,
|
|
double offset_y,
|
|
GtkDial *dial
|
|
);
|
|
|
|
static void gtk_dial_click_gesture_pressed(
|
|
GtkGestureClick *gesture,
|
|
int n_press,
|
|
double x,
|
|
double y,
|
|
GtkDial *dial
|
|
);
|
|
|
|
static gboolean gtk_dial_scroll_controller_scroll(
|
|
GtkEventControllerScroll *scroll,
|
|
double dx,
|
|
double dy,
|
|
GtkDial *dial
|
|
);
|
|
|
|
static void gtk_dial_dispose(GObject *o);
|
|
|
|
typedef enum {
|
|
GRAB_NONE,
|
|
GRAB_SLIDER
|
|
} e_grab;
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_ADJUSTMENT,
|
|
PROP_ROUND_DIGITS,
|
|
PROP_ZERO_DB,
|
|
PROP_TAPER,
|
|
LAST_PROP
|
|
};
|
|
|
|
enum {
|
|
SIGNAL_0,
|
|
VALUE_CHANGED,
|
|
MOVE_SLIDER,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL];
|
|
static GParamSpec *properties[LAST_PROP];
|
|
|
|
struct _GtkDial {
|
|
GtkWidget parent_instance;
|
|
GtkAdjustment *adj;
|
|
|
|
GtkGesture *drag_gesture, *click_gesture;
|
|
GtkEventController *scroll_controller;
|
|
e_grab grab;
|
|
double dvalp;
|
|
|
|
int round_digits;
|
|
double zero_db;
|
|
int taper;
|
|
|
|
// linear taper breakpoints array
|
|
double *taper_breakpoints;
|
|
double *taper_outputs;
|
|
int taper_breakpoints_count;
|
|
};
|
|
|
|
G_DEFINE_TYPE(GtkDial, gtk_dial, GTK_TYPE_WIDGET)
|
|
|
|
static void dial_snapshot(GtkWidget *widget, GtkSnapshot *snapshot);
|
|
static void dial_measure(
|
|
GtkWidget *widget,
|
|
GtkOrientation orientation,
|
|
int for_size,
|
|
int *minimum,
|
|
int *natural,
|
|
int *minimum_baseline,
|
|
int *natural_baseline
|
|
);
|
|
|
|
#define add_slider_binding(w_class, binding_set, keyval, mask, scroll) \
|
|
gtk_widget_class_add_binding_signal(w_class, \
|
|
keyval, mask, \
|
|
"move-slider", \
|
|
"(i)", scroll)
|
|
|
|
// BEGIN SECTION HELPERS
|
|
|
|
#define TOTAL_ROTATION_DEGREES 290
|
|
#define TOTAL_ROTATION (2 * M_PI * TOTAL_ROTATION_DEGREES / 360)
|
|
#define ANGLE_START (-M_PI / 2 - TOTAL_ROTATION / 2)
|
|
#define ANGLE_END (-M_PI / 2 + TOTAL_ROTATION / 2)
|
|
|
|
#define DRAG_FACTOR 0.5
|
|
|
|
// convert val from mn..mx to 0..1 with clamp
|
|
static double calc_valp(double val, double mn, double mx) {
|
|
if (val <= mn)
|
|
return 0.0;
|
|
if (val >= mx)
|
|
return 1.0;
|
|
|
|
return (val - mn) / (mx - mn);
|
|
}
|
|
|
|
static double taper_linear(double val, double *bp, double *out, int count) {
|
|
if (count < 3)
|
|
return val;
|
|
|
|
if (val <= bp[0])
|
|
return out[0];
|
|
|
|
for (int i = 0; i < count - 1; i++) {
|
|
if (val > bp[i + 1])
|
|
continue;
|
|
|
|
double scale = (out[i + 1] - out[i]) / (bp[i + 1] - bp[i]);
|
|
return out[i] + scale * (val - bp[i]);
|
|
}
|
|
|
|
return out[count - 1];
|
|
}
|
|
|
|
static double taper_log(double val) {
|
|
|
|
// 10^(val - 1) converts it to 0.1..1 with a nice curve
|
|
val = pow(10, val - 1);
|
|
|
|
// convert to 0..1 again
|
|
return (val - 0.1) / 0.9;
|
|
}
|
|
|
|
static double calc_taper(GtkDial *dial, double val) {
|
|
if (dial->taper == GTK_DIAL_TAPER_LINEAR)
|
|
return taper_linear(
|
|
val,
|
|
dial->taper_breakpoints,
|
|
dial->taper_outputs,
|
|
dial->taper_breakpoints_count
|
|
);
|
|
|
|
if (dial->taper == GTK_DIAL_TAPER_LOG)
|
|
return taper_log(val);
|
|
|
|
g_warning("Invalid taper value: %d", dial->taper);
|
|
|
|
return val;
|
|
}
|
|
|
|
static double calc_val(double valp, double mn, double mx) {
|
|
return (mx - mn) * valp + mn;
|
|
}
|
|
|
|
struct dial_properties {
|
|
double w;
|
|
double h;
|
|
double radius;
|
|
double slider_thickness;
|
|
double cx;
|
|
double cy;
|
|
double valp;
|
|
double angle;
|
|
double slider_cx;
|
|
double slider_cy;
|
|
int dim;
|
|
};
|
|
|
|
static int calculate_dial_height(int width) {
|
|
double radius = width / 2;
|
|
double angle = (360 - TOTAL_ROTATION_DEGREES) / 2 * M_PI / 180;
|
|
double height = radius + radius * cos(angle);
|
|
|
|
return ceil(height);
|
|
}
|
|
|
|
static double calculate_dial_radius_from_height(int height) {
|
|
double angle = (360 - TOTAL_ROTATION_DEGREES) / 2.0 * M_PI / 180.0;
|
|
return height / (1 + cos(angle));
|
|
}
|
|
|
|
static void get_dial_properties(
|
|
GtkDial *dial,
|
|
struct dial_properties *props
|
|
) {
|
|
props->w = gtk_widget_get_width(GTK_WIDGET(dial));
|
|
props->h = gtk_widget_get_height(GTK_WIDGET(dial));
|
|
|
|
double width = props->w;
|
|
if (width > DIAL_MAX_WIDTH)
|
|
width = DIAL_MAX_WIDTH;
|
|
|
|
double max_height = calculate_dial_height(DIAL_MAX_WIDTH);
|
|
double height = props->h;
|
|
if (height > max_height)
|
|
height = max_height;
|
|
|
|
double radius_from_width = width / 2;
|
|
|
|
double radius_from_height = calculate_dial_radius_from_height(height);
|
|
|
|
props->radius = radius_from_width < radius_from_height ?
|
|
radius_from_width : radius_from_height;
|
|
props->radius -= 0.5;
|
|
|
|
props->cx = props->w / 2;
|
|
double angle = (360 - TOTAL_ROTATION_DEGREES) / 2.0 * M_PI / 180.0;
|
|
double y_offset = props->radius * cos(angle);
|
|
props->cy = (props->h / 2.0) + (props->radius - y_offset) / 2.0 - 0.5;
|
|
|
|
props->slider_thickness = 16;
|
|
|
|
double mn = dial->adj ? gtk_adjustment_get_lower(dial->adj) : 0;
|
|
double mx = dial->adj ? gtk_adjustment_get_upper(dial->adj) : 1;
|
|
double value = dial->adj ? gtk_adjustment_get_value(dial->adj) : 0.25;
|
|
props->valp = calc_taper(dial, calc_valp(value, mn, mx));
|
|
|
|
props->angle = calc_val(props->valp, ANGLE_START, ANGLE_END);
|
|
double slider_radius = props->radius - props->slider_thickness / 2;
|
|
props->slider_cx = cos(props->angle) * slider_radius + props->cx;
|
|
props->slider_cy = sin(props->angle) * slider_radius + props->cy;
|
|
|
|
props->dim = !gtk_widget_is_sensitive(GTK_WIDGET(dial));
|
|
}
|
|
|
|
static double pdist2(double x1, double y1, double x2, double y2) {
|
|
double dx = x2 - x1;
|
|
double dy = y2 - y1;
|
|
|
|
return dx * dx + dy * dy;
|
|
}
|
|
|
|
static gboolean circle_contains_point(
|
|
double cx,
|
|
double cy,
|
|
double r,
|
|
double px,
|
|
double py
|
|
) {
|
|
return pdist2(cx, cy, px, py) <= r * r;
|
|
}
|
|
|
|
// END SECTION HELPERS
|
|
|
|
static void gtk_dial_class_init(GtkDialClass *klass) {
|
|
GtkWidgetClass *w_class = GTK_WIDGET_CLASS(klass);
|
|
GObjectClass *g_class = G_OBJECT_CLASS(klass);
|
|
GtkWidgetClass *p_class = GTK_WIDGET_CLASS(gtk_dial_parent_class);
|
|
|
|
g_class->set_property = >k_dial_set_property;
|
|
g_class->get_property = >k_dial_get_property;
|
|
g_class->dispose = >k_dial_dispose;
|
|
|
|
w_class->size_allocate = p_class->size_allocate;
|
|
w_class->measure = &dial_measure;
|
|
w_class->snapshot = &dial_snapshot;
|
|
w_class->grab_focus = p_class->grab_focus;
|
|
w_class->focus = p_class->focus;
|
|
|
|
klass->move_slider = >k_dial_move_slider;
|
|
klass->value_changed = NULL;
|
|
|
|
gtk_widget_class_set_css_name(w_class, "dial");
|
|
|
|
/**
|
|
* GtkDial:adjustment: (attributes org.gtk.Method.get=gtk_dial_get_adjustment org.gtk.Method.set=gtk_dial_set_adjustment)
|
|
*
|
|
* The GtkAdjustment that contains the current value of this dial object.
|
|
*/
|
|
properties[PROP_ADJUSTMENT] = g_param_spec_object(
|
|
"adjustment",
|
|
"Adjustment",
|
|
"The GtkAdjustment that contains the current value of this dial object",
|
|
GTK_TYPE_ADJUSTMENT,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT
|
|
);
|
|
|
|
/**
|
|
* GtkDial:round_digits: (attributes org.gtk.Method.get=gtk_dial_get_round_digits org.gtk.Method.set=gtk_dial_set_round_digits)
|
|
*
|
|
* Limits the number of decimal points this GtkDial will store (default 0: integers).
|
|
*/
|
|
properties[PROP_ROUND_DIGITS] = g_param_spec_int(
|
|
"round_digits",
|
|
"RoundDigits",
|
|
"Limits the number of decimal points this GtkDial will store",
|
|
-1, 1000,
|
|
-1,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT
|
|
);
|
|
|
|
/**
|
|
* GtkDial:zero_db: (attributes org.gtk.Method.get=gtk_dial_get_zero_db org.gtk.Method.set=gtk_dial_set_zero_db)
|
|
*
|
|
* The zero-dB value of the dial.
|
|
*/
|
|
properties[PROP_ZERO_DB] = g_param_spec_double(
|
|
"zero_db",
|
|
"ZerodB",
|
|
"The zero-dB value of the dial",
|
|
-G_MAXDOUBLE, G_MAXDOUBLE,
|
|
0.0,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT
|
|
);
|
|
|
|
/**
|
|
* GtkDial:taper: (attributes org.gtk.Method.get=gtk_dial_get_taper org.gtk.Method.set=gtk_dial_set_taper)
|
|
*
|
|
* The taper of the dial.
|
|
*/
|
|
properties[PROP_TAPER] = g_param_spec_int(
|
|
"taper",
|
|
"Taper",
|
|
"The taper of the dial",
|
|
GTK_DIAL_TAPER_LINEAR, GTK_DIAL_TAPER_LOG,
|
|
GTK_DIAL_TAPER_LINEAR,
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT
|
|
);
|
|
|
|
g_object_class_install_properties(g_class, LAST_PROP, properties);
|
|
|
|
/**
|
|
* GtkDial::value-changed:
|
|
* @dial: the `GtkDial` that received the signal
|
|
*
|
|
* Emitted when the dial value changes.
|
|
*/
|
|
signals[VALUE_CHANGED] = g_signal_new(
|
|
"value-changed",
|
|
G_TYPE_FROM_CLASS(g_class),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET(GtkDialClass, value_changed),
|
|
NULL, NULL,
|
|
NULL,
|
|
G_TYPE_NONE, 0
|
|
);
|
|
|
|
/**
|
|
* GtkDial::move-slider:
|
|
* @Dial: the `GtkDial` that received the signal
|
|
* @step: how to move the slider
|
|
*
|
|
* Virtual function that moves the slider.
|
|
*
|
|
* Used for keybindings.
|
|
*/
|
|
signals[MOVE_SLIDER] = g_signal_new(
|
|
"move-slider",
|
|
G_TYPE_FROM_CLASS(g_class),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET(GtkDialClass, move_slider),
|
|
NULL, NULL,
|
|
NULL,
|
|
G_TYPE_NONE, 1,
|
|
GTK_TYPE_SCROLL_TYPE
|
|
);
|
|
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Left, 0, GTK_SCROLL_STEP_LEFT);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Down, 0, GTK_SCROLL_STEP_LEFT);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Right, 0, GTK_SCROLL_STEP_RIGHT);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Up, 0, GTK_SCROLL_STEP_RIGHT);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Page_Up, 0, GTK_SCROLL_PAGE_RIGHT);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Page_Down, 0, GTK_SCROLL_PAGE_LEFT);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_Home, 0, GTK_SCROLL_START);
|
|
add_slider_binding(w_class, binding_set, GDK_KEY_End, 0, GTK_SCROLL_END);
|
|
}
|
|
|
|
static void gtk_dial_focus_change_cb(
|
|
GtkEventControllerFocus *controller, GtkDial *dial
|
|
) {
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
}
|
|
|
|
static void gtk_dial_init(GtkDial *dial) {
|
|
gtk_widget_set_focusable(GTK_WIDGET(dial), TRUE);
|
|
|
|
dial->adj = NULL;
|
|
|
|
dial->grab = GRAB_NONE;
|
|
dial->drag_gesture = gtk_gesture_drag_new();
|
|
gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(dial->drag_gesture), 0);
|
|
g_signal_connect(
|
|
dial->drag_gesture, "drag-begin",
|
|
G_CALLBACK(gtk_dial_drag_gesture_begin), dial
|
|
);
|
|
g_signal_connect(
|
|
dial->drag_gesture, "drag-update",
|
|
G_CALLBACK(gtk_dial_drag_gesture_update), dial
|
|
);
|
|
g_signal_connect(
|
|
dial->drag_gesture, "drag-end",
|
|
G_CALLBACK(gtk_dial_drag_gesture_end), dial
|
|
);
|
|
gtk_widget_add_controller(
|
|
GTK_WIDGET(dial), GTK_EVENT_CONTROLLER(dial->drag_gesture)
|
|
);
|
|
|
|
dial->click_gesture = gtk_gesture_click_new();
|
|
gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(dial->click_gesture), 0);
|
|
g_signal_connect(
|
|
dial->click_gesture, "pressed",
|
|
G_CALLBACK(gtk_dial_click_gesture_pressed), dial
|
|
);
|
|
gtk_widget_add_controller(
|
|
GTK_WIDGET(dial), GTK_EVENT_CONTROLLER(dial->click_gesture)
|
|
);
|
|
gtk_gesture_group(dial->click_gesture, dial->drag_gesture);
|
|
|
|
dial->scroll_controller = gtk_event_controller_scroll_new(
|
|
GTK_EVENT_CONTROLLER_SCROLL_BOTH_AXES
|
|
);
|
|
g_signal_connect(
|
|
dial->scroll_controller, "scroll",
|
|
G_CALLBACK(gtk_dial_scroll_controller_scroll), dial
|
|
);
|
|
gtk_widget_add_controller(GTK_WIDGET(dial), dial->scroll_controller);
|
|
|
|
GtkEventController *controller = gtk_event_controller_focus_new();
|
|
g_signal_connect(
|
|
controller, "enter", G_CALLBACK(gtk_dial_focus_change_cb), dial
|
|
);
|
|
g_signal_connect(
|
|
controller, "leave", G_CALLBACK(gtk_dial_focus_change_cb), dial
|
|
);
|
|
gtk_widget_add_controller(GTK_WIDGET(dial), controller);
|
|
}
|
|
|
|
static void dial_measure(
|
|
GtkWidget *widget,
|
|
GtkOrientation orientation,
|
|
int for_size,
|
|
int *minimum,
|
|
int *natural,
|
|
int *minimum_baseline,
|
|
int *natural_baseline
|
|
) {
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL) {
|
|
*minimum = DIAL_MIN_WIDTH;
|
|
*natural = DIAL_MAX_WIDTH;
|
|
} else {
|
|
*minimum = calculate_dial_height(DIAL_MIN_WIDTH);
|
|
*natural = calculate_dial_height(DIAL_MAX_WIDTH);
|
|
}
|
|
*minimum_baseline = -1;
|
|
*natural_baseline = -1;
|
|
}
|
|
|
|
// internal replacement for cairo_set_source_rgba() that dims the
|
|
// color if the widget is insensitive
|
|
static void cairo_set_source_rgba_dim(
|
|
cairo_t *cr,
|
|
double r,
|
|
double g,
|
|
double b,
|
|
double a,
|
|
int dim
|
|
) {
|
|
if (dim)
|
|
cairo_set_source_rgba(cr, r * 0.5, g * 0.5, b * 0.5, a);
|
|
else
|
|
cairo_set_source_rgba(cr, r, g, b, a);
|
|
}
|
|
|
|
// internal replacement for cairo_pattern_add_color_stop_rgb() that
|
|
// dims the color if the widget is insensitive
|
|
static void cairo_add_stop_rgb_dim(
|
|
cairo_pattern_t *pat,
|
|
double offset,
|
|
double r,
|
|
double g,
|
|
double b,
|
|
int dim
|
|
) {
|
|
if (dim)
|
|
cairo_pattern_add_color_stop_rgb(pat, offset, r * 0.5, g * 0.5, b * 0.5);
|
|
else
|
|
cairo_pattern_add_color_stop_rgb(pat, offset, r, g, b);
|
|
}
|
|
|
|
static void draw_slider(
|
|
cairo_t *cr,
|
|
struct dial_properties *p,
|
|
double radius,
|
|
double thickness,
|
|
double alpha
|
|
) {
|
|
cairo_arc(cr, p->cx, p->cy, radius, ANGLE_START, p->angle);
|
|
cairo_set_line_width(cr, thickness);
|
|
cairo_set_source_rgba_dim(cr, 1, 1, 1, alpha, p->dim);
|
|
cairo_stroke(cr);
|
|
}
|
|
|
|
static void dial_snapshot(GtkWidget *widget, GtkSnapshot *snapshot) {
|
|
GtkDial *dial = GTK_DIAL(widget);
|
|
|
|
struct dial_properties p;
|
|
get_dial_properties(dial, &p);
|
|
|
|
cairo_t *cr = gtk_snapshot_append_cairo(
|
|
snapshot,
|
|
&GRAPHENE_RECT_INIT(0, 0, p.w, p.h)
|
|
);
|
|
|
|
cairo_set_line_cap(cr, CAIRO_LINE_CAP_ROUND);
|
|
cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
|
|
|
|
double slider_radius = p.radius - p.slider_thickness / 2;
|
|
double background_radius = slider_radius + p.slider_thickness / 4;
|
|
|
|
// background line
|
|
cairo_arc(cr, p.cx, p.cy, slider_radius, ANGLE_START, ANGLE_END);
|
|
cairo_set_line_width(cr, 2);
|
|
cairo_set_source_rgba_dim(cr, 1, 1, 1, 0.17, p.dim);
|
|
cairo_stroke(cr);
|
|
|
|
if (p.valp > 0.0) {
|
|
// outside value shadow
|
|
draw_slider(cr, &p, background_radius, p.slider_thickness / 2, 0.1);
|
|
|
|
// value blur 2
|
|
draw_slider(cr, &p, slider_radius, 6, 0.3);
|
|
}
|
|
|
|
// draw line to zero db
|
|
double zero_db = gtk_dial_get_zero_db(dial);
|
|
if (zero_db != 0.0) {
|
|
double zero_db_valp = calc_taper(
|
|
dial,
|
|
calc_valp(
|
|
zero_db,
|
|
gtk_adjustment_get_lower(dial->adj),
|
|
gtk_adjustment_get_upper(dial->adj)
|
|
)
|
|
);
|
|
|
|
double zero_db_angle = calc_val(zero_db_valp, ANGLE_START, ANGLE_END);
|
|
double zero_db_x = cos(zero_db_angle) * slider_radius + p.cx;
|
|
double zero_db_y = sin(zero_db_angle) * slider_radius + p.cy;
|
|
|
|
cairo_move_to(cr, p.cx, p.cy);
|
|
cairo_line_to(cr, zero_db_x, zero_db_y);
|
|
cairo_set_line_width(cr, 2);
|
|
cairo_set_source_rgba_dim(cr, 1, 1, 1, 0.17, p.dim);
|
|
cairo_stroke(cr);
|
|
}
|
|
|
|
// marker when at min or max
|
|
if (gtk_dial_get_value(dial) == gtk_adjustment_get_lower(dial->adj) ||
|
|
gtk_dial_get_value(dial) == gtk_adjustment_get_upper(dial->adj)) {
|
|
cairo_move_to(cr, p.cx, p.cy);
|
|
cairo_line_to(cr, p.slider_cx, p.slider_cy);
|
|
cairo_set_line_width(cr, 2);
|
|
cairo_set_source_rgba_dim(cr, 1, 1, 1, 0.5, p.dim);
|
|
cairo_stroke(cr);
|
|
}
|
|
|
|
if (p.valp > 0.0) {
|
|
// value blur 1
|
|
draw_slider(cr, &p, slider_radius, 4, 0.5);
|
|
|
|
// value
|
|
draw_slider(cr, &p, slider_radius, 2, 1);
|
|
}
|
|
|
|
// fill the circle
|
|
cairo_pattern_t *pat;
|
|
pat = cairo_pattern_create_radial(
|
|
p.cx + 5, p.cy + 5, 0, p.cx, p.cy, p.radius
|
|
);
|
|
if (gtk_widget_has_focus(GTK_WIDGET(dial))) {
|
|
cairo_add_stop_rgb_dim(pat, 0.0, 0.30, 0.30, 0.33, p.dim);
|
|
cairo_add_stop_rgb_dim(pat, 0.4, 0.30, 0.30, 0.33, p.dim);
|
|
cairo_add_stop_rgb_dim(pat, 1.0, 0.50, 0.50, 0.53, p.dim);
|
|
} else {
|
|
cairo_add_stop_rgb_dim(pat, 0.0, 0.18, 0.18, 0.20, p.dim);
|
|
cairo_add_stop_rgb_dim(pat, 0.4, 0.18, 0.18, 0.20, p.dim);
|
|
cairo_add_stop_rgb_dim(pat, 1.0, 0.40, 0.40, 0.42, p.dim);
|
|
}
|
|
cairo_set_source(cr, pat);
|
|
|
|
cairo_arc(cr, p.cx, p.cy, p.radius - p.slider_thickness, 0, 2 * M_PI);
|
|
cairo_fill(cr);
|
|
|
|
// draw the circle
|
|
cairo_pattern_t *pat2;
|
|
pat2 = cairo_pattern_create_linear(
|
|
p.cx - p.radius / 2,
|
|
p.cy - p.radius / 2,
|
|
p.cx + p.radius / 2,
|
|
p.cy + p.radius / 2
|
|
);
|
|
cairo_add_stop_rgb_dim(pat2, 0, 0.9, 0.9, 0.9, p.dim);
|
|
cairo_add_stop_rgb_dim(pat2, 1, 0.3, 0.3, 0.3, p.dim);
|
|
cairo_set_source(cr, pat2);
|
|
|
|
cairo_arc(cr, p.cx, p.cy, p.radius - p.slider_thickness, 0, 2 * M_PI);
|
|
cairo_set_line_width(cr, 2);
|
|
cairo_stroke(cr);
|
|
|
|
cairo_pattern_destroy(pat);
|
|
cairo_pattern_destroy(pat2);
|
|
|
|
cairo_destroy(cr);
|
|
}
|
|
|
|
GtkWidget *gtk_dial_new(GtkAdjustment *adjustment) {
|
|
g_return_val_if_fail(
|
|
adjustment == NULL || GTK_IS_ADJUSTMENT(adjustment),
|
|
NULL
|
|
);
|
|
|
|
return g_object_new(
|
|
GTK_TYPE_DIAL,
|
|
"adjustment", adjustment,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
GtkWidget *gtk_dial_new_with_range(
|
|
double min,
|
|
double max,
|
|
double step,
|
|
double page
|
|
) {
|
|
GtkAdjustment *adj;
|
|
int digits;
|
|
|
|
g_return_val_if_fail(min < max, NULL);
|
|
|
|
adj = gtk_adjustment_new(min, min, max, step, page, 0);
|
|
|
|
if (step == 0.0) {
|
|
digits = -1;
|
|
} else if (fabs(step) >= 1.0) {
|
|
digits = 0;
|
|
} else {
|
|
digits = abs((int)floor(log10(fabs(step))));
|
|
if (digits > 5)
|
|
digits = 5;
|
|
}
|
|
|
|
return g_object_new(
|
|
GTK_TYPE_DIAL,
|
|
"adjustment", adj,
|
|
"round_digits", digits,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
static void gtk_dial_set_property(
|
|
GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec
|
|
) {
|
|
GtkDial *dial = GTK_DIAL(object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_ADJUSTMENT:
|
|
gtk_dial_set_adjustment(dial, g_value_get_object(value));
|
|
break;
|
|
case PROP_ROUND_DIGITS:
|
|
gtk_dial_set_round_digits(dial, g_value_get_int(value));
|
|
break;
|
|
case PROP_ZERO_DB:
|
|
gtk_dial_set_zero_db(dial, g_value_get_double(value));
|
|
break;
|
|
case PROP_TAPER:
|
|
gtk_dial_set_taper(dial, g_value_get_int(value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void gtk_dial_get_property(
|
|
GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec
|
|
) {
|
|
GtkDial *dial = GTK_DIAL(object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_ADJUSTMENT:
|
|
g_value_set_object(value, dial->adj);
|
|
break;
|
|
case PROP_ROUND_DIGITS:
|
|
g_value_set_int(value, dial->round_digits);
|
|
break;
|
|
case PROP_ZERO_DB:
|
|
g_value_set_double(value, dial->zero_db);
|
|
break;
|
|
case PROP_TAPER:
|
|
g_value_set_int(value, dial->taper);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
double gtk_dial_get_value(GtkDial *dial) {
|
|
return gtk_adjustment_get_value(dial->adj);
|
|
}
|
|
|
|
void gtk_dial_set_value(GtkDial *dial, double value) {
|
|
set_value(dial, value);
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
}
|
|
|
|
void gtk_dial_set_round_digits(GtkDial *dial, int round_digits) {
|
|
dial->round_digits = round_digits;
|
|
gtk_dial_set_value(dial, gtk_dial_get_value(dial));
|
|
}
|
|
|
|
int gtk_dial_get_round_digits(GtkDial *dial) {
|
|
return dial->round_digits;
|
|
}
|
|
|
|
void gtk_dial_set_zero_db(GtkDial *dial, double zero_db) {
|
|
dial->zero_db = zero_db;
|
|
}
|
|
|
|
double gtk_dial_get_zero_db(GtkDial *dial) {
|
|
return dial->zero_db;
|
|
}
|
|
|
|
void gtk_dial_set_taper(GtkDial *dial, int taper) {
|
|
dial->taper = taper;
|
|
}
|
|
|
|
int gtk_dial_get_taper(GtkDial *dial) {
|
|
return dial->taper;
|
|
}
|
|
|
|
void gtk_dial_set_taper_linear_breakpoints(
|
|
GtkDial *dial,
|
|
const double *breakpoints,
|
|
const double *outputs,
|
|
int count
|
|
) {
|
|
free(dial->taper_breakpoints);
|
|
free(dial->taper_outputs);
|
|
dial->taper_breakpoints = NULL;
|
|
dial->taper_outputs = NULL;
|
|
dial->taper_breakpoints_count = 0;
|
|
|
|
if (count < 1)
|
|
return;
|
|
|
|
int total_count = count + 2;
|
|
|
|
dial->taper_breakpoints = malloc(total_count * sizeof(double));
|
|
dial->taper_outputs = malloc(total_count * sizeof(double));
|
|
|
|
dial->taper_breakpoints[0] = 0;
|
|
dial->taper_outputs[0] = 0;
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
dial->taper_breakpoints[i + 1] = breakpoints[i];
|
|
dial->taper_outputs[i + 1] = outputs[i];
|
|
}
|
|
|
|
dial->taper_breakpoints[total_count - 1] = 1;
|
|
dial->taper_outputs[total_count - 1] = 1;
|
|
|
|
dial->taper_breakpoints_count = total_count;
|
|
}
|
|
|
|
void gtk_dial_set_adjustment(GtkDial *dial, GtkAdjustment *adj) {
|
|
if (!(adj == NULL || GTK_IS_ADJUSTMENT(adj)))
|
|
return;
|
|
if (dial->adj)
|
|
g_object_unref(dial->adj);
|
|
dial->adj = adj;
|
|
g_object_ref_sink(dial->adj);
|
|
g_signal_emit(dial, signals[VALUE_CHANGED], 0);
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
}
|
|
|
|
GtkAdjustment *gtk_dial_get_adjustment(GtkDial *dial) {
|
|
return dial->adj;
|
|
}
|
|
|
|
static void set_value(GtkDial *dial, double newval) {
|
|
if (dial->round_digits >= 0) {
|
|
double power;
|
|
int i;
|
|
|
|
i = dial->round_digits;
|
|
power = 1;
|
|
while (i--)
|
|
power *= 10;
|
|
|
|
newval = floor((newval * power) + 0.5) / power;
|
|
}
|
|
|
|
gtk_adjustment_set_value(dial->adj, newval);
|
|
g_signal_emit(dial, signals[VALUE_CHANGED], 0);
|
|
}
|
|
|
|
static void step_back(GtkDial *dial) {
|
|
double newval;
|
|
|
|
newval = gtk_adjustment_get_value(dial->adj) - gtk_adjustment_get_step_increment(dial->adj);
|
|
set_value(dial, newval);
|
|
}
|
|
|
|
static void step_forward(GtkDial *dial) {
|
|
double newval;
|
|
|
|
newval = gtk_adjustment_get_value(dial->adj) + gtk_adjustment_get_step_increment(dial->adj);
|
|
set_value(dial, newval);
|
|
}
|
|
|
|
static void page_back(GtkDial *dial) {
|
|
double newval;
|
|
|
|
newval = gtk_adjustment_get_value(dial->adj) - gtk_adjustment_get_page_increment(dial->adj);
|
|
set_value(dial, newval);
|
|
}
|
|
|
|
static void page_forward(GtkDial *dial) {
|
|
double newval;
|
|
|
|
newval = gtk_adjustment_get_value(dial->adj) + gtk_adjustment_get_page_increment(dial->adj);
|
|
set_value(dial, newval);
|
|
}
|
|
|
|
static void scroll_begin(GtkDial *dial) {
|
|
double newval = gtk_adjustment_get_lower(dial->adj);
|
|
|
|
set_value(dial, newval);
|
|
}
|
|
|
|
static void scroll_end(GtkDial *dial) {
|
|
double newval = gtk_adjustment_get_upper(dial->adj) - gtk_adjustment_get_page_size(dial->adj);
|
|
|
|
set_value(dial, newval);
|
|
}
|
|
|
|
static gboolean should_invert_move(GtkDial *dial, GtkOrientation o) {
|
|
return FALSE;
|
|
}
|
|
|
|
static void gtk_dial_move_slider(GtkDial *dial, GtkScrollType scroll) {
|
|
switch (scroll) {
|
|
case GTK_SCROLL_STEP_LEFT:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_HORIZONTAL))
|
|
step_forward(dial);
|
|
else
|
|
step_back(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_STEP_UP:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_VERTICAL))
|
|
step_forward(dial);
|
|
else
|
|
step_back(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_STEP_RIGHT:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_HORIZONTAL))
|
|
step_back(dial);
|
|
else
|
|
step_forward(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_STEP_DOWN:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_VERTICAL))
|
|
step_back(dial);
|
|
else
|
|
step_forward(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_STEP_BACKWARD:
|
|
step_back(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_STEP_FORWARD:
|
|
step_forward(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_PAGE_LEFT:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_HORIZONTAL))
|
|
page_forward(dial);
|
|
else
|
|
page_back(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_PAGE_UP:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_VERTICAL))
|
|
page_forward(dial);
|
|
else
|
|
page_back(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_PAGE_RIGHT:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_HORIZONTAL))
|
|
page_back(dial);
|
|
else
|
|
page_forward(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_PAGE_DOWN:
|
|
if (should_invert_move(dial, GTK_ORIENTATION_VERTICAL))
|
|
page_back(dial);
|
|
else
|
|
page_forward(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_PAGE_BACKWARD:
|
|
page_back(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_PAGE_FORWARD:
|
|
page_forward(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_START:
|
|
scroll_begin(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_END:
|
|
scroll_end(dial);
|
|
break;
|
|
|
|
case GTK_SCROLL_JUMP:
|
|
case GTK_SCROLL_NONE:
|
|
default:
|
|
break;
|
|
}
|
|
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
}
|
|
|
|
static void gtk_dial_drag_gesture_begin(
|
|
GtkGestureDrag *gesture,
|
|
double offset_x,
|
|
double offset_y,
|
|
GtkDial *dial
|
|
) {
|
|
dial->dvalp = calc_valp(
|
|
gtk_dial_get_value(dial),
|
|
gtk_adjustment_get_lower(dial->adj),
|
|
gtk_adjustment_get_upper(dial->adj)
|
|
);
|
|
gtk_gesture_set_state(dial->drag_gesture, GTK_EVENT_SEQUENCE_CLAIMED);
|
|
}
|
|
|
|
static void gtk_dial_drag_gesture_update(
|
|
GtkGestureDrag *gesture,
|
|
double offset_x,
|
|
double offset_y,
|
|
GtkDial *dial
|
|
) {
|
|
double start_x, start_y;
|
|
|
|
gtk_gesture_drag_get_start_point(gesture, &start_x, &start_y);
|
|
|
|
struct dial_properties p;
|
|
get_dial_properties(dial, &p);
|
|
|
|
double valp = dial->dvalp - DRAG_FACTOR * (offset_y / p.h);
|
|
valp = CLAMP(valp, 0.0, 1.0);
|
|
|
|
double val = calc_val(
|
|
valp,
|
|
gtk_adjustment_get_lower(dial->adj),
|
|
gtk_adjustment_get_upper(dial->adj)
|
|
);
|
|
|
|
set_value(dial, val);
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
}
|
|
|
|
static void gtk_dial_drag_gesture_end(
|
|
GtkGestureDrag *gesture,
|
|
double offset_x,
|
|
double offset_y,
|
|
GtkDial *dial
|
|
) {
|
|
dial->grab = GRAB_NONE;
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
}
|
|
|
|
static void gtk_dial_click_gesture_pressed(
|
|
GtkGestureClick *gesture,
|
|
int n_press,
|
|
double x,
|
|
double y,
|
|
GtkDial *dial
|
|
) {
|
|
|
|
// on double (or more) click, toggle between lower and zero_db value
|
|
if (n_press >= 2) {
|
|
double lower = gtk_adjustment_get_lower(dial->adj);
|
|
|
|
if (gtk_dial_get_value(dial) != lower)
|
|
set_value(dial, lower);
|
|
else
|
|
set_value(dial, dial->zero_db);
|
|
|
|
return;
|
|
}
|
|
|
|
if (gtk_widget_get_focus_on_click(GTK_WIDGET(dial)) &&
|
|
!gtk_widget_has_focus(GTK_WIDGET(dial)))
|
|
gtk_widget_grab_focus(GTK_WIDGET(dial));
|
|
|
|
struct dial_properties p;
|
|
|
|
get_dial_properties(dial, &p);
|
|
if (circle_contains_point(p.slider_cx, p.slider_cy, p.radius, x, y))
|
|
dial->grab = GRAB_SLIDER;
|
|
else
|
|
dial->grab = GRAB_NONE;
|
|
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
gtk_gesture_set_state(GTK_GESTURE(gesture), GTK_EVENT_SEQUENCE_CLAIMED);
|
|
}
|
|
|
|
static gboolean gtk_dial_scroll_controller_scroll(
|
|
GtkEventControllerScroll *scroll,
|
|
double dx,
|
|
double dy,
|
|
GtkDial *dial
|
|
) {
|
|
double delta = dx ? dx : dy;
|
|
double absolute_delta = fabs(delta);
|
|
|
|
if (absolute_delta > 1)
|
|
delta *= absolute_delta;
|
|
|
|
double step = -gtk_adjustment_get_step_increment(dial->adj) * delta;
|
|
|
|
set_value(dial, gtk_adjustment_get_value(dial->adj) + step);
|
|
gtk_widget_queue_draw(GTK_WIDGET(dial));
|
|
|
|
return GDK_EVENT_STOP;
|
|
}
|
|
|
|
void gtk_dial_dispose(GObject *o) {
|
|
GtkDial *dial = GTK_DIAL(o);
|
|
|
|
free(dial->taper_breakpoints);
|
|
dial->taper_breakpoints = NULL;
|
|
free(dial->taper_outputs);
|
|
dial->taper_outputs = NULL;
|
|
dial->taper_breakpoints_count = 0;
|
|
|
|
g_object_unref(dial->adj);
|
|
dial->adj = NULL;
|
|
G_OBJECT_CLASS(gtk_dial_parent_class)->dispose(o);
|
|
}
|