mapeditor.cpp
author insilmaril
Wed, 09 Sep 2009 12:57:06 +0000
changeset 793 cac93797c580
parent 792 7d67be709091
child 794 d922fb6ea482
permissions -rw-r--r--
more fixes for collision detection
insilmaril@0
     1
#include "mapeditor.h"
insilmaril@0
     2
insilmaril@0
     3
#include <iostream>
insilmaril@95
     4
#include <cstdlib>
insilmaril@0
     5
#include <typeinfo>
insilmaril@0
     6
insilmaril@720
     7
#include <QObject>
insilmaril@720
     8
insilmaril@750
     9
#include "branchitem.h"
insilmaril@792
    10
#include "geometry.h"
insilmaril@175
    11
#include "mainwindow.h"
insilmaril@175
    12
#include "misc.h"
insilmaril@314
    13
#include "warningdialog.h"
insilmaril@790
    14
#include "xlinkitem.h"
insilmaril@0
    15
insilmaril@0
    16
insilmaril@0
    17
extern int statusbarTime;
insilmaril@0
    18
extern Main *mainWindow;
insilmaril@164
    19
extern QString tmpVymDir;
insilmaril@166
    20
extern QString clipboardDir;
insilmaril@491
    21
extern QString clipboardFile;
insilmaril@166
    22
extern bool clipboardEmpty;
insilmaril@473
    23
extern bool debug;
insilmaril@0
    24
insilmaril@366
    25
extern QMenu* branchContextMenu;
insilmaril@366
    26
extern QMenu* branchAddContextMenu;
insilmaril@366
    27
extern QMenu* branchRemoveContextMenu;
insilmaril@366
    28
extern QMenu* branchLinksContextMenu;
insilmaril@366
    29
extern QMenu* branchXLinksContextMenuEdit;
insilmaril@366
    30
extern QMenu* branchXLinksContextMenuFollow;
insilmaril@366
    31
extern QMenu* floatimageContextMenu;
insilmaril@366
    32
extern QMenu* canvasContextMenu;
insilmaril@366
    33
insilmaril@0
    34
extern Settings settings;
insilmaril@269
    35
extern QString iconPath;
insilmaril@0
    36
insilmaril@0
    37
///////////////////////////////////////////////////////////////////////
insilmaril@0
    38
///////////////////////////////////////////////////////////////////////
insilmaril@720
    39
MapEditor::MapEditor( VymModel *vm) 
insilmaril@0
    40
{
insilmaril@0
    41
	//cout << "Constructor ME "<<this<<endl;
insilmaril@720
    42
	mapScene= new QGraphicsScene(NULL);
insilmaril@408
    43
	mapScene->setBackgroundBrush (QBrush(Qt::white, Qt::SolidPattern));
insilmaril@408
    44
insilmaril@767
    45
	zoomFactor=zoomFactorTarget=1;
insilmaril@767
    46
insilmaril@720
    47
	model=vm;
insilmaril@645
    48
	model->setScene (mapScene);
insilmaril@720
    49
	model->registerEditor(this);
insilmaril@727
    50
	model->makeDefault();	// No changes in model so far
insilmaril@408
    51
insilmaril@408
    52
    setScene (mapScene);
insilmaril@0
    53
insilmaril@0
    54
    printer=NULL;
insilmaril@0
    55
insilmaril@366
    56
	// Create bitmap cursors, platform dependant
insilmaril@404
    57
	HandOpenCursor=QCursor (QPixmap(iconPath+"cursorhandopen.png"),1,1);		
insilmaril@404
    58
	PickColorCursor=QCursor ( QPixmap(iconPath+"cursorcolorpicker.png"), 5,27 ); 
insilmaril@404
    59
	CopyCursor=QCursor ( QPixmap(iconPath+"cursorcopy.png"), 1,1 ); 
insilmaril@404
    60
	XLinkCursor=QCursor ( QPixmap(iconPath+"cursorxlink.png"), 1,7 ); 
insilmaril@398
    61
insilmaril@769
    62
	//setFocusPolicy (Qt::StrongFocus);	//FIXME-3
insilmaril@0
    63
insilmaril@0
    64
	pickingColor=false;
insilmaril@91
    65
	drawingLink=false;
insilmaril@93
    66
	copyingObj=false;
insilmaril@0
    67
insilmaril@0
    68
    editingBO=NULL;
insilmaril@0
    69
    movingObj=NULL;
insilmaril@0
    70
insilmaril@0
    71
	printFrame=true;
insilmaril@0
    72
	printFooter=true;
insilmaril@0
    73
insilmaril@417
    74
	setAcceptDrops (true);	
insilmaril@0
    75
insilmaril@746
    76
	//model->reposition();	//FIXME-3 really still needed?
insilmaril@381
    77
insilmaril@531
    78
insilmaril@769
    79
	// Shortcuts and actions
insilmaril@769
    80
	QAction *a;
insilmaril@769
    81
    a = new QAction("Select upper branch", this);
insilmaril@769
    82
	a->setShortcut (Qt::Key_Up );
insilmaril@769
    83
	a->setShortcutContext (Qt::WidgetShortcut);
insilmaril@769
    84
	addAction (a);
insilmaril@769
    85
    connect( a, SIGNAL( triggered() ), this, SLOT( cursorUp() ) );
insilmaril@769
    86
insilmaril@769
    87
    a = new QAction( "Select lower branch",this);
insilmaril@769
    88
	a->setShortcut ( Qt::Key_Down );
insilmaril@769
    89
	a->setShortcutContext (Qt::WidgetShortcut);
insilmaril@769
    90
	addAction (a);
insilmaril@769
    91
    connect( a, SIGNAL( triggered() ), this, SLOT( cursorDown() ) );
insilmaril@769
    92
insilmaril@769
    93
    a = new QAction( "Select left branch", this);
insilmaril@769
    94
	a->setShortcut (Qt::Key_Left );
insilmaril@769
    95
//	a->setShortcutContext (Qt::WindowShortcut);
insilmaril@769
    96
	a->setShortcutContext (Qt::WidgetWithChildrenShortcut);
insilmaril@769
    97
	addAction (a);
insilmaril@769
    98
    connect( a, SIGNAL( triggered() ), this, SLOT( cursorLeft() ) );
insilmaril@769
    99
insilmaril@769
   100
    a = new QAction( "Select child branch", this);
insilmaril@769
   101
	a->setShortcut (Qt::Key_Right);
insilmaril@769
   102
	a->setShortcutContext (Qt::WidgetWithChildrenShortcut);
insilmaril@769
   103
	addAction (a);
insilmaril@769
   104
    connect( a, SIGNAL( triggered() ), this, SLOT( cursorRight() ) );
insilmaril@769
   105
insilmaril@769
   106
    a = new QAction(  "Select first branch", this);
insilmaril@769
   107
	a->setShortcut (Qt::Key_Home );
insilmaril@769
   108
	a->setShortcutContext (Qt::WidgetWithChildrenShortcut);
insilmaril@769
   109
	addAction (a);
insilmaril@769
   110
    connect( a, SIGNAL( triggered() ), this, SLOT( cursorFirst() ) );
insilmaril@769
   111
insilmaril@769
   112
    a = new QAction( "Select last branch",this);
insilmaril@769
   113
	a->setShortcut ( Qt::Key_End );
insilmaril@769
   114
	a->setShortcutContext (Qt::WidgetWithChildrenShortcut);
insilmaril@769
   115
	addAction (a);
insilmaril@769
   116
    connect( a, SIGNAL( triggered() ), this, SLOT( cursorLast() ) );
insilmaril@769
   117
insilmaril@720
   118
	// Action to embed LineEdit for heading in Scene
insilmaril@720
   119
	editingHeading=false;
insilmaril@720
   120
	lineEdit=new QLineEdit;
insilmaril@720
   121
	lineEdit->hide();
insilmaril@720
   122
	QGraphicsProxyWidget *pw=scene()->addWidget (lineEdit);
insilmaril@772
   123
	pw->setZValue (Z_LINEEDIT);
insilmaril@676
   124
insilmaril@769
   125
	a = new QAction( tr( "Edit heading","MapEditor" ), this);
insilmaril@720
   126
	a->setShortcut ( Qt::Key_Return );					//Edit heading
insilmaril@720
   127
	addAction (a);
insilmaril@720
   128
    connect( a, SIGNAL( triggered() ), this, SLOT( editHeading() ) );
insilmaril@720
   129
	a = new QAction( tr( "Edit heading","MapEditor" ), this);
insilmaril@720
   130
	a->setShortcut ( Qt::Key_Enter);					//Edit heading
insilmaril@720
   131
	addAction (a);
insilmaril@720
   132
    connect( a, SIGNAL( triggered() ), this, SLOT( editHeading() ) );
insilmaril@628
   133
insilmaril@728
   134
	// Selections
insilmaril@728
   135
	selectionColor =QColor (255,255,0);
insilmaril@728
   136
	
insilmaril@728
   137
insilmaril@746
   138
	// Attributes	//FIXME-2 testing only...
insilmaril@638
   139
	QString k;
insilmaril@638
   140
	AttributeDef *ad;
insilmaril@628
   141
	attrTable= new AttributeTable();
insilmaril@638
   142
	k="A - StringList";
insilmaril@638
   143
	ad=attrTable->addKey (k,StringList);
insilmaril@638
   144
	if (ad)
insilmaril@638
   145
	{
insilmaril@638
   146
		QStringList sl;
insilmaril@638
   147
		sl <<"val 1"<<"val 2"<< "val 3";
insilmaril@638
   148
		ad->setValue (QVariant (sl));
insilmaril@638
   149
	}
insilmaril@638
   150
	//attrTable->addValue ("Key A","P 1");
insilmaril@638
   151
	//attrTable->addValue ("Key A","P 2");
insilmaril@638
   152
	//attrTable->addValue ("Key A","P 3");
insilmaril@638
   153
	//attrTable->addValue ("Key A","P 4");
insilmaril@638
   154
	k="B - FreeString";
insilmaril@638
   155
	ad=attrTable->addKey (k,FreeString);
insilmaril@638
   156
	if (ad)
insilmaril@638
   157
	{
insilmaril@638
   158
		//attrTable->addValue ("Key B","w1");
insilmaril@638
   159
		//attrTable->addValue ("Key B","w2");
insilmaril@638
   160
	}
insilmaril@638
   161
	k="C - UniqueString";
insilmaril@638
   162
	ad=attrTable->addKey (k,UniqueString);
insilmaril@638
   163
	if (ad)
insilmaril@638
   164
	{
insilmaril@638
   165
	//attrTable->addKey ("Key Prio");
insilmaril@638
   166
	//attrTable->addValue ("Key Prio","Prio 1");
insilmaril@638
   167
	//attrTable->addValue ("Key Prio","Prio 2");
insilmaril@638
   168
	}
insilmaril@0
   169
}
insilmaril@0
   170
insilmaril@0
   171
MapEditor::~MapEditor()
insilmaril@0
   172
{
insilmaril@746
   173
	//cout <<"Destructor MapEditor for "<<model->getMapName().toStdString()<<endl;
insilmaril@720
   174
	model->unregisterEditor(this);
insilmaril@0
   175
}
insilmaril@0
   176
insilmaril@645
   177
VymModel* MapEditor::getModel()
insilmaril@0
   178
{
insilmaril@645
   179
    return model;
insilmaril@0
   180
}
insilmaril@0
   181
insilmaril@408
   182
QGraphicsScene * MapEditor::getScene()
insilmaril@0
   183
{
insilmaril@408
   184
    return mapScene;
insilmaril@0
   185
}
insilmaril@0
   186
insilmaril@772
   187
void MapEditor::scrollTo (const QModelIndex &index)
insilmaril@772
   188
{
insilmaril@772
   189
	if (index.isValid())
insilmaril@772
   190
	{
insilmaril@776
   191
		LinkableMapObj* lmo=NULL;
insilmaril@776
   192
		TreeItem *ti= static_cast<TreeItem*>(index.internalPointer());
insilmaril@776
   193
		if (ti->getType()==TreeItem::Image ||ti->isBranchLikeType() )
insilmaril@779
   194
			lmo=((MapItem*)ti)->getLMO();
insilmaril@772
   195
		if (lmo) setScrollBarPosTarget (lmo->getBBox() );
insilmaril@772
   196
	}
insilmaril@772
   197
}
insilmaril@772
   198
insilmaril@767
   199
void MapEditor::setScrollBarPosTarget (const QRectF &rect)
insilmaril@767
   200
{
insilmaril@767
   201
	// Code copied from Qt sources
insilmaril@767
   202
	int xmargin=50;
insilmaril@767
   203
	int ymargin=50;
insilmaril@767
   204
insilmaril@767
   205
    qreal width = viewport()->width();
insilmaril@767
   206
    qreal height = viewport()->height();
insilmaril@767
   207
    QRectF viewRect = matrix().mapRect(rect);
insilmaril@767
   208
insilmaril@767
   209
    qreal left = horizontalScrollBar()->value();
insilmaril@767
   210
    qreal right = left + width;
insilmaril@767
   211
    qreal top = verticalScrollBar()->value();
insilmaril@767
   212
    qreal bottom = top + height;
insilmaril@767
   213
insilmaril@767
   214
    if (viewRect.left() <= left + xmargin) {
insilmaril@767
   215
        // need to scroll from the left
insilmaril@767
   216
  //      if (!d->leftIndent)
insilmaril@767
   217
            scrollBarPosTarget.setX(int(viewRect.left() - xmargin - 0.5));
insilmaril@767
   218
    }
insilmaril@767
   219
    if (viewRect.right() >= right - xmargin) {
insilmaril@767
   220
        // need to scroll from the right
insilmaril@767
   221
//        if (!d->leftIndent)
insilmaril@767
   222
            scrollBarPosTarget.setX(int(viewRect.right() - width + xmargin + 0.5));
insilmaril@767
   223
    }
insilmaril@767
   224
    if (viewRect.top() <= top + ymargin) {
insilmaril@767
   225
        // need to scroll from the top
insilmaril@767
   226
   //     if (!d->topIndent)
insilmaril@767
   227
            scrollBarPosTarget.setY(int(viewRect.top() - ymargin - 0.5));
insilmaril@767
   228
    }
insilmaril@767
   229
    if (viewRect.bottom() >= bottom - ymargin) {
insilmaril@767
   230
        // need to scroll from the bottom
insilmaril@767
   231
//        if (!d->topIndent)
insilmaril@767
   232
            scrollBarPosTarget.setY(int(viewRect.bottom() - height + ymargin + 0.5));
insilmaril@767
   233
    }
insilmaril@767
   234
insilmaril@767
   235
	if (scrollBarPosAnimation.state()==QtAbstractAnimation::Running)
insilmaril@767
   236
		scrollBarPosAnimation.stop();
insilmaril@767
   237
	scrollBarPosAnimation.setTargetObject (this);
insilmaril@767
   238
	scrollBarPosAnimation.setPropertyName ("scrollBarPos");
insilmaril@767
   239
	scrollBarPosAnimation.setDuration(1000);
insilmaril@767
   240
	scrollBarPosAnimation.setEasingCurve ( QtEasingCurve::OutQuint);
insilmaril@767
   241
	scrollBarPosAnimation.setStartValue(
insilmaril@767
   242
		QPointF (horizontalScrollBar()->value() ,
insilmaril@767
   243
		         verticalScrollBar()->value() ) );
insilmaril@767
   244
	scrollBarPosAnimation.setEndValue(scrollBarPosTarget);
insilmaril@767
   245
	scrollBarPosAnimation.start();
insilmaril@767
   246
}
insilmaril@767
   247
insilmaril@767
   248
QPointF MapEditor::getScrollBarPosTarget()
insilmaril@767
   249
{
insilmaril@767
   250
    return scrollBarPosTarget;
insilmaril@767
   251
}
insilmaril@767
   252
insilmaril@767
   253
insilmaril@767
   254
void MapEditor::setScrollBarPos(const QPointF &p)
insilmaril@767
   255
{
insilmaril@767
   256
    scrollBarPos=p;
insilmaril@767
   257
	horizontalScrollBar()->setValue(int(p.x()));
insilmaril@767
   258
	verticalScrollBar()->setValue(int(p.y()));
insilmaril@767
   259
}
insilmaril@767
   260
insilmaril@767
   261
QPointF MapEditor::getScrollBarPos()
insilmaril@767
   262
{
insilmaril@767
   263
    return scrollBarPos;
insilmaril@767
   264
}
insilmaril@767
   265
insilmaril@767
   266
void MapEditor::setZoomFactorTarget (const qreal &zft)
insilmaril@767
   267
{
insilmaril@767
   268
	zoomFactorTarget=zft;
insilmaril@767
   269
	if (zoomAnimation.state()==QtAbstractAnimation::Running)
insilmaril@767
   270
		zoomAnimation.stop();
insilmaril@767
   271
	//zoomAnimation=QtPropertyAnimation(this, "zoomFactor");
insilmaril@767
   272
	zoomAnimation.setTargetObject (this);
insilmaril@767
   273
	zoomAnimation.setPropertyName ("zoomFactor");
insilmaril@767
   274
	zoomAnimation.setDuration(1000);
insilmaril@767
   275
	zoomAnimation.setEasingCurve ( QtEasingCurve::OutQuint);
insilmaril@767
   276
	zoomAnimation.setStartValue(zoomFactor);
insilmaril@767
   277
	zoomAnimation.setEndValue(zft);
insilmaril@767
   278
	zoomAnimation.start();
insilmaril@767
   279
}
insilmaril@767
   280
insilmaril@767
   281
qreal MapEditor::getZoomFactorTarget()
insilmaril@767
   282
{
insilmaril@767
   283
    return zoomFactorTarget;
insilmaril@767
   284
}
insilmaril@767
   285
insilmaril@767
   286
insilmaril@767
   287
void MapEditor::setZoomFactor(const qreal &zf)
insilmaril@767
   288
{
insilmaril@767
   289
    zoomFactor=zf;
insilmaril@767
   290
	setMatrix (QMatrix(zf, 0, 0, zf, 0, 0),false );
insilmaril@767
   291
}
insilmaril@767
   292
insilmaril@767
   293
qreal MapEditor::getZoomFactor()
insilmaril@767
   294
{
insilmaril@767
   295
    return zoomFactor;
insilmaril@767
   296
}
insilmaril@767
   297
insilmaril@0
   298
void MapEditor::print()
insilmaril@0
   299
{
insilmaril@0
   300
	if ( !printer ) 
insilmaril@0
   301
	{
insilmaril@0
   302
		printer = new QPrinter;
insilmaril@0
   303
		printer->setColorMode (QPrinter::Color);
insilmaril@366
   304
		printer->setPrinterName (settings.value("/mainwindow/printerName",printer->printerName()).toString());
insilmaril@412
   305
		printer->setOutputFormat((QPrinter::OutputFormat)settings.value("/mainwindow/printerFormat",printer->outputFormat()).toInt());
insilmaril@412
   306
		printer->setOutputFileName(settings.value("/mainwindow/printerFileName",printer->outputFileName()).toString());
insilmaril@0
   307
	}
insilmaril@0
   308
insilmaril@782
   309
	QRectF totalBBox=getTotalBBox();
insilmaril@0
   310
insilmaril@0
   311
	// Try to set orientation automagically
insilmaril@0
   312
	// Note: Interpretation of generated postscript is amibiguous, if 
insilmaril@0
   313
	// there are problems with landscape mode, see
insilmaril@0
   314
	// http://sdb.suse.de/de/sdb/html/jsmeix_print-cups-landscape-81.html
insilmaril@0
   315
insilmaril@0
   316
	if (totalBBox.width()>totalBBox.height())
insilmaril@0
   317
		// recommend landscape
insilmaril@0
   318
		printer->setOrientation (QPrinter::Landscape);
insilmaril@0
   319
	else	
insilmaril@0
   320
		// recommend portrait
insilmaril@0
   321
		printer->setOrientation (QPrinter::Portrait);
insilmaril@0
   322
insilmaril@0
   323
	if ( printer->setup(this) ) 
insilmaril@0
   324
	// returns false, if printing is canceled
insilmaril@0
   325
	{
insilmaril@0
   326
		QPainter pp(printer);
insilmaril@0
   327
insilmaril@412
   328
		pp.setRenderHint(QPainter::Antialiasing,true);
insilmaril@412
   329
insilmaril@0
   330
		// Don't print the visualisation of selection
insilmaril@720
   331
		model->unselect();
insilmaril@0
   332
insilmaril@408
   333
		QRectF mapRect=totalBBox;
insilmaril@411
   334
		QGraphicsRectItem *frame=NULL;
insilmaril@412
   335
insilmaril@417
   336
		if (printFrame) 
insilmaril@0
   337
		{
insilmaril@417
   338
			// Print frame around map
insilmaril@417
   339
			mapRect.setRect (totalBBox.x()-10, totalBBox.y()-10, 
insilmaril@417
   340
				totalBBox.width()+20, totalBBox.height()+20);
insilmaril@417
   341
			frame=mapScene->addRect (mapRect, QPen(Qt::black),QBrush(Qt::NoBrush));
insilmaril@417
   342
			frame->setZValue(0);
insilmaril@417
   343
			frame->show();    
insilmaril@417
   344
		}		
insilmaril@417
   345
insilmaril@417
   346
insilmaril@412
   347
		double paperAspect = (double)printer->width()   / (double)printer->height();
insilmaril@412
   348
		double   mapAspect = (double)mapRect.width() / (double)mapRect.height();
insilmaril@417
   349
		int viewBottom;
insilmaril@0
   350
		if (mapAspect>=paperAspect)
insilmaril@0
   351
		{
insilmaril@0
   352
			// Fit horizontally to paper width
insilmaril@417
   353
			//pp.setViewport(0,0, printer->width(),(int)(printer->width()/mapAspect) );	
insilmaril@417
   354
			viewBottom=(int)(printer->width()/mapAspect);	
insilmaril@0
   355
		}	else
insilmaril@0
   356
		{
insilmaril@0
   357
			// Fit vertically to paper height
insilmaril@417
   358
			//pp.setViewport(0,0,(int)(printer->height()*mapAspect),printer->height());	
insilmaril@417
   359
			viewBottom=printer->height();	
insilmaril@0
   360
		}	
insilmaril@417
   361
		
insilmaril@417
   362
		if (printFooter) 
insilmaril@417
   363
		{
insilmaril@417
   364
			// Print footer below map
insilmaril@417
   365
			QFont font;		
insilmaril@417
   366
			font.setPointSize(10);
insilmaril@417
   367
			pp.setFont (font);
insilmaril@417
   368
			QRectF footerBox(0,viewBottom,printer->width(),15);
insilmaril@746
   369
			// FIXME-3 fileName not any longer available here: pp.drawText ( footerBox,Qt::AlignLeft,"VYM - " +fileName);
insilmaril@417
   370
			pp.drawText ( footerBox, Qt::AlignRight, QDate::currentDate().toString(Qt::TextDate));
insilmaril@417
   371
		}
insilmaril@412
   372
		mapScene->render (
insilmaril@412
   373
			&pp, 
insilmaril@412
   374
			QRectF (0,0,printer->width(),printer->height()-15),
insilmaril@412
   375
			QRectF(mapRect.x(),mapRect.y(),mapRect.width(),mapRect.height())
insilmaril@412
   376
		);
insilmaril@412
   377
		
insilmaril@412
   378
		// Viewport has paper dimension
insilmaril@412
   379
		if (frame)  delete (frame);
insilmaril@0
   380
insilmaril@0
   381
		// Restore selection
insilmaril@720
   382
		model->reselect();
insilmaril@119
   383
insilmaril@119
   384
		// Save settings in vymrc
insilmaril@366
   385
		settings.writeEntry("/mainwindow/printerName",printer->printerName());
insilmaril@412
   386
		settings.writeEntry("/mainwindow/printerFormat",printer->outputFormat());
insilmaril@412
   387
		settings.writeEntry("/mainwindow/printerFileName",printer->outputFileName());
insilmaril@0
   388
	}
insilmaril@0
   389
}
insilmaril@0
   390
insilmaril@782
   391
QRectF MapEditor::getTotalBBox()	//FIXME-2 needed e.g. for image export
insilmaril@782
   392
{
insilmaril@782
   393
	QRectF r;
insilmaril@782
   394
/*
insilmaril@782
   395
	for (int i=0;i<rootItem->branchCount(); i++)
insilmaril@782
   396
		r=addBBox (rootItem->getBranchNum(i)->getTotalBBox(), r);
insilmaril@782
   397
*/ 
insilmaril@782
   398
	return r;	
insilmaril@782
   399
}
insilmaril@782
   400
insilmaril@782
   401
insilmaril@782
   402
QPixmap MapEditor::getPixmap()
insilmaril@782
   403
{
insilmaril@782
   404
	QRectF mapRect=getTotalBBox();
insilmaril@782
   405
	QPixmap pix((int)mapRect.width()+2,(int)mapRect.height()+1);
insilmaril@782
   406
	QPainter pp (&pix);
insilmaril@782
   407
	
insilmaril@782
   408
	pp.setRenderHints(renderHints());
insilmaril@782
   409
insilmaril@782
   410
	// Don't print the visualisation of selection
insilmaril@782
   411
	model->unselect();
insilmaril@782
   412
insilmaril@782
   413
	mapScene->render (	&pp, 
insilmaril@782
   414
		QRectF(0,0,mapRect.width()+2,mapRect.height()+2),
insilmaril@782
   415
		QRectF(mapRect.x(),mapRect.y(),mapRect.width(),mapRect.height() ));
insilmaril@782
   416
insilmaril@782
   417
	// Restore selection
insilmaril@782
   418
	model->reselect();
insilmaril@782
   419
	
insilmaril@782
   420
	return pix;
insilmaril@782
   421
}
insilmaril@782
   422
insilmaril@782
   423
insilmaril@408
   424
void MapEditor::setAntiAlias (bool b)
insilmaril@408
   425
{
insilmaril@408
   426
	setRenderHint(QPainter::Antialiasing,b);
insilmaril@408
   427
}
insilmaril@408
   428
insilmaril@408
   429
void MapEditor::setSmoothPixmap(bool b)
insilmaril@408
   430
{
insilmaril@408
   431
	setRenderHint(QPainter::SmoothPixmapTransform,b);
insilmaril@408
   432
}
insilmaril@408
   433
insilmaril@754
   434
TreeItem* MapEditor::findMapItem (QPointF p,TreeItem *exclude)
insilmaril@754
   435
{
insilmaril@754
   436
	// Start with mapcenter, no images allowed at rootItem
insilmaril@754
   437
	int i=0;
insilmaril@754
   438
	BranchItem *bi=model->getRootItem()->getFirstBranch();
insilmaril@754
   439
	TreeItem *found=NULL;
insilmaril@754
   440
	while (bi)
insilmaril@754
   441
	{
insilmaril@754
   442
		found=bi->findMapItem (p, exclude);
insilmaril@754
   443
		if (found) return found;
insilmaril@754
   444
		i++;
insilmaril@754
   445
		bi=model->getRootItem()->getBranchNum(i);
insilmaril@754
   446
	}
insilmaril@754
   447
	return NULL;
insilmaril@754
   448
}
insilmaril@754
   449
insilmaril@628
   450
AttributeTable* MapEditor::attributeTable()
insilmaril@628
   451
{
insilmaril@628
   452
	return attrTable;
insilmaril@628
   453
}
insilmaril@628
   454
insilmaril@527
   455
void MapEditor::testFunction1()
insilmaril@0
   456
{
insilmaril@788
   457
	/*
insilmaril@767
   458
	// Code copied from Qt sources
insilmaril@767
   459
	QRectF rect=model->getSelectedBranchObj()->getBBox();
insilmaril@767
   460
	int xmargin=50;
insilmaril@767
   461
	int ymargin=50;
insilmaril@741
   462
insilmaril@767
   463
    qreal width = viewport()->width();
insilmaril@767
   464
    qreal height = viewport()->height();
insilmaril@767
   465
    QRectF viewRect = matrix().mapRect(rect);
insilmaril@741
   466
insilmaril@767
   467
    qreal left = horizontalScrollBar()->value();
insilmaril@767
   468
    qreal right = left + width;
insilmaril@767
   469
    qreal top = verticalScrollBar()->value();
insilmaril@767
   470
    qreal bottom = top + height;
insilmaril@723
   471
insilmaril@767
   472
    if (viewRect.left() <= left + xmargin) {
insilmaril@767
   473
        // need to scroll from the left
insilmaril@767
   474
  //      if (!d->leftIndent)
insilmaril@767
   475
            horizontalScrollBar()->setValue(int(viewRect.left() - xmargin - 0.5));
insilmaril@767
   476
    }
insilmaril@767
   477
    if (viewRect.right() >= right - xmargin) {
insilmaril@767
   478
        // need to scroll from the right
insilmaril@767
   479
//        if (!d->leftIndent)
insilmaril@767
   480
            horizontalScrollBar()->setValue(int(viewRect.right() - width + xmargin + 0.5));
insilmaril@767
   481
    }
insilmaril@767
   482
    if (viewRect.top() <= top + ymargin) {
insilmaril@767
   483
        // need to scroll from the top
insilmaril@767
   484
   //     if (!d->topIndent)
insilmaril@767
   485
            verticalScrollBar()->setValue(int(viewRect.top() - ymargin - 0.5));
insilmaril@767
   486
    }
insilmaril@767
   487
    if (viewRect.bottom() >= bottom - ymargin) {
insilmaril@767
   488
        // need to scroll from the bottom
insilmaril@767
   489
//        if (!d->topIndent)
insilmaril@767
   490
            verticalScrollBar()->setValue(int(viewRect.bottom() - height + ymargin + 0.5));
insilmaril@767
   491
    }
insilmaril@767
   492
	cout << "test1:  hor="<<horizontalScrollBar()->value()<<endl;
insilmaril@767
   493
	cout << "test1:  ver="<<verticalScrollBar()->value()<<endl;
insilmaril@767
   494
}
insilmaril@788
   495
insilmaril@788
   496
*/
insilmaril@741
   497
/*
insilmaril@767
   498
	 QtPropertyAnimation *animation=new QtPropertyAnimation(this, "sceneRect");
insilmaril@767
   499
	 animation->setDuration(5000);
insilmaril@767
   500
	 //animation->setEasingCurve ( QtEasingCurve::OutElastic);
insilmaril@767
   501
	 animation->setEasingCurve ( QtEasingCurve::OutQuint);
insilmaril@767
   502
	 animation->setStartValue(sceneRect() );
insilmaril@767
   503
	 animation->setEndValue(QRectF(50, 50, 1000, 1000));
insilmaril@740
   504
insilmaril@767
   505
	 animation->start();
insilmaril@767
   506
*/	 
insilmaril@767
   507
/*
insilmaril@767
   508
	QDialog *dia= new QDialog (this);
insilmaril@767
   509
	dia->setGeometry (50,50,10,10);
insilmaril@723
   510
insilmaril@767
   511
     dia->show();
insilmaril@767
   512
     dia ->raise();
insilmaril@767
   513
insilmaril@767
   514
	 QtPropertyAnimation *animation=new QtPropertyAnimation(dia, "geometry");
insilmaril@767
   515
	 animation->setDuration(1000);
insilmaril@767
   516
	 //animation->setEasingCurve ( QtEasingCurve::OutElastic);
insilmaril@767
   517
	 animation->setEasingCurve ( QtEasingCurve::OutQuint);
insilmaril@767
   518
	 animation->setStartValue(QRect(50, 50, 10, 10));
insilmaril@767
   519
	 animation->setEndValue(QRect(250, 250, 100, 100));
insilmaril@767
   520
insilmaril@767
   521
	 animation->start();
insilmaril@767
   522
 */
insilmaril@723
   523
insilmaril@791
   524
/* FIXME-4 Hide hidden stuff temporary, maybe add this as regular function somewhere
insilmaril@260
   525
	if (hidemode==HideNone)
insilmaril@185
   526
	{
insilmaril@260
   527
		setHideTmpMode (HideExport);
insilmaril@260
   528
		mapCenter->calcBBoxSizeWithChilds();
insilmaril@408
   529
		QRectF totalBBox=mapCenter->getTotalBBox();
insilmaril@408
   530
		QRectF mapRect=totalBBox;
insilmaril@260
   531
		QCanvasRectangle *frame=NULL;
insilmaril@260
   532
insilmaril@260
   533
		cout << "  map has =("<<totalBBox.x()<<","<<totalBBox.y()<<","<<totalBBox.width()<<","<<totalBBox.height()<<")\n";
insilmaril@260
   534
	
insilmaril@260
   535
		mapRect.setRect (totalBBox.x(), totalBBox.y(), 
insilmaril@260
   536
			totalBBox.width(), totalBBox.height());
insilmaril@408
   537
		frame=new QCanvasRectangle (mapRect,mapScene);
insilmaril@260
   538
		frame->setBrush (QColor(white));
insilmaril@260
   539
		frame->setPen (QColor(black));
insilmaril@408
   540
		frame->setZValue(0);
insilmaril@260
   541
		frame->show();    
insilmaril@260
   542
	}	
insilmaril@260
   543
	else	
insilmaril@260
   544
	{
insilmaril@260
   545
		setHideTmpMode (HideNone);
insilmaril@260
   546
	}	
insilmaril@260
   547
	cout <<"  hidemode="<<hidemode<<endl;
insilmaril@301
   548
	*/
insilmaril@645
   549
insilmaril@691
   550
/*
insilmaril@628
   551
	// Toggle hidemode
insilmaril@628
   552
	if (hidemode==HideExport)
insilmaril@628
   553
		setHideTmpMode (HideNone);
insilmaril@628
   554
	else	
insilmaril@628
   555
		setHideTmpMode (HideExport);
insilmaril@691
   556
*/		
insilmaril@792
   557
insilmaril@792
   558
}
insilmaril@792
   559
	
insilmaril@792
   560
void MapEditor::testFunction2()
insilmaril@792
   561
{
insilmaril@792
   562
	// Create list with all bounding polygons
insilmaril@792
   563
	QList <MapObj*> mapobjects;
insilmaril@792
   564
	QList <ConvexPolygon> polys; 
insilmaril@793
   565
	ConvexPolygon p;
insilmaril@792
   566
	QList <Vector> vectors;
insilmaril@792
   567
	QList <Vector> orgpos;
insilmaril@793
   568
	QStringList headings;	//FIXME-3 testing only
insilmaril@793
   569
	Vector v;
insilmaril@792
   570
	BranchItem *bi;
insilmaril@793
   571
	BranchItem *bi2;
insilmaril@792
   572
	BranchObj *bo;
insilmaril@793
   573
	BranchItem *ri=model->getRootItem();
insilmaril@792
   574
	for (int i=0;i<ri->branchCount();++i)
insilmaril@792
   575
	{
insilmaril@792
   576
		bi=ri->getBranchNum (i);
insilmaril@792
   577
		bo=(BranchObj*)bi->getLMO();
insilmaril@792
   578
		if (bo)
insilmaril@792
   579
		{
insilmaril@792
   580
			mapobjects.append (bo);
insilmaril@793
   581
			p=bo->getBoundingPolygon();
insilmaril@793
   582
			p.calcCentroid();
insilmaril@793
   583
			polys.append(p);
insilmaril@792
   584
			vectors.append (QPointF(0,0));
insilmaril@793
   585
			orgpos.append (p.at(0));
insilmaril@793
   586
			headings.append (bi->getHeading());
insilmaril@793
   587
		}
insilmaril@793
   588
		for (int j=0;j<bi->branchCount();++j)
insilmaril@793
   589
		{
insilmaril@793
   590
			bi2=bi->getBranchNum (j);
insilmaril@793
   591
			bo=(BranchObj*)bi2->getLMO();
insilmaril@793
   592
			if (bo)
insilmaril@793
   593
			{
insilmaril@793
   594
				mapobjects.append (bo);
insilmaril@793
   595
				p=bo->getBoundingPolygon();
insilmaril@793
   596
				p.calcCentroid();
insilmaril@793
   597
				polys.append(p);
insilmaril@793
   598
				vectors.append (QPointF(0,0));
insilmaril@793
   599
				orgpos.append (p.at(0));
insilmaril@793
   600
				headings.append (bi2->getHeading());
insilmaril@793
   601
			}	
insilmaril@793
   602
		}
insilmaril@792
   603
	}
insilmaril@792
   604
insilmaril@792
   605
	// Iterate moving bounding polygons until we have no more collisions
insilmaril@792
   606
	int collisions=1;
insilmaril@792
   607
	while (collisions>0)
insilmaril@792
   608
	{
insilmaril@792
   609
		collisions=0;
insilmaril@792
   610
		for (int i=0; i<polys.size()-1; ++i)
insilmaril@792
   611
		{
insilmaril@792
   612
			for (int j=i+1; j<polys.size();++j)
insilmaril@792
   613
			{
insilmaril@792
   614
				if (polygonCollision (polys.at(i),polys.at(j), QPointF(0,0)).intersect )
insilmaril@792
   615
				{
insilmaril@792
   616
					collisions++;
insilmaril@793
   617
					v=polys.at(j).centroid()-polys.at(i).centroid();
insilmaril@792
   618
					// Move also away if centroids are identical
insilmaril@792
   619
					if (v.isNull()) 
insilmaril@792
   620
					{
insilmaril@793
   621
						//cout << "v==0="<<polys[i].centroid()<<polys[j].centroid()<<" "<<v<<"  "<<headings[i].toStdString()<<" - "<<headings[j].toStdString()<<"  ";
insilmaril@792
   622
						v.setX (rand()%200 -100);
insilmaril@792
   623
						v.setY (rand()%200 -100);
insilmaril@793
   624
						//cout << v;
insilmaril@792
   625
					}
insilmaril@792
   626
					v.normalize();
insilmaril@792
   627
					v.scale (2);
insilmaril@793
   628
					//cout <<  "  "<<v<<endl;
insilmaril@792
   629
					vectors[j]=v;
insilmaril@792
   630
					vectors[i]=v;
insilmaril@792
   631
					vectors[i].invert();
insilmaril@793
   632
				}  
insilmaril@792
   633
			}
insilmaril@792
   634
		}
insilmaril@792
   635
		for (int i=0;i<vectors.size();i++)
insilmaril@793
   636
		{
insilmaril@793
   637
			//cout << " v="<<vectors[i]<<" "<<headings[i].toStdString()<<endl;
insilmaril@792
   638
			polys[i].translate (vectors[i]);
insilmaril@793
   639
		}
insilmaril@792
   640
		//cout << "Collisions: "<<collisions<<endl;
insilmaril@793
   641
		//collisions=0;
insilmaril@792
   642
	}	
insilmaril@792
   643
insilmaril@792
   644
	// Finally move the real objects and update 
insilmaril@792
   645
	for (int i=0;i<polys.size();i++)
insilmaril@792
   646
	{
insilmaril@792
   647
		Vector v=polys[i].at(0)-orgpos[i];
insilmaril@792
   648
		mapobjects[i]->moveBy(v.x(),v.y() );
insilmaril@792
   649
	}	
insilmaril@792
   650
	model->reposition();	
insilmaril@105
   651
}
insilmaril@105
   652
insilmaril@787
   653
BranchItem* MapEditor::getBranchDirectAbove (BranchItem *bi)
insilmaril@787
   654
{
insilmaril@787
   655
	if (bi)
insilmaril@787
   656
	{
insilmaril@787
   657
		int i=bi->num();
insilmaril@787
   658
		if (i>0) return bi->parent()->getBranchNum(i-1);
insilmaril@787
   659
	}
insilmaril@787
   660
	return NULL;
insilmaril@787
   661
}
insilmaril@787
   662
insilmaril@787
   663
BranchItem* MapEditor::getBranchAbove (BranchItem *selbi)
insilmaril@787
   664
{
insilmaril@787
   665
	if (selbi)
insilmaril@787
   666
	{
insilmaril@787
   667
		int dz=selbi->depth();	// original depth
insilmaril@787
   668
		bool invert=false;
insilmaril@787
   669
		if (selbi->getLMO()->getOrientation()==LinkableMapObj::LeftOfCenter)
insilmaril@787
   670
			invert=true;
insilmaril@787
   671
insilmaril@787
   672
		BranchItem *bi;
insilmaril@787
   673
insilmaril@787
   674
		// Look for branch with same parent but directly above
insilmaril@787
   675
		if (dz==1 && invert)
insilmaril@787
   676
			bi=getBranchDirectBelow(selbi);
insilmaril@787
   677
		else
insilmaril@787
   678
			bi=getBranchDirectAbove (selbi);
insilmaril@787
   679
insilmaril@787
   680
		if (bi) 
insilmaril@787
   681
			// direct predecessor
insilmaril@787
   682
			return bi;
insilmaril@787
   683
insilmaril@787
   684
		// Go towards center and look for predecessor
insilmaril@787
   685
		while (selbi->depth()>0)
insilmaril@787
   686
		{
insilmaril@787
   687
			selbi=(BranchItem*)(selbi->parent());
insilmaril@787
   688
			if (selbi->depth()==1 && invert)
insilmaril@787
   689
				bi=getBranchDirectBelow (selbi);
insilmaril@787
   690
			else
insilmaril@787
   691
				bi=getBranchDirectAbove (selbi);
insilmaril@787
   692
			if (bi)
insilmaril@787
   693
			{
insilmaril@787
   694
				// turn 
insilmaril@787
   695
				selbi=bi;
insilmaril@787
   696
				while (selbi->depth()<dz)
insilmaril@787
   697
				{
insilmaril@787
   698
					// try to get back to original depth dz
insilmaril@787
   699
					bi=selbi->getLastBranch();
insilmaril@787
   700
					if (!bi) 
insilmaril@787
   701
					{
insilmaril@787
   702
						return selbi;
insilmaril@787
   703
					}
insilmaril@787
   704
					selbi=bi;
insilmaril@787
   705
				}
insilmaril@787
   706
				return selbi;
insilmaril@787
   707
			}
insilmaril@787
   708
		}
insilmaril@787
   709
	}
insilmaril@787
   710
	return NULL;
insilmaril@787
   711
}
insilmaril@787
   712
insilmaril@787
   713
BranchItem* MapEditor::getBranchDirectBelow(BranchItem *bi)
insilmaril@787
   714
{
insilmaril@787
   715
	if (bi)
insilmaril@787
   716
	{
insilmaril@787
   717
		int i=bi->num();
insilmaril@787
   718
		if (i+1<bi->parent()->branchCount()) return bi->parent()->getBranchNum(i+1);
insilmaril@787
   719
	}
insilmaril@787
   720
	return NULL;
insilmaril@787
   721
}
insilmaril@787
   722
insilmaril@787
   723
BranchItem* MapEditor::getBranchBelow (BranchItem *selbi)
insilmaril@787
   724
{
insilmaril@787
   725
	if (selbi)
insilmaril@787
   726
	{
insilmaril@787
   727
		BranchItem *bi;
insilmaril@787
   728
		int dz=selbi->depth();	// original depth
insilmaril@787
   729
		bool invert=false;
insilmaril@787
   730
		if (selbi->getLMO()->getOrientation()==LinkableMapObj::LeftOfCenter)
insilmaril@787
   731
			invert=true;
insilmaril@787
   732
insilmaril@787
   733
insilmaril@787
   734
		// Look for branch with same parent but directly below
insilmaril@787
   735
		if (dz==1 && invert)
insilmaril@787
   736
			bi=getBranchDirectAbove (selbi);
insilmaril@787
   737
		else
insilmaril@787
   738
			bi=getBranchDirectBelow (selbi);
insilmaril@787
   739
		if (bi) 
insilmaril@787
   740
			// direct successor
insilmaril@787
   741
			return bi;
insilmaril@787
   742
insilmaril@787
   743
insilmaril@787
   744
		// Go towards center and look for neighbour
insilmaril@787
   745
		while (selbi->depth()>0)
insilmaril@787
   746
		{
insilmaril@787
   747
			selbi=(BranchItem*)(selbi->parent());
insilmaril@787
   748
			if (selbi->depth()==1 && invert)
insilmaril@787
   749
				bi=getBranchDirectAbove (selbi);
insilmaril@787
   750
			else
insilmaril@787
   751
				bi=getBranchDirectBelow (selbi);
insilmaril@787
   752
			if (bi)
insilmaril@787
   753
			{
insilmaril@787
   754
				// turn 
insilmaril@787
   755
				selbi=bi;
insilmaril@787
   756
				while (selbi->depth()<dz)
insilmaril@787
   757
				{
insilmaril@787
   758
					// try to get back to original depth dz
insilmaril@787
   759
					bi=selbi->getFirstBranch();
insilmaril@787
   760
					if (!bi) 
insilmaril@787
   761
					{
insilmaril@787
   762
						return selbi;
insilmaril@787
   763
					}
insilmaril@787
   764
					selbi=bi;
insilmaril@787
   765
				}
insilmaril@787
   766
				return selbi;
insilmaril@787
   767
			}
insilmaril@787
   768
		}
insilmaril@787
   769
	}
insilmaril@787
   770
	return NULL;
insilmaril@787
   771
}
insilmaril@787
   772
insilmaril@787
   773
BranchItem* MapEditor::getLeftBranch (BranchItem *bi)
insilmaril@787
   774
{
insilmaril@787
   775
	if (bi)
insilmaril@787
   776
	{
insilmaril@787
   777
		if (bi->depth()==0)
insilmaril@787
   778
			// Special case: use alternative selection index
insilmaril@787
   779
			return bi->getLastSelectedBranchAlt();	
insilmaril@787
   780
		if (bi->getBranchObj()->getOrientation()==LinkableMapObj::RightOfCenter)	
insilmaril@787
   781
			// right of center
insilmaril@787
   782
			return (BranchItem*)(bi->parent());
insilmaril@787
   783
		else
insilmaril@787
   784
			// left of center
insilmaril@787
   785
			if (bi->getType()== TreeItem::Branch )
insilmaril@787
   786
				return bi->getLastSelectedBranch();
insilmaril@787
   787
	}
insilmaril@787
   788
	return NULL;
insilmaril@787
   789
}
insilmaril@787
   790
insilmaril@787
   791
BranchItem* MapEditor::getRightBranch(BranchItem *bi)
insilmaril@787
   792
{
insilmaril@787
   793
	if (bi)
insilmaril@787
   794
	{
insilmaril@787
   795
		if (bi->depth()==0) return bi->getLastSelectedBranch();	
insilmaril@787
   796
		if (bi->getBranchObj()->getOrientation()==LinkableMapObj::LeftOfCenter)	
insilmaril@787
   797
			// left of center
insilmaril@787
   798
			return (BranchItem*)(bi->parent());
insilmaril@787
   799
		else
insilmaril@787
   800
			// right of center
insilmaril@787
   801
			if (bi->getType()== TreeItem::Branch )
insilmaril@787
   802
				return (BranchItem*)bi->getLastSelectedBranch();
insilmaril@787
   803
	}
insilmaril@787
   804
	return NULL;
insilmaril@787
   805
}
insilmaril@787
   806
insilmaril@787
   807
insilmaril@787
   808
insilmaril@769
   809
void MapEditor::cursorUp()
insilmaril@769
   810
{
insilmaril@787
   811
	BranchItem *bi=model->getSelectedBranch();
insilmaril@787
   812
	if (bi) model->select (getBranchAbove(bi));
insilmaril@769
   813
}
insilmaril@769
   814
insilmaril@769
   815
void MapEditor::cursorDown()	
insilmaril@769
   816
insilmaril@769
   817
{
insilmaril@787
   818
	BranchItem *bi=model->getSelectedBranch();
insilmaril@787
   819
	if (bi) model->select (getBranchBelow(bi));
insilmaril@769
   820
}
insilmaril@769
   821
insilmaril@769
   822
void MapEditor::cursorLeft()
insilmaril@769
   823
{
insilmaril@787
   824
	BranchItem *bi=getLeftBranch (model->getSelectedBranch());
insilmaril@787
   825
	if (bi) model->select (bi);
insilmaril@769
   826
}
insilmaril@769
   827
insilmaril@769
   828
void MapEditor::cursorRight()	
insilmaril@769
   829
{
insilmaril@787
   830
	BranchItem *bi=getRightBranch (model->getSelectedBranch());
insilmaril@787
   831
	if (bi) model->select (bi);
insilmaril@769
   832
}
insilmaril@769
   833
insilmaril@769
   834
void MapEditor::cursorFirst()	
insilmaril@769
   835
{
insilmaril@769
   836
	model->selectFirstBranch();
insilmaril@769
   837
}
insilmaril@769
   838
insilmaril@769
   839
void MapEditor::cursorLast()	
insilmaril@769
   840
{
insilmaril@769
   841
	model->selectLastBranch();
insilmaril@769
   842
}
insilmaril@769
   843
insilmaril@769
   844
insilmaril@720
   845
void MapEditor::editHeading()
insilmaril@720
   846
{
insilmaril@720
   847
	if (editingHeading)
insilmaril@720
   848
	{
insilmaril@720
   849
		editHeadingFinished();
insilmaril@720
   850
		return;
insilmaril@720
   851
	}
insilmaril@753
   852
	BranchObj *bo=model->getSelectedBranchObj();
insilmaril@787
   853
	BranchItem *bi=model->getSelectedBranch();
insilmaril@773
   854
	if (bo)	
insilmaril@720
   855
	{
insilmaril@720
   856
		model->setSelectionBlocked(true);
insilmaril@720
   857
insilmaril@753
   858
		lineEdit->setText (bi->getHeading());
insilmaril@720
   859
		QPoint p = mapTo (this,bo->getAbsPos().toPoint() );
insilmaril@720
   860
		lineEdit->setGeometry(p.x(),p.y(),230,25);
insilmaril@720
   861
		lineEdit->selectAll();
insilmaril@720
   862
		lineEdit->show();
insilmaril@720
   863
		lineEdit->setFocus();
insilmaril@720
   864
		lineEdit->grabKeyboard();
insilmaril@720
   865
		editingHeading=true;
insilmaril@720
   866
	}
insilmaril@720
   867
insilmaril@720
   868
}
insilmaril@720
   869
void MapEditor::editHeadingFinished()
insilmaril@720
   870
{
insilmaril@720
   871
	editingHeading=false;
insilmaril@720
   872
	lineEdit->releaseKeyboard();
insilmaril@720
   873
	model->setHeading (lineEdit->text() );
insilmaril@720
   874
	model->setSelectionBlocked(false);
insilmaril@720
   875
	lineEdit->hide();
insilmaril@720
   876
insilmaril@720
   877
	// Maybe reselect previous branch 
insilmaril@720
   878
	mainWindow->editHeadingFinished (model);
insilmaril@720
   879
}
insilmaril@720
   880
insilmaril@720
   881
insilmaril@408
   882
void MapEditor::contextMenuEvent ( QContextMenuEvent * e )
insilmaril@0
   883
{
insilmaril@0
   884
	// Lineedits are already closed by preceding
insilmaril@0
   885
	// mouseEvent, we don't need to close here.
insilmaril@0
   886
insilmaril@408
   887
    QPointF p = mapToScene(e->pos());
insilmaril@754
   888
    TreeItem *ti=findMapItem (p, NULL);
insilmaril@754
   889
    LinkableMapObj* lmo=NULL;
insilmaril@779
   890
	if (ti) lmo=((MapItem*)ti)->getLMO();
insilmaril@0
   891
	
insilmaril@0
   892
    if (lmo) 
insilmaril@0
   893
	{	// MapObj was found
insilmaril@735
   894
		if (model->getSelectedLMO() != lmo)
insilmaril@0
   895
		{
insilmaril@0
   896
			// select the MapObj
insilmaril@720
   897
			model->select(lmo);
insilmaril@0
   898
		}
insilmaril@0
   899
		// Context Menu 
insilmaril@753
   900
		if (model->getSelectedBranchObj() ) 
insilmaril@0
   901
		{
insilmaril@424
   902
			// Context Menu on branch or mapcenter
insilmaril@424
   903
			branchContextMenu->popup(e->globalPos() );
insilmaril@424
   904
		} else
insilmaril@424
   905
		{
insilmaril@787
   906
			if (model->getSelectedImage() )
insilmaril@0
   907
			{
insilmaril@0
   908
				// Context Menu on floatimage
insilmaril@0
   909
				floatimageContextMenu->popup(e->globalPos() );
insilmaril@0
   910
			}	
insilmaril@0
   911
		}	
insilmaril@0
   912
	} else 
insilmaril@0
   913
	{ // No MapObj found, we are on the Canvas itself
insilmaril@408
   914
		// Context Menu on scene
insilmaril@723
   915
		
insilmaril@723
   916
		// Open context menu synchronously to position new mapcenter
insilmaril@723
   917
		model->setContextPos (p);
insilmaril@723
   918
		canvasContextMenu->exec(e->globalPos() );
insilmaril@723
   919
		model->unsetContextPos ();
insilmaril@0
   920
    } 
insilmaril@329
   921
	e->accept();
insilmaril@0
   922
}
insilmaril@0
   923
insilmaril@398
   924
void MapEditor::keyPressEvent(QKeyEvent* e)
insilmaril@398
   925
{
insilmaril@411
   926
	if (e->modifiers() & Qt::ControlModifier)
insilmaril@398
   927
	{
insilmaril@398
   928
		switch (mainWindow->getModMode())
insilmaril@398
   929
		{
insilmaril@473
   930
			case Main::ModModeColor: 
insilmaril@398
   931
				setCursor (PickColorCursor);
insilmaril@398
   932
				break;
insilmaril@473
   933
			case Main::ModModeCopy: 
insilmaril@398
   934
				setCursor (CopyCursor);
insilmaril@398
   935
				break;
insilmaril@473
   936
			case Main::ModModeXLink: 
insilmaril@398
   937
				setCursor (XLinkCursor);
insilmaril@398
   938
				break;
insilmaril@398
   939
			default :
insilmaril@398
   940
				setCursor (Qt::ArrowCursor);
insilmaril@398
   941
				break;
insilmaril@398
   942
		} 
insilmaril@398
   943
	}	
insilmaril@398
   944
}
insilmaril@398
   945
insilmaril@398
   946
void MapEditor::keyReleaseEvent(QKeyEvent* e)
insilmaril@398
   947
{
insilmaril@411
   948
	if (!(e->modifiers() & Qt::ControlModifier))
insilmaril@398
   949
		setCursor (Qt::ArrowCursor);
insilmaril@398
   950
}
insilmaril@398
   951
insilmaril@408
   952
void MapEditor::mousePressEvent(QMouseEvent* e)
insilmaril@0
   953
{
insilmaril@329
   954
	// Ignore right clicks, these will go to context menus
insilmaril@366
   955
	if (e->button() == Qt::RightButton )
insilmaril@329
   956
	{
insilmaril@329
   957
		e->ignore();
insilmaril@329
   958
		return;
insilmaril@329
   959
	}
insilmaril@329
   960
insilmaril@491
   961
	//Ignore clicks while editing heading
insilmaril@720
   962
	if (model->isSelectionBlocked() ) 
insilmaril@491
   963
	{
insilmaril@491
   964
		e->ignore();
insilmaril@491
   965
		return;
insilmaril@491
   966
	}
insilmaril@491
   967
insilmaril@408
   968
    QPointF p = mapToScene(e->pos());
insilmaril@754
   969
    TreeItem *ti=findMapItem (p, NULL);
insilmaril@754
   970
    LinkableMapObj* lmo=NULL;
insilmaril@779
   971
	if (ti) lmo=((MapItem*)ti)->getLMO();
insilmaril@0
   972
	
insilmaril@790
   973
	
insilmaril@329
   974
	e->accept();
insilmaril@329
   975
insilmaril@442
   976
	//Take care of  system flags _or_ modifier modes
insilmaril@267
   977
	//
insilmaril@773
   978
	if (lmo && ti->isBranchLikeType() )
insilmaril@267
   979
	{
insilmaril@267
   980
		QString foname=((BranchObj*)lmo)->getSystemFlagName(p);
insilmaril@267
   981
		if (!foname.isEmpty())
insilmaril@267
   982
		{
insilmaril@267
   983
			// systemFlag clicked
insilmaril@750
   984
			model->select (lmo);	// FIXME-3 was selectInt
insilmaril@267
   985
			if (foname=="url") 
insilmaril@267
   986
			{
insilmaril@366
   987
				if (e->state() & Qt::ControlModifier)
insilmaril@267
   988
					mainWindow->editOpenURLTab();
insilmaril@267
   989
				else	
insilmaril@267
   990
					mainWindow->editOpenURL();
insilmaril@267
   991
			}	
insilmaril@780
   992
			else if (foname=="system-vymLink")
insilmaril@267
   993
			{
insilmaril@267
   994
				mainWindow->editOpenVymLink();
insilmaril@267
   995
				// tabWidget may change, better return now
insilmaril@267
   996
				// before segfaulting...
insilmaril@267
   997
			} else if (foname=="note")
insilmaril@267
   998
				mainWindow->windowToggleNoteEditor();
insilmaril@267
   999
			else if (foname=="hideInExport")		
insilmaril@720
  1000
				model->toggleHideExport();
insilmaril@746
  1001
			// FIXME-3 needed? xelection.update();	
insilmaril@267
  1002
			return;	
insilmaril@267
  1003
		} 
insilmaril@720
  1004
	}	
insilmaril@442
  1005
	// No system flag clicked, take care of modmodes (CTRL-Click)
insilmaril@366
  1006
	if (e->state() & Qt::ControlModifier)
insilmaril@0
  1007
	{
insilmaril@473
  1008
		if (mainWindow->getModMode()==Main::ModModeColor)
insilmaril@83
  1009
		{
insilmaril@83
  1010
				pickingColor=true;
insilmaril@398
  1011
				setCursor (PickColorCursor);
insilmaril@83
  1012
				return;
insilmaril@83
  1013
		} 
insilmaril@473
  1014
		if (mainWindow->getModMode()==Main::ModModeXLink)
insilmaril@83
  1015
		{	
insilmaril@790
  1016
			BranchItem *bi_begin=model->getSelectedBranch();
insilmaril@790
  1017
			if (bi_begin)	
insilmaril@83
  1018
			{
insilmaril@83
  1019
				drawingLink=true;
insilmaril@790
  1020
				tmpXLink=model->createXLink(bi_begin,true);
insilmaril@720
  1021
				tmpXLink->setColor(model->getMapDefXLinkColor());
insilmaril@720
  1022
				tmpXLink->setWidth(model->getMapDefXLinkWidth());
insilmaril@790
  1023
				tmpXLink->setEnd   (p);
insilmaril@95
  1024
				tmpXLink->updateXLink();
insilmaril@83
  1025
				return;
insilmaril@89
  1026
			} 
insilmaril@267
  1027
		}
insilmaril@442
  1028
	}	// End of modmodes
insilmaril@442
  1029
insilmaril@0
  1030
    if (lmo) 
insilmaril@267
  1031
	{	
insilmaril@738
  1032
	/*
insilmaril@729
  1033
		cout << "ME::mouse pressed\n";
insilmaril@729
  1034
		cout << "  lmo="<<lmo<<endl;
insilmaril@776
  1035
		cout << "   ti="<<ti->getHeadingStd()<<endl;
insilmaril@738
  1036
	*/
insilmaril@442
  1037
		// Select the clicked object
insilmaril@729
  1038
insilmaril@776
  1039
		// Get clicked LMO
insilmaril@738
  1040
		model->select (ti);
insilmaril@442
  1041
insilmaril@0
  1042
		// Left Button	    Move Branches
insilmaril@366
  1043
		if (e->button() == Qt::LeftButton )
insilmaril@0
  1044
		{
insilmaril@424
  1045
			movingObj_start.setX( p.x() - lmo->x() );	
insilmaril@424
  1046
			movingObj_start.setY( p.y() - lmo->y() );	
insilmaril@129
  1047
			movingObj_orgPos.setX (lmo->x() );
insilmaril@129
  1048
			movingObj_orgPos.setY (lmo->y() );
insilmaril@720
  1049
			lmo->setRelPos();
insilmaril@366
  1050
			movingObj_orgRelPos=lmo->getRelPos();
insilmaril@129
  1051
insilmaril@93
  1052
			// If modMode==copy, then we want to "move" the _new_ object around
insilmaril@93
  1053
			// then we need the offset from p to the _old_ selection, because of tmp
insilmaril@473
  1054
			if (mainWindow->getModMode()==Main::ModModeCopy &&
insilmaril@366
  1055
				e->state() & Qt::ControlModifier)
insilmaril@93
  1056
			{
insilmaril@787
  1057
				BranchItem *bi=model->getSelectedBranch();
insilmaril@750
  1058
				if (bi)
insilmaril@93
  1059
				{
insilmaril@93
  1060
					copyingObj=true;
insilmaril@750
  1061
					//FIXME-2   TreeItem::addBranch (BranchItem still missing) 
insilmaril@787
  1062
					//bi->addBranch (model->getSelectedBranch());
insilmaril@720
  1063
					model->unselect();
insilmaril@750
  1064
					model->select(bi->getLastBranch());
insilmaril@645
  1065
					model->reposition();
insilmaril@93
  1066
				}
insilmaril@421
  1067
			} 
insilmaril@421
  1068
insilmaril@735
  1069
			movingObj=model->getSelectedLMO();	
insilmaril@0
  1070
		} else
insilmaril@0
  1071
			// Middle Button    Toggle Scroll
insilmaril@0
  1072
			// (On Mac OS X this won't work, but we still have 
insilmaril@0
  1073
			// a button in the toolbar)
insilmaril@366
  1074
			if (e->button() == Qt::MidButton )
insilmaril@720
  1075
				model->toggleScroll();
insilmaril@746
  1076
		// model->updateActions(); FIXME-3 needed?
insilmaril@746
  1077
		// FIXME-3 needed? xelection.update();
insilmaril@0
  1078
	} else 
insilmaril@408
  1079
	{ // No MapObj found, we are on the scene itself
insilmaril@408
  1080
		// Left Button	    move Pos of sceneView
insilmaril@366
  1081
		if (e->button() == Qt::LeftButton )
insilmaril@0
  1082
		{
insilmaril@0
  1083
			movingObj=NULL;	// move Content not Obj
insilmaril@0
  1084
			movingObj_start=e->globalPos();
insilmaril@408
  1085
			movingCont_start=QPointF (
insilmaril@408
  1086
				horizontalScrollBar()->value(),
insilmaril@408
  1087
				verticalScrollBar()->value());
insilmaril@408
  1088
			movingVec=QPointF(0,0);
insilmaril@398
  1089
			setCursor(HandOpenCursor);
insilmaril@0
  1090
		} 
insilmaril@0
  1091
    } 
insilmaril@0
  1092
}
insilmaril@0
  1093
insilmaril@408
  1094
void MapEditor::mouseMoveEvent(QMouseEvent* e)
insilmaril@0
  1095
{
insilmaril@408
  1096
    QPointF p = mapToScene(e->pos());
insilmaril@754
  1097
	TreeItem *seli=model->getSelectedItem();
insilmaril@777
  1098
	LinkableMapObj* lmosel=NULL;	
insilmaril@776
  1099
	if (seli && (seli->isBranchLikeType() ||seli->getType()==TreeItem::Image))
insilmaril@779
  1100
		lmosel=((MapItem*)seli)->getLMO();
insilmaril@83
  1101
insilmaril@0
  1102
    // Move the selected MapObj
insilmaril@424
  1103
    if ( lmosel && movingObj) 
insilmaril@0
  1104
    {	
insilmaril@421
  1105
		// reset cursor if we are moving and don't copy
insilmaril@473
  1106
		if (mainWindow->getModMode()!=Main::ModModeCopy)
insilmaril@421
  1107
			setCursor (Qt::ArrowCursor);
insilmaril@421
  1108
insilmaril@408
  1109
		// To avoid jumping of the sceneView, only 
insilmaril@763
  1110
		// show selection, if not tmp linked
insilmaril@424
  1111
		if (!lmosel->hasParObjTmp())
insilmaril@763
  1112
			model->emitShowSelection();
insilmaril@0
  1113
		
insilmaril@83
  1114
		// Now move the selection, but add relative position 
insilmaril@83
  1115
		// (movingObj_start) where selection was chosen with 
insilmaril@83
  1116
		// mousepointer. (This avoids flickering resp. jumping 
insilmaril@0
  1117
		// of selection back to absPos)
insilmaril@0
  1118
		
insilmaril@0
  1119
		// Check if we could link 
insilmaril@777
  1120
		TreeItem *ti=findMapItem (p, seli);
insilmaril@777
  1121
		BranchItem *dsti=NULL;
insilmaril@754
  1122
		LinkableMapObj* dst=NULL;
insilmaril@777
  1123
		if (ti && ti!=seli && ti->isBranchLikeType())
insilmaril@777
  1124
		{
insilmaril@777
  1125
			dsti=(BranchItem*)ti;
insilmaril@777
  1126
			dst=dsti->getLMO(); 
insilmaril@777
  1127
		} else
insilmaril@772
  1128
			dsti=NULL;
insilmaril@0
  1129
		
insilmaril@0
  1130
insilmaril@777
  1131
		if (lmosel && seli->getType()==TreeItem::Image)	
insilmaril@0
  1132
		{
insilmaril@776
  1133
			FloatObj *fio=(FloatImageObj*)lmosel;
insilmaril@424
  1134
			fio->move   (p.x() -movingObj_start.x(), p.y()-movingObj_start.y() );		
insilmaril@424
  1135
			fio->setRelPos();
insilmaril@779
  1136
			fio->updateLinkGeometry(); //no need for reposition, if we update link here
insilmaril@775
  1137
			model->emitSelectionChanged();	// position has changed
insilmaril@0
  1138
insilmaril@0
  1139
			// Relink float to new mapcenter or branch, if shift is pressed	
insilmaril@0
  1140
			// Only relink, if selection really has a new parent
insilmaril@777
  1141
			if ( e->modifiers()==Qt::ShiftModifier && dsti &&  dsti != seli->parent()  )
insilmaril@0
  1142
			{
insilmaril@773
  1143
				// Also save the move which was done so far
insilmaril@792
  1144
				QString pold=qpointFToString(movingObj_orgRelPos);
insilmaril@792
  1145
				QString pnow=qpointFToString(fio->getRelPos());
insilmaril@773
  1146
				model->saveState(
insilmaril@777
  1147
					seli,
insilmaril@773
  1148
					"moveRel "+pold,
insilmaril@777
  1149
					seli,
insilmaril@773
  1150
					"moveRel "+pnow,
insilmaril@773
  1151
					QString("Move %1 to relative position %2").arg(model->getObjectName(fio)).arg(pnow));
insilmaril@773
  1152
				fio->getParObj()->requestReposition();
insilmaril@773
  1153
				model->reposition();
insilmaril@420
  1154
insilmaril@777
  1155
				model->relinkImage ((ImageItem*) seli,dsti);
insilmaril@777
  1156
				model->select (seli);
insilmaril@777
  1157
				//movingObj=lmosel;	//FIXME-3
insilmaril@773
  1158
				//movingObj_orgRelPos=lmosel->getRelPos();	
insilmaril@420
  1159
insilmaril@773
  1160
				model->reposition();
insilmaril@0
  1161
			}
insilmaril@388
  1162
		} else	
insilmaril@388
  1163
		{	// selection != a FloatObj
insilmaril@755
  1164
			if (seli->depth()==0)		//FIXME-1 also moved mapcenters could be linked, but not working here...
insilmaril@0
  1165
			{
insilmaril@388
  1166
				// Move MapCenter
insilmaril@388
  1167
				if (e->buttons()== Qt::LeftButton && e->modifiers()==Qt::ShiftModifier) 
insilmaril@773
  1168
					((BranchObj*)lmosel)->moveBy(
insilmaril@773
  1169
						QPointF(p.x() -movingObj_start.x(), 
insilmaril@773
  1170
						p.y()-movingObj_start.y()) );		
insilmaril@0
  1171
				else	
insilmaril@645
  1172
					lmosel->move   (p.x() -movingObj_start.x(), p.y()-movingObj_start.y() );		
insilmaril@645
  1173
				model->updateRelPositions();
insilmaril@0
  1174
			} else
insilmaril@0
  1175
			{	
insilmaril@755
  1176
				if (seli->depth()==1)
insilmaril@0
  1177
				{
insilmaril@388
  1178
					// Move mainbranch
insilmaril@0
  1179
					lmosel->move(p.x() -movingObj_start.x(), p.y()-movingObj_start.y() );		
insilmaril@388
  1180
					lmosel->setRelPos();
insilmaril@0
  1181
				} else
insilmaril@0
  1182
				{
insilmaril@388
  1183
					// Move ordinary branch
insilmaril@473
  1184
					if (lmosel->getOrientation() == LinkableMapObj::LeftOfCenter)
insilmaril@0
  1185
						// Add width of bbox here, otherwise alignRelTo will cause jumping around
insilmaril@720
  1186
						lmosel->move(p.x() -movingObj_start.x() , //lmosel->getBBox().width(), 
insilmaril@175
  1187
							p.y()-movingObj_start.y() +lmosel->getTopPad() );		
insilmaril@0
  1188
					else	
insilmaril@175
  1189
						lmosel->move(p.x() -movingObj_start.x(), p.y()-movingObj_start.y() -lmosel->getTopPad());
insilmaril@720
  1190
					lmosel->setRelPos();	
insilmaril@0
  1191
				} 
insilmaril@0
  1192
insilmaril@431
  1193
				// Maybe we can relink temporary?
insilmaril@772
  1194
				if (dsti)
insilmaril@0
  1195
				{
insilmaril@388
  1196
					if (e->modifiers()==Qt::ControlModifier)
insilmaril@0
  1197
					{
insilmaril@754
  1198
						// Special case: CTRL to link below dst
insilmaril@754
  1199
						lmosel->setParObjTmp (dst,p,+1);
insilmaril@0
  1200
					}
insilmaril@388
  1201
					else if (e->modifiers()==Qt::ShiftModifier)
insilmaril@754
  1202
						lmosel->setParObjTmp (dst,p,-1);
insilmaril@0
  1203
					else
insilmaril@754
  1204
						lmosel->setParObjTmp (dst,p,0);
insilmaril@0
  1205
				} else	
insilmaril@0
  1206
				{
insilmaril@93
  1207
					lmosel->unsetParObjTmp();
insilmaril@0
  1208
				}		
insilmaril@428
  1209
				// reposition subbranch
insilmaril@428
  1210
				lmosel->reposition();	
insilmaril@0
  1211
			} // depth>0
insilmaril@0
  1212
insilmaril@732
  1213
			QItemSelection sel=model->getSelectionModel()->selection();
insilmaril@735
  1214
			updateSelection(sel,sel);	// position has changed
insilmaril@732
  1215
insilmaril@0
  1216
		} // no FloatImageObj
insilmaril@0
  1217
insilmaril@408
  1218
		scene()->update();
insilmaril@0
  1219
		return;
insilmaril@0
  1220
	} // selection && moving_obj
insilmaril@0
  1221
		
insilmaril@83
  1222
	// Draw a link from one branch to another
insilmaril@83
  1223
	if (drawingLink)
insilmaril@83
  1224
	{
insilmaril@95
  1225
		 tmpXLink->setEnd (p);
insilmaril@95
  1226
		 tmpXLink->updateXLink();
insilmaril@83
  1227
	}	 
insilmaril@83
  1228
	
insilmaril@408
  1229
    // Move sceneView 
insilmaril@411
  1230
    if (!movingObj && !pickingColor &&!drawingLink && e->buttons() == Qt::LeftButton ) 
insilmaril@0
  1231
	{
insilmaril@408
  1232
		QPointF p=e->globalPos();
insilmaril@0
  1233
		movingVec.setX(-p.x() + movingObj_start.x() );
insilmaril@0
  1234
		movingVec.setY(-p.y() + movingObj_start.y() );
insilmaril@408
  1235
		horizontalScrollBar()->setSliderPosition((int)( movingCont_start.x()+movingVec.x() ));
insilmaril@408
  1236
		verticalScrollBar()->setSliderPosition((int)( movingCont_start.y()+movingVec.y() ) );
insilmaril@0
  1237
    }
insilmaril@0
  1238
}
insilmaril@0
  1239
insilmaril@0
  1240
insilmaril@408
  1241
void MapEditor::mouseReleaseEvent(QMouseEvent* e)
insilmaril@0
  1242
{
insilmaril@408
  1243
    QPointF p = mapToScene(e->pos());
insilmaril@754
  1244
	TreeItem *seli=model->getSelectedItem();
insilmaril@754
  1245
insilmaril@772
  1246
	TreeItem *dsti=NULL;
insilmaril@772
  1247
	if (seli) dsti=findMapItem(p, seli);
insilmaril@755
  1248
	LinkableMapObj* dst=NULL;
insilmaril@772
  1249
	if (dsti && dsti->isBranchLikeType ()) 
insilmaril@779
  1250
		dst=((MapItem*)dsti)->getLMO();	
insilmaril@772
  1251
	else
insilmaril@772
  1252
		dsti=NULL;
insilmaril@754
  1253
insilmaril@754
  1254
insilmaril@0
  1255
	// Have we been picking color?
insilmaril@0
  1256
	if (pickingColor)
insilmaril@0
  1257
	{
insilmaril@0
  1258
		pickingColor=false;
insilmaril@366
  1259
		setCursor (Qt::ArrowCursor);
insilmaril@0
  1260
		// Check if we are over another branch
insilmaril@772
  1261
		if (dst) 
insilmaril@0
  1262
		{	
insilmaril@366
  1263
			if (e->state() & Qt::ShiftModifier)
insilmaril@720
  1264
				model->colorBranch (((BranchObj*)dst)->getColor());
insilmaril@0
  1265
			else	
insilmaril@720
  1266
				model->colorSubtree (((BranchObj*)dst)->getColor());
insilmaril@0
  1267
		} 
insilmaril@0
  1268
		return;
insilmaril@0
  1269
	}
insilmaril@83
  1270
insilmaril@83
  1271
	// Have we been drawing a link?
insilmaril@491
  1272
	if (drawingLink)	
insilmaril@83
  1273
	{
insilmaril@83
  1274
		drawingLink=false;
insilmaril@83
  1275
		// Check if we are over another branch
insilmaril@772
  1276
		if (dsti)
insilmaril@83
  1277
		{	
insilmaril@790
  1278
			tmpXLink->setEnd ( ((BranchItem*)dsti) );
insilmaril@95
  1279
			tmpXLink->updateXLink();
insilmaril@790
  1280
			tmpXLink->activate(); 
insilmaril@790
  1281
			//FIXME-0 model->saveStateComplete(QString("Activate xLink from %1 to %2").arg(model->getObjectName(tmpXLink->getBegin())).arg(model->getObjectName(tmpXLink->getEnd())) );	
insilmaril@83
  1282
		} else
insilmaril@83
  1283
		{
insilmaril@790
  1284
			model->deleteItem(tmpXLink);
insilmaril@95
  1285
			tmpXLink=NULL;
insilmaril@83
  1286
		}
insilmaril@83
  1287
		return;
insilmaril@83
  1288
	}
insilmaril@83
  1289
	
insilmaril@0
  1290
    // Have we been moving something?
insilmaril@754
  1291
    if ( seli && movingObj ) 
insilmaril@0
  1292
    {	
insilmaril@776
  1293
		if (seli->getType()==TreeItem::Image)
insilmaril@175
  1294
		{
insilmaril@779
  1295
			FloatImageObj *fio=(FloatImageObj*)( ((MapItem*)seli)->getLMO());
insilmaril@776
  1296
			if(fio)
insilmaril@776
  1297
			{
insilmaril@776
  1298
				// Moved FloatObj. Maybe we need to reposition
insilmaril@792
  1299
				QString pold=qpointFToString(movingObj_orgRelPos);
insilmaril@792
  1300
				QString pnow=qpointFToString(fio->getRelPos());
insilmaril@776
  1301
				model->saveState(
insilmaril@776
  1302
					seli,
insilmaril@776
  1303
					"moveRel "+pold,
insilmaril@776
  1304
					seli,
insilmaril@776
  1305
					"moveRel "+pnow,
insilmaril@776
  1306
					QString("Move %1 to relative position %2").arg(model->getObjectName(seli)).arg(pnow));
insilmaril@424
  1307
insilmaril@785
  1308
				cout << "ME::release mouse\n";
insilmaril@776
  1309
				fio->getParObj()->requestReposition();
insilmaril@776
  1310
				model->reposition();
insilmaril@776
  1311
			}	
insilmaril@776
  1312
		}
insilmaril@175
  1313
insilmaril@787
  1314
		BranchItem *bi=model->getSelectedBranch();
insilmaril@753
  1315
		if (bi && bi->depth()==0)
insilmaril@497
  1316
		{	
insilmaril@753
  1317
            if (movingObj_orgPos != bi->getBranchObj()->getAbsPos())	// FIXME-3 check getBO here...
insilmaril@625
  1318
            {
insilmaril@792
  1319
                QString pold=qpointFToString(movingObj_orgPos);
insilmaril@792
  1320
                QString pnow=qpointFToString(bi->getBranchObj()->getAbsPos());		// FIXME-3 check getBO here...
insilmaril@753
  1321
insilmaril@720
  1322
                model->saveState(
insilmaril@753
  1323
                    bi,
insilmaril@625
  1324
                    "move "+pold,
insilmaril@753
  1325
                    bi,
insilmaril@625
  1326
                    "move "+pnow,
insilmaril@753
  1327
                    QString("Move mapcenter %1 to position %2").arg(model->getObjectName(bi)).arg(pnow));
insilmaril@625
  1328
            }
insilmaril@388
  1329
		}
insilmaril@645
  1330
	
insilmaril@773
  1331
		if (seli->isBranchLikeType() ) //(seli->getType() == TreeItem::Branch )
insilmaril@388
  1332
		{	// A branch was moved
insilmaril@777
  1333
			LinkableMapObj* lmosel=NULL;		
insilmaril@779
  1334
			lmosel=((MapItem*)seli)->getLMO();
insilmaril@760
  1335
				
insilmaril@0
  1336
			// save the position in case we link to mapcenter
insilmaril@424
  1337
			QPointF savePos=QPointF (lmosel->getAbsPos()  );
insilmaril@0
  1338
insilmaril@0
  1339
			// Reset the temporary drawn link to the original one
insilmaril@424
  1340
			lmosel->unsetParObjTmp();
insilmaril@0
  1341
insilmaril@366
  1342
			// For Redo we may need to save original selection
insilmaril@773
  1343
			QString preSelStr=model->getSelectString(seli);
insilmaril@93
  1344
insilmaril@94
  1345
			copyingObj=false;	
insilmaril@772
  1346
			if (dsti)
insilmaril@93
  1347
			{
insilmaril@691
  1348
				// We have a destination, relink to that
insilmaril@691
  1349
insilmaril@753
  1350
				BranchObj* bsel=model->getSelectedBranchObj();
insilmaril@366
  1351
insilmaril@684
  1352
				QString preParStr=model->getSelectString (bsel->getParObj());
insilmaril@755
  1353
				QString preNum=QString::number (seli->num(),10);
insilmaril@366
  1354
				QString preDstParStr;
insilmaril@767
  1355
				bool relinked;
insilmaril@366
  1356
insilmaril@366
  1357
				if (e->state() & Qt::ShiftModifier && dst->getParObj())
insilmaril@366
  1358
				{	// Link above dst
insilmaril@684
  1359
					preDstParStr=model->getSelectString (dst->getParObj());
insilmaril@767
  1360
					relinked=model->relinkBranch ((BranchItem*)seli,(BranchItem*)dsti->parent(),((BranchItem*)dsti)->num());
insilmaril@366
  1361
				} else 
insilmaril@366
  1362
				if (e->state() & Qt::ControlModifier && dst->getParObj())
insilmaril@366
  1363
				{
insilmaril@366
  1364
					// Link below dst
insilmaril@684
  1365
					preDstParStr=model->getSelectString (dst->getParObj());
insilmaril@767
  1366
					relinked=model->relinkBranch ((BranchItem*)seli,(BranchItem*)dsti->parent(),((BranchItem*)dsti)->num()+1);
insilmaril@366
  1367
				} else	
insilmaril@366
  1368
				{	// Append to dst
insilmaril@684
  1369
					preDstParStr=model->getSelectString(dst);
insilmaril@767
  1370
					relinked=model->relinkBranch ((BranchItem*)seli,(BranchItem*)dsti);
insilmaril@755
  1371
					if (dsti->depth()==0) bsel->move (savePos);
insilmaril@366
  1372
				} 
insilmaril@767
  1373
				if (relinked)
insilmaril@767
  1374
				{
insilmaril@767
  1375
					QString postSelStr=model->getSelectString(lmosel);
insilmaril@767
  1376
					QString postNum=QString::number (seli->num(),10);
insilmaril@366
  1377
insilmaril@776
  1378
					QString undoCom="relinkTo (\""+ 
insilmaril@767
  1379
						preParStr+ "\"," + preNum  +"," + 
insilmaril@767
  1380
						QString ("%1,%2").arg(movingObj_orgPos.x()).arg(movingObj_orgPos.y())+ ")";
insilmaril@366
  1381
insilmaril@776
  1382
					QString redoCom="relinkTo (\""+ 
insilmaril@767
  1383
						preDstParStr + "\"," + postNum + "," +
insilmaril@767
  1384
						QString ("%1,%2").arg(savePos.x()).arg(savePos.y())+ ")";
insilmaril@366
  1385
insilmaril@767
  1386
					model->saveState (
insilmaril@767
  1387
						postSelStr,undoCom,
insilmaril@767
  1388
						preSelStr, redoCom,
insilmaril@767
  1389
						QString("Relink %1 to %2").arg(model->getObjectName(bsel)).arg(model->getObjectName(dst)) );
insilmaril@691
  1390
insilmaril@767
  1391
				}
insilmaril@163
  1392
			} else
insilmaril@691
  1393
			{
insilmaril@691
  1394
				// No destination, undo  temporary move
insilmaril@691
  1395
insilmaril@755
  1396
				if (seli->depth()==1)
insilmaril@366
  1397
				{
insilmaril@366
  1398
					// The select string might be different _after_ moving around.
insilmaril@366
  1399
					// Therefor reposition and then use string of old selection, too
insilmaril@645
  1400
					model->reposition();
insilmaril@366
  1401
insilmaril@625
  1402
                    QPointF rp(lmosel->getRelPos());
insilmaril@625
  1403
                    if (rp != movingObj_orgRelPos)
insilmaril@625
  1404
                    {
insilmaril@792
  1405
                        QString ps=qpointFToString(rp);
insilmaril@720
  1406
                        model->saveState(
insilmaril@792
  1407
                            model->getSelectString(lmosel), "moveRel "+qpointFToString(movingObj_orgRelPos), 
insilmaril@625
  1408
                            preSelStr, "moveRel "+ps, 
insilmaril@726
  1409
                            QString("Move %1 to relative position %2").arg(model->getObjectName(lmosel)).arg(ps));
insilmaril@625
  1410
                    }
insilmaril@366
  1411
				}
insilmaril@691
  1412
insilmaril@691
  1413
				// Draw the original link, before selection was moved around
insilmaril@755
  1414
				if (settings.value("/animation/use",false).toBool() && seli->depth()>1) 
insilmaril@691
  1415
				{
insilmaril@720
  1416
					lmosel->setRelPos();	// calc relPos first for starting point
insilmaril@753
  1417
					QPointF dst=bi->getBranchObj()->getParObj()->getChildPos();		// FIXME-3 check getBO here...
insilmaril@720
  1418
			//		if (lmosel->getOrientation()==LinkableMapObj::LeftOfCenter) dst.setX (dst.x()+lmosel->width() );
insilmaril@720
  1419
					
insilmaril@691
  1420
					model->startAnimation(
insilmaril@723
  1421
						(BranchObj*)lmosel,
insilmaril@691
  1422
						lmosel->getRelPos(),
insilmaril@720
  1423
						movingObj_orgRelPos
insilmaril@720
  1424
//						QPointF (movingObj_orgPos.x() - dst.x(), movingObj_orgPos.y() - dst.y() )
insilmaril@691
  1425
					);	
insilmaril@691
  1426
				} else	
insilmaril@691
  1427
					model->reposition();
insilmaril@691
  1428
			}
insilmaril@0
  1429
		}
insilmaril@775
  1430
		model->emitSelectionChanged();  //FIXME-3 needed? at least not after pos of selection has changed...
insilmaril@408
  1431
		// Finally resize scene, if needed
insilmaril@408
  1432
		scene()->update();
insilmaril@0
  1433
		movingObj=NULL;		
insilmaril@366
  1434
insilmaril@366
  1435
		// Just make sure, that actions are still ok,e.g. the move branch up/down buttons...
insilmaril@746
  1436
		// model->updateActions(); FIXME-3 neeeded? 
insilmaril@0
  1437
	} else 
insilmaril@83
  1438
		// maybe we moved View: set old cursor
insilmaril@366
  1439
		setCursor (Qt::ArrowCursor);
insilmaril@83
  1440
    
insilmaril@0
  1441
}
insilmaril@0
  1442
insilmaril@408
  1443
void MapEditor::mouseDoubleClickEvent(QMouseEvent* e)
insilmaril@0
  1444
{
insilmaril@720
  1445
	if (model->isSelectionBlocked() ) 
insilmaril@491
  1446
	{
insilmaril@491
  1447
		e->ignore();
insilmaril@491
  1448
		return;
insilmaril@491
  1449
	}
insilmaril@491
  1450
insilmaril@366
  1451
	if (e->button() == Qt::LeftButton )
insilmaril@0
  1452
	{
insilmaril@408
  1453
		QPointF p = mapToScene(e->pos());
insilmaril@754
  1454
		TreeItem *ti=findMapItem (p, NULL);
insilmaril@754
  1455
		if (ti) {	// MapObj was found
insilmaril@0
  1456
			// First select the MapObj than edit heading
insilmaril@754
  1457
			model->select (ti);
insilmaril@720
  1458
			editHeading();
insilmaril@0
  1459
		}
insilmaril@0
  1460
	}
insilmaril@0
  1461
}
insilmaril@0
  1462
insilmaril@0
  1463
void MapEditor::resizeEvent (QResizeEvent* e)
insilmaril@0
  1464
{
insilmaril@408
  1465
	QGraphicsView::resizeEvent( e );
insilmaril@0
  1466
}
insilmaril@0
  1467
insilmaril@412
  1468
void MapEditor::dragEnterEvent(QDragEnterEvent *event)
insilmaril@119
  1469
{
insilmaril@417
  1470
	//for (unsigned int i=0;event->format(i);i++) // Debug mime type
insilmaril@417
  1471
	//	cerr << event->format(i) << endl;
insilmaril@417
  1472
insilmaril@417
  1473
	if (event->mimeData()->hasImage())
insilmaril@417
  1474
		event->acceptProposedAction();
insilmaril@417
  1475
	else	
insilmaril@417
  1476
		if (event->mimeData()->hasUrls())
insilmaril@417
  1477
			event->acceptProposedAction();
insilmaril@417
  1478
}
insilmaril@417
  1479
insilmaril@531
  1480
void MapEditor::dragMoveEvent(QDragMoveEvent *)
insilmaril@417
  1481
{
insilmaril@417
  1482
}
insilmaril@417
  1483
insilmaril@417
  1484
void MapEditor::dragLeaveEvent(QDragLeaveEvent *event)
insilmaril@417
  1485
{
insilmaril@417
  1486
	event->accept();
insilmaril@119
  1487
}
insilmaril@129
  1488
insilmaril@412
  1489
void MapEditor::dropEvent(QDropEvent *event)
insilmaril@129
  1490
{
insilmaril@787
  1491
	BranchItem *selbi=model->getSelectedBranch();
insilmaril@753
  1492
	if (selbi)
insilmaril@128
  1493
	{
insilmaril@505
  1494
		if (debug)
insilmaril@505
  1495
			foreach (QString format,event->mimeData()->formats()) 
insilmaril@658
  1496
				cout << "MapEditor: Dropped format: "<<qPrintable (format)<<endl;
insilmaril@497
  1497
insilmaril@497
  1498
insilmaril@417
  1499
		QList <QUrl> uris;
insilmaril@417
  1500
		if (event->mimeData()->hasImage()) 
insilmaril@128
  1501
		{
insilmaril@417
  1502
			 QVariant imageData = event->mimeData()->imageData();
insilmaril@720
  1503
			 model->addFloatImage (qvariant_cast<QPixmap>(imageData));
insilmaril@417
  1504
		} else
insilmaril@417
  1505
		if (event->mimeData()->hasUrls())
insilmaril@417
  1506
			uris=event->mimeData()->urls();
insilmaril@417
  1507
insilmaril@417
  1508
		if (uris.count()>0)
insilmaril@128
  1509
		{
insilmaril@417
  1510
			QStringList files;
insilmaril@417
  1511
			QString s;
insilmaril@417
  1512
			QString heading;
insilmaril@753
  1513
			BranchItem *bi;
insilmaril@428
  1514
			for (int i=0; i<uris.count();i++)
insilmaril@408
  1515
			{
insilmaril@428
  1516
				// Workaround to avoid adding empty branches
insilmaril@428
  1517
				if (!uris.at(i).toString().isEmpty())
insilmaril@417
  1518
				{
insilmaril@775
  1519
					bi=model->addNewBranch();
insilmaril@753
  1520
					if (bi)
insilmaril@417
  1521
					{
insilmaril@753
  1522
						   /* FIXME-2 
insilmaril@428
  1523
						s=uris.at(i).toLocalFile();
insilmaril@428
  1524
						if (!s.isEmpty()) 
insilmaril@428
  1525
						{
insilmaril@625
  1526
						   QString file = QDir::fromNativeSeparators(s);
insilmaril@428
  1527
						   heading = QFileInfo(file).baseName();
insilmaril@428
  1528
						   files.append(file);
insilmaril@428
  1529
						   if (file.endsWith(".vym", false))
insilmaril@753
  1530
							   bi->setVymLink(file);
insilmaril@428
  1531
						   else
insilmaril@753
  1532
							   bi->setURL(uris.at(i).toString());
insilmaril@428
  1533
					   } else 
insilmaril@428
  1534
					   {
insilmaril@428
  1535
						   bo->setURL(uris.at(i).toString());
insilmaril@428
  1536
					   }
insilmaril@753
  1537
							 */  
insilmaril@428
  1538
insilmaril@428
  1539
					   if (!heading.isEmpty())
insilmaril@753
  1540
						   bi->setHeading(heading);
insilmaril@428
  1541
					   else
insilmaril@753
  1542
						   bi->setHeading(uris.at(i).toString());
insilmaril@753
  1543
						   
insilmaril@428
  1544
					}
insilmaril@417
  1545
				}
insilmaril@417
  1546
			}
insilmaril@645
  1547
			model->reposition();
insilmaril@417
  1548
		}
insilmaril@417
  1549
	}	
insilmaril@417
  1550
	event->acceptProposedAction();
insilmaril@417
  1551
}
insilmaril@417
  1552
insilmaril@776
  1553
void MapEditor::updateSelection(QItemSelection newsel,QItemSelection oldsel)
insilmaril@728
  1554
{
insilmaril@779
  1555
	// Note: Here we are prepared for multiple selections, though this 
insilmaril@779
  1556
	// is not yet implemented elsewhere
insilmaril@779
  1557
insilmaril@776
  1558
	// Here in MapEditor we can only select Branches and Images
insilmaril@786
  1559
	QList <TreeItem*> treeItemsNew;
insilmaril@786
  1560
	QList <TreeItem*> treeItemsOld;
insilmaril@786
  1561
insilmaril@776
  1562
	QModelIndex ix;
insilmaril@776
  1563
	foreach (ix,newsel.indexes() )
insilmaril@776
  1564
	{
insilmaril@776
  1565
		TreeItem *ti= static_cast<TreeItem*>(ix.internalPointer());
insilmaril@787
  1566
		if (ti->isBranchLikeType() || ti->getType()==TreeItem::Image )
insilmaril@786
  1567
			if (!treeItemsNew.contains(ti)) treeItemsNew.append (ti);
insilmaril@786
  1568
	}
insilmaril@786
  1569
	foreach (ix,oldsel.indexes() )
insilmaril@786
  1570
	{
insilmaril@786
  1571
		TreeItem *ti= static_cast<TreeItem*>(ix.internalPointer());
insilmaril@787
  1572
		if (ti->isBranchLikeType() || ti->getType()==TreeItem::Image )
insilmaril@786
  1573
			if (!treeItemsOld.contains(ti)) treeItemsOld.append (ti);
insilmaril@776
  1574
	}
insilmaril@776
  1575
insilmaril@779
  1576
	// Trim list of selection rectangles 
insilmaril@786
  1577
	while (treeItemsNew.count() < selboxList.count() )
insilmaril@776
  1578
		delete selboxList.takeFirst();
insilmaril@776
  1579
insilmaril@775
  1580
	// Take care to tmp scroll/unscroll
insilmaril@775
  1581
	if (!oldsel.isEmpty())
insilmaril@775
  1582
	{
insilmaril@775
  1583
		QModelIndex ix=oldsel.indexes().first(); 
insilmaril@775
  1584
		if (ix.isValid() )
insilmaril@775
  1585
		{
insilmaril@775
  1586
			TreeItem *ti= static_cast<TreeItem*>(ix.internalPointer());
insilmaril@779
  1587
			if (ti)
insilmaril@775
  1588
			{
insilmaril@779
  1589
				if (ti->isBranchLikeType() )
insilmaril@779
  1590
				{
insilmaril@779
  1591
					// reset tmp scrolled branches
insilmaril@779
  1592
					BranchItem *bi=(BranchItem*)ti;
insilmaril@779
  1593
					bi->resetTmpUnscroll();
insilmaril@779
  1594
				}
insilmaril@779
  1595
				if (ti->isBranchLikeType() || ti->getType()==TreeItem::Image)
insilmaril@779
  1596
					// Hide link if not needed
insilmaril@779
  1597
					((MapItem*)ti)->getLMO()->updateVisibility();
insilmaril@775
  1598
			}
insilmaril@775
  1599
		}
insilmaril@775
  1600
	}
insilmaril@775
  1601
insilmaril@786
  1602
	if (!treeItemsNew.isEmpty())
insilmaril@775
  1603
	{
insilmaril@775
  1604
		QModelIndex ix=newsel.indexes().first(); 
insilmaril@775
  1605
		if (ix.isValid() )
insilmaril@775
  1606
		{
insilmaril@775
  1607
			// Temporary unscroll if necessary
insilmaril@775
  1608
			TreeItem *ti= static_cast<TreeItem*>(ix.internalPointer());
insilmaril@775
  1609
			if (ti->isBranchLikeType() )
insilmaril@775
  1610
			{
insilmaril@775
  1611
				BranchItem *bi=(BranchItem*)ti;
insilmaril@775
  1612
				if (bi->hasScrolledParent(bi) )
insilmaril@775
  1613
					bi->tmpUnscroll();
insilmaril@775
  1614
			}
insilmaril@775
  1615
			scrollTo (ix);
insilmaril@779
  1616
			if (ti->isBranchLikeType() || ti->getType()==TreeItem::Image)
insilmaril@779
  1617
				// Show link if needed
insilmaril@779
  1618
				((MapItem*)ti)->getLMO()->updateVisibility();
insilmaril@775
  1619
		}
insilmaril@775
  1620
	}
insilmaril@775
  1621
insilmaril@731
  1622
	// Reduce rectangles
insilmaril@786
  1623
	while (treeItemsNew.count() < selboxList.count() )
insilmaril@731
  1624
		delete selboxList.takeFirst();
insilmaril@731
  1625
insilmaril@731
  1626
	// Add additonal rectangles
insilmaril@731
  1627
	QGraphicsRectItem *sb;
insilmaril@786
  1628
	while (treeItemsNew.count() > selboxList.count() )
insilmaril@728
  1629
	{
insilmaril@731
  1630
		sb = mapScene->addRect(
insilmaril@731
  1631
			QRectF(0,0,0,0), 
insilmaril@731
  1632
			QPen(selectionColor),
insilmaril@731
  1633
			selectionColor);
insilmaril@731
  1634
		sb->setZValue(Z_SELBOX);
insilmaril@731
  1635
		sb->show();
insilmaril@731
  1636
		selboxList.append (sb);
insilmaril@728
  1637
	}
insilmaril@731
  1638
insilmaril@786
  1639
insilmaril@786
  1640
insilmaril@731
  1641
	// Reposition rectangles
insilmaril@731
  1642
	QRectF bbox;
insilmaril@732
  1643
	QModelIndex index;
insilmaril@732
  1644
insilmaril@732
  1645
	LinkableMapObj *lmo;
insilmaril@786
  1646
	for (int i=0; i<treeItemsNew.count();++i)
insilmaril@731
  1647
	{
insilmaril@786
  1648
		lmo=((MapItem*)treeItemsNew.at(i) )->getLMO();
insilmaril@731
  1649
		bbox=lmo->getBBox();
insilmaril@786
  1650
		sb=selboxList.at(i);
insilmaril@731
  1651
		sb->setRect (
insilmaril@731
  1652
			bbox.x(),bbox.y(), 
insilmaril@731
  1653
			bbox.width(), bbox.height());
insilmaril@731
  1654
		sb->setPen (selectionColor);	
insilmaril@731
  1655
		sb->setBrush (selectionColor);	
insilmaril@731
  1656
		i++;
insilmaril@731
  1657
	}
insilmaril@775
  1658
insilmaril@775
  1659
	scene()->update();
insilmaril@729
  1660
}
insilmaril@728
  1661
insilmaril@753
  1662
void MapEditor::updateData (const QModelIndex &sel)
insilmaril@729
  1663
{
insilmaril@753
  1664
	TreeItem *ti= static_cast<TreeItem*>(sel.internalPointer());
insilmaril@732
  1665
insilmaril@746
  1666
/* testing
insilmaril@753
  1667
	cout << "ME::updateData\n";
insilmaril@732
  1668
insilmaril@753
  1669
	cout << "  ti="<<ti<<endl;
insilmaril@753
  1670
	cout << "  h="<<ti->getHeading().toStdString()<<endl;
insilmaril@729
  1671
	*/
insilmaril@753
  1672
	
insilmaril@753
  1673
	if (ti->isBranchLikeType())
insilmaril@753
  1674
	{
insilmaril@785
  1675
	//	cout << "  ->updating...\n";
insilmaril@779
  1676
		BranchObj *bo=(BranchObj*) ( ((MapItem*)ti)->getLMO());
insilmaril@767
  1677
		bo->updateData();
insilmaril@753
  1678
	}
insilmaril@726
  1679
}
insilmaril@412
  1680
insilmaril@728
  1681
void MapEditor::setSelectionColor (QColor col)
insilmaril@728
  1682
{
insilmaril@728
  1683
	selectionColor=col;
insilmaril@732
  1684
	QItemSelection sel=model->getSelectionModel()->selection();
insilmaril@732
  1685
	updateSelection(sel,sel);
insilmaril@728
  1686
}
insilmaril@728
  1687
insilmaril@753
  1688
insilmaril@728
  1689
QColor MapEditor::getSelectionColor ()
insilmaril@728
  1690
{
insilmaril@728
  1691
	return selectionColor;
insilmaril@728
  1692
}
insilmaril@728
  1693
insilmaril@728
  1694