123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242 |
- /*
- * gui_tool.c - GUI, tool bar
- *
- * Written 2009-2012 by Werner Almesberger
- * Copyright 2009-2012 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 <stdlib.h>
- #include <math.h>
- #include <assert.h>
- #include <gtk/gtk.h>
- #include "util.h"
- #include "inst.h"
- #include "meas.h"
- #include "obj.h"
- #include "gui_util.h"
- #include "gui_style.h"
- #include "gui_inst.h"
- #include "gui_over.h"
- #include "gui_canvas.h"
- #include "gui_status.h"
- #include "gui.h"
- #include "gui_meas.h"
- #include "gui_tool.h"
- #include "icons/arc.xpm"
- #include "icons/circ.xpm"
- #include "icons/frame.xpm"
- #include "icons/line.xpm"
- #include "icons/meas.xpm"
- #include "icons/meas_x.xpm"
- #include "icons/meas_y.xpm"
- #include "icons/pad.xpm"
- #include "icons/rpad.xpm"
- #include "icons/hole.xpm"
- #include "icons/point.xpm"
- #include "icons/delete.xpm"
- #include "icons/delete_off.xpm"
- #include "icons/rect.xpm"
- #include "icons/vec.xpm"
- static GtkWidget *ev_point, *ev_delete;
- static GtkWidget *active_tool;
- static struct tool_ops *active_ops = NULL;
- static struct inst *hover_inst = NULL;
- static GtkWidget *delete_image[2];
- static struct drag_state {
- struct inst *inst; /* non-NULL if dragging an existing object */
- struct inst *new; /* non-NULL if dragging a new object */
- int anchors_n; /* number of anchors, 0 if no moving */
- int anchor_i; /* current anchor */
- struct vec **anchors[3];
- } drag = {
- .new = NULL,
- .anchors_n = 0,
- };
- static struct coord last_canvas_pos;
- static struct vec *new_vec(struct inst *base)
- {
- struct vec *vec, **walk;
- vec = alloc_type(struct vec);
- vec->nul_tag = 0;
- vec->name = NULL;
- vec->base = inst_get_vec(base);
- vec->next = NULL;
- vec->frame = active_frame;
- for (walk = &active_frame->vecs; *walk; walk = &(*walk)->next);
- *walk = vec;
- return vec;
- }
- struct obj *new_obj_unconnected(enum obj_type type, struct inst *base)
- {
- struct obj *obj;
- obj = alloc_type(struct obj);
- obj->type = type;
- obj->name = NULL;
- obj->frame = active_frame;
- obj->base = inst_get_vec(base);
- obj->next = NULL;
- obj->lineno = 0;
- return obj;
- }
- void connect_obj(struct frame *frame, struct obj *obj)
- {
- struct obj **walk;
- obj->frame = frame;
- for (walk = &frame->objs; *walk; walk = &(*walk)->next);
- *walk = obj;
- }
- static struct obj *new_obj(enum obj_type type, struct inst *base)
- {
- struct obj *obj;
- obj = new_obj_unconnected(type, base);
- connect_obj(active_frame, obj);
- return obj;
- }
- /* ----- shared functions -------------------------------------------------- */
- struct pix_buf *draw_move_line_common(struct inst *inst,
- struct coord end, struct coord pos, int i)
- {
- struct coord from, to;
- struct pix_buf *buf;
- from = translate(inst->base);
- to = translate(end);
- pos = translate(pos);
- switch (i) {
- case 0:
- from = pos;
- break;
- case 1:
- to = pos;
- break;
- default:
- abort();
- }
- buf = save_pix_buf(DA, from.x, from.y, to.x, to.y, 1);
- gdk_draw_line(DA, gc_drag, from.x, from.y, to.x, to.y);
- return buf;
- }
- static struct pix_buf *draw_move_rect_common(struct inst *inst,
- struct coord other, struct coord pos, int i)
- {
- struct coord min, max;
- struct pix_buf *buf;
- min = translate(inst->base);
- max = translate(other);
- pos = translate(pos);
- switch (i) {
- case 0:
- min = pos;
- break;
- case 1:
- max = pos;
- break;
- default:
- abort();
- }
- sort_coord(&min, &max);
- buf = save_pix_buf(DA, min.x, min.y, max.x, max.y, 1);
- gdk_draw_rectangle(DA, gc_drag, FALSE,
- min.x, min.y, max.x-min.x, max.y-min.y);
- return buf;
- }
- static struct pix_buf *hover_common(GdkGC *gc, struct coord center, unit_type r)
- {
- struct pix_buf *buf;
- center = translate(center);
- buf = save_pix_buf(DA,
- center.x-r, center.y-r, center.x+r, center.y+r, 2);
- draw_circle(DA, gc, FALSE, center.x, center.y, VEC_EYE_R);
- return buf;
- }
- /* ----- delete ------------------------------------------------------------ */
- static void tool_selected_delete(void)
- {
- if (selected_inst) {
- tool_dehover();
- inst_delete(selected_inst);
- change_world();
- }
- tool_reset();
- }
- static struct tool_ops delete_ops = {
- .tool_selected = tool_selected_delete,
- };
- void tool_selected_inst(struct inst *inst)
- {
- set_image(ev_delete, delete_image[inst != NULL]);
- }
- /* ----- vec --------------------------------------------------------------- */
- static struct coord gridify(struct coord base, struct coord pos)
- {
- struct coord new;
- unit_type unit;
- switch (curr_unit) {
- case curr_unit_mm:
- case curr_unit_auto:
- unit = mm_to_units(0.1);
- break;
- case curr_unit_mil:
- unit = mil_to_units(10);
- break;
- default:
- abort();
- }
- new.x = pos.x-((pos.x-base.x) % unit);
- new.y = pos.y-((pos.y-base.y) % unit);
- if (new.x != base.x || new.y != base.y)
- return new;
- if (fabs(pos.x-base.x) > fabs(pos.y-base.y))
- new.x += pos.x > base.x ? unit : -unit;
- else
- new.y += pos.y > base.y ? unit : -unit;
- return new;
- }
- static struct pix_buf *drag_new_vec(struct inst *from, struct coord to)
- {
- struct coord pos;
- struct pix_buf *buf;
- pos = inst_get_point(from);
- to = gridify(pos, to);
- status_set_type_x(NULL, "dX =");
- status_set_type_y(NULL, "dX =");
- /* @@@ use status_set_xy */
- switch (curr_unit) {
- case curr_unit_mm:
- case curr_unit_auto:
- status_set_x(NULL, "%lg mm", units_to_mm(to.x-pos.x));
- status_set_y(NULL, "%lg mm", units_to_mm(to.y-pos.y));
- break;
- case curr_unit_mil:
- status_set_x(NULL, "%lg mil", units_to_mil(to.x-pos.x));
- status_set_y(NULL, "%lg mil", units_to_mil(to.y-pos.y));
- break;
- default:
- abort();
- }
- pos = translate(pos);
- to = translate(to);
- buf = save_pix_buf(DA, pos.x, pos.y, to.x, to.y, 1);
- gdk_draw_line(DA, gc_drag, pos.x, pos.y, to.x, to.y);
- return buf;
- }
- struct pix_buf *draw_move_vec(struct inst *inst, struct coord pos, int i)
- {
- return draw_move_line_common(inst,
- add_vec(sub_vec(inst->u.vec.end, inst->base), pos), pos, i);
- }
- struct pix_buf *gui_hover_vec(struct inst *self)
- {
- return hover_common(gc_vec[mode_hover],
- self->u.vec.end, VEC_EYE_R);
- }
- static int end_new_raw_vec(struct inst *from, struct coord to)
- {
- struct vec *vec;
- struct coord pos;
- vec = new_vec(from);
- pos = inst_get_point(from);
- to = gridify(pos, to);
- switch (curr_unit) {
- case curr_unit_mm:
- case curr_unit_auto:
- vec->x = new_num(make_mm(units_to_mm(to.x-pos.x)));
- vec->y = new_num(make_mm(units_to_mm(to.y-pos.y)));
- break;
- case curr_unit_mil:
- vec->x = new_num(make_mil(units_to_mil(to.x-pos.x)));
- vec->y = new_num(make_mil(units_to_mil(to.y-pos.y)));
- break;
- default:
- abort();
- }
- return 1;
- }
- static struct tool_ops vec_ops = {
- .drag_new = drag_new_vec,
- .end_new_raw = end_new_raw_vec,
- };
- /* ----- line -------------------------------------------------------------- */
- struct pix_buf *drag_new_line(struct inst *from, struct coord to)
- {
- struct coord pos;
- struct pix_buf *buf;
- pos = translate(inst_get_point(from));
- to = translate(to);
- buf = save_pix_buf(DA, pos.x, pos.y, to.x, to.y, 1);
- gdk_draw_line(DA, gc_drag, pos.x, pos.y, to.x, to.y);
- return buf;
- }
- struct pix_buf *draw_move_line(struct inst *inst, struct coord pos, int i)
- {
- return draw_move_line_common(inst, inst->u.rect.end, pos, i);
- }
- static int end_new_line(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (from == to)
- return 0;
- obj = new_obj(ot_line, from);
- obj->u.line.other = inst_get_vec(to);
- obj->u.line.width = NULL;
- return 1;
- }
- static struct tool_ops line_ops = {
- .drag_new = drag_new_line,
- .end_new = end_new_line,
- };
- /* ----- rect -------------------------------------------------------------- */
- static struct pix_buf *drag_new_rect(struct inst *from, struct coord to)
- {
- struct coord pos;
- struct pix_buf *buf;
- pos = translate(inst_get_point(from));
- to = translate(to);
- sort_coord(&pos, &to);
- buf = save_pix_buf(DA, pos.x, pos.y, to.x, to.y, 1);
- gdk_draw_rectangle(DA, gc_drag, FALSE,
- pos.x, pos.y, to.x-pos.x, to.y-pos.y);
- return buf;
- }
- struct pix_buf *draw_move_rect(struct inst *inst, struct coord pos, int i)
- {
- return draw_move_rect_common(inst, inst->u.rect.end, pos, i);
- }
- static int end_new_rect(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (from == to)
- return 0;
- obj = new_obj(ot_rect, from);
- obj->u.rect.other = inst_get_vec(to);
- obj->u.rect.width = NULL;
- return 1;
- }
- static struct tool_ops rect_ops = {
- .drag_new = drag_new_rect,
- .end_new = end_new_rect,
- };
- /* ----- pad --------------------------------------------------------------- */
- static int end_new_pad(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (from == to)
- return 0;
- obj = new_obj(ot_pad, from);
- obj->u.pad.other = inst_get_vec(to);
- obj->u.pad.name = stralloc("?");
- obj->u.pad.rounded = 0;
- obj->u.pad.type = pt_normal;
- return 1;
- }
- struct pix_buf *draw_move_pad(struct inst *inst, struct coord pos, int i)
- {
- return draw_move_rect_common(inst, inst->u.pad.other, pos, i);
- }
- static struct tool_ops pad_ops = {
- .drag_new = drag_new_rect,
- .end_new = end_new_pad,
- };
- /* ----- rounded pad ------------------------------------------------------- */
- static int end_new_rpad(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (from == to)
- return 0;
- obj = new_obj(ot_pad, from);
- obj->u.pad.other = inst_get_vec(to);
- obj->u.pad.name = stralloc("?");
- obj->u.pad.rounded = 1;
- obj->u.pad.type = pt_normal;
- return 1;
- }
- struct pix_buf *draw_move_rpad(struct inst *inst, struct coord pos, int i)
- {
- return draw_move_rect_common(inst, inst->u.pad.other, pos, i);
- }
- static struct tool_ops rpad_ops = {
- .drag_new = drag_new_rect,
- .end_new = end_new_rpad,
- };
- /* ----- hole -------------------------------------------------------------- */
- static int end_new_hole(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (from == to)
- return 0;
- obj = new_obj(ot_hole, from);
- obj->u.hole.other = inst_get_vec(to);
- return 1;
- }
- struct pix_buf *draw_move_hole(struct inst *inst, struct coord pos, int i)
- {
- return draw_move_rect_common(inst, inst->u.hole.other, pos, i);
- }
- static struct tool_ops hole_ops = {
- .drag_new = drag_new_rect,
- .end_new = end_new_hole,
- };
- /* ----- circ -------------------------------------------------------------- */
- static struct pix_buf *drag_new_circ(struct inst *from, struct coord to)
- {
- struct coord pos;
- struct pix_buf *buf;
- double r;
- pos = translate(inst_get_point(from));
- to = translate(to);
- r = hypot(to.x-pos.x, to.y-pos.y);
- buf = save_pix_buf(DA, pos.x-r, pos.y-r, pos.x+r, pos.y+r, 1);
- draw_circle(DA, gc_drag, FALSE, pos.x, pos.y, r);
- return buf;
- }
- static int end_new_circ(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (from == to)
- return 0;
- obj = new_obj(ot_arc, from);
- obj->u.arc.start = inst_get_vec(to);
- obj->u.arc.end = inst_get_vec(to);
- obj->u.arc.width = NULL;
- return 1;
- }
- struct pix_buf *draw_move_arc(struct inst *inst, struct coord pos, int i)
- {
- struct coord c, from, to, end;
- double r, r_save, a1, a2;
- struct pix_buf *buf;
- c = translate(inst->base);
- from =
- translate(rotate_r(inst->base, inst->u.arc.r, inst->u.arc.a1));
- to =
- translate(rotate_r(inst->base, inst->u.arc.r, inst->u.arc.a2));
- pos = translate(pos);
- switch (i) {
- case 0:
- c = pos;
- break;
- case 2:
- from = pos;
- break;
- case 1:
- to = pos;
- break;
- default:
- abort();
- }
- r = hypot(from.x-c.x, from.y-c.y);
- /*
- * the screen coordinate system is reversed with y growing downward,
- * so we have to negate the angles.
- */
- a1 = -theta(c, from);
- a2 = -theta(c, to);
- if (a2 < a1)
- a2 += 360.0;
- if (i != 2) {
- r_save = r;
- } else {
- r_save = hypot(to.x-c.x, to.y-c.y);
- if (r > r_save)
- r_save = r;
- }
- buf = save_pix_buf(DA,
- c.x-r_save, c.y-r_save, c.x+r_save, c.y+r_save, 1);
- draw_arc(DA, gc_drag, FALSE, c.x, c.y, r, a1, a2);
- if (i == 1) {
- end = rotate_r(c, r_save, -a2);
- gdk_draw_line(DA, gc_drag, c.x, c.y, end.x, end.y);
- }
- return buf;
- }
- void do_move_to_arc(struct inst *inst, struct inst *to, int i)
- {
- struct vec *vec = inst_get_vec(to);
- struct obj *obj = inst->obj;
- switch (i) {
- case 0:
- obj->base = vec;
- break;
- case 2:
- obj->u.arc.start = vec;
- break;
- case 1:
- obj->u.arc.end = vec;
- break;
- default:
- abort();
- }
- }
- static struct tool_ops circ_ops = {
- .drag_new = drag_new_circ,
- .end_new = end_new_circ,
- };
- /* ----- frame helper ------------------------------------------------------ */
- int is_parent_of(const struct frame *p, const struct frame *c)
- {
- const struct obj *obj;
- if (p == c)
- return 1;
- for (obj = p->objs; obj; obj = obj->next)
- if (obj->type == ot_frame)
- if (is_parent_of(obj->u.frame.ref, c))
- return 1;
- return 0;
- }
- /* ----- frame ------------------------------------------------------------- */
- static struct frame *locked_frame = NULL;
- struct pix_buf *draw_move_frame(struct inst *inst, struct coord pos, int i)
- {
- struct pix_buf *buf;
- int r = FRAME_EYE_R2;
- pos = translate(pos);
- buf = save_pix_buf(DA, pos.x-r, pos.y-r, pos.x+r, pos.y+r, 1);
- draw_arc(DA, gc_drag, FALSE, pos.x, pos.y, r, 0, 360);
- return buf;
- }
- struct pix_buf *gui_hover_frame(struct inst *self)
- {
- return hover_common(gc_frame[mode_hover],
- self->base, FRAME_EYE_R2);
- }
- static int end_new_frame(struct inst *from, struct inst *to)
- {
- struct obj *obj;
- if (!locked_frame || is_parent_of(locked_frame, active_frame))
- return 0;
- obj = new_obj(ot_frame, from);
- obj->u.frame.ref = locked_frame;
- obj->u.frame.lineno = 0;
- if (!locked_frame->active_ref)
- locked_frame->active_ref = obj;
- locked_frame = NULL;
- tool_reset();
- return 1;
- }
- static struct tool_ops frame_ops = {
- .tool_selected = NULL,
- .drag_new = NULL,
- .end_new = end_new_frame,
- };
- /* ----- moving references ------------------------------------------------- */
- #if 0
- static int may_move(struct inst *curr)
- {
- if (!selected_inst)
- return 0;
- if (drag.anchors_n)
- return 0; /* already moving something else */
- drag.anchors_n = inst_anchors(selected_inst, drag.anchors);
- for (drag.anchor_i = 0; drag.anchor_i != drag.anchors_n;
- drag.anchor_i++)
- if (*drag.anchors[drag.anchor_i] == inst_get_vec(curr))
- return 1;
- drag.anchors_n = 0;
- return 0;
- }
- #endif
- static int would_be_equal(const struct drag_state *state,
- int a, int b, struct inst *curr)
- {
- const struct vec *va;
- const struct vec *vb;
- va = a == state->anchor_i ? inst_get_vec(curr) : *state->anchors[a];
- vb = b == state->anchor_i ? inst_get_vec(curr) : *state->anchors[b];
- return va == vb;
- }
- static int may_move_to(const struct drag_state *state, struct inst *curr)
- {
- assert(drag.inst);
- assert(state->anchors_n);
- switch (state->anchors_n) {
- case 3:
- if (would_be_equal(state, 0, 2, curr))
- return 0;
- /* fall through */
- case 2:
- if (would_be_equal(state, 0, 1, curr))
- return 0;
- /* fall through */
- case 1:
- return 1;
- default:
- abort();
- }
- }
- static void do_move_to(struct drag_state *state, struct inst *curr)
- {
- assert(state->inst->ops->find_point || may_move_to(state, curr));
- *state->anchors[state->anchor_i] = inst_get_vec(curr);
- }
- /* ----- hover ------------------------------------------------------------- */
- static struct pix_buf *hover_save_and_draw(void *user)
- {
- return inst_hover(hover_inst);
- }
- void tool_dehover(void)
- {
- if (!hover_inst)
- return;
- over_leave();
- hover_inst = NULL;
- }
- /*
- * When hovering, we have to consider the following states:
- *
- * selected (selected_inst != NULL)
- * | dragging new (drag.new != NULL)
- * | | dragging existing (drag.anchors_n != 0)
- * | | | tool selected (active_ops)
- * | | | |
- * Y N N N highlight if inst_find_point_selected else don't
- * Y N N Y highlight if inst_find_point_selected else fall over to tool
- * - Y N - highlight if inst_find_point / active_ops->find_point else don't
- * - N Y - highlight if may_move_to else don't
- * - Y Y - invalid state
- * N N N Y highlight if inst_find_point / active_ops->find_point else don't
- * N N N N don't highlight
- */
- static struct inst *get_hover_inst(struct coord pos)
- {
- struct inst *inst;
- int i;
- if (drag.new) {
- if (active_ops->find_point)
- return active_ops->find_point(pos);
- return inst_find_point(pos);
- }
- if (drag.anchors_n) {
- if (drag.inst->ops->find_point)
- return drag.inst->ops->find_point(drag.inst, pos);
- inst = inst_find_point(pos);
- if (!inst)
- return NULL;
- return may_move_to(&drag, inst) ? inst : NULL;
- }
- if (selected_inst) {
- i = inst_find_point_selected(pos, &inst);
- if (i != -1)
- return inst;
- }
- if (!active_ops)
- return NULL;
- if (active_ops->find_point)
- return active_ops->find_point(pos);
- return inst_find_point(pos);
- }
- int tool_hover(struct coord pos)
- {
- struct inst *curr;
- curr = get_hover_inst(pos);
- #if 0
- if ((drag.new && curr == drag.new) || (drag.inst && curr == drag.inst))
- return;
- if (curr && !active_ops) {
- if (drag.anchors_n) {
- if (!may_move_to(&drag, curr))
- curr = NULL;
- } else {
- if (!may_move(curr))
- curr = NULL;
- drag.anchors_n = 0;
- }
- }
- got:
- #endif
- if (curr == hover_inst)
- return !!curr;
- if (hover_inst) {
- over_leave();
- hover_inst = NULL;
- }
- if (!curr)
- return 0;
- hover_inst = curr;
- over_enter(hover_save_and_draw, NULL);
- return 1;
- }
- /* ----- frame drag and drop ----------------------------------------------- */
- /*
- * When dragging a frame, we temporarily replace the selected tool (if any)
- * with the frame tool.
- */
- static struct tool_ops *pushed_ops;
- void tool_push_frame(struct frame *frame)
- {
- pushed_ops = active_ops;
- locked_frame = frame;
- active_ops = &frame_ops;
- /*
- * We don't need to call tool_selected since, with drag and drop, the
- * frame tools doesn't need activation anymore.
- */
- }
- static int do_place_frame(struct frame *frame, struct coord pos)
- {
- if (!get_hover_inst(pos))
- return 0;
- tool_consider_drag(pos);
- return 1;
- }
- /*
- * Gtk calls drag-leave, drag-end, and only then drag-drop. So we'll already
- * have cleaned up in tool_pop_frame before we get here. In order to place the
- * frame, we need to activate the frame tool again.
- *
- * @@@ bug: there's a tool_reset in this path, so we'll lose the widget of the
- * tool that's really active. This problem will vanish when scrapping the
- * old-style frame referenes.
- */
- int tool_place_frame(struct frame *frame, struct coord pos)
- {
- int ok;
- active_ops = &frame_ops;
- ok = do_place_frame(frame, pos);
- active_ops = pushed_ops;
- return ok;
- }
- void tool_pop_frame(void)
- {
- if (!active_tool)
- return;
- active_ops = pushed_ops;
- /*
- * We don't need to call tool_selected since the only tool that could
- * use this would be the delete tool, and there the semantics would be
- * undesirable. Also, the delete tool never stays active, so it can't
- * appear together with drag and drop anyway.
- */
- }
- /* ----- tooltip ----------------------------------------------------------- */
- const char *tool_tip(struct coord pos)
- {
- struct inst *inst;
- inst = get_hover_inst(pos);
- if (!inst)
- return NULL;
- /*
- * The logic below follows exactly what happens in get_hover_inst.
- */
- if (drag.new)
- return "End here";
- if (drag.anchors_n)
- return "Move here";
- if (selected_inst)
- return "Move this point";
- return "Start here";
- }
- /* ----- mouse actions ----------------------------------------------------- */
- static struct pix_buf *drag_save_and_draw(void *user, struct coord to)
- {
- if (drag.new) {
- assert(active_ops);
- return active_ops->drag_new(drag.new, to);
- } else {
- return inst_draw_move(drag.inst, to, drag.anchor_i);
- }
- }
- /*
- * When considering dragging, we have the following states:
- *
- * selected (selected_inst != NULL)
- * | tool selected (active_ops)
- * | |
- * N N don't
- * Y - if we could drag, drag_new/end_new, else fall over to tool
- * N Y else single-click creation, else drag_new/end_new
- */
- int tool_consider_drag(struct coord pos)
- {
- struct inst *curr;
- assert(!drag.new);
- assert(!drag.anchors_n);
- last_canvas_pos = translate(pos);
- if (!selected_inst && !active_ops)
- return 0;
- if (selected_inst) {
- drag.anchor_i = inst_find_point_selected(pos, NULL);
- if (drag.anchor_i != -1) {
- tool_dehover();
- drag.inst = selected_inst;
- drag.new = NULL;
- drag.anchors_n =
- inst_anchors(selected_inst, drag.anchors);
- over_begin(drag_save_and_draw, NULL, pos);
- inst_begin_drag_move(selected_inst, drag.anchor_i);
- return 1;
- }
- }
- if (!active_ops)
- return 0;
- curr = get_hover_inst(pos);
- if (!curr)
- return 0;
- tool_dehover();
- if (active_ops->drag_new) {
- if (active_ops->begin_drag_new)
- active_ops->begin_drag_new(curr);
- drag.inst = NULL;
- drag.new = curr;
- over_begin(drag_save_and_draw, NULL, pos);
- return 1;
- }
- /* object is created without dragging */
- if (active_ops->end_new(curr, NULL))
- return -1;
- return 0;
- }
- void tool_drag(struct coord to)
- {
- last_canvas_pos = translate(to);
- over_move(to);
- }
- void tool_cancel_drag(void)
- {
- if (drag.anchors_n && drag.inst->ops->end_drag_move)
- drag.inst->ops->end_drag_move();
- drag.new = NULL;
- active_ops = NULL;
- drag.anchors_n = 0;
- over_end();
- tool_dehover();
- tool_reset();
- }
- int tool_end_drag(struct coord to)
- {
- struct drag_state state = drag;
- struct inst *end;
- struct tool_ops *ops = active_ops;
- tool_cancel_drag();
- if (state.new && ops->end_new_raw)
- return ops->end_new_raw(state.new, to);
- if (state.new && ops->find_point) {
- end = ops->find_point(to);
- } else {
- if (state.inst && state.inst->ops->find_point)
- end = state.inst->ops->find_point(state.inst, to);
- else
- end = inst_find_point(to);
- }
- if (!end) {
- if (state.new && ops->cancel_drag_new)
- ops->cancel_drag_new();
- return 0;
- }
- if (state.new)
- return ops->end_new(state.new, end);
- /* if we got the point from find_point, it's authoritative */
- if (!state.inst->ops->find_point && !may_move_to(&state, end))
- return 0;
- if (!inst_do_move_to(state.inst, end, state.anchor_i))
- do_move_to(&state, end);
- return 1;
- }
- void tool_redraw(void)
- {
- struct coord pos;
- over_reset();
- hover_inst = NULL;
- if (!drag.new && !drag.anchors_n)
- return;
- pos = canvas_to_coord(last_canvas_pos.x, last_canvas_pos.y);
- tool_hover(pos);
- over_begin(drag_save_and_draw, NULL, pos);
- }
- /* ----- Retrieve icons by instance characteristics ------------------------ */
- GtkWidget *get_icon_by_inst(const struct inst *inst)
- {
- char **image;
- switch (inst->prio) {
- case ip_frame:
- image = xpm_frame;
- break;
- case ip_pad_copper:
- case ip_pad_special:
- image = inst->obj->u.pad.rounded ? xpm_rpad : xpm_pad;
- break;
- case ip_hole:
- image = xpm_hole;
- break;
- case ip_circ:
- image = xpm_circ;
- break;
- case ip_arc:
- image = xpm_arc;
- break;
- case ip_rect:
- image = xpm_rect;
- break;
- case ip_meas:
- switch (inst->obj->u.meas.type) {
- case mt_xy_next:
- case mt_xy_max:
- image = xpm_meas;
- break;
- case mt_x_next:
- case mt_x_max:
- image = xpm_meas_x;
- break;
- case mt_y_next:
- case mt_y_max:
- image = xpm_meas_y;
- break;
- default:
- abort();
- }
- break;
- case ip_line:
- image = xpm_line;
- break;
- case ip_vec:
- image = xpm_vec;
- break;
- default:
- abort();
- }
- return make_transparent_image(DA, image, NULL);
- }
- /* ----- tool bar creation ------------------------------------------------- */
- static void tool_select(GtkWidget *evbox, struct tool_ops *ops)
- {
- GdkColor col;
- if (active_tool) {
- if (active_ops && active_ops->tool_deselected)
- active_ops->tool_deselected();
- col = get_color(COLOR_TOOL_UNSELECTED);
- gtk_widget_modify_bg(active_tool, GTK_STATE_NORMAL, &col);
- active_tool = NULL;
- }
- col = get_color(COLOR_TOOL_SELECTED);
- gtk_widget_modify_bg(evbox, GTK_STATE_NORMAL, &col);
- active_tool = evbox;
- active_ops = ops;
- if (ops && ops->tool_selected)
- ops->tool_selected();
- }
- void tool_reset(void)
- {
- over_reset();
- hover_inst = NULL;
- tool_select(ev_point, NULL);
- }
- static gboolean tool_button_press_event(GtkWidget *widget,
- GdkEventButton *event, gpointer data)
- {
- switch (event->button) {
- case 1:
- tool_select(widget, data);
- break;
- }
- return TRUE;
- }
- static void tool_separator(GtkWidget *bar)
- {
- GtkToolItem *item;
- item = gtk_separator_tool_item_new();
- gtk_separator_tool_item_set_draw(GTK_SEPARATOR_TOOL_ITEM(item), FALSE);
- gtk_toolbar_insert(GTK_TOOLBAR(bar), item, -1);
- }
- GtkWidget *gui_setup_tools(GdkDrawable *drawable)
- {
- GtkWidget *bar;
- bar = gtk_toolbar_new();
- gtk_toolbar_set_style(GTK_TOOLBAR(bar), GTK_TOOLBAR_ICONS);
- gtk_toolbar_set_orientation(GTK_TOOLBAR(bar),
- GTK_ORIENTATION_VERTICAL);
- ev_point = tool_button(bar, drawable, xpm_point,
- "Select and move items",
- tool_button_press_event, NULL);
- ev_delete = tool_button(bar, drawable, NULL, NULL,
- tool_button_press_event, &delete_ops);
- tool_separator(bar);
- tool_button(bar, drawable, xpm_vec,
- "Add a vector",
- tool_button_press_event, &vec_ops);
- tool_button(bar, drawable, xpm_pad,
- "Add a rectangular pad",
- tool_button_press_event, &pad_ops);
- tool_button(bar, drawable, xpm_rpad,
- "Add a rounded pad",
- tool_button_press_event, &rpad_ops);
- tool_button(bar, drawable, xpm_hole,
- "Add a hole",
- tool_button_press_event, &hole_ops);
- tool_button(bar, drawable, xpm_line,
- "Add a silk screen line",
- tool_button_press_event, &line_ops);
- tool_button(bar, drawable, xpm_rect,
- "Add a silk screen rectangle",
- tool_button_press_event, &rect_ops);
- tool_button(bar, drawable, xpm_circ,
- "Add a silk screen circle or arc",
- tool_button_press_event, &circ_ops);
- tool_separator(bar);
- tool_button(bar, drawable, xpm_meas,
- "Add a measurement",
- tool_button_press_event, &tool_meas_ops);
- tool_button(bar, drawable, xpm_meas_x,
- "Add a horizontal measurement",
- tool_button_press_event, &tool_meas_ops_x);
- tool_button(bar, drawable, xpm_meas_y,
- "Add a vertical measurement",
- tool_button_press_event, &tool_meas_ops_y);
- delete_image[0] = gtk_widget_ref(make_image(drawable, xpm_delete_off,
- NULL));
- delete_image[1] = gtk_widget_ref(make_image(drawable, xpm_delete,
- "Delete the selected item"));
- set_image(ev_delete, delete_image[0]);
- tool_reset();
- return bar;
- }
- void gui_cleanup_tools(void)
- {
- g_object_unref(delete_image[0]);
- g_object_unref(delete_image[1]);
- }
|