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