Browse Source

Fix scaling issues

master
eyenseo 7 years ago
parent
commit
19e6bb9acf
  1. 203
      unlock_indicator.c
  2. 1
      unlock_indicator.h

203
unlock_indicator.c

@ -42,7 +42,7 @@ extern bool debug_mode;
/* The current position in the input buffer. Useful to determine if any /* The current position in the input buffer. Useful to determine if any
* characters of the password have already been entered or not. */ * characters of the password have already been entered or not. */
int input_position; extern int input_position;
/* The lock window. */ /* The lock window. */
extern xcb_window_t win; extern xcb_window_t win;
@ -202,10 +202,10 @@ static cairo_font_face_t *font_faces[5] = {
* Pro 13" Retina screen, the scaling factor is 227/96 = 2.36. * Pro 13" Retina screen, the scaling factor is 227/96 = 2.36.
* *
*/ */
static double scaling_factor(void) { static double calculate_scaling_factor(void) {
const int dpi = (double)screen->height_in_pixels * 25.4 / const double dpi = (double)screen->height_in_pixels * 25.4 /
(double)screen->height_in_millimeters; (double)screen->height_in_millimeters;
return (dpi / 96.0); return dpi / 96.0;
} }
static cairo_font_face_t *get_font_face(int which) { static cairo_font_face_t *get_font_face(int which) {
@ -418,7 +418,6 @@ static void draw_bar(cairo_t *ctx, double x, double y, double bar_offset) {
static void draw_indic(cairo_t *ctx, double ind_x, double ind_y) { static void draw_indic(cairo_t *ctx, double ind_x, double ind_y) {
if (unlock_indicator && if (unlock_indicator &&
(unlock_state >= STATE_KEY_PRESSED || auth_state > STATE_AUTH_IDLE || show_indicator)) { (unlock_state >= STATE_KEY_PRESSED || auth_state > STATE_AUTH_IDLE || show_indicator)) {
cairo_scale(ctx, scaling_factor(), scaling_factor());
/* Draw a (centered) circle with transparent background. */ /* Draw a (centered) circle with transparent background. */
cairo_set_line_width(ctx, RING_WIDTH); cairo_set_line_width(ctx, RING_WIDTH);
cairo_arc(ctx, ind_x, ind_y, BUTTON_RADIUS, 0, 2 * M_PI); cairo_arc(ctx, ind_x, ind_y, BUTTON_RADIUS, 0, 2 * M_PI);
@ -596,16 +595,24 @@ void init_colors_once(void) {
colorgen_rgb(&tmp_rgb, color, &rgb16); colorgen_rgb(&tmp_rgb, color, &rgb16);
} }
te_expr *compile_expression(const char *const from, const char *expression, const te_variable *variables, int var_count) {
int te_err = 0;
te_expr *expr = te_compile(expression, variables, var_count, &te_err);
if (te_err) {
fprintf(stderr, "Failed to reason about '%s' given by '%s'\n", expression, from);
exit(1);
}
return expr;
}
/* /*
* Draws global image with fill color onto a pixmap with the given * Draws global image with fill color onto a pixmap with the given
* resolution and returns it. * resolution and returns it.
* *
*/ */
xcb_pixmap_t draw_image(uint32_t *resolution) { xcb_pixmap_t draw_image(uint32_t *resolution) {
const double scaling_factor = calculate_scaling_factor();
xcb_pixmap_t bg_pixmap = XCB_NONE; xcb_pixmap_t bg_pixmap = XCB_NONE;
int button_diameter_physical = ceil(scaling_factor() * BUTTON_DIAMETER);
DEBUG("scaling_factor is %.f, physical diameter is %d px\n",
scaling_factor(), button_diameter_physical);
if (!vistype) if (!vistype)
vistype = get_root_visual_type(screen); vistype = get_root_visual_type(screen);
@ -617,6 +624,7 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
*/ */
cairo_surface_t *output = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, resolution[0], resolution[1]); cairo_surface_t *output = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, resolution[0], resolution[1]);
cairo_t *ctx = cairo_create(output); cairo_t *ctx = cairo_create(output);
cairo_scale(ctx, scaling_factor, scaling_factor);
// cairo_set_font_face(ctx, get_font_face(0)); // cairo_set_font_face(ctx, get_font_face(0));
@ -784,87 +792,74 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
} }
// initialize positioning vars // initialize positioning vars
double ix, iy; double screen_x = 0, screen_y = 0,
double x, y; width = 0, height = 0;
double screen_x, screen_y;
double w, h; double mod_x = 0, mod_y = 0,
double tx = 0; date_x = 0, date_y = 0,
double ty = 0; bar_x = 0, bar_y = 0,
double dx = 0; layout_x = 0, layout_y = 0,
double dy = 0; status_x = 0, status_y = 0,
time_x = 0, time_y = 0,
double radius = BUTTON_RADIUS; indicator_x = 0, indicator_y = 0;
double radius = (circle_radius + ring_width);
double bar_offset = 0; double bar_offset = 0;
int button_diameter_physical = ceil(scaling_factor * BUTTON_DIAMETER);
DEBUG("scaling_factor is %.f, physical diameter is %d px\n",
scaling_factor, button_diameter_physical);
int te_x_err;
int te_y_err;
// variable mapping for evaluating the clock position expression // variable mapping for evaluating the clock position expression
const unsigned int vars_size = 11;
te_variable vars[] = te_variable vars[] =
{{"w", &w}, {{"w", &width},
{"h", &h}, {"h", &height},
{"x", &screen_x}, {"x", &screen_x},
{"y", &screen_y}, {"y", &screen_y},
{"ix", &ix}, {"ix", &indicator_x},
{"iy", &iy}, {"iy", &indicator_y},
{"tx", &tx}, {"tx", &time_x},
{"ty", &ty}, {"ty", &time_y},
{"dx", &dx}, {"dx", &date_x},
{"dy", &dy}, {"dy", &date_y},
{"r", &radius}}; {"r", &radius}};
#define NUM_VARS 11
te_expr *te_ind_x_expr = compile_expression("--indpos", ind_x_expr, vars, vars_size);
te_expr *te_ind_x_expr = te_compile(ind_x_expr, vars, NUM_VARS, &te_x_err); te_expr *te_ind_y_expr = compile_expression("--indpos", ind_y_expr, vars, vars_size);
te_expr *te_ind_y_expr = te_compile(ind_y_expr, vars, NUM_VARS, &te_y_err); te_expr *te_time_x_expr = compile_expression("--timepos", time_x_expr, vars, vars_size);
te_expr *te_time_x_expr = te_compile(time_x_expr, vars, NUM_VARS, &te_x_err); te_expr *te_time_y_expr = compile_expression("--timepos", time_y_expr, vars, vars_size);
te_expr *te_time_y_expr = te_compile(time_y_expr, vars, NUM_VARS, &te_y_err); te_expr *te_date_x_expr = compile_expression("--datepos", date_x_expr, vars, vars_size);
te_expr *te_date_x_expr = te_compile(date_x_expr, vars, NUM_VARS, &te_x_err); te_expr *te_date_y_expr = compile_expression("--datepos", date_y_expr, vars, vars_size);
te_expr *te_date_y_expr = te_compile(date_y_expr, vars, NUM_VARS, &te_y_err); te_expr *te_layout_x_expr = compile_expression("--layoutpos", layout_x_expr, vars, vars_size);
te_expr *te_layout_x_expr = te_compile(layout_x_expr, vars, NUM_VARS, &te_x_err); te_expr *te_layout_y_expr = compile_expression("--layoutpos", layout_y_expr, vars, vars_size);
te_expr *te_layout_y_expr = te_compile(layout_y_expr, vars, NUM_VARS, &te_y_err); te_expr *te_status_x_expr = compile_expression("--statuspos", status_x_expr, vars, vars_size);
te_expr *te_status_x_expr = te_compile(status_x_expr, vars, NUM_VARS, &te_x_err); te_expr *te_status_y_expr = compile_expression("--statuspos", status_y_expr, vars, vars_size);
te_expr *te_status_y_expr = te_compile(status_y_expr, vars, NUM_VARS, &te_y_err); te_expr *te_modif_x_expr = compile_expression("--modifpos", modif_x_expr, vars, vars_size);
te_expr *te_modif_x_expr = te_compile(modif_x_expr, vars, NUM_VARS, &te_x_err); te_expr *te_modif_y_expr = compile_expression("--modifpos", modif_y_expr, vars, vars_size);
te_expr *te_modif_y_expr = te_compile(modif_y_expr, vars, NUM_VARS, &te_y_err); te_expr *te_bar_expr = compile_expression("--bar-position", bar_expr, vars, vars_size);
te_expr *te_bar_expr = te_compile(bar_expr, vars, NUM_VARS, &te_x_err);
double time_x = 0, time_y = 0,
date_x = 0, date_y = 0,
mod_x = 0, mod_y = 0,
layout_x = 0, layout_y = 0,
status_x = 0, status_y = 0;
if (screen_number < 0 || !(screen_number < xr_screens)) { if (screen_number < 0 || !(screen_number < xr_screens)) {
screen_number = 0; screen_number = 0;
} }
if (xr_screens > 0 && !bar_enabled) { if (xr_screens > 0 && !bar_enabled) {
w = xr_resolutions[screen_number].width; width = xr_resolutions[screen_number].width / scaling_factor;
h = xr_resolutions[screen_number].height; height = xr_resolutions[screen_number].height / scaling_factor;
screen_x = xr_resolutions[screen_number].x; screen_x = xr_resolutions[screen_number].x / scaling_factor;
screen_y = xr_resolutions[screen_number].y; screen_y = xr_resolutions[screen_number].y / scaling_factor;
if (te_ind_x_expr && te_ind_y_expr) { if (te_ind_x_expr && te_ind_y_expr) {
ix = 0; indicator_x = te_eval(te_ind_x_expr);
iy = 0; indicator_y = te_eval(te_ind_y_expr);
ix = te_eval(te_ind_x_expr);
iy = te_eval(te_ind_y_expr);
} else { } else {
ix = xr_resolutions[screen_number].x + (xr_resolutions[screen_number].width / 2); indicator_x = screen_x + (width / 2);
iy = xr_resolutions[screen_number].y + (xr_resolutions[screen_number].height / 2); indicator_y = screen_y + (height / 2);
} }
bar_x = indicator_x - (button_diameter_physical / 2);
x = ix - (button_diameter_physical / 2); bar_y = indicator_y - (button_diameter_physical / 2);
y = iy - (button_diameter_physical / 2); time_x = te_eval(te_time_x_expr);
time_y = te_eval(te_time_y_expr);
tx = 0; date_x = te_eval(te_date_x_expr);
ty = 0; date_y = te_eval(te_date_y_expr);
tx = te_eval(te_time_x_expr);
ty = te_eval(te_time_y_expr);
time_x = tx;
time_y = ty;
dx = te_eval(te_date_x_expr);
dy = te_eval(te_date_y_expr);
date_x = dx;
date_y = dy;
layout_x = te_eval(te_layout_x_expr); layout_x = te_eval(te_layout_x_expr);
layout_y = te_eval(te_layout_y_expr); layout_y = te_eval(te_layout_y_expr);
status_x = te_eval(te_status_x_expr); status_x = te_eval(te_status_x_expr);
@ -875,21 +870,17 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
/* We have no information about the screen sizes/positions, so we just /* We have no information about the screen sizes/positions, so we just
* place the unlock indicator in the middle of the X root window and * place the unlock indicator in the middle of the X root window and
* hope for the best. */ * hope for the best. */
w = last_resolution[0]; width = last_resolution[0] / scaling_factor;
h = last_resolution[1]; height = last_resolution[1] / scaling_factor;
ix = last_resolution[0] / 2; indicator_x = width / 2;
iy = last_resolution[1] / 2; indicator_y = height / 2;
x = ix - (button_diameter_physical / 2); bar_x = indicator_x - (button_diameter_physical / 2);
y = iy - (button_diameter_physical / 2); bar_y = indicator_y - (button_diameter_physical / 2);
if (te_time_x_expr && te_time_y_expr) { if (te_time_x_expr && te_time_y_expr) {
tx = te_eval(te_time_x_expr); time_x = te_eval(te_time_x_expr);
ty = te_eval(te_time_y_expr); time_y = te_eval(te_time_y_expr);
time_x = tx; date_x = te_eval(te_date_x_expr);
time_y = ty; date_y = te_eval(te_date_y_expr);
dx = te_eval(te_date_x_expr);
dy = te_eval(te_date_y_expr);
date_x = dx;
date_y = dy;
layout_x = te_eval(te_layout_x_expr); layout_x = te_eval(te_layout_x_expr);
layout_y = te_eval(te_layout_y_expr); layout_y = te_eval(te_layout_y_expr);
status_x = te_eval(te_status_x_expr); status_x = te_eval(te_status_x_expr);
@ -898,22 +889,18 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
mod_y = te_eval(te_modif_y_expr); mod_y = te_eval(te_modif_y_expr);
} }
} else { } else {
w = xr_resolutions[screen_number].width; width = xr_resolutions[screen_number].width / scaling_factor;
h = xr_resolutions[screen_number].height; height = xr_resolutions[screen_number].height / scaling_factor;
ix = w / 2; indicator_x = width / 2;
iy = h / 2; indicator_y = height / 2;
bar_x = indicator_x - (button_diameter_physical / 2);
bar_y = indicator_y - (button_diameter_physical / 2);
bar_offset = te_eval(te_bar_expr); bar_offset = te_eval(te_bar_expr);
tx = 0; time_x = te_eval(te_time_x_expr);
ty = 0; time_y = te_eval(te_time_y_expr);
tx = te_eval(te_time_x_expr); date_x = te_eval(te_date_x_expr);
ty = te_eval(te_time_y_expr); date_y = te_eval(te_date_y_expr);
time_x = tx;
time_y = ty;
dx = te_eval(te_date_x_expr);
dy = te_eval(te_date_y_expr);
date_x = dx;
date_y = dy;
layout_x = te_eval(te_layout_x_expr); layout_x = te_eval(te_layout_x_expr);
layout_y = te_eval(te_layout_y_expr); layout_y = te_eval(te_layout_y_expr);
status_x = te_eval(te_status_x_expr); status_x = te_eval(te_status_x_expr);
@ -936,6 +923,14 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
te_free(te_modif_y_expr); te_free(te_modif_y_expr);
te_free(te_bar_expr); te_free(te_bar_expr);
DEBUG("Indicator at %fx%f\n", indicator_x, indicator_y);
DEBUG("Bar at %fx%f\n", bar_x, bar_y);
DEBUG("Time at %fx%f\n", time_x, time_y);
DEBUG("Date at %fx%f\n", date_x, date_y);
DEBUG("Layout at %fx%f\n", layout_x, layout_y);
DEBUG("Status at %fx%f\n", status_x, status_y);
DEBUG("Mod at %fx%f\n", mod_x, mod_y);
status_text.x = status_x; status_text.x = status_x;
status_text.y = status_y; status_text.y = status_y;
@ -952,7 +947,7 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
// indicator stuff // indicator stuff
if (!bar_enabled) { if (!bar_enabled) {
draw_indic(ctx, ix, iy); draw_indic(ctx, indicator_x, indicator_y);
} else { } else {
if (unlock_state == STATE_KEY_ACTIVE || if (unlock_state == STATE_KEY_ACTIVE ||
unlock_state == STATE_BACKSPACE_ACTIVE) { unlock_state == STATE_BACKSPACE_ACTIVE) {
@ -977,7 +972,7 @@ xcb_pixmap_t draw_image(uint32_t *resolution) {
break; break;
} }
} }
draw_bar(ctx, x, y, bar_offset); draw_bar(ctx, bar_x, bar_y, bar_offset);
} }
draw_text(ctx, status_text); draw_text(ctx, status_text);

1
unlock_indicator.h

@ -2,6 +2,7 @@
#define _UNLOCK_INDICATOR_H #define _UNLOCK_INDICATOR_H
#include <ev.h> #include <ev.h>
#include <xcb/xcb.h>
typedef struct rgb { typedef struct rgb {
double red; double red;

Loading…
Cancel
Save