graph_displayer_canvas.cc
author hegyi
Thu, 16 Jun 2005 18:58:15 +0000
branchgui
changeset 20 a3bd39d50930
parent 19 164783ceb9be
child 21 44bb92014108
permissions -rwxr-xr-x
First of all: revision 1981 is mine, what is important me because I was born in 1981. But what is new in my revision? If you drag nodes with left button, edge-breakpoints do not change location. If you drag nodes by right button, they do, they take up their base situation at the halfpoint of the edge.
ladanyi@6
     1
#include <graph_displayer_canvas.h>
hegyi@17
     2
#include <broken_edge.h>
ladanyi@6
     3
#include <math.h>
ladanyi@6
     4
hegyi@20
     5
GraphDisplayerCanvas::GraphDisplayerCanvas(Graph & gr, CoordinatesMap & cm, MapStorage & ms):g(gr),nodesmap(g),edgesmap(g),edgetextmap(g),displayed_graph(*(root()), 0, 0),mapstorage(ms),isbutton(0),active_item(NULL),target_item(NULL)
ladanyi@6
     6
{
hegyi@9
     7
  
hegyi@11
     8
  actual_handler=displayed_graph.signal_event().connect(sigc::mem_fun(*this, &GraphDisplayerCanvas::create_edge_event_handler), false);
hegyi@9
     9
hegyi@12
    10
  active_node=INVALID;
hegyi@12
    11
  active_edge=INVALID;
hegyi@12
    12
ladanyi@6
    13
  //set_center_scroll_region(true);
ladanyi@6
    14
ladanyi@6
    15
  //first edges are drawn, to hide joining with nodes later
ladanyi@6
    16
ladanyi@6
    17
  for (EdgeIt i(g); i!=INVALID; ++i)
ladanyi@6
    18
  {
ladanyi@6
    19
ladanyi@6
    20
    //drawing green lines, coordinates are from cm
ladanyi@6
    21
ladanyi@6
    22
    Gnome::Canvas::Points coos;
ladanyi@6
    23
    coos.push_back(Gnome::Art::Point(cm[g.source(i)].x,cm[g.source(i)].y));
ladanyi@6
    24
    coos.push_back(Gnome::Art::Point(cm[g.target(i)].x,cm[g.target(i)].y));
ladanyi@6
    25
    
hegyi@17
    26
    edgesmap[i]=new BrokenEdge(displayed_graph, coos);
ladanyi@6
    27
    *(edgesmap[i]) << Gnome::Canvas::Properties::fill_color("green");
ladanyi@6
    28
    edgesmap[i]->property_width_pixels().set_value(10);    
ladanyi@6
    29
    
ladanyi@6
    30
    //initializing edge-text as well, to empty string
ladanyi@6
    31
ladanyi@6
    32
    double x1, x2, y1, y2;
ladanyi@6
    33
    edgesmap[i]->get_bounds(x1, y1, x2, y2);
ladanyi@6
    34
    
ladanyi@6
    35
    edgetextmap[i]=new Gnome::Canvas::Text(displayed_graph,(x1+x2)/2, (y1+y2)/2, "");
ladanyi@6
    36
    edgetextmap[i]->property_fill_color().set_value("black");
ladanyi@6
    37
  }
ladanyi@6
    38
ladanyi@6
    39
  //afterwards nodes come to be drawn
ladanyi@6
    40
ladanyi@6
    41
  NodeIt i(g);
ladanyi@6
    42
  int maxx=0, maxy=0, minx=(int)cm[i].x, miny=(int)cm[i].y;
ladanyi@6
    43
ladanyi@6
    44
  for (; i!=INVALID; ++i)
ladanyi@6
    45
  {
ladanyi@6
    46
    //minimum and maximum is gathered to be able to zoom to the graph correctly (whole figure should be seen)
ladanyi@6
    47
ladanyi@6
    48
    if(cm[i].x>maxx)maxx=(int)cm[i].x;
ladanyi@6
    49
    if(cm[i].y>maxy)maxy=(int)cm[i].y;
ladanyi@6
    50
    if(cm[i].x<minx)minx=(int)cm[i].x;
ladanyi@6
    51
    if(cm[i].y<miny)miny=(int)cm[i].y;
ladanyi@6
    52
ladanyi@6
    53
    //drawing bule nodes, with black line around them
ladanyi@6
    54
ladanyi@6
    55
    nodesmap[i]=new Gnome::Canvas::Ellipse(displayed_graph, cm[i].x-20, cm[i].y-20, cm[i].x+20, cm[i].y+20);
ladanyi@6
    56
    *(nodesmap[i]) << Gnome::Canvas::Properties::fill_color("blue");
ladanyi@6
    57
    *(nodesmap[i]) << Gnome::Canvas::Properties::outline_color("black");
hegyi@9
    58
    //!!!!!!! (nodesmap[i])->signal_event().connect(sigc::bind(sigc::mem_fun(*this, &GraphDisplayerCanvas::event_handler),i));
ladanyi@6
    59
  }
ladanyi@6
    60
ladanyi@6
    61
  updateScrollRegion();
ladanyi@6
    62
}
ladanyi@6
    63
ladanyi@6
    64
GraphDisplayerCanvas::~GraphDisplayerCanvas()
ladanyi@6
    65
{
ladanyi@6
    66
ladanyi@6
    67
  //writing out the end state of the graph
ladanyi@6
    68
  //\todo all the maps has to be write out!
ladanyi@6
    69
ladanyi@6
    70
  Graph::NodeMap <int> id(g);
ladanyi@6
    71
  Graph::NodeMap <double> xc(g);
ladanyi@6
    72
  Graph::NodeMap <double> yc(g);
ladanyi@6
    73
  
ladanyi@6
    74
  int j=1;
ladanyi@6
    75
  
ladanyi@6
    76
  for (NodeIt i(g); i!=INVALID; ++i)
ladanyi@6
    77
  {
ladanyi@6
    78
    double x1,y1,x2,y2;
ladanyi@6
    79
    nodesmap[i]->get_bounds(x1, y1, x2, y2);
ladanyi@6
    80
    
ladanyi@6
    81
    id[i]=j++;
ladanyi@6
    82
    xc[i]=(x1+x2)/2;
ladanyi@6
    83
    yc[i]=(y1+y2)/2;
ladanyi@6
    84
  }
ladanyi@6
    85
ladanyi@6
    86
  GraphWriter<Graph> writer(std::cout,g);
ladanyi@6
    87
  
ladanyi@6
    88
  writer.writeNodeMap("id", id);
ladanyi@6
    89
  writer.writeNodeMap("coordinates_x", xc);
ladanyi@6
    90
  writer.writeNodeMap("coordinates_y", yc);
ladanyi@6
    91
  writer.run();
ladanyi@6
    92
}
ladanyi@6
    93
ladanyi@6
    94
int GraphDisplayerCanvas::changeLineWidth (std::string mapname)
ladanyi@6
    95
{
ladanyi@6
    96
  for (EdgeIt i(g); i!=INVALID; ++i)
ladanyi@6
    97
  {
ladanyi@6
    98
    int w=(int)(*(mapstorage.edgemap_storage)[mapname])[i];
ladanyi@6
    99
    edgesmap[i]->property_width_pixels().set_value(w);
ladanyi@6
   100
  }
ladanyi@6
   101
  return 0;
ladanyi@6
   102
};
ladanyi@6
   103
ladanyi@6
   104
int GraphDisplayerCanvas::changeColor (std::string mapname)
ladanyi@6
   105
{  
ladanyi@6
   106
ladanyi@6
   107
  //function maps the range of the maximum and
ladanyi@6
   108
  //the minimum of the nodemap to the range of
ladanyi@6
   109
  //green in RGB
ladanyi@6
   110
ladanyi@6
   111
  for (EdgeIt i(g); i!=INVALID; ++i)
ladanyi@6
   112
  {
ladanyi@6
   113
    double w=(*(mapstorage.edgemap_storage)[mapname])[i];
ladanyi@6
   114
    double max=mapstorage.maxOfEdgeMap(mapname);
ladanyi@6
   115
    double min=mapstorage.minOfEdgeMap(mapname);
ladanyi@6
   116
      
ladanyi@6
   117
    //std::cout<<w<<" "<<max<<" "<<min<<" "<<100*(w-min)/(max-min)<<std::endl;
ladanyi@6
   118
    Gdk::Color color;
ladanyi@6
   119
    if(max!=min)
ladanyi@6
   120
    {
ladanyi@6
   121
      color.set_rgb_p (0, 100*(w-min)/(max-min), 0);
ladanyi@6
   122
    }
ladanyi@6
   123
    else
ladanyi@6
   124
    {
ladanyi@6
   125
      color.set_rgb_p (0, 100, 0);
ladanyi@6
   126
    }
ladanyi@6
   127
ladanyi@6
   128
    edgesmap[i]->property_fill_color_gdk().set_value(color);
ladanyi@6
   129
  }
ladanyi@6
   130
  return 0;
ladanyi@6
   131
};
ladanyi@6
   132
ladanyi@6
   133
int GraphDisplayerCanvas::changeText (std::string mapname)
ladanyi@6
   134
{
ladanyi@6
   135
ladanyi@6
   136
  //the number in the map will be written on the edge
ladanyi@6
   137
  //EXCEPT when the name of the map is Text, because
ladanyi@6
   138
  //in that case empty string will be written, because
ladanyi@6
   139
  //that is the deleter map
ladanyi@6
   140
  //\todo isn't it a bit woodcutter?
ladanyi@6
   141
ladanyi@6
   142
  for (EdgeIt i(g); i!=INVALID; ++i)
ladanyi@6
   143
  {
ladanyi@6
   144
    if(mapname!="Text")
ladanyi@6
   145
    {
ladanyi@6
   146
      double number=(*(mapstorage.edgemap_storage)[mapname])[i];
ladanyi@6
   147
      int length=(int)(floor(log(number)/log(10)))+1;
ladanyi@6
   148
      int maxpos=(int)(pow(10,length-1));
ladanyi@6
   149
      int strl=length+1+RANGE;
ladanyi@6
   150
      char * str=new char[strl];
ladanyi@6
   151
      str[length]='.';
ladanyi@6
   152
      str[strl]='\0';
ladanyi@6
   153
      
ladanyi@6
   154
      for(int j=0;j<strl;j++)
ladanyi@6
   155
      {
ladanyi@6
   156
	if(j!=length)
ladanyi@6
   157
        {
ladanyi@6
   158
	  int digit=(int)(number/maxpos);
ladanyi@6
   159
	  str[j]=(digit+'0');
ladanyi@6
   160
	  number-=digit*maxpos;
ladanyi@6
   161
	  number*=10;
ladanyi@6
   162
        }
ladanyi@6
   163
      }
ladanyi@6
   164
      
ladanyi@6
   165
      edgetextmap[i]->property_text().set_value(str);
ladanyi@6
   166
    }
ladanyi@6
   167
    else
ladanyi@6
   168
    {
ladanyi@6
   169
      edgetextmap[i]->property_text().set_value("");
ladanyi@6
   170
    }
ladanyi@6
   171
  }
ladanyi@6
   172
  return 0;
ladanyi@6
   173
};
ladanyi@6
   174
ladanyi@6
   175
bool GraphDisplayerCanvas::event_handler(GdkEvent* e, Node n)
ladanyi@6
   176
{
ladanyi@6
   177
  switch(e->type)
ladanyi@6
   178
  {
ladanyi@6
   179
    case GDK_BUTTON_PRESS:
ladanyi@6
   180
      //we mark the location of the event to be able to calculate parameters of dragging
ladanyi@6
   181
      clicked_x=e->button.x;
ladanyi@6
   182
      clicked_y=e->button.y;
ladanyi@6
   183
      active_item=(get_item_at(e->button.x, e->button.y));
hegyi@20
   184
      isbutton=1;
ladanyi@6
   185
      break;
ladanyi@6
   186
    case GDK_BUTTON_RELEASE:
hegyi@20
   187
      isbutton=0;
ladanyi@6
   188
      active_item=NULL;
ladanyi@7
   189
      updateScrollRegion();
ladanyi@6
   190
      break;
ladanyi@6
   191
    case GDK_MOTION_NOTIFY:
ladanyi@6
   192
      //we only have to do sg. if the mouse button is pressed
ladanyi@6
   193
      if(isbutton)
ladanyi@6
   194
      {
ladanyi@6
   195
	//new coordinates will be the old values,
ladanyi@6
   196
	//because the item will be moved to the
ladanyi@6
   197
	//new coordinate therefore the new movement
ladanyi@6
   198
	//has to be calculated from here
ladanyi@6
   199
ladanyi@6
   200
        double dx=e->motion.x-clicked_x;
ladanyi@6
   201
        double dy=e->motion.y-clicked_y;
ladanyi@6
   202
        active_item->move(dx, dy);
ladanyi@6
   203
        clicked_x=e->motion.x;
ladanyi@6
   204
        clicked_y=e->motion.y;
ladanyi@6
   205
ladanyi@6
   206
	//all the edges connected to the moved point has to be redrawn
ladanyi@6
   207
ladanyi@6
   208
        EdgeIt e;
ladanyi@6
   209
        g.firstOut(e,n);
ladanyi@6
   210
        for(;e!=INVALID;g.nextOut(e))
ladanyi@6
   211
        {
ladanyi@6
   212
            Gnome::Canvas::Points coos;
ladanyi@6
   213
            double x1, x2, y1, y2;
ladanyi@6
   214
ladanyi@6
   215
            nodesmap[g.source(e)]->get_bounds(x1, y1, x2, y2);
ladanyi@6
   216
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
ladanyi@6
   217
ladanyi@6
   218
            nodesmap[g.target(e)]->get_bounds(x1, y1, x2, y2);
ladanyi@6
   219
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
ladanyi@6
   220
ladanyi@6
   221
            edgesmap[e]->property_points().set_value(coos);
ladanyi@6
   222
ladanyi@6
   223
	    edgesmap[e]->get_bounds(x1, y1, x2, y2);
ladanyi@6
   224
ladanyi@6
   225
	    edgetextmap[e]->property_x().set_value((x1+x2)/2);
ladanyi@6
   226
	    edgetextmap[e]->property_y().set_value((y1+y2)/2);
ladanyi@6
   227
        }
ladanyi@6
   228
ladanyi@6
   229
        g.firstIn(e,n);
ladanyi@6
   230
        for(;e!=INVALID;g.nextIn(e))
ladanyi@6
   231
        {
ladanyi@6
   232
            Gnome::Canvas::Points coos;
ladanyi@6
   233
            double x1, x2, y1, y2;
ladanyi@6
   234
ladanyi@6
   235
            nodesmap[g.source(e)]->get_bounds(x1, y1, x2, y2);
ladanyi@6
   236
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
ladanyi@6
   237
ladanyi@6
   238
            nodesmap[g.target(e)]->get_bounds(x1, y1, x2, y2);
ladanyi@6
   239
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
ladanyi@6
   240
ladanyi@6
   241
            edgesmap[e]->property_points().set_value(coos);
ladanyi@6
   242
ladanyi@6
   243
	    edgesmap[e]->get_bounds(x1, y1, x2, y2);
ladanyi@6
   244
ladanyi@6
   245
	    edgetextmap[e]->property_x().set_value((x1+x2)/2);
ladanyi@6
   246
	    edgetextmap[e]->property_y().set_value((y1+y2)/2);
ladanyi@6
   247
        }
ladanyi@6
   248
      }
ladanyi@6
   249
    default: break;
ladanyi@6
   250
  }
ladanyi@6
   251
  return true;
ladanyi@6
   252
}
ladanyi@6
   253
ladanyi@6
   254
bool GraphDisplayerCanvas::on_expose_event(GdkEventExpose *event)
ladanyi@6
   255
{
ladanyi@6
   256
  Gnome::Canvas::CanvasAA::on_expose_event(event);
ladanyi@6
   257
  //usleep(10000);
ladanyi@6
   258
  //rezoom();
ladanyi@6
   259
  return true;
ladanyi@6
   260
}
ladanyi@6
   261
ladanyi@6
   262
void GraphDisplayerCanvas::zoomIn()
ladanyi@6
   263
{
ladanyi@6
   264
  set_pixels_per_unit(
ladanyi@6
   265
      (1.0 + (double) zoom_step / 100.0) * get_pixels_per_unit());
ladanyi@6
   266
}
ladanyi@6
   267
ladanyi@6
   268
void GraphDisplayerCanvas::zoomOut()
ladanyi@6
   269
{
ladanyi@6
   270
  set_pixels_per_unit(
ladanyi@6
   271
      (1.0 - (double) zoom_step / 100.0) * get_pixels_per_unit());
ladanyi@6
   272
}
ladanyi@6
   273
ladanyi@6
   274
void GraphDisplayerCanvas::zoomFit()
ladanyi@6
   275
{
ladanyi@6
   276
  // get the height and width of the canvas
ladanyi@6
   277
  Gtk::Allocation a = get_allocation();
ladanyi@6
   278
  int aw = a.get_width();
ladanyi@6
   279
  int ah = a.get_height();
ladanyi@6
   280
  // add some space
ladanyi@6
   281
  aw -= 5; if (aw < 0) aw = 0;
ladanyi@6
   282
  ah -= 5; if (ah < 0) ah = 0;
ladanyi@6
   283
ladanyi@6
   284
  // get the bounding box of the graph
ladanyi@6
   285
  double wx1, wy1, wx2, wy2;
ladanyi@6
   286
  Gnome::Canvas::Item* pCanvasItem = root();
ladanyi@6
   287
  pCanvasItem->get_bounds(wx1, wy1, wx2, wy2);
ladanyi@6
   288
ladanyi@6
   289
  // fit the graph to the window
ladanyi@7
   290
  double ppu1 = (double) aw / fabs(wx2 - wx1);
ladanyi@7
   291
  double ppu2 = (double) ah / fabs(wy2 - wy1);
ladanyi@7
   292
  set_pixels_per_unit((ppu1 < ppu2) ? ppu1 : ppu2);
ladanyi@6
   293
}
ladanyi@6
   294
ladanyi@6
   295
void GraphDisplayerCanvas::zoom100()
ladanyi@6
   296
{
ladanyi@6
   297
  set_pixels_per_unit(1.0);
ladanyi@6
   298
}
ladanyi@6
   299
ladanyi@6
   300
void GraphDisplayerCanvas::updateScrollRegion()
ladanyi@6
   301
{
ladanyi@6
   302
  double wx1, wy1, wx2, wy2;
ladanyi@6
   303
  Gnome::Canvas::Item* pCanvasItem = root();
ladanyi@6
   304
  pCanvasItem->get_bounds(wx1, wy1, wx2, wy2);
ladanyi@7
   305
  set_scroll_region(wx1, wy1, wx2, wy2);
ladanyi@6
   306
}
hegyi@9
   307
hegyi@9
   308
void GraphDisplayerCanvas::changeEditorialTool(int newtool)
hegyi@9
   309
{
hegyi@9
   310
  actual_handler.disconnect();
hegyi@9
   311
hegyi@9
   312
  switch(newtool)
hegyi@9
   313
    {
hegyi@9
   314
    case MOVE:
hegyi@9
   315
      actual_handler=displayed_graph.signal_event().connect(sigc::mem_fun(*this, &GraphDisplayerCanvas::move_event_handler), false);
hegyi@9
   316
      break;
hegyi@12
   317
hegyi@12
   318
      //it has to assigned to canvas, because all the canvas has to be monitored, not only the elements of the already drawn group
hegyi@9
   319
    case CREATE_NODE:
hegyi@9
   320
      actual_handler=signal_event().connect(sigc::mem_fun(*this, &GraphDisplayerCanvas::create_node_event_handler), false);
hegyi@9
   321
      break;
hegyi@12
   322
hegyi@9
   323
    case CREATE_EDGE:
hegyi@9
   324
      actual_handler=displayed_graph.signal_event().connect(sigc::mem_fun(*this, &GraphDisplayerCanvas::create_edge_event_handler), false);
hegyi@9
   325
      break;
hegyi@13
   326
hegyi@13
   327
    case ERASER:
hegyi@13
   328
      actual_handler=displayed_graph.signal_event().connect(sigc::mem_fun(*this, &GraphDisplayerCanvas::eraser_event_handler), false);
hegyi@13
   329
      break;
hegyi@13
   330
hegyi@9
   331
    default:
hegyi@9
   332
      break;
hegyi@9
   333
    }
hegyi@9
   334
}
hegyi@9
   335
hegyi@9
   336
bool GraphDisplayerCanvas::move_event_handler(GdkEvent* e)
hegyi@9
   337
{
hegyi@9
   338
  switch(e->type)
hegyi@9
   339
  {
hegyi@9
   340
    case GDK_BUTTON_PRESS:
hegyi@9
   341
      //we mark the location of the event to be able to calculate parameters of dragging
hegyi@9
   342
      clicked_x=e->button.x;
hegyi@9
   343
      clicked_y=e->button.y;
hegyi@9
   344
      active_item=(get_item_at(e->button.x, e->button.y));
hegyi@9
   345
      active_node=INVALID;
hegyi@9
   346
      for (NodeIt i(g); i!=INVALID; ++i)
hegyi@9
   347
	{
hegyi@9
   348
	  if(nodesmap[i]==active_item)
hegyi@9
   349
	    {
hegyi@9
   350
	      active_node=i;
hegyi@9
   351
	    }
hegyi@9
   352
	}
hegyi@20
   353
      switch(e->button.button)
hegyi@20
   354
	{
hegyi@20
   355
	case 3:      
hegyi@20
   356
	  isbutton=3;
hegyi@20
   357
	  break;
hegyi@20
   358
	default:
hegyi@20
   359
	  isbutton=1;
hegyi@20
   360
	  break;
hegyi@20
   361
	}
hegyi@9
   362
      break;
hegyi@9
   363
    case GDK_BUTTON_RELEASE:
hegyi@20
   364
      isbutton=0;
hegyi@9
   365
      active_item=NULL;
hegyi@12
   366
      active_node=INVALID;
hegyi@9
   367
      updateScrollRegion();
hegyi@9
   368
      break;
hegyi@9
   369
    case GDK_MOTION_NOTIFY:
hegyi@12
   370
      //we only have to do sg. if the mouse button is pressed AND the click was on a node that was found in the set of nodes
hegyi@12
   371
      if(active_node!=INVALID)
hegyi@9
   372
      {
hegyi@9
   373
	//new coordinates will be the old values,
hegyi@9
   374
	//because the item will be moved to the
hegyi@9
   375
	//new coordinate therefore the new movement
hegyi@9
   376
	//has to be calculated from here
hegyi@9
   377
hegyi@9
   378
        double dx=e->motion.x-clicked_x;
hegyi@9
   379
        double dy=e->motion.y-clicked_y;
hegyi@9
   380
hegyi@9
   381
        active_item->move(dx, dy);
hegyi@9
   382
hegyi@9
   383
        clicked_x=e->motion.x;
hegyi@9
   384
        clicked_y=e->motion.y;
hegyi@9
   385
hegyi@9
   386
	//all the edges connected to the moved point has to be redrawn
hegyi@20
   387
        EdgeIt ei;
hegyi@9
   388
hegyi@20
   389
        g.firstOut(ei,active_node);
hegyi@9
   390
hegyi@20
   391
        for(;ei!=INVALID;g.nextOut(ei))
hegyi@9
   392
        {
hegyi@9
   393
            Gnome::Canvas::Points coos;
hegyi@9
   394
            double x1, x2, y1, y2;
hegyi@9
   395
hegyi@20
   396
            nodesmap[g.source(ei)]->get_bounds(x1, y1, x2, y2);
hegyi@9
   397
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
hegyi@9
   398
hegyi@20
   399
            nodesmap[g.target(ei)]->get_bounds(x1, y1, x2, y2);
hegyi@9
   400
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
hegyi@9
   401
hegyi@20
   402
	    if(isbutton==3)
hegyi@20
   403
	      {
hegyi@20
   404
		edgesmap[ei]->set_points(coos);
hegyi@20
   405
	      }
hegyi@20
   406
	    else
hegyi@20
   407
	      {
hegyi@20
   408
		edgesmap[ei]->set_points(coos,true);
hegyi@20
   409
	      }
hegyi@9
   410
hegyi@20
   411
	    edgesmap[ei]->get_bounds(x1, y1, x2, y2);
hegyi@9
   412
hegyi@20
   413
	    edgetextmap[ei]->property_x().set_value((x1+x2)/2);
hegyi@20
   414
	    edgetextmap[ei]->property_y().set_value((y1+y2)/2);
hegyi@9
   415
        }
hegyi@9
   416
hegyi@20
   417
        g.firstIn(ei,active_node);
hegyi@20
   418
        for(;ei!=INVALID;g.nextIn(ei))
hegyi@9
   419
        {
hegyi@9
   420
            Gnome::Canvas::Points coos;
hegyi@9
   421
            double x1, x2, y1, y2;
hegyi@9
   422
hegyi@20
   423
            nodesmap[g.source(ei)]->get_bounds(x1, y1, x2, y2);
hegyi@9
   424
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
hegyi@9
   425
hegyi@20
   426
            nodesmap[g.target(ei)]->get_bounds(x1, y1, x2, y2);
hegyi@9
   427
            coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
hegyi@9
   428
hegyi@20
   429
	    if(isbutton==3)
hegyi@20
   430
	      {
hegyi@20
   431
		edgesmap[ei]->set_points(coos);
hegyi@20
   432
	      }
hegyi@20
   433
	    else
hegyi@20
   434
	      {
hegyi@20
   435
		edgesmap[ei]->set_points(coos,true);
hegyi@20
   436
	      }
hegyi@9
   437
hegyi@20
   438
	    edgesmap[ei]->get_bounds(x1, y1, x2, y2);
hegyi@9
   439
hegyi@20
   440
	    edgetextmap[ei]->property_x().set_value((x1+x2)/2);
hegyi@20
   441
	    edgetextmap[ei]->property_y().set_value((y1+y2)/2);
hegyi@9
   442
        }
hegyi@9
   443
      }
hegyi@9
   444
    default: break;
hegyi@9
   445
  }
hegyi@9
   446
hegyi@9
   447
  return true;
hegyi@9
   448
}
hegyi@9
   449
hegyi@9
   450
bool GraphDisplayerCanvas::create_node_event_handler(GdkEvent* e)
hegyi@9
   451
{
hegyi@9
   452
  switch(e->type)
hegyi@9
   453
    {
hegyi@12
   454
hegyi@12
   455
      //draw the new node in red at the clicked place
hegyi@9
   456
    case GDK_BUTTON_PRESS:
hegyi@20
   457
      isbutton=1;
hegyi@9
   458
hegyi@9
   459
      active_node=NodeIt(g,g.addNode());
hegyi@9
   460
hegyi@9
   461
      window_to_world (e->button.x, e->button.y, clicked_x, clicked_y);
hegyi@9
   462
hegyi@9
   463
      nodesmap[active_node]=new Gnome::Canvas::Ellipse(displayed_graph, clicked_x-20, clicked_y-20, clicked_x+20, clicked_y+20);
hegyi@9
   464
      active_item=(Gnome::Canvas::Item *)(nodesmap[active_node]);
hegyi@9
   465
      *(nodesmap[active_node]) << Gnome::Canvas::Properties::fill_color("red");
hegyi@9
   466
      *(nodesmap[active_node]) << Gnome::Canvas::Properties::outline_color("black");
hegyi@9
   467
      (nodesmap[active_node])->show();
hegyi@9
   468
      break;
hegyi@12
   469
hegyi@12
   470
      //move the new node
hegyi@9
   471
    case GDK_MOTION_NOTIFY:
hegyi@9
   472
      {
hegyi@9
   473
	double world_motion_x, world_motion_y;
hegyi@9
   474
	GdkEvent * generated=new GdkEvent();
hegyi@9
   475
	window_to_world (e->motion.x, e->motion.y, world_motion_x, world_motion_y);
hegyi@9
   476
	generated->motion.x=world_motion_x;
hegyi@9
   477
	generated->motion.y=world_motion_y;
hegyi@9
   478
	generated->type=GDK_MOTION_NOTIFY;
hegyi@9
   479
	move_event_handler(generated);      
hegyi@9
   480
	break;
hegyi@9
   481
      }
hegyi@12
   482
hegyi@12
   483
      //finalize the new node
hegyi@9
   484
    case GDK_BUTTON_RELEASE:
hegyi@20
   485
      isbutton=0;
hegyi@9
   486
      *active_item << Gnome::Canvas::Properties::fill_color("blue");
hegyi@9
   487
      active_item=NULL;
hegyi@12
   488
      active_node=INVALID;
hegyi@9
   489
      updateScrollRegion();
hegyi@9
   490
      break;
hegyi@9
   491
    default:
hegyi@9
   492
      break;
hegyi@9
   493
    }
hegyi@9
   494
  return false;
hegyi@9
   495
}
hegyi@9
   496
hegyi@9
   497
bool GraphDisplayerCanvas::create_edge_event_handler(GdkEvent* e)
hegyi@9
   498
{
hegyi@11
   499
  switch(e->type)
hegyi@11
   500
    {
hegyi@11
   501
    case GDK_BUTTON_PRESS:
hegyi@12
   502
      //in edge creatino right button has special meaning
hegyi@12
   503
      if(e->button.button!=3)
hegyi@11
   504
	{
hegyi@12
   505
	  //there is not yet selected node
hegyi@12
   506
	  if(active_node==INVALID)
hegyi@11
   507
	    {
hegyi@12
   508
	      //we mark the location of the event to be able to calculate parameters of dragging
hegyi@12
   509
	      clicked_x=e->button.x;
hegyi@12
   510
	      clicked_y=e->button.y;
hegyi@12
   511
	      active_item=(get_item_at(e->button.x, e->button.y));
hegyi@12
   512
	      active_node=INVALID;
hegyi@12
   513
	      for (NodeIt i(g); i!=INVALID; ++i)
hegyi@11
   514
		{
hegyi@12
   515
		  if(nodesmap[i]==active_item)
hegyi@12
   516
		    {
hegyi@12
   517
		      active_node=i;
hegyi@12
   518
		    }
hegyi@12
   519
		}
hegyi@12
   520
	      //the clicked item is really a node
hegyi@12
   521
	      if(active_node!=INVALID)
hegyi@12
   522
		{
hegyi@12
   523
		  *(nodesmap[active_node]) << Gnome::Canvas::Properties::fill_color("red");
hegyi@20
   524
		  isbutton=1;
hegyi@12
   525
		}
hegyi@12
   526
	      //clicked item was not a node. It could be e.g. edge.
hegyi@12
   527
	      else
hegyi@12
   528
		{
hegyi@12
   529
		  active_item=NULL;
hegyi@11
   530
		}
hegyi@11
   531
	    }
hegyi@12
   532
	  //we only have to do sg. if the mouse button
hegyi@12
   533
	  // is pressed already once AND the click was
hegyi@12
   534
	  // on a node that was found in the set of 
hegyi@12
   535
	  //nodes, and now we only search for the second 
hegyi@12
   536
	  //node
hegyi@12
   537
	  else
hegyi@11
   538
	    {
hegyi@12
   539
	      target_item=(get_item_at(e->button.x, e->button.y));
hegyi@12
   540
	      Graph::NodeIt target_node=INVALID;
hegyi@12
   541
	      for (NodeIt i(g); i!=INVALID; ++i)
hegyi@11
   542
		{
hegyi@12
   543
		  if(nodesmap[i]==target_item)
hegyi@12
   544
		    {
hegyi@12
   545
		      target_node=i;
hegyi@12
   546
		    }
hegyi@12
   547
		}
hegyi@12
   548
	      //the clicked item is a node, the edge can be drawn
hegyi@12
   549
	      if(target_node!=INVALID)
hegyi@12
   550
		{
hegyi@12
   551
		  *(nodesmap[target_node]) << Gnome::Canvas::Properties::fill_color("red");
hegyi@12
   552
hegyi@12
   553
		  //creating new edge
hegyi@12
   554
		  active_edge=EdgeIt(g,g.addEdge(active_node, target_node));
hegyi@12
   555
	  
hegyi@12
   556
		  //calculating coordinates of new edge
hegyi@12
   557
		  Gnome::Canvas::Points coos;
hegyi@12
   558
		  double x1, x2, y1, y2;
hegyi@12
   559
	  
hegyi@12
   560
		  active_item->get_bounds(x1, y1, x2, y2);
hegyi@12
   561
		  coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
hegyi@12
   562
hegyi@12
   563
		  target_item->get_bounds(x1, y1, x2, y2);
hegyi@12
   564
		  coos.push_back(Gnome::Art::Point((x1+x2)/2,(y1+y2)/2));
hegyi@12
   565
hegyi@12
   566
		  //drawing new edge
hegyi@19
   567
		  edgesmap[active_edge]=new BrokenEdge(displayed_graph, coos);
hegyi@12
   568
		  *(edgesmap[active_edge]) << Gnome::Canvas::Properties::fill_color("green");
hegyi@12
   569
		  edgesmap[active_edge]->property_width_pixels().set_value(10);
hegyi@12
   570
hegyi@12
   571
		  //redraw nodes to blank terminations of the new edge
hegyi@12
   572
		  target_item->raise_to_top();
hegyi@12
   573
		  active_item->raise_to_top();
hegyi@12
   574
hegyi@12
   575
		  //initializing edge-text as well, to empty string
hegyi@12
   576
		  edgesmap[active_edge]->get_bounds(x1, y1, x2, y2);
hegyi@12
   577
		  edgetextmap[active_edge]=new Gnome::Canvas::Text(displayed_graph,(x1+x2)/2, (y1+y2)/2, "");
hegyi@12
   578
		  edgetextmap[active_edge]->property_fill_color().set_value("black");
hegyi@12
   579
		}
hegyi@12
   580
	      //clicked item was not a node. it could be an e.g. edge. we do not deal with it furthermore.
hegyi@12
   581
	      else
hegyi@12
   582
		{
hegyi@12
   583
		  target_item=NULL;
hegyi@11
   584
		}
hegyi@11
   585
	    }
hegyi@11
   586
	}
hegyi@11
   587
      break;
hegyi@11
   588
    case GDK_BUTTON_RELEASE:
hegyi@20
   589
      isbutton=0;
hegyi@12
   590
      //we clear settings in two cases
hegyi@12
   591
      //1: the edge is ready (target_item has valid value)
hegyi@12
   592
      //2: the edge creation is cancelled with right button
hegyi@12
   593
      if((target_item)||(e->button.button==3))
hegyi@11
   594
	{
hegyi@12
   595
	  if(active_item)
hegyi@12
   596
	    {
hegyi@12
   597
	      *active_item << Gnome::Canvas::Properties::fill_color("blue");
hegyi@12
   598
	      active_item=NULL;
hegyi@12
   599
	    }
hegyi@12
   600
	  if(target_item)
hegyi@12
   601
	    {
hegyi@12
   602
	      *target_item << Gnome::Canvas::Properties::fill_color("blue");
hegyi@12
   603
	      target_item=NULL;
hegyi@12
   604
	    }
hegyi@12
   605
	  active_node=INVALID;
hegyi@12
   606
	  active_edge=INVALID;
hegyi@11
   607
	}
hegyi@11
   608
      break;
hegyi@11
   609
    default:
hegyi@11
   610
      break;
hegyi@11
   611
    }
hegyi@9
   612
  return false;
hegyi@9
   613
}
hegyi@9
   614
hegyi@13
   615
bool GraphDisplayerCanvas::eraser_event_handler(GdkEvent* e)
hegyi@13
   616
{
hegyi@13
   617
  switch(e->type)
hegyi@13
   618
    {
hegyi@13
   619
    case GDK_BUTTON_PRESS:
hegyi@13
   620
      active_item=(get_item_at(e->button.x, e->button.y));
hegyi@13
   621
      active_node=INVALID;
hegyi@13
   622
      active_edge=INVALID;
hegyi@13
   623
      for (NodeIt i(g); i!=INVALID; ++i)
hegyi@13
   624
	{
hegyi@13
   625
	  if(nodesmap[i]==active_item)
hegyi@13
   626
	    {
hegyi@13
   627
	      active_node=i;
hegyi@13
   628
	    }
hegyi@13
   629
	}
hegyi@13
   630
      if(active_node==INVALID)
hegyi@13
   631
	{
hegyi@13
   632
	  for (EdgeIt i(g); i!=INVALID; ++i)
hegyi@13
   633
	    {
hegyi@13
   634
	      if(edgesmap[i]==active_item)
hegyi@13
   635
		{
hegyi@13
   636
		  active_edge=i;
hegyi@13
   637
		}
hegyi@13
   638
	    }
hegyi@13
   639
	}
hegyi@13
   640
    *active_item << Gnome::Canvas::Properties::fill_color("red");
hegyi@13
   641
      break;
hegyi@13
   642
hegyi@13
   643
    case GDK_BUTTON_RELEASE:
hegyi@13
   644
      if(active_item==(get_item_at(e->button.x, e->button.y)))
hegyi@13
   645
	{
hegyi@13
   646
	  if(active_node!=INVALID)
hegyi@13
   647
	    {
hegyi@14
   648
hegyi@16
   649
	      //collecting edges to delete
hegyi@16
   650
	      EdgeIt e;
hegyi@16
   651
	      std::set<Graph::Edge> edges_to_delete;
hegyi@16
   652
hegyi@16
   653
	      g.firstOut(e,active_node);
hegyi@16
   654
	      for(;e!=INVALID;g.nextOut(e))
hegyi@13
   655
		{
hegyi@16
   656
		      edges_to_delete.insert(e);
hegyi@16
   657
		}
hegyi@16
   658
hegyi@16
   659
	      g.firstIn(e,active_node);
hegyi@16
   660
	      for(;e!=INVALID;g.nextIn(e))
hegyi@16
   661
		{
hegyi@16
   662
		      edges_to_delete.insert(e);
hegyi@16
   663
		}
hegyi@16
   664
hegyi@16
   665
	      //deleting collected edges
hegyi@16
   666
	      for(std::set<Graph::Edge>::iterator edge_set_it=edges_to_delete.begin();edge_set_it!=edges_to_delete.end();edge_set_it++)
hegyi@16
   667
		{
hegyi@16
   668
		  delete_item(*edge_set_it);
hegyi@13
   669
		}
hegyi@14
   670
	      delete_item(active_node);
hegyi@14
   671
	    }
hegyi@16
   672
	  //a simple edge was chosen
hegyi@14
   673
	  else
hegyi@14
   674
	    {
hegyi@14
   675
	      delete_item(active_edge);
hegyi@14
   676
	    }
hegyi@13
   677
hegyi@13
   678
	  
hegyi@13
   679
	}
hegyi@16
   680
      //pointer was moved, deletion is cancelled
hegyi@13
   681
      else
hegyi@13
   682
	{
hegyi@13
   683
	  if(active_node!=INVALID)
hegyi@13
   684
	    {
hegyi@13
   685
	      *active_item << Gnome::Canvas::Properties::fill_color("blue");
hegyi@13
   686
	    }
hegyi@13
   687
	  else
hegyi@13
   688
	    {
hegyi@13
   689
	      *active_item << Gnome::Canvas::Properties::fill_color("green");
hegyi@13
   690
	    }
hegyi@13
   691
	}
hegyi@16
   692
      //reseting datas
hegyi@13
   693
      active_item=NULL;
hegyi@13
   694
      active_edge=INVALID;
hegyi@13
   695
      active_node=INVALID;
hegyi@13
   696
      break;
hegyi@13
   697
hegyi@13
   698
    case GDK_MOTION_NOTIFY:
hegyi@13
   699
      break;
hegyi@13
   700
hegyi@13
   701
    default:
hegyi@13
   702
      break;
hegyi@13
   703
    }
hegyi@13
   704
  return true;
hegyi@13
   705
}
hegyi@13
   706
hegyi@14
   707
void GraphDisplayerCanvas::delete_item(NodeIt node_to_delete)
hegyi@13
   708
{
hegyi@14
   709
  delete(nodesmap[node_to_delete]);
hegyi@14
   710
  g.erase(node_to_delete);
hegyi@13
   711
}
hegyi@13
   712
hegyi@14
   713
void GraphDisplayerCanvas::delete_item(EdgeIt edge_to_delete)
hegyi@14
   714
{
hegyi@14
   715
  delete(edgesmap[edge_to_delete]);
hegyi@14
   716
  g.erase(edge_to_delete);
hegyi@14
   717
}
hegyi@14
   718
hegyi@16
   719
void GraphDisplayerCanvas::delete_item(Graph::Edge edge_to_delete)
hegyi@16
   720
{
hegyi@16
   721
  delete(edgesmap[edge_to_delete]);
hegyi@16
   722
  g.erase(edge_to_delete);
hegyi@16
   723
}
hegyi@16
   724