Bug fix.
1 // This example was started by Guillaume Laurent.
2 // It has become a place to dump code that tests parts of the
3 // gnomemm canvas code. Little thought has been given to the
4 // actual on-screen output.
6 #include <libgnomecanvasmm.h>
7 #include <libgnomecanvasmm/polygon.h>
12 #include <lemon/list_graph.h>
13 #include <lemon/graph_reader.h>
14 #include <lemon/graph_utils.h>
15 #include <lemon/maps.h>
16 #include <lemon/error.h>
19 using namespace lemon;
21 typedef xy<double> Coordinates;
22 typedef ListGraph Graph;
23 typedef Graph::NodeMap<Coordinates> CoordinatesMap;
24 typedef Graph::Node Node;
25 typedef Graph::EdgeIt EdgeIt;
26 typedef Graph::NodeIt NodeIt;
28 class GraphDisplayerCanvas : public Gnome::Canvas::CanvasAA
30 typedef Gnome::Canvas::CanvasAA Parent;
33 GraphDisplayerCanvas(Graph &, CoordinatesMap &);
34 virtual ~GraphDisplayerCanvas();
38 ///Event handler function that handles dragging nodes of displayed_graph
39 bool event_handler(GdkEvent* e, int b);
41 ///Event handler function that handles dragging displayed_graph
42 bool tri_mover(GdkEvent* e);
44 ///Coordinates of Weight Point of tirangle
45 Gnome::Art::Point * wp;
46 ///Array of nodes of planefigure
47 Gnome::Canvas::Ellipse ** nodes;
48 ///Sides of planefigure
49 Gnome::Canvas::Polygon * sides;
50 ///Group of graphical elements of displayed_graph
51 Gnome::Canvas::Group displayed_graph;
53 ///Indicates whether the button of mouse is pressed or not
56 ///Number Of Elements - the number of nodes
59 ///Array of coordinates
62 ///At this location was the mousebutton pressed.
63 ///It helps to calculate the distance of dragging.
64 double clicked_x, clicked_y;
66 ///Remembers which Gnome::Canvas::Item was pressed.
67 ///this variable is needed, because
68 ///1. we cannot query the item at he cursor as fast as it could not cause a Segmentation Fault
69 ///2. we would like to handle only ony item per movement, therefore quering it is not a working solution
70 Gnome::Canvas::Item * active_item;
75 ///When we click on the weight point we can drag the whole planefigure. This function resolves it.
76 bool GraphDisplayerCanvas::tri_mover(GdkEvent* e)
80 case GDK_BUTTON_PRESS:
81 clicked_x=e->button.x;
82 clicked_y=e->button.y;
85 case GDK_BUTTON_RELEASE:
89 case GDK_MOTION_NOTIFY:
92 double dx=e->motion.x-clicked_x;
93 double dy=e->motion.y-clicked_y;
95 Gnome::Canvas::Points coos;
97 for(int i=0;i<=noe;i++)
99 nodes[i]->move(dx,dy);
101 double x=(coordinates[2*i]+=dx);
102 double y=(coordinates[2*i+1]+=dy);
104 if(i!=noe)coos.push_back(Gnome::Art::Point(x,y));
108 clicked_x=e->motion.x;
109 clicked_y=e->motion.y;
111 sides->property_points().set_value(coos);
118 ///This function moves only one node of displayed_graph,
119 ///but recalculate the location of weight point,
120 ///and also redraw the sides of the planefigure.
121 bool GraphDisplayerCanvas::event_handler(GdkEvent* e, int b)
125 case GDK_BUTTON_PRESS:
126 clicked_x=e->button.x;
127 clicked_y=e->button.y;
128 active_item=(get_item_at(e->button.x, e->button.y));
131 case GDK_BUTTON_RELEASE:
135 case GDK_MOTION_NOTIFY:
138 //double x1, y1, x2, y2;
139 //(get_item_at(e->motion.x, e->motion.y))->get_bounds(x1, y1, x2, y2);
140 //printf("Item coos: %d %d %d %d\n", (int)x1, (int)y1, (int)x2, (int)y2);
141 //printf("Mouse is moved! %d %d\n",(int)e->motion.x,(int)e->motion.y);
142 double dx=e->motion.x-clicked_x;
143 double dy=e->motion.y-clicked_y;
144 active_item->move(dx, dy);
146 coordinates[2*b]+=dx;
147 coordinates[2*b+1]+=dy;
149 Gnome::Canvas::Points coos;
154 for(int i=0;i<noe;i++)
156 coos.push_back(Gnome::Art::Point(coordinates[2*i], coordinates[2*i+1]));
158 x_wp+=coordinates[2*i];
159 y_wp+=coordinates[2*i+1];
162 sides->property_points().set_value(coos);
167 dx=x_wp-coordinates[noe*2];
168 dy=y_wp-coordinates[noe*2+1];
169 nodes[noe]->move(dx, dy);
171 coordinates[noe*2]+=dx;
172 coordinates[noe*2+1]+=dy;
174 clicked_x=e->motion.x;
175 clicked_y=e->motion.y;
182 GraphDisplayerCanvas::GraphDisplayerCanvas(Graph & g, CoordinatesMap & cm):displayed_graph(*(root()), 0, 0),isbutton(false),active_item(NULL)
184 nodes=new Gnome::Canvas::Ellipse* [countNodes(g)];
187 for (NodeIt i(g); i!=INVALID; ++i)
189 nodes[sorszam]= new Gnome::Canvas::Ellipse(displayed_graph, cm[i].x-20, cm[i].y-20, cm[i].x+20, cm[i].y+20);
190 *(nodes[sorszam]) << Gnome::Canvas::Properties::fill_color("blue");
191 *(nodes[sorszam]) << Gnome::Canvas::Properties::outline_color("black");
192 (nodes[sorszam])->signal_event().connect(sigc::bind(sigc::mem_fun(*this, &GraphDisplayerCanvas::event_handler),sorszam));
196 for (EdgeIt i(g); i!=INVALID; ++i)
203 GraphDisplayerCanvas::~GraphDisplayerCanvas()
209 class MainWin : public Gtk::Window
212 MainWin(const std::string& title, Graph &, CoordinatesMap &);
216 GraphDisplayerCanvas gd_canvas;
219 MainWin::MainWin(const std::string& title, Graph & graph, CoordinatesMap & cm):gd_canvas(graph, cm)
223 set_default_size(900,600);
229 ///This class is responsible for being able
230 ///to read xy datastructure from file. It is
231 ///based on BaseMap. The set method sets the
232 ///appropriate value in the final xy NodeMap
233 ///that was given to the constructor.
234 class CoordReaderMap: public MapBase <Node, double>
240 CoordReaderMap(char xory, CoordinatesMap & coordmap);
241 void set(Node node, double coord);
244 ///The constructor expects for an xy NodeMap,
245 ///and we have to tell it, for which value
246 ///of the xy vector is responsible the actual
248 CoordReaderMap::CoordReaderMap(char xory, CoordinatesMap & coordmap): cm(coordmap)
257 throw UninitializedParameter() ;
261 ///set method sets the appropriate value in the
262 ///xy type NodeMap that is under construction
263 void CoordReaderMap::set(Node node, double coord)
274 throw UninitializedParameter() ;
280 int main(int argc, char *argv[])
284 std::cerr << "USAGE: gd <input filename.lgf>" << endl;
288 Coordinates coosvector;
292 CoordinatesMap cm(g);
294 //we create one object to read x coordinates
295 //and one to read y coordinate of nodes and write them to cm NodeMap.
296 CoordReaderMap xreader('x',cm);
297 CoordReaderMap yreader('y',cm);
299 std::ifstream is(argv[1]);
301 GraphReader<Graph> reader(is, g);
302 reader.addNodeMap("coordinates_x", xreader);
303 reader.addNodeMap("coordinates_y", yreader);
306 for (NodeIt i(g); i!=INVALID; ++i)
307 std::cout << " " << g.id(i) << " " << cm[i];
308 std::cout << std::endl;
310 Gnome::Canvas::init();
311 Gtk::Main app(argc, argv);
313 MainWin mainwin("Displayed Graph", g, cm);