2016-08-18 03:07:43 -03:00
|
|
|
/*
|
|
|
|
* gui/glabel.c - Global label pop-up
|
|
|
|
*
|
|
|
|
* Written 2016 by Werner Almesberger
|
|
|
|
* Copyright 2016 by Werner Almesberger
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
|
|
|
|
#include "misc/util.h"
|
|
|
|
#include "kicad/dwg.h"
|
|
|
|
#include "gui/style.h"
|
2016-08-18 18:14:57 -03:00
|
|
|
#include "gui/input.h"
|
2016-08-18 03:07:43 -03:00
|
|
|
#include "gui/aoi.h"
|
|
|
|
#include "gui/over.h"
|
|
|
|
#include "gui/common.h"
|
|
|
|
|
|
|
|
|
|
|
|
/* small offset to hide rounding errors */
|
|
|
|
#define CHEAT 1
|
|
|
|
|
|
|
|
|
|
|
|
struct glabel_aoi_ctx {
|
|
|
|
const struct gui_sheet *sheet;
|
|
|
|
const struct sch_obj *obj;
|
|
|
|
struct dwg_bbox bbox;
|
|
|
|
struct overlay *over;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2016-08-18 17:55:54 -03:00
|
|
|
#define GLABEL_W 100
|
|
|
|
|
|
|
|
|
2016-08-18 04:01:45 -03:00
|
|
|
/* ----- Tools ------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
|
|
|
static void eeschema_coord(const struct gui_ctx *ctx,
|
|
|
|
int x, int y, int *rx, int *ry)
|
|
|
|
{
|
|
|
|
GtkAllocation alloc;
|
|
|
|
|
|
|
|
gtk_widget_get_allocation(ctx->da, &alloc);
|
2016-08-22 06:41:03 -03:00
|
|
|
*rx = ((x - ctx->x) * ctx->scale) + alloc.width / 2;
|
|
|
|
*ry = ((y - ctx->y) * ctx->scale) + alloc.height / 2;
|
2016-08-18 04:01:45 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ----- AoIs -------------------------------------------------------------- */
|
|
|
|
|
|
|
|
|
2016-08-18 03:07:43 -03:00
|
|
|
static void glabel_dest_click(void *user)
|
|
|
|
{
|
|
|
|
struct gui_sheet *sheet = user;
|
|
|
|
|
|
|
|
go_to_sheet(sheet->ctx, sheet);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void dehover_glabel(struct gui_ctx *ctx)
|
|
|
|
{
|
|
|
|
overlay_remove_all(&ctx->pop_overlays);
|
2016-08-18 17:13:36 -03:00
|
|
|
overlay_remove_all(&ctx->pop_underlays);
|
2016-08-19 05:50:26 -03:00
|
|
|
ctx->pop_origin = NULL;
|
2016-08-18 03:07:43 -03:00
|
|
|
redraw(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-18 17:55:54 -03:00
|
|
|
static void add_dest_header(struct gui_ctx *ctx, const char *label)
|
|
|
|
{
|
|
|
|
struct overlay_style style = {
|
|
|
|
.font = BOLD_FONT,
|
|
|
|
.wmin = GLABEL_W,
|
|
|
|
.wmax = GLABEL_W,
|
|
|
|
.radius = 0,
|
2016-08-18 18:17:44 -03:00
|
|
|
.pad = 0,
|
|
|
|
.skip = 6,
|
|
|
|
.fg = { 0.5, 0.0, 0.0, 1.0 },
|
|
|
|
.bg = { 0.0, 0.0, 0.0, 0.0 },
|
2016-08-18 17:55:54 -03:00
|
|
|
.frame = { 1.0, 1.0, 1.0, 1.0 }, /* debugging */
|
|
|
|
.width = 0,
|
|
|
|
};
|
|
|
|
struct overlay *over;
|
|
|
|
|
|
|
|
over = overlay_add(&ctx->pop_overlays, NULL, NULL, NULL, NULL);
|
|
|
|
overlay_text(over, "%s", label);
|
|
|
|
overlay_style(over, &style);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-18 14:32:46 -03:00
|
|
|
static void add_dest_overlay(struct gui_ctx *ctx, const char *label,
|
|
|
|
struct gui_sheet *sheet, unsigned n)
|
2016-08-18 03:07:43 -03:00
|
|
|
{
|
|
|
|
struct overlay_style style = {
|
|
|
|
.font = BOLD_FONT,
|
2016-08-18 17:55:54 -03:00
|
|
|
.wmin = GLABEL_W,
|
|
|
|
.wmax = GLABEL_W,
|
2016-08-18 03:07:43 -03:00
|
|
|
.radius = 0,
|
2016-08-18 18:17:44 -03:00
|
|
|
.pad = 0,
|
|
|
|
.skip = 4,
|
2016-08-18 03:07:43 -03:00
|
|
|
.fg = { 0.0, 0.0, 0.0, 1.0 },
|
2016-08-18 17:13:36 -03:00
|
|
|
.bg = { 0.0, 0.0, 0.0, 0.0 },
|
2016-08-18 03:07:43 -03:00
|
|
|
.frame = { 1.0, 1.0, 1.0, 1.0 }, /* debugging */
|
|
|
|
.width = 0,
|
|
|
|
};
|
2016-08-18 14:32:46 -03:00
|
|
|
const struct sch_obj *obj;
|
|
|
|
struct overlay *over;
|
|
|
|
|
|
|
|
if (sheet == ctx->curr_sheet)
|
2016-08-18 17:27:20 -03:00
|
|
|
style.fg = RGBA(0.5, 0.5, 0.5, 1.0);
|
|
|
|
|
2016-08-18 14:32:46 -03:00
|
|
|
for (obj = sheet->sch->objs; obj; obj = obj->next) {
|
|
|
|
if (obj->type != sch_obj_glabel)
|
|
|
|
continue;
|
|
|
|
if (strcmp(obj->u.text.s, label))
|
|
|
|
continue;
|
|
|
|
over = overlay_add(&ctx->pop_overlays,
|
|
|
|
&ctx->aois, NULL, glabel_dest_click, sheet);
|
2016-08-19 05:09:00 -03:00
|
|
|
overlay_text(over, "%d %s", n,
|
|
|
|
sheet->sch->title ? sheet->sch->title : "(unnamed)");
|
2016-08-18 14:32:46 -03:00
|
|
|
overlay_style(over, &style);
|
2016-08-18 17:13:36 -03:00
|
|
|
break;
|
2016-08-18 14:32:46 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-21 22:31:52 -03:00
|
|
|
static bool pop_hover(void *user, bool on, int dx, int dy)
|
2016-08-18 18:14:57 -03:00
|
|
|
{
|
|
|
|
struct gui_ctx *ctx = user;
|
|
|
|
|
|
|
|
if (!on)
|
|
|
|
dehover_glabel(ctx);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-18 17:13:36 -03:00
|
|
|
static void add_dest_frame(struct gui_ctx *ctx)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
|
|
|
|
overlay_size_all(ctx->pop_overlays,
|
|
|
|
gtk_widget_get_pango_context(ctx->da), 0, 1, &w, &h);
|
|
|
|
|
|
|
|
struct overlay_style style = {
|
|
|
|
.font = BOLD_FONT,
|
|
|
|
.wmin = w,
|
|
|
|
.hmin = h,
|
|
|
|
.radius = 0,
|
|
|
|
.pad = GLABEL_STACK_PADDING,
|
|
|
|
.skip = 0,
|
|
|
|
.fg = { 0.0, 0.0, 0.0, 1.0 },
|
|
|
|
.bg = { 0.9, 0.9, 0.3, 0.8 },
|
|
|
|
.frame = { 0.0, 0.0, 0.0, 1.0 }, /* debugging */
|
|
|
|
.width = 1,
|
|
|
|
};
|
|
|
|
struct overlay *over;
|
|
|
|
|
2016-08-18 18:14:57 -03:00
|
|
|
over = overlay_add(&ctx->pop_underlays, &ctx->aois,
|
|
|
|
pop_hover, NULL, ctx);
|
2016-08-18 17:13:36 -03:00
|
|
|
overlay_text_raw(over, "");
|
|
|
|
overlay_style(over, &style);
|
2016-08-19 03:33:26 -03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This makes it all work. When we receive a click while hovering, it
|
|
|
|
* goes to the hovering overlay if that overlay accepts clicks.
|
|
|
|
* However, if the overlay accepting the click is different, we first
|
|
|
|
* de-hover.
|
|
|
|
*
|
|
|
|
* Now, in the case of the frame overlay, dehovering would destroy the
|
|
|
|
* destination overlays right before trying to deliver the click.
|
|
|
|
*
|
|
|
|
* We solve this by declaring the frame overlay to be "related" to the
|
|
|
|
* destination overlays. This suppresses dehovering.
|
|
|
|
*/
|
|
|
|
overlay_set_related_all(ctx->pop_overlays, over);
|
2016-08-18 17:13:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-21 22:31:52 -03:00
|
|
|
static bool hover_glabel(void *user, bool on, int dx, int dy)
|
2016-08-18 14:32:46 -03:00
|
|
|
{
|
|
|
|
struct glabel_aoi_ctx *aoi_ctx = user;
|
|
|
|
struct gui_ctx *ctx = aoi_ctx->sheet->ctx;
|
|
|
|
const struct gui_sheet *curr_sheet = ctx->curr_sheet;
|
|
|
|
const struct dwg_bbox *bbox = &aoi_ctx->bbox;
|
|
|
|
|
|
|
|
if (!on) {
|
|
|
|
dehover_glabel(ctx);
|
|
|
|
return 1;
|
|
|
|
}
|
2016-08-19 05:50:26 -03:00
|
|
|
if (ctx->pop_underlays) {
|
|
|
|
if (ctx->pop_origin == aoi_ctx)
|
|
|
|
return 0;
|
|
|
|
dehover_glabel(ctx);
|
|
|
|
}
|
|
|
|
|
2016-08-18 14:32:46 -03:00
|
|
|
GtkAllocation alloc;
|
2016-08-18 03:07:43 -03:00
|
|
|
int sx, sy, ex, ey, mx, my;
|
|
|
|
unsigned n = 0;
|
|
|
|
struct gui_sheet *sheet;
|
|
|
|
|
2016-08-19 06:48:44 -03:00
|
|
|
ctx->glabel = aoi_ctx->obj->u.text.s;
|
2016-08-19 05:50:26 -03:00
|
|
|
ctx->pop_origin = aoi_ctx;
|
|
|
|
|
2016-08-18 03:07:43 -03:00
|
|
|
aoi_dehover();
|
|
|
|
overlay_remove_all(&ctx->pop_overlays);
|
2016-08-18 17:13:36 -03:00
|
|
|
overlay_remove_all(&ctx->pop_underlays);
|
2016-08-18 17:55:54 -03:00
|
|
|
|
|
|
|
add_dest_header(ctx, aoi_ctx->obj->u.text.s);
|
2016-08-18 14:32:46 -03:00
|
|
|
for (sheet = ctx->new_hist->sheets; sheet; sheet = sheet->next)
|
|
|
|
add_dest_overlay(ctx, aoi_ctx->obj->u.text.s, sheet, ++n);
|
2016-08-18 17:13:36 -03:00
|
|
|
add_dest_frame(ctx);
|
2016-08-18 03:07:43 -03:00
|
|
|
|
|
|
|
eeschema_coord(ctx,
|
|
|
|
bbox->x - curr_sheet->xmin, bbox->y - curr_sheet->ymin,
|
|
|
|
&sx, &sy);
|
|
|
|
eeschema_coord(ctx, bbox->x + bbox->w - curr_sheet->xmin,
|
|
|
|
bbox->y + bbox->h - curr_sheet->ymin, &ex, &ey);
|
|
|
|
|
|
|
|
gtk_widget_get_allocation(ctx->da, &alloc);
|
|
|
|
mx = (sx + ex) / 2;
|
|
|
|
my = (sy + ey) / 2;
|
2016-08-19 05:22:42 -03:00
|
|
|
if (mx < alloc.width / 2) {
|
|
|
|
ctx->pop_x = sx - CHEAT;
|
|
|
|
ctx->pop_dx = 1;
|
|
|
|
} else {
|
|
|
|
ctx->pop_x = ex + CHEAT;
|
|
|
|
ctx->pop_dx = -1;
|
|
|
|
}
|
|
|
|
if (my < alloc.height / 2) {
|
|
|
|
ctx->pop_y = sy - CHEAT;
|
|
|
|
ctx->pop_dy = 1;
|
|
|
|
} else {
|
|
|
|
ctx->pop_y = ey + CHEAT;
|
|
|
|
ctx->pop_dy = -1;
|
|
|
|
}
|
2016-08-18 03:07:43 -03:00
|
|
|
|
2016-08-19 06:05:42 -03:00
|
|
|
/*
|
|
|
|
* @@@ The idea is to get input to trigger hovering over the pop-up.
|
|
|
|
* However, this doesn't work because the overlay has not been drawn
|
|
|
|
* yet and therefore has not created its AoI. We therefore only get a
|
|
|
|
* chance to begin hovering at the next motion update, which may
|
|
|
|
* already be outside the pop-up.
|
|
|
|
*
|
|
|
|
* Probably the only way to fix this is by making overlay_add do the
|
|
|
|
* layout calculation and create the AoI immediately.
|
|
|
|
*
|
|
|
|
* Another problem occurs as deep zoom levels, when the label is larger
|
|
|
|
* than the pop-up. Then we can trigger pop-up creation from a location
|
|
|
|
* that will be outside the pop-up.
|
|
|
|
*
|
|
|
|
* We could fix this by aligning the pop-up with the mouse position
|
|
|
|
* instead the box, either in general, or in this specific case. Not
|
|
|
|
* sure if it's worth the trouble, though.
|
|
|
|
*
|
2016-08-20 18:29:54 -03:00
|
|
|
* Another way to avoid the problem would be to size the pop-up such
|
|
|
|
* that it always includes the mouse position. But that could lead to
|
|
|
|
* rather weird-looking results at deep high zoom levels.
|
|
|
|
*
|
|
|
|
* Yet another option would be to move the mouse pointer onto the
|
|
|
|
* pop-up. The problem with this is that forced mouse pointer movement
|
|
|
|
* is not appreciated by all users.
|
|
|
|
*
|
2016-08-19 06:05:42 -03:00
|
|
|
* Both issues result in a "hanging" pop-up because AoI (and input)
|
|
|
|
* don't even know we're hovering. The pop-up can be cleared by
|
|
|
|
* - hovering into it,
|
2016-08-20 18:29:54 -03:00
|
|
|
* - hovering over some other glabel,
|
|
|
|
* - clicking, or
|
|
|
|
* - pressing Escape.
|
2016-08-19 06:05:42 -03:00
|
|
|
*/
|
2016-08-18 18:14:57 -03:00
|
|
|
input_update();
|
2016-08-18 03:07:43 -03:00
|
|
|
redraw(ctx);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void add_glabel_aoi(struct gui_sheet *sheet, const struct sch_obj *obj)
|
|
|
|
{
|
|
|
|
const struct dwg_bbox *bbox = &obj->u.text.bbox;
|
|
|
|
struct glabel_aoi_ctx *aoi_ctx = alloc_type(struct glabel_aoi_ctx);
|
|
|
|
|
|
|
|
struct aoi cfg = {
|
|
|
|
.x = bbox->x,
|
|
|
|
.y = bbox->y,
|
|
|
|
.w = bbox->w,
|
|
|
|
.h = bbox->h,
|
|
|
|
.hover = hover_glabel,
|
|
|
|
.user = aoi_ctx,
|
|
|
|
};
|
|
|
|
|
|
|
|
aoi_ctx->sheet = sheet;
|
|
|
|
aoi_ctx->obj = obj;
|
|
|
|
aoi_ctx->bbox = *bbox;
|
|
|
|
|
|
|
|
aoi_add(&sheet->aois, &cfg);
|
|
|
|
}
|