annotate kdiff3/src/mergeresultwindow.cpp @ 69:8febbfb1148c

KDiff3 0.9.89
author joachim99
date Mon, 10 Apr 2006 08:40:51 +0000
parents d7cafcda8c99
children 5bbfe4784324
rev   line source
joachim99@8 1 /***************************************************************************
joachim99@8 2 mergeresultwindow.cpp - description
joachim99@8 3 -------------------
joachim99@8 4 begin : Sun Apr 14 2002
joachim99@69 5 copyright : (C) 2002-2006 by Joachim Eibl
joachim99@69 6 email : joachim.eibl at gmx.de
joachim99@8 7 ***************************************************************************/
joachim99@8 8
joachim99@8 9 /***************************************************************************
joachim99@8 10 * *
joachim99@8 11 * This program is free software; you can redistribute it and/or modify *
joachim99@8 12 * it under the terms of the GNU General Public License as published by *
joachim99@8 13 * the Free Software Foundation; either version 2 of the License, or *
joachim99@8 14 * (at your option) any later version. *
joachim99@8 15 * *
joachim99@8 16 ***************************************************************************/
joachim99@8 17
joachim99@69 18 #include "mergeresultwindow.h"
joachim99@69 19 #include "optiondialog.h"
joachim99@69 20
joachim99@8 21 #include <qpainter.h>
joachim99@8 22 #include <qapplication.h>
joachim99@8 23 #include <qclipboard.h>
joachim99@8 24 #include <qdir.h>
joachim99@8 25 #include <qfile.h>
joachim99@8 26 #include <qcursor.h>
joachim99@8 27 #include <qpopupmenu.h>
joachim99@69 28 #include <qstatusbar.h>
joachim99@69 29 #include <qregexp.h>
joachim99@69 30
joachim99@8 31 #include <klocale.h>
joachim99@8 32 #include <kmessagebox.h>
joachim99@58 33 #include <iostream>
joachim99@8 34
joachim99@8 35 int g_bAutoSolve = true;
joachim99@8 36
joachim99@53 37 #undef leftInfoWidth
joachim99@8 38 #define leftInfoWidth 3
joachim99@8 39
joachim99@8 40 MergeResultWindow::MergeResultWindow(
joachim99@8 41 QWidget* pParent,
joachim99@66 42 OptionDialog* pOptionDialog,
joachim99@66 43 QStatusBar* pStatusBar
joachim99@8 44 )
joachim99@8 45 : QWidget( pParent, 0, WRepaintNoErase )
joachim99@8 46 {
joachim99@8 47 setFocusPolicy( QWidget::ClickFocus );
joachim99@8 48
joachim99@8 49 m_firstLine = 0;
joachim99@8 50 m_firstColumn = 0;
joachim99@8 51 m_nofColumns = 0;
joachim99@8 52 m_nofLines = 0;
joachim99@58 53 m_totalSize = 0;
joachim99@8 54 m_bMyUpdate = false;
joachim99@8 55 m_bInsertMode = true;
joachim99@8 56 m_scrollDeltaX = 0;
joachim99@8 57 m_scrollDeltaY = 0;
joachim99@8 58 m_bModified = false;
joachim99@69 59 m_eOverviewMode=Overview::eOMNormal;
joachim99@8 60
joachim99@8 61 m_fileName = "";
joachim99@8 62 m_pldA = 0;
joachim99@8 63 m_pldB = 0;
joachim99@8 64 m_pldC = 0;
joachim99@69 65 m_sizeA = 0;
joachim99@69 66 m_sizeB = 0;
joachim99@69 67 m_sizeC = 0;
joachim99@8 68
joachim99@8 69 m_pDiff3LineList = 0;
joachim99@8 70 m_pTotalDiffStatus = 0;
joachim99@66 71 m_pStatusBar = pStatusBar;
joachim99@58 72
joachim99@8 73 m_pOptionDialog = pOptionDialog;
joachim99@58 74 m_bPaintingAllowed = false;
joachim99@69 75 m_delayedDrawTimer = 0;
joachim99@8 76
joachim99@8 77 m_cursorXPos=0;
joachim99@8 78 m_cursorOldXPos=0;
joachim99@8 79 m_cursorYPos=0;
joachim99@8 80 m_bCursorOn = true;
joachim99@69 81 m_bCursorUpdate = false;
joachim99@8 82 connect( &m_cursorTimer, SIGNAL(timeout()), this, SLOT( slotCursorUpdate() ) );
joachim99@8 83 m_cursorTimer.start( 500 /*ms*/, true /*single shot*/ );
joachim99@8 84 m_selection.reset();
joachim99@8 85
joachim99@8 86 setMinimumSize( QSize(20,20) );
joachim99@27 87 setFont( m_pOptionDialog->m_font );
joachim99@8 88 }
joachim99@8 89
joachim99@8 90 void MergeResultWindow::init(
joachim99@69 91 const LineData* pLineDataA, int sizeA,
joachim99@69 92 const LineData* pLineDataB, int sizeB,
joachim99@69 93 const LineData* pLineDataC, int sizeC,
joachim99@8 94 const Diff3LineList* pDiff3LineList,
joachim99@66 95 TotalDiffStatus* pTotalDiffStatus,
joachim99@8 96 QString fileName
joachim99@8 97 )
joachim99@8 98 {
joachim99@8 99 m_firstLine = 0;
joachim99@8 100 m_firstColumn = 0;
joachim99@8 101 m_nofColumns = 0;
joachim99@8 102 m_nofLines = 0;
joachim99@8 103 m_bMyUpdate = false;
joachim99@8 104 m_bInsertMode = true;
joachim99@8 105 m_scrollDeltaX = 0;
joachim99@8 106 m_scrollDeltaY = 0;
joachim99@8 107 m_bModified = false;
joachim99@8 108
joachim99@8 109 m_fileName = fileName;
joachim99@8 110 m_pldA = pLineDataA;
joachim99@8 111 m_pldB = pLineDataB;
joachim99@8 112 m_pldC = pLineDataC;
joachim99@69 113 m_sizeA = sizeA;
joachim99@69 114 m_sizeB = sizeB;
joachim99@69 115 m_sizeC = sizeC;
joachim99@8 116
joachim99@8 117 m_pDiff3LineList = pDiff3LineList;
joachim99@8 118 m_pTotalDiffStatus = pTotalDiffStatus;
joachim99@8 119
joachim99@8 120 m_selection.reset();
joachim99@8 121 m_cursorXPos=0;
joachim99@8 122 m_cursorOldXPos=0;
joachim99@8 123 m_cursorYPos=0;
joachim99@8 124
joachim99@8 125 merge( g_bAutoSolve, -1 );
joachim99@8 126 g_bAutoSolve = true;
joachim99@8 127 update();
joachim99@66 128 updateSourceMask();
joachim99@68 129
joachim99@68 130 int wsc;
joachim99@69 131 int nofUnsolved = getNrOfUnsolvedConflicts(&wsc);
joachim99@68 132 m_pStatusBar->message( i18n("Number of remaining unsolved conflicts: %1 (of which %2 are whitespace)")
joachim99@69 133 .arg(nofUnsolved).arg(wsc) );
joachim99@8 134 }
joachim99@8 135
joachim99@69 136 void MergeResultWindow::reset()
joachim99@69 137 {
joachim99@69 138 m_pDiff3LineList = 0;
joachim99@69 139 m_pTotalDiffStatus = 0;
joachim99@69 140 m_pldA = 0;
joachim99@69 141 m_pldB = 0;
joachim99@69 142 m_pldC = 0;
joachim99@69 143 m_fileName = "";
joachim99@69 144 }
joachim99@8 145
joachim99@8 146 // Calculate the merge information for the given Diff3Line.
joachim99@8 147 // Results will be stored in mergeDetails, bConflict, bLineRemoved and src.
joachim99@8 148 void mergeOneLine(
joachim99@8 149 const Diff3Line& d, e_MergeDetails& mergeDetails, bool& bConflict,
joachim99@8 150 bool& bLineRemoved, int& src, bool bTwoInputs
joachim99@8 151 )
joachim99@8 152 {
joachim99@8 153 mergeDetails = eDefault;
joachim99@8 154 bConflict = false;
joachim99@8 155 bLineRemoved = false;
joachim99@8 156 src = 0;
joachim99@8 157
joachim99@8 158 if ( bTwoInputs ) // Only two input files
joachim99@8 159 {
joachim99@8 160 if ( d.lineA!=-1 && d.lineB!=-1 )
joachim99@8 161 {
joachim99@8 162 if ( d.pFineAB == 0 )
joachim99@8 163 {
joachim99@8 164 mergeDetails = eNoChange; src = A;
joachim99@8 165 }
joachim99@8 166 else
joachim99@8 167 {
joachim99@8 168 mergeDetails = eBChanged; bConflict = true;
joachim99@8 169 }
joachim99@8 170 }
joachim99@8 171 else
joachim99@8 172 {
joachim99@8 173 if ( d.lineA!=-1 && d.lineB==-1 )
joachim99@8 174 {
joachim99@8 175 mergeDetails = eBDeleted; bConflict = true;
joachim99@8 176 }
joachim99@8 177 else if ( d.lineA==-1 && d.lineB!=-1 )
joachim99@8 178 {
joachim99@8 179 mergeDetails = eBDeleted; bConflict = true;
joachim99@8 180 }
joachim99@8 181 }
joachim99@8 182 return;
joachim99@8 183 }
joachim99@8 184
joachim99@8 185 // A is base.
joachim99@8 186 if ( d.lineA!=-1 && d.lineB!=-1 && d.lineC!=-1 )
joachim99@8 187 {
joachim99@8 188 if ( d.pFineAB == 0 && d.pFineBC == 0 && d.pFineCA == 0)
joachim99@8 189 {
joachim99@8 190 mergeDetails = eNoChange; src = A;
joachim99@8 191 }
joachim99@8 192 else if( d.pFineAB == 0 && d.pFineBC != 0 && d.pFineCA != 0 )
joachim99@8 193 {
joachim99@8 194 mergeDetails = eCChanged; src = C;
joachim99@8 195 }
joachim99@8 196 else if( d.pFineAB != 0 && d.pFineBC != 0 && d.pFineCA == 0 )
joachim99@8 197 {
joachim99@8 198 mergeDetails = eBChanged; src = B;
joachim99@8 199 }
joachim99@8 200 else if( d.pFineAB != 0 && d.pFineBC == 0 && d.pFineCA != 0 )
joachim99@8 201 {
joachim99@8 202 mergeDetails = eBCChangedAndEqual; src = C;
joachim99@8 203 }
joachim99@8 204 else if( d.pFineAB != 0 && d.pFineBC != 0 && d.pFineCA != 0 )
joachim99@8 205 {
joachim99@8 206 mergeDetails = eBCChanged; bConflict = true;
joachim99@8 207 }
joachim99@8 208 else
joachim99@8 209 assert(false);
joachim99@8 210 }
joachim99@8 211 else if ( d.lineA!=-1 && d.lineB!=-1 && d.lineC==-1 )
joachim99@8 212 {
joachim99@8 213 if( d.pFineAB != 0 )
joachim99@8 214 {
joachim99@8 215 mergeDetails = eBChanged_CDeleted; bConflict = true;
joachim99@8 216 }
joachim99@8 217 else
joachim99@8 218 {
joachim99@8 219 mergeDetails = eCDeleted; bLineRemoved = true; src = C;
joachim99@8 220 }
joachim99@8 221 }
joachim99@8 222 else if ( d.lineA!=-1 && d.lineB==-1 && d.lineC!=-1 )
joachim99@8 223 {
joachim99@8 224 if( d.pFineCA != 0 )
joachim99@8 225 {
joachim99@8 226 mergeDetails = eCChanged_BDeleted; bConflict = true;
joachim99@8 227 }
joachim99@8 228 else
joachim99@8 229 {
joachim99@8 230 mergeDetails = eBDeleted; bLineRemoved = true; src = B;
joachim99@8 231 }
joachim99@8 232 }
joachim99@8 233 else if ( d.lineA==-1 && d.lineB!=-1 && d.lineC!=-1 )
joachim99@8 234 {
joachim99@8 235 if( d.pFineBC != 0 )
joachim99@8 236 {
joachim99@8 237 mergeDetails = eBCAdded; bConflict = true;
joachim99@8 238 }
joachim99@8 239 else // B==C
joachim99@8 240 {
joachim99@8 241 mergeDetails = eBCAddedAndEqual; src = C;
joachim99@8 242 }
joachim99@8 243 }
joachim99@8 244 else if ( d.lineA==-1 && d.lineB==-1 && d.lineC!= -1 )
joachim99@8 245 {
joachim99@8 246 mergeDetails = eCAdded; src = C;
joachim99@8 247 }
joachim99@8 248 else if ( d.lineA==-1 && d.lineB!=-1 && d.lineC== -1 )
joachim99@8 249 {
joachim99@8 250 mergeDetails = eBAdded; src = B;
joachim99@8 251 }
joachim99@8 252 else if ( d.lineA!=-1 && d.lineB==-1 && d.lineC==-1 )
joachim99@8 253 {
joachim99@8 254 mergeDetails = eBCDeleted; bLineRemoved = true; src = C;
joachim99@8 255 }
joachim99@8 256 else
joachim99@8 257 assert(false);
joachim99@8 258 }
joachim99@8 259
joachim99@8 260 bool MergeResultWindow::sameKindCheck( const MergeLine& ml1, const MergeLine& ml2 )
joachim99@8 261 {
joachim99@8 262 if ( ml1.bConflict && ml2.bConflict )
joachim99@8 263 {
joachim99@51 264 // Both lines have conflicts: If one is only a white space conflict and
joachim99@51 265 // the other one is a real conflict, then this line returns false.
joachim99@8 266 return ml1.id3l->bAEqC == ml2.id3l->bAEqC && ml1.id3l->bAEqB == ml2.id3l->bAEqB;
joachim99@8 267 }
joachim99@8 268 else
joachim99@8 269 return (
joachim99@8 270 !ml1.bConflict && !ml2.bConflict && ml1.bDelta && ml2.bDelta && ml1.srcSelect == ml2.srcSelect ||
joachim99@8 271 !ml1.bDelta && !ml2.bDelta
joachim99@8 272 );
joachim99@8 273 }
joachim99@8 274
joachim99@51 275 void MergeResultWindow::merge(bool bAutoSolve, int defaultSelector, bool bConflictsOnly, bool bWhiteSpaceOnly )
joachim99@51 276 {
joachim99@51 277 if ( !bConflictsOnly )
joachim99@8 278 {
joachim99@51 279 if(m_bModified)
joachim99@8 280 {
joachim99@51 281 int result = KMessageBox::warningYesNo(this,
joachim99@51 282 i18n("The output has been modified.\n"
joachim99@51 283 "If you continue your changes will be lost."),
joachim99@51 284 i18n("Warning"), i18n("C&ontinue"), i18n("&Cancel"));
joachim99@51 285 if ( result==KMessageBox::No )
joachim99@51 286 return;
joachim99@8 287 }
joachim99@8 288
joachim99@51 289 m_mergeLineList.clear();
joachim99@58 290 m_totalSize = 0;
joachim99@51 291 int lineIdx = 0;
joachim99@51 292 Diff3LineList::const_iterator it;
joachim99@51 293 for( it=m_pDiff3LineList->begin(); it!=m_pDiff3LineList->end(); ++it, ++lineIdx )
joachim99@8 294 {
joachim99@51 295 const Diff3Line& d = *it;
joachim99@51 296
joachim99@51 297 MergeLine ml;
joachim99@51 298 bool bLineRemoved;
joachim99@51 299 mergeOneLine( d, ml.mergeDetails, ml.bConflict, bLineRemoved, ml.srcSelect, m_pldC==0 );
joachim99@51 300
joachim99@51 301 // Automatic solving for only whitespace changes.
joachim99@51 302 if ( ml.bConflict &&
joachim99@51 303 ( m_pldC==0 && (d.bAEqB || d.bWhiteLineA && d.bWhiteLineB) ||
joachim99@51 304 m_pldC!=0 && (d.bAEqB && d.bAEqC || d.bWhiteLineA && d.bWhiteLineB && d.bWhiteLineC ) ) )
joachim99@51 305 {
joachim99@51 306 ml.bWhiteSpaceConflict = true;
joachim99@51 307 }
joachim99@51 308
joachim99@51 309 ml.d3lLineIdx = lineIdx;
joachim99@51 310 ml.bDelta = ml.srcSelect != A;
joachim99@51 311 ml.id3l = it;
joachim99@51 312 ml.srcRangeLength = 1;
joachim99@51 313
joachim99@51 314 MergeLine* back = m_mergeLineList.empty() ? 0 : &m_mergeLineList.back();
joachim99@51 315
joachim99@51 316 bool bSame = back!=0 && sameKindCheck( ml, *back );
joachim99@51 317 if( bSame )
joachim99@51 318 {
joachim99@51 319 ++back->srcRangeLength;
joachim99@51 320 if ( back->bWhiteSpaceConflict && !ml.bWhiteSpaceConflict )
joachim99@51 321 back->bWhiteSpaceConflict = false;
joachim99@51 322 }
joachim99@51 323 else
joachim99@51 324 {
joachim99@51 325 if (back!=0 && back->bWhiteSpaceConflict )
joachim99@51 326 {
joachim99@51 327 if ( m_pldC==0 && m_pOptionDialog->m_whiteSpace2FileMergeDefault != 0 ) // Only two inputs
joachim99@51 328 {
joachim99@51 329 back->srcSelect = m_pOptionDialog->m_whiteSpace2FileMergeDefault;
joachim99@51 330 back->bConflict = false;
joachim99@51 331 }
joachim99@51 332 else if ( m_pldC!=0 && m_pOptionDialog->m_whiteSpace3FileMergeDefault != 0 )
joachim99@51 333 {
joachim99@51 334 back->srcSelect = m_pOptionDialog->m_whiteSpace3FileMergeDefault;
joachim99@51 335 back->bConflict = false;
joachim99@51 336 }
joachim99@51 337 }
joachim99@58 338 ml.mergeEditLineList.setTotalSizePtr(&m_totalSize);
joachim99@51 339 m_mergeLineList.push_back( ml );
joachim99@51 340 }
joachim99@51 341
joachim99@51 342 if ( ! ml.bConflict )
joachim99@51 343 {
joachim99@51 344 MergeLine& tmpBack = m_mergeLineList.back();
joachim99@69 345 MergeEditLine mel(ml.id3l);
joachim99@69 346 mel.setSource( ml.srcSelect, bLineRemoved );
joachim99@51 347 tmpBack.mergeEditLineList.push_back(mel);
joachim99@51 348 }
joachim99@51 349 else if ( back==0 || ! back->bConflict || !bSame )
joachim99@51 350 {
joachim99@51 351 MergeLine& tmpBack = m_mergeLineList.back();
joachim99@69 352 MergeEditLine mel(ml.id3l);
joachim99@51 353 mel.setConflict();
joachim99@51 354 tmpBack.mergeEditLineList.push_back(mel);
joachim99@51 355 }
joachim99@8 356 }
joachim99@8 357 }
joachim99@8 358
joachim99@8 359 if ( !bAutoSolve )
joachim99@8 360 {
joachim99@8 361 // Change all auto selections
joachim99@8 362 MergeLineList::iterator mlIt;
joachim99@8 363 for( mlIt=m_mergeLineList.begin(); mlIt!=m_mergeLineList.end(); ++mlIt )
joachim99@8 364 {
joachim99@8 365 MergeLine& ml = *mlIt;
joachim99@51 366 bool bConflict = ml.mergeEditLineList.empty() || ml.mergeEditLineList.begin()->isConflict();
joachim99@51 367 if ( ml.bDelta && ( !bConflictsOnly || bConflict ) && (!bWhiteSpaceOnly || ml.bWhiteSpaceConflict ))
joachim99@8 368 {
joachim99@8 369 ml.mergeEditLineList.clear();
joachim99@8 370 if ( defaultSelector==-1 && ml.bDelta )
joachim99@8 371 {
joachim99@69 372 MergeEditLine mel(ml.id3l);;
joachim99@8 373 mel.setConflict();
joachim99@8 374 ml.bConflict = true;
joachim99@8 375 ml.mergeEditLineList.push_back(mel);
joachim99@8 376 }
joachim99@8 377 else
joachim99@8 378 {
joachim99@8 379 Diff3LineList::const_iterator d3llit=ml.id3l;
joachim99@8 380 int j;
joachim99@8 381
joachim99@8 382 for( j=0; j<ml.srcRangeLength; ++j )
joachim99@8 383 {
joachim99@69 384 MergeEditLine mel(d3llit);
joachim99@69 385 mel.setSource( defaultSelector, false );
joachim99@51 386
joachim99@8 387 int srcLine = defaultSelector==1 ? d3llit->lineA :
joachim99@8 388 defaultSelector==2 ? d3llit->lineB :
joachim99@8 389 defaultSelector==3 ? d3llit->lineC : -1;
joachim99@51 390
joachim99@8 391 if ( srcLine != -1 )
joachim99@8 392 {
joachim99@8 393 ml.mergeEditLineList.push_back(mel);
joachim99@8 394 }
joachim99@8 395
joachim99@8 396 ++d3llit;
joachim99@8 397 }
joachim99@8 398
joachim99@8 399 if ( ml.mergeEditLineList.empty() ) // Make a line nevertheless
joachim99@8 400 {
joachim99@69 401 MergeEditLine mel(ml.id3l);
joachim99@8 402 mel.setRemoved( defaultSelector );
joachim99@8 403 ml.mergeEditLineList.push_back(mel);
joachim99@8 404 }
joachim99@8 405 }
joachim99@8 406 }
joachim99@8 407 }
joachim99@8 408 }
joachim99@8 409
joachim99@8 410 MergeLineList::iterator mlIt;
joachim99@8 411 for( mlIt=m_mergeLineList.begin(); mlIt!=m_mergeLineList.end(); ++mlIt )
joachim99@8 412 {
joachim99@8 413 MergeLine& ml = *mlIt;
joachim99@8 414 // Remove all lines that are empty, because no src lines are there.
joachim99@8 415
joachim99@8 416 int oldSrcLine = -1;
joachim99@8 417 int oldSrc = -1;
joachim99@8 418 MergeEditLineList::iterator melIt;
joachim99@8 419 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); )
joachim99@8 420 {
joachim99@8 421 MergeEditLine& mel = *melIt;
joachim99@8 422 int melsrc = mel.src();
joachim99@8 423
joachim99@69 424 int srcLine = mel.isRemoved() ? -1 :
joachim99@69 425 melsrc==1 ? mel.id3l()->lineA :
joachim99@8 426 melsrc==2 ? mel.id3l()->lineB :
joachim99@8 427 melsrc==3 ? mel.id3l()->lineC : -1;
joachim99@8 428
joachim99@69 429 // At least one line remains because oldSrc != melsrc for first line in list
joachim99@69 430 // Other empty lines will be removed
joachim99@8 431 if ( srcLine == -1 && oldSrcLine==-1 && oldSrc == melsrc )
joachim99@8 432 melIt = ml.mergeEditLineList.erase( melIt );
joachim99@8 433 else
joachim99@8 434 ++melIt;
joachim99@8 435
joachim99@8 436 oldSrcLine = srcLine;
joachim99@8 437 oldSrc = melsrc;
joachim99@8 438 }
joachim99@8 439 }
joachim99@8 440
joachim99@69 441 if ( bAutoSolve && !bConflictsOnly )
joachim99@69 442 {
joachim99@69 443 if ( m_pOptionDialog->m_bRunHistoryAutoMergeOnMergeStart )
joachim99@69 444 slotMergeHistory();
joachim99@69 445 if ( m_pOptionDialog->m_bRunRegExpAutoMergeOnMergeStart )
joachim99@69 446 slotRegExpAutoMerge();
joachim99@69 447 if ( m_pldC != 0 && ! doRelevantChangesExist() )
joachim99@69 448 emit noRelevantChangesDetected();
joachim99@69 449 }
joachim99@69 450
joachim99@66 451 int nrOfSolvedConflicts = 0;
joachim99@66 452 int nrOfUnsolvedConflicts = 0;
joachim99@66 453 int nrOfWhiteSpaceConflicts = 0;
joachim99@66 454
joachim99@66 455 MergeLineList::iterator i;
joachim99@66 456 for ( i = m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@66 457 {
joachim99@66 458 if ( i->bConflict )
joachim99@66 459 ++nrOfUnsolvedConflicts;
joachim99@66 460 else if ( i->bDelta )
joachim99@66 461 ++nrOfSolvedConflicts;
joachim99@68 462
joachim99@66 463 if ( i->bWhiteSpaceConflict )
joachim99@66 464 ++nrOfWhiteSpaceConflicts;
joachim99@66 465 }
joachim99@68 466
joachim99@66 467 m_pTotalDiffStatus->nofUnsolvedConflicts = nrOfUnsolvedConflicts;
joachim99@66 468 m_pTotalDiffStatus->nofSolvedConflicts = nrOfSolvedConflicts;
joachim99@66 469 m_pTotalDiffStatus->nofWhitespaceConflicts = nrOfWhiteSpaceConflicts;
joachim99@66 470
joachim99@66 471
joachim99@8 472 m_cursorXPos=0;
joachim99@8 473 m_cursorOldXPos=0;
joachim99@8 474 m_cursorYPos=0;
joachim99@66 475 //m_firstLine = 0; // Must not set line/column without scrolling there
joachim99@66 476 //m_firstColumn = 0;
joachim99@51 477
joachim99@8 478 m_bModified = false;
joachim99@66 479
joachim99@66 480 m_currentMergeLineIt = m_mergeLineList.begin();
joachim99@66 481 slotGoTop();
joachim99@66 482
joachim99@8 483 updateAvailabilities();
joachim99@8 484 update();
joachim99@8 485 }
joachim99@8 486
joachim99@8 487 void MergeResultWindow::setFirstLine(int firstLine)
joachim99@8 488 {
joachim99@8 489 m_firstLine = max2(0,firstLine);
joachim99@8 490 update();
joachim99@8 491 }
joachim99@8 492
joachim99@8 493 void MergeResultWindow::setFirstColumn(int firstCol)
joachim99@8 494 {
joachim99@8 495 m_firstColumn = max2(0,firstCol);
joachim99@8 496 update();
joachim99@8 497 }
joachim99@8 498
joachim99@8 499 int MergeResultWindow::getNofColumns()
joachim99@8 500 {
joachim99@8 501 return m_nofColumns;
joachim99@8 502 }
joachim99@8 503
joachim99@8 504 int MergeResultWindow::getNofLines()
joachim99@8 505 {
joachim99@58 506 return m_totalSize;
joachim99@8 507 }
joachim99@8 508
joachim99@8 509 int MergeResultWindow::getNofVisibleColumns()
joachim99@8 510 {
joachim99@8 511 QFontMetrics fm = fontMetrics();
joachim99@8 512 return width()/fm.width('W')-4;
joachim99@8 513 }
joachim99@8 514
joachim99@8 515 int MergeResultWindow::getNofVisibleLines()
joachim99@8 516 {
joachim99@8 517 QFontMetrics fm = fontMetrics();
joachim99@8 518 return (height()-3)/fm.height()-2;
joachim99@8 519 }
joachim99@8 520
joachim99@8 521 void MergeResultWindow::resizeEvent( QResizeEvent* e )
joachim99@8 522 {
joachim99@8 523 QWidget::resizeEvent(e);
joachim99@8 524 emit resizeSignal();
joachim99@8 525 }
joachim99@8 526
joachim99@69 527 Overview::e_OverviewMode MergeResultWindow::getOverviewMode()
joachim99@69 528 {
joachim99@69 529 return m_eOverviewMode;
joachim99@69 530 }
joachim99@69 531
joachim99@69 532 void MergeResultWindow::setOverviewMode( Overview::e_OverviewMode eOverviewMode )
joachim99@69 533 {
joachim99@69 534 m_eOverviewMode = eOverviewMode;
joachim99@69 535 }
joachim99@69 536
joachim99@69 537 // Check whether we should ignore current delta when moving to next/previous delta
joachim99@69 538 bool MergeResultWindow::checkOverviewIgnore(MergeLineList::iterator &i)
joachim99@69 539 {
joachim99@69 540 if (m_eOverviewMode == Overview::eOMNormal) return false;
joachim99@69 541 if (m_eOverviewMode == Overview::eOMAvsB)
joachim99@69 542 return i->mergeDetails == eCAdded || i->mergeDetails == eCDeleted || i->mergeDetails == eCChanged;
joachim99@69 543 if (m_eOverviewMode == Overview::eOMAvsC)
joachim99@69 544 return i->mergeDetails == eBAdded || i->mergeDetails == eBDeleted || i->mergeDetails == eBChanged;
joachim99@69 545 if (m_eOverviewMode == Overview::eOMBvsC)
joachim99@69 546 return i->mergeDetails == eBCAddedAndEqual || i->mergeDetails == eBCDeleted || i->mergeDetails == eBCChangedAndEqual;
joachim99@69 547 return false;
joachim99@69 548 }
joachim99@69 549
joachim99@8 550 // Go to prev/next delta/conflict or first/last delta.
joachim99@8 551 void MergeResultWindow::go( e_Direction eDir, e_EndPoint eEndPoint )
joachim99@8 552 {
joachim99@8 553 assert( eDir==eUp || eDir==eDown );
joachim99@8 554 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@51 555 bool bSkipWhiteConflicts = ! m_pOptionDialog->m_bShowWhiteSpace;
joachim99@8 556 if( eEndPoint==eEnd )
joachim99@8 557 {
joachim99@8 558 if (eDir==eUp) i = m_mergeLineList.begin(); // first mergeline
joachim99@8 559 else i = --m_mergeLineList.end(); // last mergeline
joachim99@8 560
joachim99@51 561 while ( i!=m_mergeLineList.end() && ! i->bDelta )
joachim99@8 562 {
joachim99@8 563 if ( eDir==eUp ) ++i; // search downwards
joachim99@8 564 else --i; // search upwards
joachim99@8 565 }
joachim99@8 566 }
joachim99@8 567 else if ( eEndPoint == eDelta && i!=m_mergeLineList.end())
joachim99@8 568 {
joachim99@8 569 do
joachim99@8 570 {
joachim99@8 571 if ( eDir==eUp ) --i;
joachim99@8 572 else ++i;
joachim99@8 573 }
joachim99@69 574 while ( i!=m_mergeLineList.end() && ( i->bDelta == false || checkOverviewIgnore(i) || bSkipWhiteConflicts && i->bWhiteSpaceConflict ) );
joachim99@8 575 }
joachim99@51 576 else if ( eEndPoint == eConflict && i!=m_mergeLineList.end() )
joachim99@8 577 {
joachim99@8 578 do
joachim99@8 579 {
joachim99@8 580 if ( eDir==eUp ) --i;
joachim99@8 581 else ++i;
joachim99@8 582 }
joachim99@51 583 while ( i!=m_mergeLineList.end() && (i->bConflict == false || bSkipWhiteConflicts && i->bWhiteSpaceConflict ) );
joachim99@8 584 }
joachim99@51 585 else if ( i!=m_mergeLineList.end() && eEndPoint == eUnsolvedConflict )
joachim99@8 586 {
joachim99@8 587 do
joachim99@8 588 {
joachim99@8 589 if ( eDir==eUp ) --i;
joachim99@8 590 else ++i;
joachim99@8 591 }
joachim99@8 592 while ( i!=m_mergeLineList.end() && ! i->mergeEditLineList.begin()->isConflict() );
joachim99@8 593 }
joachim99@8 594
joachim99@53 595 if ( isVisible() )
joachim99@53 596 setFocus();
joachim99@68 597
joachim99@66 598 setFastSelector( i );
joachim99@8 599 }
joachim99@8 600
joachim99@8 601 bool MergeResultWindow::isDeltaAboveCurrent()
joachim99@8 602 {
joachim99@51 603 bool bSkipWhiteConflicts = ! m_pOptionDialog->m_bShowWhiteSpace;
joachim99@66 604 if (m_mergeLineList.empty()) return false;
joachim99@8 605 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@66 606 if (i == m_mergeLineList.begin()) return false;
joachim99@66 607 do
joachim99@8 608 {
joachim99@66 609 --i;
joachim99@51 610 if ( i->bDelta && !( bSkipWhiteConflicts && i->bWhiteSpaceConflict ) ) return true;
joachim99@68 611 }
joachim99@66 612 while (i!=m_mergeLineList.begin());
joachim99@68 613
joachim99@8 614 return false;
joachim99@8 615 }
joachim99@8 616
joachim99@8 617 bool MergeResultWindow::isDeltaBelowCurrent()
joachim99@8 618 {
joachim99@51 619 bool bSkipWhiteConflicts = ! m_pOptionDialog->m_bShowWhiteSpace;
joachim99@66 620 if (m_mergeLineList.empty()) return false;
joachim99@68 621
joachim99@8 622 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@66 623 if (i!=m_mergeLineList.end())
joachim99@8 624 {
joachim99@66 625 ++i;
joachim99@66 626 for( ; i!=m_mergeLineList.end(); ++i )
joachim99@66 627 {
joachim99@66 628 if ( i->bDelta && !( bSkipWhiteConflicts && i->bWhiteSpaceConflict ) ) return true;
joachim99@66 629 }
joachim99@8 630 }
joachim99@8 631 return false;
joachim99@8 632 }
joachim99@8 633
joachim99@8 634 bool MergeResultWindow::isConflictAboveCurrent()
joachim99@8 635 {
joachim99@66 636 if (m_mergeLineList.empty()) return false;
joachim99@8 637 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@66 638 if (i == m_mergeLineList.begin()) return false;
joachim99@68 639
joachim99@66 640 do
joachim99@8 641 {
joachim99@66 642 --i;
joachim99@8 643 if ( i->bConflict ) return true;
joachim99@66 644 }
joachim99@66 645 while (i!=m_mergeLineList.begin());
joachim99@66 646
joachim99@8 647 return false;
joachim99@8 648 }
joachim99@8 649
joachim99@8 650 bool MergeResultWindow::isConflictBelowCurrent()
joachim99@8 651 {
joachim99@8 652 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@66 653 if (m_mergeLineList.empty()) return false;
joachim99@66 654
joachim99@66 655 if (i!=m_mergeLineList.end())
joachim99@8 656 {
joachim99@66 657 ++i;
joachim99@66 658 for( ; i!=m_mergeLineList.end(); ++i )
joachim99@66 659 {
joachim99@66 660 if ( i->bConflict ) return true;
joachim99@66 661 }
joachim99@8 662 }
joachim99@69 663 return false;
joachim99@8 664 }
joachim99@8 665
joachim99@8 666 bool MergeResultWindow::isUnsolvedConflictAboveCurrent()
joachim99@8 667 {
joachim99@66 668 if (m_mergeLineList.empty()) return false;
joachim99@8 669 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@66 670 if (i == m_mergeLineList.begin()) return false;
joachim99@66 671
joachim99@66 672 do
joachim99@8 673 {
joachim99@66 674 --i;
joachim99@8 675 if ( i->mergeEditLineList.begin()->isConflict() ) return true;
joachim99@66 676 }
joachim99@66 677 while (i!=m_mergeLineList.begin());
joachim99@66 678
joachim99@8 679 return false;
joachim99@8 680 }
joachim99@8 681
joachim99@8 682 bool MergeResultWindow::isUnsolvedConflictBelowCurrent()
joachim99@8 683 {
joachim99@8 684 MergeLineList::iterator i = m_currentMergeLineIt;
joachim99@66 685 if (m_mergeLineList.empty()) return false;
joachim99@66 686
joachim99@66 687 if (i!=m_mergeLineList.end())
joachim99@8 688 {
joachim99@66 689 ++i;
joachim99@66 690 for( ; i!=m_mergeLineList.end(); ++i )
joachim99@66 691 {
joachim99@66 692 if ( i->mergeEditLineList.begin()->isConflict() ) return true;
joachim99@66 693 }
joachim99@8 694 }
joachim99@8 695 return false;
joachim99@8 696 }
joachim99@8 697
joachim99@8 698 void MergeResultWindow::slotGoTop()
joachim99@8 699 {
joachim99@8 700 go( eUp, eEnd );
joachim99@8 701 }
joachim99@8 702
joachim99@8 703 void MergeResultWindow::slotGoCurrent()
joachim99@8 704 {
joachim99@8 705 setFastSelector( m_currentMergeLineIt );
joachim99@8 706 }
joachim99@8 707
joachim99@8 708 void MergeResultWindow::slotGoBottom()
joachim99@8 709 {
joachim99@8 710 go( eDown, eEnd );
joachim99@8 711 }
joachim99@8 712
joachim99@8 713 void MergeResultWindow::slotGoPrevDelta()
joachim99@8 714 {
joachim99@8 715 go( eUp, eDelta );
joachim99@8 716 }
joachim99@8 717
joachim99@8 718 void MergeResultWindow::slotGoNextDelta()
joachim99@8 719 {
joachim99@8 720 go( eDown, eDelta );
joachim99@8 721 }
joachim99@8 722
joachim99@8 723 void MergeResultWindow::slotGoPrevConflict()
joachim99@8 724 {
joachim99@8 725 go( eUp, eConflict );
joachim99@8 726 }
joachim99@8 727
joachim99@8 728 void MergeResultWindow::slotGoNextConflict()
joachim99@8 729 {
joachim99@8 730 go( eDown, eConflict );
joachim99@8 731 }
joachim99@8 732
joachim99@8 733 void MergeResultWindow::slotGoPrevUnsolvedConflict()
joachim99@8 734 {
joachim99@8 735 go( eUp, eUnsolvedConflict );
joachim99@8 736 }
joachim99@8 737
joachim99@8 738 void MergeResultWindow::slotGoNextUnsolvedConflict()
joachim99@8 739 {
joachim99@8 740 go( eDown, eUnsolvedConflict );
joachim99@8 741 }
joachim99@8 742
joachim99@8 743 /** The line is given as a index in the Diff3LineList.
joachim99@8 744 The function calculates the corresponding iterator. */
joachim99@8 745 void MergeResultWindow::slotSetFastSelectorLine( int line )
joachim99@8 746 {
joachim99@8 747 MergeLineList::iterator i;
joachim99@8 748 for ( i = m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@8 749 {
joachim99@8 750 if ( line>=i->d3lLineIdx && line < i->d3lLineIdx + i->srcRangeLength )
joachim99@8 751 {
joachim99@68 752 //if ( i->bDelta )
joachim99@8 753 {
joachim99@8 754 setFastSelector( i );
joachim99@8 755 }
joachim99@8 756 break;
joachim99@8 757 }
joachim99@8 758 }
joachim99@8 759 }
joachim99@8 760
joachim99@68 761 int MergeResultWindow::getNrOfUnsolvedConflicts( int* pNrOfWhiteSpaceConflicts )
joachim99@8 762 {
joachim99@8 763 int nrOfUnsolvedConflicts = 0;
joachim99@68 764 if (pNrOfWhiteSpaceConflicts!=0)
joachim99@68 765 *pNrOfWhiteSpaceConflicts = 0;
joachim99@8 766
joachim99@8 767 MergeLineList::iterator mlIt = m_mergeLineList.begin();
joachim99@8 768 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 769 {
joachim99@8 770 MergeLine& ml = *mlIt;
joachim99@8 771 MergeEditLineList::iterator melIt = ml.mergeEditLineList.begin();
joachim99@8 772 if ( melIt->isConflict() )
joachim99@68 773 {
joachim99@8 774 ++nrOfUnsolvedConflicts;
joachim99@68 775 if ( ml.bWhiteSpaceConflict && pNrOfWhiteSpaceConflicts!=0 )
joachim99@68 776 ++ *pNrOfWhiteSpaceConflicts;
joachim99@68 777 }
joachim99@8 778 }
joachim99@8 779
joachim99@8 780 return nrOfUnsolvedConflicts;
joachim99@8 781 }
joachim99@8 782
joachim99@8 783 void MergeResultWindow::showNrOfConflicts()
joachim99@8 784 {
joachim99@69 785 int nrOfConflicts = 0;
joachim99@8 786 MergeLineList::iterator i;
joachim99@8 787 for ( i = m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@8 788 {
joachim99@69 789 if ( i->bConflict || i->bDelta )
joachim99@69 790 ++nrOfConflicts;
joachim99@8 791 }
joachim99@8 792 QString totalInfo;
joachim99@8 793 if ( m_pTotalDiffStatus->bBinaryAEqB && m_pTotalDiffStatus->bBinaryAEqC )
joachim99@8 794 totalInfo += i18n("All input files are binary equal.");
joachim99@8 795 else if ( m_pTotalDiffStatus->bTextAEqB && m_pTotalDiffStatus->bTextAEqC )
joachim99@8 796 totalInfo += i18n("All input files contain the same text.");
joachim99@8 797 else {
joachim99@8 798 if ( m_pTotalDiffStatus->bBinaryAEqB ) totalInfo += i18n("Files A and B are binary equal.\n");
joachim99@8 799 else if ( m_pTotalDiffStatus->bTextAEqB ) totalInfo += i18n("Files A and B have equal text. \n");
joachim99@8 800 if ( m_pTotalDiffStatus->bBinaryAEqC ) totalInfo += i18n("Files A and C are binary equal.\n");
joachim99@8 801 else if ( m_pTotalDiffStatus->bTextAEqC ) totalInfo += i18n("Files A and C have equal text. \n");
joachim99@8 802 if ( m_pTotalDiffStatus->bBinaryBEqC ) totalInfo += i18n("Files B and C are binary equal.\n");
joachim99@8 803 else if ( m_pTotalDiffStatus->bTextBEqC ) totalInfo += i18n("Files B and C have equal text. \n");
joachim99@8 804 }
joachim99@69 805
joachim99@69 806 int nrOfUnsolvedConflicts = getNrOfUnsolvedConflicts();
joachim99@69 807
joachim99@8 808 KMessageBox::information( this,
joachim99@69 809 i18n("Total number of conflicts: ") + QString::number(nrOfConflicts) +
joachim99@69 810 i18n("\nNr of automatically solved conflicts: ") + QString::number(nrOfConflicts-nrOfUnsolvedConflicts) +
joachim99@8 811 i18n("\nNr of unsolved conflicts: ") + QString::number(nrOfUnsolvedConflicts) +
joachim99@8 812 "\n"+totalInfo,
joachim99@8 813 i18n("Conflicts")
joachim99@8 814 );
joachim99@8 815 }
joachim99@8 816
joachim99@8 817 void MergeResultWindow::setFastSelector(MergeLineList::iterator i)
joachim99@8 818 {
joachim99@8 819 if ( i==m_mergeLineList.end() )
joachim99@8 820 return;
joachim99@8 821 m_currentMergeLineIt = i;
joachim99@8 822 emit setFastSelectorRange( i->d3lLineIdx, i->srcRangeLength );
joachim99@8 823
joachim99@8 824 int line1 = 0;
joachim99@8 825
joachim99@8 826 MergeLineList::iterator mlIt = m_mergeLineList.begin();
joachim99@8 827 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 828 {
joachim99@8 829 if(mlIt==m_currentMergeLineIt)
joachim99@8 830 break;
joachim99@8 831 line1 += mlIt->mergeEditLineList.size();
joachim99@8 832 }
joachim99@8 833
joachim99@8 834 int nofLines = m_currentMergeLineIt->mergeEditLineList.size();
joachim99@8 835 int newFirstLine = getBestFirstLine( line1, nofLines, m_firstLine, getNofVisibleLines() );
joachim99@8 836 if ( newFirstLine != m_firstLine )
joachim99@8 837 {
joachim99@8 838 scroll( 0, newFirstLine - m_firstLine );
joachim99@8 839 }
joachim99@8 840
joachim99@8 841 if ( m_selection.isEmpty() )
joachim99@8 842 {
joachim99@8 843 m_cursorXPos = 0;
joachim99@8 844 m_cursorOldXPos = 0;
joachim99@8 845 m_cursorYPos = line1;
joachim99@8 846 }
joachim99@8 847
joachim99@8 848 update();
joachim99@66 849 updateSourceMask();
joachim99@8 850 emit updateAvailabilities();
joachim99@8 851 }
joachim99@8 852
joachim99@8 853 void MergeResultWindow::choose( int selector )
joachim99@8 854 {
joachim99@66 855 if ( m_currentMergeLineIt==m_mergeLineList.end() )
joachim99@66 856 return;
joachim99@68 857
joachim99@8 858 setModified();
joachim99@8 859
joachim99@8 860 // First find range for which this change works.
joachim99@8 861 MergeLine& ml = *m_currentMergeLineIt;
joachim99@8 862
joachim99@8 863 MergeEditLineList::iterator melIt;
joachim99@8 864
joachim99@8 865 // Now check if selector is active for this range already.
joachim99@8 866 bool bActive = false;
joachim99@8 867
joachim99@8 868 // Remove unneeded lines in the range.
joachim99@8 869 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); )
joachim99@8 870 {
joachim99@8 871 MergeEditLine& mel = *melIt;
joachim99@8 872 if ( mel.src()==selector )
joachim99@8 873 bActive = true;
joachim99@8 874
joachim99@8 875 if ( mel.src()==selector || !mel.isEditableText() || mel.isModified() )
joachim99@8 876 melIt = ml.mergeEditLineList.erase( melIt );
joachim99@8 877 else
joachim99@8 878 ++melIt;
joachim99@8 879 }
joachim99@8 880
joachim99@8 881 if ( !bActive ) // Selected source wasn't active.
joachim99@8 882 { // Append the lines from selected source here at rangeEnd.
joachim99@8 883 Diff3LineList::const_iterator d3llit=ml.id3l;
joachim99@8 884 int j;
joachim99@8 885
joachim99@8 886 for( j=0; j<ml.srcRangeLength; ++j )
joachim99@8 887 {
joachim99@69 888 MergeEditLine mel(d3llit);
joachim99@69 889 mel.setSource( selector, false );
joachim99@8 890 ml.mergeEditLineList.push_back(mel);
joachim99@8 891
joachim99@8 892 ++d3llit;
joachim99@8 893 }
joachim99@8 894 }
joachim99@8 895
joachim99@8 896 if ( ! ml.mergeEditLineList.empty() )
joachim99@8 897 {
joachim99@8 898 // Remove all lines that are empty, because no src lines are there.
joachim99@8 899 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); )
joachim99@8 900 {
joachim99@8 901 MergeEditLine& mel = *melIt;
joachim99@8 902
joachim99@8 903 int srcLine = mel.src()==1 ? mel.id3l()->lineA :
joachim99@8 904 mel.src()==2 ? mel.id3l()->lineB :
joachim99@8 905 mel.src()==3 ? mel.id3l()->lineC : -1;
joachim99@8 906
joachim99@8 907 if ( srcLine == -1 )
joachim99@8 908 melIt = ml.mergeEditLineList.erase( melIt );
joachim99@8 909 else
joachim99@8 910 ++melIt;
joachim99@8 911 }
joachim99@8 912 }
joachim99@8 913
joachim99@8 914 if ( ml.mergeEditLineList.empty() )
joachim99@8 915 {
joachim99@8 916 // Insert a dummy line:
joachim99@69 917 MergeEditLine mel(ml.id3l);
joachim99@8 918
joachim99@8 919 if ( bActive ) mel.setConflict(); // All src entries deleted => conflict
joachim99@8 920 else mel.setRemoved(selector); // No lines in corresponding src found.
joachim99@8 921
joachim99@8 922 ml.mergeEditLineList.push_back(mel);
joachim99@8 923 }
joachim99@68 924
joachim99@58 925 if ( m_cursorYPos >= m_totalSize )
joachim99@58 926 {
joachim99@58 927 m_cursorYPos = m_totalSize-1;
joachim99@58 928 m_cursorXPos = 0;
joachim99@58 929 }
joachim99@8 930
joachim99@8 931 update();
joachim99@66 932 updateSourceMask();
joachim99@8 933 emit updateAvailabilities();
joachim99@68 934 int wsc;
joachim99@69 935 int nofUnsolved = getNrOfUnsolvedConflicts(&wsc);
joachim99@68 936 m_pStatusBar->message( i18n("Number of remaining unsolved conflicts: %1 (of which %2 are whitespace)")
joachim99@69 937 .arg(nofUnsolved).arg(wsc) );
joachim99@8 938 }
joachim99@8 939
joachim99@51 940 // bConflictsOnly: automatically choose for conflicts only (true) or for everywhere (false)
joachim99@51 941 void MergeResultWindow::chooseGlobal(int selector, bool bConflictsOnly, bool bWhiteSpaceOnly )
joachim99@8 942 {
joachim99@8 943 resetSelection();
joachim99@8 944
joachim99@51 945 merge( false, selector, bConflictsOnly, bWhiteSpaceOnly );
joachim99@8 946 emit modified();
joachim99@8 947 update();
joachim99@68 948 int wsc;
joachim99@69 949 int nofUnsolved = getNrOfUnsolvedConflicts(&wsc);
joachim99@68 950 m_pStatusBar->message( i18n("Number of remaining unsolved conflicts: %1 (of which %2 are whitespace)")
joachim99@69 951 .arg(nofUnsolved).arg(wsc) );
joachim99@8 952 }
joachim99@8 953
joachim99@8 954 void MergeResultWindow::slotAutoSolve()
joachim99@8 955 {
joachim99@8 956 resetSelection();
joachim99@8 957 merge( true, -1 );
joachim99@8 958 emit modified();
joachim99@8 959 update();
joachim99@68 960 int wsc;
joachim99@69 961 int nofUnsolved = getNrOfUnsolvedConflicts(&wsc);
joachim99@68 962 m_pStatusBar->message( i18n("Number of remaining unsolved conflicts: %1 (of which %2 are whitespace)")
joachim99@69 963 .arg(nofUnsolved).arg(wsc) );
joachim99@8 964 }
joachim99@8 965
joachim99@8 966 void MergeResultWindow::slotUnsolve()
joachim99@8 967 {
joachim99@8 968 resetSelection();
joachim99@8 969 merge( false, -1 );
joachim99@8 970 emit modified();
joachim99@8 971 update();
joachim99@68 972 int wsc;
joachim99@69 973 int nofUnsolved = getNrOfUnsolvedConflicts(&wsc);
joachim99@68 974 m_pStatusBar->message( i18n("Number of remaining unsolved conflicts: %1 (of which %2 are whitespace)")
joachim99@69 975 .arg(nofUnsolved).arg(wsc) );
joachim99@69 976 }
joachim99@69 977
joachim99@69 978
joachim99@69 979 static void findHistoryRange( bool bThreeFiles, const Diff3LineList* pD3LList,
joachim99@69 980 Diff3LineList::const_iterator& iBegin, Diff3LineList::const_iterator& iEnd, int& idxBegin, int& idxEnd )
joachim99@69 981 {
joachim99@69 982 QRegExp historyStart(".*\\$Log.*\\$.*");
joachim99@69 983 QString historyLead;
joachim99@69 984 // Search for start of history
joachim99@69 985 for( iBegin = pD3LList->begin(), idxBegin=0; iBegin!=pD3LList->end(); ++iBegin, ++idxBegin )
joachim99@69 986 {
joachim99@69 987 if ( historyStart.exactMatch( iBegin->getString(A) ) &&
joachim99@69 988 historyStart.exactMatch( iBegin->getString(B) ) &&
joachim99@69 989 ( !bThreeFiles || historyStart.exactMatch( iBegin->getString(C) ) ) )
joachim99@69 990 {
joachim99@69 991 historyLead = iBegin->getString(A).section(' ', 0, 0, QString::SectionIncludeLeadingSep);
joachim99@69 992 break;
joachim99@69 993 }
joachim99@69 994 }
joachim99@69 995 // Search for end of history
joachim99@69 996 for( iEnd = iBegin, idxEnd = idxBegin; iEnd!=pD3LList->end(); ++iEnd, ++idxEnd )
joachim99@69 997 {
joachim99@69 998 QString sA = iEnd->getString(A);
joachim99@69 999 QString sB = iEnd->getString(B);
joachim99@69 1000 QString sC = iEnd->getString(C);
joachim99@69 1001 if ( ! ((sA.isNull() || historyLead == sA.section(' ',0,0,QString::SectionIncludeLeadingSep)) &&
joachim99@69 1002 (sB.isNull() || historyLead == sB.section(' ',0,0,QString::SectionIncludeLeadingSep)) &&
joachim99@69 1003 (!bThreeFiles || sC.isNull() || historyLead == sC.section(' ',0,0,QString::SectionIncludeLeadingSep))
joachim99@69 1004 ))
joachim99@69 1005 {
joachim99@69 1006 break; // End of the history
joachim99@69 1007 }
joachim99@69 1008 }
joachim99@69 1009 }
joachim99@69 1010
joachim99@69 1011 bool findParenthesesGroups( const QString& s, QStringList& sl )
joachim99@69 1012 {
joachim99@69 1013 sl.clear();
joachim99@69 1014 int i=0;
joachim99@69 1015 std::list<int> startPosStack;
joachim99@69 1016 int length = s.length();
joachim99@69 1017 for( i=0; i<length; ++i )
joachim99@69 1018 {
joachim99@69 1019 if ( s[i]=='\\' && i+1<length && ( s[i+1]=='\\' || s[i+1]=='(' || s[i+1]==')' ) )
joachim99@69 1020 {
joachim99@69 1021 ++i;
joachim99@69 1022 continue;
joachim99@69 1023 }
joachim99@69 1024 if ( s[i]=='(' )
joachim99@69 1025 {
joachim99@69 1026 startPosStack.push_back(i);
joachim99@69 1027 }
joachim99@69 1028 else if ( s[i]==')' )
joachim99@69 1029 {
joachim99@69 1030 if (startPosStack.empty())
joachim99@69 1031 return false; // Parentheses don't match
joachim99@69 1032 int startPos = startPosStack.back();
joachim99@69 1033 startPosStack.pop_back();
joachim99@69 1034 sl.push_back( s.mid( startPos+1, i-startPos-1 ) );
joachim99@69 1035 }
joachim99@69 1036 }
joachim99@69 1037 return startPosStack.empty(); // false if parentheses don't match
joachim99@69 1038 }
joachim99@69 1039
joachim99@69 1040 QString calcHistorySortKey( const QString& keyOrder, QRegExp& matchedRegExpr, const QStringList& parenthesesGroupList )
joachim99@69 1041 {
joachim99@69 1042 QStringList keyOrderList = QStringList::split(',', keyOrder );
joachim99@69 1043 QString key;
joachim99@69 1044 for ( QStringList::iterator keyIt = keyOrderList.begin(); keyIt!=keyOrderList.end(); ++keyIt )
joachim99@69 1045 {
joachim99@69 1046 if ( (*keyIt).isEmpty() )
joachim99@69 1047 continue;
joachim99@69 1048 bool bOk=false;
joachim99@69 1049 int groupIdx = (*keyIt).toInt(&bOk);
joachim99@69 1050 if (!bOk || groupIdx<0 || groupIdx >(int)parenthesesGroupList.size() )
joachim99@69 1051 continue;
joachim99@69 1052 QString s = matchedRegExpr.cap( groupIdx );
joachim99@69 1053 if ( groupIdx == 0 )
joachim99@69 1054 {
joachim99@69 1055 key += s + " ";
joachim99@69 1056 continue;
joachim99@69 1057 }
joachim99@69 1058
joachim99@69 1059 QString groupRegExp = parenthesesGroupList[groupIdx-1];
joachim99@69 1060 if( groupRegExp.find('|')<0 || groupRegExp.find('(')>=0 )
joachim99@69 1061 {
joachim99@69 1062 bool bOk = false;
joachim99@69 1063 int i = s.toInt( &bOk );
joachim99@69 1064 if ( bOk && i>=0 && i<10000 )
joachim99@69 1065 s.sprintf("%04d", i); // This should help for correct sorting of numbers.
joachim99@69 1066 key += s + " ";
joachim99@69 1067 }
joachim99@69 1068 else
joachim99@69 1069 {
joachim99@69 1070 // Assume that the groupRegExp consists of something like "Jan|Feb|Mar|Apr"
joachim99@69 1071 // s is the string that managed to match.
joachim99@69 1072 // Now we want to know at which position it occurred. e.g. Jan=0, Feb=1, Mar=2, etc.
joachim99@69 1073 QStringList sl = QStringList::split( '|', groupRegExp );
joachim99@69 1074 int idx = sl.findIndex( s );
joachim99@69 1075 if (idx<0)
joachim99@69 1076 {
joachim99@69 1077 // Didn't match
joachim99@69 1078 }
joachim99@69 1079 else
joachim99@69 1080 {
joachim99@69 1081 QString sIdx;
joachim99@69 1082 sIdx.sprintf("%02d", idx+1 ); // Up to 99 words in the groupRegExp (more than 12 aren't expected)
joachim99@69 1083 key += sIdx + " ";
joachim99@69 1084 }
joachim99@69 1085 }
joachim99@69 1086 }
joachim99@69 1087 return key;
joachim99@69 1088 }
joachim99@69 1089
joachim99@69 1090 void MergeResultWindow::collectHistoryInformation(
joachim99@69 1091 int src, Diff3LineList::const_iterator iHistoryBegin, Diff3LineList::const_iterator iHistoryEnd,
joachim99@69 1092 HistoryMap& historyMap,
joachim99@69 1093 std::list< HistoryMap::iterator >& hitList // list of iterators
joachim99@69 1094 )
joachim99@69 1095 {
joachim99@69 1096 std::list< HistoryMap::iterator >::iterator itHitListFront = hitList.begin();
joachim99@69 1097 Diff3LineList::const_iterator id3l = iHistoryBegin;
joachim99@69 1098 QString historyLead;
joachim99@69 1099 {
joachim99@69 1100 const LineData* pld = id3l->getLineData(src);
joachim99@69 1101 QString s( pld->pLine, pld->size );
joachim99@69 1102 historyLead = s.section(' ',0,0,QString::SectionIncludeLeadingSep);
joachim99@69 1103 }
joachim99@69 1104 QRegExp historyStart = m_pOptionDialog->m_historyStartRegExp;
joachim99@69 1105 ++id3l; // Skip line with "$Log ... $"
joachim99@69 1106 QRegExp newHistoryEntry = m_pOptionDialog->m_historyEntryStartRegExp;
joachim99@69 1107 QStringList parenthesesGroups;
joachim99@69 1108 findParenthesesGroups( m_pOptionDialog->m_historyEntryStartRegExp, parenthesesGroups );
joachim99@69 1109 QString key;
joachim99@69 1110 MergeEditLineList melList;
joachim99@69 1111 bool bPrevLineIsEmpty = true;
joachim99@69 1112 bool bUseRegExp = !m_pOptionDialog->m_historyEntryStartRegExp.isEmpty();
joachim99@69 1113 for(; id3l != iHistoryEnd; ++id3l )
joachim99@69 1114 {
joachim99@69 1115 const LineData* pld = id3l->getLineData(src);
joachim99@69 1116 if ( !pld ) continue;
joachim99@69 1117 QString s( pld->pLine, pld->size );
joachim99@69 1118 if (historyLead.isNull()) historyLead = s.section(' ',0,0,QString::SectionIncludeLeadingSep);
joachim99@69 1119 QString sLine = s.mid(historyLead.length());
joachim99@69 1120 if ( ( !bUseRegExp && !sLine.stripWhiteSpace().isEmpty() && bPrevLineIsEmpty )
joachim99@69 1121 || bUseRegExp && newHistoryEntry.exactMatch( sLine )
joachim99@69 1122 )
joachim99@69 1123 {
joachim99@69 1124 if ( !key.isEmpty() && !melList.empty() )
joachim99@69 1125 {
joachim99@69 1126 // Only insert new HistoryMapEntry if key not found; in either case p.first is a valid iterator to element key.
joachim99@69 1127 std::pair<HistoryMap::iterator, bool> p = historyMap.insert(HistoryMap::value_type(key,HistoryMapEntry()));
joachim99@69 1128 HistoryMapEntry& hme = p.first->second;
joachim99@69 1129 if ( src==A ) hme.mellA = melList;
joachim99@69 1130 if ( src==B ) hme.mellB = melList;
joachim99@69 1131 if ( src==C ) hme.mellC = melList;
joachim99@69 1132 if ( p.second ) // Not in list yet?
joachim99@69 1133 {
joachim99@69 1134 hitList.insert( itHitListFront, p.first );
joachim99@69 1135 }
joachim99@69 1136 }
joachim99@69 1137
joachim99@69 1138 if ( ! bUseRegExp )
joachim99@69 1139 key = sLine;
joachim99@69 1140 else
joachim99@69 1141 key = calcHistorySortKey(m_pOptionDialog->m_historyEntryStartSortKeyOrder,newHistoryEntry,parenthesesGroups);
joachim99@69 1142
joachim99@69 1143 melList.clear();
joachim99@69 1144 melList.push_back( MergeEditLine(id3l,src) );
joachim99@69 1145 }
joachim99@69 1146 else if ( ! historyStart.exactMatch( s ) )
joachim99@69 1147 {
joachim99@69 1148 melList.push_back( MergeEditLine(id3l,src) );
joachim99@69 1149 }
joachim99@69 1150
joachim99@69 1151 bPrevLineIsEmpty = sLine.stripWhiteSpace().isEmpty();
joachim99@69 1152 }
joachim99@69 1153 if ( !key.isEmpty() )
joachim99@69 1154 {
joachim99@69 1155 // Only insert new HistoryMapEntry if key not found; in either case p.first is a valid iterator to element key.
joachim99@69 1156 std::pair<HistoryMap::iterator, bool> p = historyMap.insert(HistoryMap::value_type(key,HistoryMapEntry()));
joachim99@69 1157 HistoryMapEntry& hme = p.first->second;
joachim99@69 1158 if ( src==A ) hme.mellA = melList;
joachim99@69 1159 if ( src==B ) hme.mellB = melList;
joachim99@69 1160 if ( src==C ) hme.mellC = melList;
joachim99@69 1161 if ( p.second ) // Not in list yet?
joachim99@69 1162 {
joachim99@69 1163 hitList.insert( itHitListFront, p.first );
joachim99@69 1164 }
joachim99@69 1165 }
joachim99@69 1166 // End of the history
joachim99@69 1167 }
joachim99@69 1168
joachim99@69 1169 MergeResultWindow::MergeEditLineList& MergeResultWindow::HistoryMapEntry::choice( bool bThreeInputs )
joachim99@69 1170 {
joachim99@69 1171 if ( !bThreeInputs )
joachim99@69 1172 return mellA.empty() ? mellB : mellA;
joachim99@69 1173 else
joachim99@69 1174 {
joachim99@69 1175 if ( mellA.empty() )
joachim99@69 1176 return mellC.empty() ? mellB : mellC; // A doesn't exist, return one that exists
joachim99@69 1177 else if ( ! mellB.empty() && ! mellC.empty() )
joachim99@69 1178 { // A, B and C exist
joachim99@69 1179 return mellA;
joachim99@69 1180 }
joachim99@69 1181 else
joachim99@69 1182 return mellB.empty() ? mellB : mellC; // A exists, return the one that doesn't exist
joachim99@69 1183 }
joachim99@69 1184 }
joachim99@69 1185
joachim99@69 1186 bool MergeResultWindow::HistoryMapEntry::staysInPlace( bool bThreeInputs, Diff3LineList::const_iterator& iHistoryEnd )
joachim99@69 1187 {
joachim99@69 1188 // The entry should stay in place if the decision made by the automerger is correct.
joachim99@69 1189 Diff3LineList::const_iterator& iHistoryLast = iHistoryEnd;
joachim99@69 1190 --iHistoryLast;
joachim99@69 1191 if ( !bThreeInputs )
joachim99@69 1192 {
joachim99@69 1193 if ( !mellA.empty() && !mellB.empty() && mellA.begin()->id3l()==mellB.begin()->id3l() &&
joachim99@69 1194 mellA.back().id3l() == iHistoryLast && mellB.back().id3l() == iHistoryLast )
joachim99@69 1195 {
joachim99@69 1196 iHistoryEnd = mellA.begin()->id3l();
joachim99@69 1197 return true;
joachim99@69 1198 }
joachim99@69 1199 else
joachim99@69 1200 {
joachim99@69 1201 return false;
joachim99@69 1202 }
joachim99@69 1203 }
joachim99@69 1204 else
joachim99@69 1205 {
joachim99@69 1206 if ( !mellA.empty() && !mellB.empty() && !mellC.empty()
joachim99@69 1207 && mellA.begin()->id3l()==mellB.begin()->id3l() && mellA.begin()->id3l()==mellC.begin()->id3l()
joachim99@69 1208 && mellA.back().id3l() == iHistoryLast && mellB.back().id3l() == iHistoryLast && mellC.back().id3l() == iHistoryLast )
joachim99@69 1209 {
joachim99@69 1210 iHistoryEnd = mellA.begin()->id3l();
joachim99@69 1211 return true;
joachim99@69 1212 }
joachim99@69 1213 else
joachim99@69 1214 {
joachim99@69 1215 return false;
joachim99@69 1216 }
joachim99@69 1217 }
joachim99@69 1218 }
joachim99@69 1219
joachim99@69 1220 void MergeResultWindow::slotMergeHistory()
joachim99@69 1221 {
joachim99@69 1222 Diff3LineList::const_iterator iD3LHistoryBegin;
joachim99@69 1223 Diff3LineList::const_iterator iD3LHistoryEnd;
joachim99@69 1224 int d3lHistoryBeginLineIdx = -1;
joachim99@69 1225 int d3lHistoryEndLineIdx = -1;
joachim99@69 1226
joachim99@69 1227 // Search for history start, history end in the diff3LineList
joachim99@69 1228 findHistoryRange( m_pldC!=0, m_pDiff3LineList, iD3LHistoryBegin, iD3LHistoryEnd, d3lHistoryBeginLineIdx, d3lHistoryEndLineIdx );
joachim99@69 1229
joachim99@69 1230 if ( iD3LHistoryBegin != m_pDiff3LineList->end() )
joachim99@69 1231 {
joachim99@69 1232 // Now collect the historyMap information
joachim99@69 1233 HistoryMap historyMap;
joachim99@69 1234 std::list< HistoryMap::iterator > hitList;
joachim99@69 1235 if (m_pldC==0)
joachim99@69 1236 {
joachim99@69 1237 collectHistoryInformation( A, iD3LHistoryBegin, iD3LHistoryEnd, historyMap, hitList );
joachim99@69 1238 collectHistoryInformation( B, iD3LHistoryBegin, iD3LHistoryEnd, historyMap, hitList );
joachim99@69 1239 }
joachim99@69 1240 else
joachim99@69 1241 {
joachim99@69 1242 collectHistoryInformation( A, iD3LHistoryBegin, iD3LHistoryEnd, historyMap, hitList );
joachim99@69 1243 collectHistoryInformation( B, iD3LHistoryBegin, iD3LHistoryEnd, historyMap, hitList );
joachim99@69 1244 collectHistoryInformation( C, iD3LHistoryBegin, iD3LHistoryEnd, historyMap, hitList );
joachim99@69 1245 }
joachim99@69 1246
joachim99@69 1247 Diff3LineList::const_iterator iD3LHistoryOrigEnd = iD3LHistoryEnd;
joachim99@69 1248
joachim99@69 1249 bool bHistoryMergeSorting = m_pOptionDialog->m_bHistoryMergeSorting && ! m_pOptionDialog->m_historyEntryStartSortKeyOrder.isEmpty() &&
joachim99@69 1250 ! m_pOptionDialog->m_historyEntryStartRegExp.isEmpty();
joachim99@69 1251
joachim99@69 1252 // Remove parts from the historyMap and hitList that stay in place
joachim99@69 1253 if ( bHistoryMergeSorting )
joachim99@69 1254 {
joachim99@69 1255 while ( ! historyMap.empty() )
joachim99@69 1256 {
joachim99@69 1257 HistoryMap::iterator hMapIt = historyMap.begin();
joachim99@69 1258 if( hMapIt->second.staysInPlace( m_pldC!=0, iD3LHistoryEnd ) )
joachim99@69 1259 historyMap.erase(hMapIt);
joachim99@69 1260 else
joachim99@69 1261 break;
joachim99@69 1262 }
joachim99@69 1263 }
joachim99@69 1264 else
joachim99@69 1265 {
joachim99@69 1266 while ( ! hitList.empty() )
joachim99@69 1267 {
joachim99@69 1268 HistoryMap::iterator hMapIt = hitList.back();
joachim99@69 1269 if( hMapIt->second.staysInPlace( m_pldC!=0, iD3LHistoryEnd ) )
joachim99@69 1270 hitList.pop_back();
joachim99@69 1271 else
joachim99@69 1272 break;
joachim99@69 1273 }
joachim99@69 1274 }
joachim99@69 1275 while (iD3LHistoryOrigEnd != iD3LHistoryEnd)
joachim99@69 1276 {
joachim99@69 1277 --iD3LHistoryOrigEnd;
joachim99@69 1278 --d3lHistoryEndLineIdx;
joachim99@69 1279 }
joachim99@69 1280
joachim99@69 1281 MergeLineList::iterator iMLLStart = splitAtDiff3LineIdx(d3lHistoryBeginLineIdx);
joachim99@69 1282 MergeLineList::iterator iMLLEnd = splitAtDiff3LineIdx(d3lHistoryEndLineIdx);
joachim99@69 1283 // Now join all MergeLines in the history
joachim99@69 1284 MergeLineList::iterator i = iMLLStart;
joachim99@69 1285 if ( i != iMLLEnd )
joachim99@69 1286 {
joachim99@69 1287 ++i;
joachim99@69 1288 while ( i!=iMLLEnd )
joachim99@69 1289 {
joachim99@69 1290 iMLLStart->join(*i);
joachim99@69 1291 i = m_mergeLineList.erase( i );
joachim99@69 1292 }
joachim99@69 1293 }
joachim99@69 1294 iMLLStart->mergeEditLineList.clear();
joachim99@69 1295 // Now insert the complete history into the first MergeLine of the history
joachim99@69 1296 iMLLStart->mergeEditLineList.push_back( MergeEditLine( iD3LHistoryBegin, m_pldC == 0 ? B : C ) );
joachim99@69 1297 QString lead = iD3LHistoryBegin->getString(A).section(' ',0,0,QString::SectionIncludeLeadingSep);
joachim99@69 1298 MergeEditLine mel( m_pDiff3LineList->end() );
joachim99@69 1299 mel.setString( lead );
joachim99@69 1300 iMLLStart->mergeEditLineList.push_back(mel);
joachim99@69 1301
joachim99@69 1302 if ( bHistoryMergeSorting )
joachim99@69 1303 {
joachim99@69 1304 // Create a sorted history
joachim99@69 1305 HistoryMap::reverse_iterator hmit;
joachim99@69 1306 for ( hmit = historyMap.rbegin(); hmit != historyMap.rend(); ++hmit )
joachim99@69 1307 {
joachim99@69 1308 HistoryMapEntry& hme = hmit->second;
joachim99@69 1309 MergeEditLineList& mell = hme.choice(m_pldC!=0);
joachim99@69 1310 if (!mell.empty())
joachim99@69 1311 iMLLStart->mergeEditLineList.splice( iMLLStart->mergeEditLineList.end(), mell, mell.begin(), mell.end() );
joachim99@69 1312 }
joachim99@69 1313 }
joachim99@69 1314 else
joachim99@69 1315 {
joachim99@69 1316 // Create history in order of appearance
joachim99@69 1317 std::list< HistoryMap::iterator >::iterator hlit;
joachim99@69 1318 for ( hlit = hitList.begin(); hlit != hitList.end(); ++hlit )
joachim99@69 1319 {
joachim99@69 1320 HistoryMapEntry& hme = (*hlit)->second;
joachim99@69 1321 MergeEditLineList& mell = hme.choice(m_pldC!=0);
joachim99@69 1322 if (!mell.empty())
joachim99@69 1323 iMLLStart->mergeEditLineList.splice( iMLLStart->mergeEditLineList.end(), mell, mell.begin(), mell.end() );
joachim99@69 1324 }
joachim99@69 1325 }
joachim99@69 1326 setFastSelector( iMLLStart );
joachim99@69 1327 update();
joachim99@69 1328 }
joachim99@69 1329 }
joachim99@69 1330
joachim99@69 1331 void MergeResultWindow::slotRegExpAutoMerge()
joachim99@69 1332 {
joachim99@69 1333 if ( m_pOptionDialog->m_autoMergeRegExp.isEmpty() )
joachim99@69 1334 return;
joachim99@69 1335
joachim99@69 1336 QRegExp vcsKeywords = m_pOptionDialog->m_autoMergeRegExp;
joachim99@69 1337 MergeLineList::iterator i;
joachim99@69 1338 for ( i=m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@69 1339 {
joachim99@69 1340 if (i->bConflict )
joachim99@69 1341 {
joachim99@69 1342 Diff3LineList::const_iterator id3l = i->id3l;
joachim99@69 1343 if ( vcsKeywords.exactMatch( id3l->getString(A) ) &&
joachim99@69 1344 vcsKeywords.exactMatch( id3l->getString(B) ) &&
joachim99@69 1345 (m_pldC==0 || vcsKeywords.exactMatch( id3l->getString(C) )))
joachim99@69 1346 {
joachim99@69 1347 MergeEditLine& mel = *i->mergeEditLineList.begin();
joachim99@69 1348 mel.setSource( m_pldC==0 ? B : C, false );
joachim99@69 1349 splitAtDiff3LineIdx( i->d3lLineIdx+1 );
joachim99@69 1350 }
joachim99@69 1351 }
joachim99@69 1352 }
joachim99@69 1353 update();
joachim99@69 1354 }
joachim99@69 1355
joachim99@69 1356 // This doesn't detect user modifications and should only be called after automatic merge
joachim99@69 1357 // This will only do something for three file merge.
joachim99@69 1358 // Irrelevant changes are those where all contributions from B are already contained in C.
joachim99@69 1359 // Also irrelevant are conflicts automatically solved (automerge regexp and history automerge)
joachim99@69 1360 // Precondition: The VCS-keyword would also be C.
joachim99@69 1361 bool MergeResultWindow::doRelevantChangesExist()
joachim99@69 1362 {
joachim99@69 1363 if ( m_pldC==0 || m_mergeLineList.size() <= 1 )
joachim99@69 1364 return true;
joachim99@69 1365
joachim99@69 1366 MergeLineList::iterator i;
joachim99@69 1367 for ( i=m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@69 1368 {
joachim99@69 1369 if ( ( i->bConflict && i->mergeEditLineList.begin()->src()!=C )
joachim99@69 1370 || i->srcSelect == B )
joachim99@69 1371 {
joachim99@69 1372 return true;
joachim99@69 1373 }
joachim99@69 1374 }
joachim99@69 1375
joachim99@69 1376 return false;
joachim99@69 1377 }
joachim99@69 1378
joachim99@69 1379 // Returns the iterator to the MergeLine after the split
joachim99@69 1380 MergeResultWindow::MergeLineList::iterator MergeResultWindow::splitAtDiff3LineIdx( int d3lLineIdx )
joachim99@69 1381 {
joachim99@69 1382 MergeLineList::iterator i;
joachim99@69 1383 for ( i = m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@69 1384 {
joachim99@69 1385 if ( i->d3lLineIdx==d3lLineIdx )
joachim99@69 1386 {
joachim99@69 1387 // No split needed, this is the beginning of a MergeLine
joachim99@69 1388 return i;
joachim99@69 1389 }
joachim99@69 1390 else if ( i->d3lLineIdx > d3lLineIdx )
joachim99@69 1391 {
joachim99@69 1392 // The split must be in the previous MergeLine
joachim99@69 1393 --i;
joachim99@69 1394 MergeLine& ml = *i;
joachim99@69 1395 MergeLine newML;
joachim99@69 1396 ml.split(newML,d3lLineIdx);
joachim99@69 1397 ++i;
joachim99@69 1398 return m_mergeLineList.insert( i, newML );
joachim99@69 1399 }
joachim99@69 1400 }
joachim99@69 1401 // The split must be in the previous MergeLine
joachim99@69 1402 --i;
joachim99@69 1403 MergeLine& ml = *i;
joachim99@69 1404 MergeLine newML;
joachim99@69 1405 ml.split(newML,d3lLineIdx);
joachim99@69 1406 ++i;
joachim99@69 1407 return m_mergeLineList.insert( i, newML );
joachim99@69 1408 }
joachim99@69 1409
joachim99@69 1410 void MergeResultWindow::slotSplitDiff( int firstD3lLineIdx, int lastD3lLineIdx )
joachim99@69 1411 {
joachim99@69 1412 if (lastD3lLineIdx>=0)
joachim99@69 1413 splitAtDiff3LineIdx( lastD3lLineIdx + 1 );
joachim99@69 1414 setFastSelector( splitAtDiff3LineIdx(firstD3lLineIdx) );
joachim99@69 1415 }
joachim99@69 1416
joachim99@69 1417 void MergeResultWindow::slotJoinDiffs( int firstD3lLineIdx, int lastD3lLineIdx )
joachim99@69 1418 {
joachim99@69 1419 MergeLineList::iterator i;
joachim99@69 1420 MergeLineList::iterator iMLLStart = m_mergeLineList.end();
joachim99@69 1421 MergeLineList::iterator iMLLEnd = m_mergeLineList.end();
joachim99@69 1422 for ( i=m_mergeLineList.begin(); i!=m_mergeLineList.end(); ++i )
joachim99@69 1423 {
joachim99@69 1424 MergeLine& ml = *i;
joachim99@69 1425 if ( firstD3lLineIdx >= ml.d3lLineIdx && firstD3lLineIdx < ml.d3lLineIdx + ml.srcRangeLength )
joachim99@69 1426 {
joachim99@69 1427 iMLLStart = i;
joachim99@69 1428 }
joachim99@69 1429 if ( lastD3lLineIdx >= ml.d3lLineIdx && lastD3lLineIdx < ml.d3lLineIdx + ml.srcRangeLength )
joachim99@69 1430 {
joachim99@69 1431 iMLLEnd = i;
joachim99@69 1432 ++iMLLEnd;
joachim99@69 1433 break;
joachim99@69 1434 }
joachim99@69 1435 }
joachim99@69 1436
joachim99@69 1437 bool bJoined = false;
joachim99@69 1438 for( i=iMLLStart; i!=iMLLEnd && i!=m_mergeLineList.end(); )
joachim99@69 1439 {
joachim99@69 1440 if ( i==iMLLStart )
joachim99@69 1441 {
joachim99@69 1442 ++i;
joachim99@69 1443 }
joachim99@69 1444 else
joachim99@69 1445 {
joachim99@69 1446 iMLLStart->join(*i);
joachim99@69 1447 i = m_mergeLineList.erase( i );
joachim99@69 1448 bJoined = true;
joachim99@69 1449 }
joachim99@69 1450 }
joachim99@69 1451 if (bJoined)
joachim99@69 1452 {
joachim99@69 1453 iMLLStart->mergeEditLineList.clear();
joachim99@69 1454 // Insert a conflict line as placeholder
joachim99@69 1455 iMLLStart->mergeEditLineList.push_back( MergeEditLine( iMLLStart->id3l ) );
joachim99@69 1456 }
joachim99@69 1457 setFastSelector( iMLLStart );
joachim99@8 1458 }
joachim99@8 1459
joachim99@8 1460 void MergeResultWindow::myUpdate(int afterMilliSecs)
joachim99@8 1461 {
joachim99@69 1462 killTimer(m_delayedDrawTimer);
joachim99@8 1463 m_bMyUpdate = true;
joachim99@69 1464 m_delayedDrawTimer = startTimer( afterMilliSecs );
joachim99@8 1465 }
joachim99@8 1466
joachim99@8 1467 void MergeResultWindow::timerEvent(QTimerEvent*)
joachim99@8 1468 {
joachim99@69 1469 killTimer(m_delayedDrawTimer);
joachim99@69 1470 m_delayedDrawTimer = 0;
joachim99@8 1471
joachim99@8 1472 if ( m_bMyUpdate )
joachim99@8 1473 {
joachim99@69 1474 update();
joachim99@8 1475 m_bMyUpdate = false;
joachim99@8 1476 }
joachim99@8 1477
joachim99@8 1478 if ( m_scrollDeltaX != 0 || m_scrollDeltaY != 0 )
joachim99@8 1479 {
joachim99@8 1480 m_selection.end( m_selection.lastLine + m_scrollDeltaY, m_selection.lastPos + m_scrollDeltaX );
joachim99@8 1481 emit scroll( m_scrollDeltaX, m_scrollDeltaY );
joachim99@69 1482 killTimer(m_delayedDrawTimer);
joachim99@69 1483 m_delayedDrawTimer = startTimer(50);
joachim99@8 1484 }
joachim99@8 1485 }
joachim99@8 1486
joachim99@68 1487 QString MergeResultWindow::MergeEditLine::getString( const MergeResultWindow* mrw )
joachim99@8 1488 {
joachim99@68 1489 if ( isRemoved() ) { return QString(); }
joachim99@8 1490
joachim99@8 1491 if ( ! isModified() )
joachim99@8 1492 {
joachim99@8 1493 int src = m_src;
joachim99@8 1494 const Diff3Line& d3l = *m_id3l;
joachim99@68 1495 if ( src == 0 ) { return QString(); }
joachim99@8 1496
joachim99@8 1497 const LineData* pld = 0;
joachim99@8 1498 assert( src == A || src == B || src == C );
joachim99@8 1499 if ( src == A && d3l.lineA!=-1 ) pld = &mrw->m_pldA[ d3l.lineA ];
joachim99@8 1500 else if ( src == B && d3l.lineB!=-1 ) pld = &mrw->m_pldB[ d3l.lineB ];
joachim99@8 1501 else if ( src == C && d3l.lineC!=-1 ) pld = &mrw->m_pldC[ d3l.lineC ];
joachim99@8 1502
joachim99@8 1503 if ( pld == 0 )
joachim99@8 1504 {
joachim99@8 1505 // assert(false); This is no error.
joachim99@68 1506 return QString();
joachim99@8 1507 }
joachim99@8 1508
joachim99@68 1509 return QString( pld->pLine, pld->size );
joachim99@8 1510 }
joachim99@8 1511 else
joachim99@8 1512 {
joachim99@8 1513 return m_str;
joachim99@8 1514 }
joachim99@8 1515 return 0;
joachim99@8 1516 }
joachim99@8 1517
joachim99@8 1518 /// Converts the cursor-posOnScreen into a text index, considering tabulators.
joachim99@69 1519 int convertToPosInText( const QString& s, int posOnScreen, int tabSize )
joachim99@8 1520 {
joachim99@8 1521 int localPosOnScreen = 0;
joachim99@68 1522 int size=s.length();
joachim99@8 1523 for ( int i=0; i<size; ++i )
joachim99@8 1524 {
joachim99@8 1525 if ( localPosOnScreen>=posOnScreen )
joachim99@8 1526 return i;
joachim99@8 1527
joachim99@8 1528 // All letters except tabulator have width one.
joachim99@69 1529 int letterWidth = s[i]!='\t' ? 1 : tabber( localPosOnScreen, tabSize );
joachim99@8 1530
joachim99@8 1531 localPosOnScreen += letterWidth;
joachim99@8 1532
joachim99@8 1533 if ( localPosOnScreen>posOnScreen )
joachim99@8 1534 return i;
joachim99@8 1535 }
joachim99@8 1536 return size;
joachim99@8 1537 }
joachim99@8 1538
joachim99@8 1539
joachim99@8 1540 /// Converts the index into the text to a cursor-posOnScreen considering tabulators.
joachim99@69 1541 int convertToPosOnScreen( const QString& p, int posInText, int tabSize )
joachim99@8 1542 {
joachim99@8 1543 int posOnScreen = 0;
joachim99@8 1544 for ( int i=0; i<posInText; ++i )
joachim99@8 1545 {
joachim99@8 1546 // All letters except tabulator have width one.
joachim99@69 1547 int letterWidth = p[i]!='\t' ? 1 : tabber( posOnScreen, tabSize );
joachim99@8 1548
joachim99@8 1549 posOnScreen += letterWidth;
joachim99@8 1550 }
joachim99@8 1551 return posOnScreen;
joachim99@8 1552 }
joachim99@8 1553
joachim99@8 1554 void MergeResultWindow::writeLine(
joachim99@68 1555 MyPainter& p, int line, const QString& str,
joachim99@68 1556 int srcSelect, e_MergeDetails mergeDetails, int rangeMark, bool bUserModified, bool bLineRemoved, bool bWhiteSpaceConflict
joachim99@8 1557 )
joachim99@8 1558 {
joachim99@8 1559 const QFontMetrics& fm = fontMetrics();
joachim99@8 1560 int fontHeight = fm.height();
joachim99@8 1561 int fontWidth = fm.width("W");
joachim99@8 1562 int fontAscent = fm.ascent();
joachim99@8 1563
joachim99@8 1564 int topLineYOffset = fontHeight + 3;
joachim99@8 1565 int xOffset = fontWidth * leftInfoWidth;
joachim99@8 1566
joachim99@8 1567 int yOffset = ( line-m_firstLine ) * fontHeight;
joachim99@8 1568 if ( yOffset < 0 || yOffset > height() )
joachim99@8 1569 return;
joachim99@8 1570
joachim99@8 1571 yOffset += topLineYOffset;
joachim99@8 1572
joachim99@8 1573 QString srcName = " ";
joachim99@8 1574 if ( bUserModified ) srcName = "m";
joachim99@8 1575 else if ( srcSelect == A && mergeDetails != eNoChange ) srcName = "A";
joachim99@8 1576 else if ( srcSelect == B ) srcName = "B";
joachim99@8 1577 else if ( srcSelect == C ) srcName = "C";
joachim99@8 1578
joachim99@8 1579 if ( rangeMark & 4 )
joachim99@8 1580 {
joachim99@8 1581 p.fillRect( xOffset, yOffset, width(), fontHeight, m_pOptionDialog->m_currentRangeBgColor );
joachim99@8 1582 }
joachim99@68 1583
joachim99@8 1584 if( (srcSelect > 0 || bUserModified ) && !bLineRemoved )
joachim99@8 1585 {
joachim99@8 1586 int outPos = 0;
joachim99@68 1587 QString s;
joachim99@68 1588 int size = str.length();
joachim99@8 1589 for ( int i=0; i<size; ++i )
joachim99@8 1590 {
joachim99@8 1591 int spaces = 1;
joachim99@68 1592 if ( str[i]=='\t' )
joachim99@8 1593 {
joachim99@69 1594 spaces = tabber( outPos, m_pOptionDialog->m_tabSize );
joachim99@8 1595 for( int j=0; j<spaces; ++j )
joachim99@8 1596 s+=' ';
joachim99@8 1597 }
joachim99@8 1598 else
joachim99@8 1599 {
joachim99@68 1600 s+=str[i];
joachim99@8 1601 }
joachim99@8 1602 outPos += spaces;
joachim99@8 1603 }
joachim99@8 1604
joachim99@8 1605 if ( m_selection.lineWithin( line ) )
joachim99@8 1606 {
joachim99@69 1607 int firstPosInLine = convertToPosOnScreen( str, convertToPosInText( str, m_selection.firstPosInLine(line), m_pOptionDialog->m_tabSize ),m_pOptionDialog->m_tabSize );
joachim99@69 1608 int lastPosInLine = convertToPosOnScreen( str, convertToPosInText( str, m_selection.lastPosInLine(line), m_pOptionDialog->m_tabSize ), m_pOptionDialog->m_tabSize );
joachim99@8 1609 int lengthInLine = max2(0,lastPosInLine - firstPosInLine);
joachim99@8 1610 if (lengthInLine>0) m_selection.bSelectionContainsData = true;
joachim99@8 1611
joachim99@8 1612 if ( lengthInLine < int(s.length()) )
joachim99@8 1613 { // Draw a normal line first
joachim99@8 1614 p.setPen( m_pOptionDialog->m_fgColor );
joachim99@68 1615 p.drawText( xOffset, yOffset+fontAscent, s.mid(m_firstColumn), true );
joachim99@8 1616 }
joachim99@8 1617 int firstPosInLine2 = max2( firstPosInLine, m_firstColumn );
joachim99@8 1618 int lengthInLine2 = max2(0,lastPosInLine - firstPosInLine2);
joachim99@8 1619
joachim99@8 1620 if( m_selection.lineWithin( line+1 ) )
joachim99@8 1621 p.fillRect( xOffset + fontWidth*(firstPosInLine2-m_firstColumn), yOffset,
joachim99@8 1622 width(), fontHeight, colorGroup().highlight() );
joachim99@69 1623 else if ( lengthInLine2>0 )
joachim99@8 1624 p.fillRect( xOffset + fontWidth*(firstPosInLine2-m_firstColumn), yOffset,
joachim99@8 1625 fontWidth*lengthInLine2, fontHeight, colorGroup().highlight() );
joachim99@8 1626
joachim99@8 1627 p.setPen( colorGroup().highlightedText() );
joachim99@8 1628 p.drawText( xOffset + fontWidth*(firstPosInLine2-m_firstColumn), yOffset+fontAscent,
joachim99@68 1629 s.mid(firstPosInLine2,lengthInLine2), true );
joachim99@8 1630 }
joachim99@8 1631 else
joachim99@8 1632 {
joachim99@8 1633 p.setPen( m_pOptionDialog->m_fgColor );
joachim99@68 1634 p.drawText( xOffset, yOffset+fontAscent, s.mid(m_firstColumn), true );
joachim99@8 1635 }
joachim99@8 1636
joachim99@8 1637 p.setPen( m_pOptionDialog->m_fgColor );
joachim99@8 1638 if ( m_cursorYPos==line )
joachim99@8 1639 {
joachim99@8 1640 m_cursorXPos = minMaxLimiter( m_cursorXPos, 0, outPos );
joachim99@69 1641 m_cursorXPos = convertToPosOnScreen( str, convertToPosInText( str, m_cursorXPos, m_pOptionDialog->m_tabSize ),m_pOptionDialog->m_tabSize );
joachim99@8 1642 }
joachim99@8 1643
joachim99@68 1644 p.drawText( 1, yOffset+fontAscent, srcName, true );
joachim99@8 1645 }
joachim99@8 1646 else if ( bLineRemoved )
joachim99@8 1647 {
joachim99@8 1648 p.setPen( m_pOptionDialog->m_colorForConflict );
joachim99@8 1649 p.drawText( xOffset, yOffset+fontAscent, i18n("<No src line>") );
joachim99@8 1650 p.drawText( 1, yOffset+fontAscent, srcName );
joachim99@8 1651 if ( m_cursorYPos==line ) m_cursorXPos = 0;
joachim99@8 1652 }
joachim99@8 1653 else if ( srcSelect == 0 )
joachim99@8 1654 {
joachim99@8 1655 p.setPen( m_pOptionDialog->m_colorForConflict );
joachim99@68 1656 if ( bWhiteSpaceConflict )
joachim99@68 1657 p.drawText( xOffset, yOffset+fontAscent, i18n("<Merge Conflict (Whitespace only)>") );
joachim99@68 1658 else
joachim99@68 1659 p.drawText( xOffset, yOffset+fontAscent, i18n("<Merge Conflict>") );
joachim99@8 1660 p.drawText( 1, yOffset+fontAscent, "?" );
joachim99@8 1661 if ( m_cursorYPos==line ) m_cursorXPos = 0;
joachim99@8 1662 }
joachim99@8 1663 else assert(false);
joachim99@8 1664
joachim99@8 1665 xOffset -= fontWidth;
joachim99@8 1666 p.setPen( m_pOptionDialog->m_fgColor );
joachim99@8 1667 if ( rangeMark & 1 ) // begin mark
joachim99@8 1668 {
joachim99@8 1669 p.drawLine( xOffset, yOffset+1, xOffset, yOffset+fontHeight/2 );
joachim99@8 1670 p.drawLine( xOffset, yOffset+1, xOffset-2, yOffset+1 );
joachim99@8 1671 }
joachim99@8 1672 else
joachim99@8 1673 {
joachim99@8 1674 p.drawLine( xOffset, yOffset, xOffset, yOffset+fontHeight/2 );
joachim99@8 1675 }
joachim99@8 1676
joachim99@8 1677 if ( rangeMark & 2 ) // end mark
joachim99@8 1678 {
joachim99@8 1679 p.drawLine( xOffset, yOffset+fontHeight/2, xOffset, yOffset+fontHeight-1 );
joachim99@8 1680 p.drawLine( xOffset, yOffset+fontHeight-1, xOffset-2, yOffset+fontHeight-1 );
joachim99@8 1681 }
joachim99@8 1682 else
joachim99@8 1683 {
joachim99@8 1684 p.drawLine( xOffset, yOffset+fontHeight/2, xOffset, yOffset+fontHeight );
joachim99@8 1685 }
joachim99@8 1686
joachim99@8 1687 if ( rangeMark & 4 )
joachim99@8 1688 {
joachim99@8 1689 p.fillRect( xOffset + 3, yOffset, 3, fontHeight, m_pOptionDialog->m_fgColor );
joachim99@8 1690 /* p.setPen( blue );
joachim99@8 1691 p.drawLine( xOffset+2, yOffset, xOffset+2, yOffset+fontHeight-1 );
joachim99@8 1692 p.drawLine( xOffset+3, yOffset, xOffset+3, yOffset+fontHeight-1 );*/
joachim99@8 1693 }
joachim99@8 1694 }
joachim99@8 1695
joachim99@58 1696 void MergeResultWindow::setPaintingAllowed(bool bPaintingAllowed)
joachim99@58 1697 {
joachim99@58 1698 m_bPaintingAllowed = bPaintingAllowed;
joachim99@66 1699 if ( !m_bPaintingAllowed )
joachim99@69 1700 {
joachim99@66 1701 m_currentMergeLineIt = m_mergeLineList.end();
joachim99@69 1702 reset();
joachim99@69 1703 }
joachim99@58 1704 }
joachim99@58 1705
joachim99@69 1706 void MergeResultWindow::paintEvent( QPaintEvent* )
joachim99@8 1707 {
joachim99@58 1708 if (m_pDiff3LineList==0 || !m_bPaintingAllowed) return;
joachim99@8 1709
joachim99@8 1710 bool bOldSelectionContainsData = m_selection.bSelectionContainsData;
joachim99@8 1711 const QFontMetrics& fm = fontMetrics();
joachim99@8 1712 int fontHeight = fm.height();
joachim99@8 1713 int fontWidth = fm.width("W");
joachim99@8 1714 int fontAscent = fm.ascent();
joachim99@8 1715
joachim99@69 1716 if ( !m_bCursorUpdate ) // Don't redraw everything for blinking cursor?
joachim99@8 1717 {
joachim99@8 1718 m_selection.bSelectionContainsData = false;
joachim99@8 1719 if ( size() != m_pixmap.size() )
joachim99@8 1720 m_pixmap.resize(size());
joachim99@8 1721
joachim99@68 1722 MyPainter p(&m_pixmap, m_pOptionDialog->m_bRightToLeftLanguage, width(), fontWidth);
joachim99@8 1723 p.setFont( font() );
joachim99@68 1724 p.QPainter::fillRect( rect(), m_pOptionDialog->m_bgColor );
joachim99@8 1725
joachim99@8 1726 //int visibleLines = height() / fontHeight;
joachim99@8 1727
joachim99@8 1728 { // Draw the topline
joachim99@58 1729 QString s = " " +i18n("Output") + " : " + m_fileName + " ";
joachim99@8 1730 if (m_bModified)
joachim99@8 1731 s += i18n("[Modified]");
joachim99@8 1732
joachim99@8 1733 int topLineYOffset = fontHeight + 3;
joachim99@8 1734
joachim99@8 1735 if (hasFocus())
joachim99@8 1736 {
joachim99@69 1737 p.fillRect( 0, 0, width(), topLineYOffset, Qt::lightGray /*m_pOptionDialog->m_diffBgColor*/ );
joachim99@8 1738 }
joachim99@8 1739 else
joachim99@8 1740 {
joachim99@8 1741 p.fillRect( 0, 0, width(), topLineYOffset, m_pOptionDialog->m_bgColor );
joachim99@8 1742 }
joachim99@8 1743 p.setPen( m_pOptionDialog->m_fgColor );
joachim99@68 1744 p.QPainter::drawText( 0, fontAscent+1, s );
joachim99@8 1745 p.drawLine( 0, fontHeight + 2, width(), fontHeight + 2 );
joachim99@8 1746 }
joachim99@8 1747
joachim99@8 1748 int lastVisibleLine = m_firstLine + getNofVisibleLines() + 5;
joachim99@8 1749 int nofColumns = 0;
joachim99@8 1750 int line = 0;
joachim99@8 1751 MergeLineList::iterator mlIt = m_mergeLineList.begin();
joachim99@8 1752 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 1753 {
joachim99@8 1754 MergeLine& ml = *mlIt;
joachim99@8 1755 if ( line > lastVisibleLine || line + ml.mergeEditLineList.size() < m_firstLine)
joachim99@8 1756 {
joachim99@8 1757 line += ml.mergeEditLineList.size();
joachim99@8 1758 }
joachim99@8 1759 else
joachim99@8 1760 {
joachim99@8 1761 MergeEditLineList::iterator melIt;
joachim99@8 1762 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); ++melIt )
joachim99@8 1763 {
joachim99@8 1764 if (line>=m_firstLine && line<=lastVisibleLine)
joachim99@8 1765 {
joachim99@8 1766 MergeEditLine& mel = *melIt;
joachim99@8 1767 MergeEditLineList::iterator melIt1 = melIt;
joachim99@8 1768 ++melIt1;
joachim99@8 1769
joachim99@8 1770 int rangeMark = 0;
joachim99@8 1771 if ( melIt==ml.mergeEditLineList.begin() ) rangeMark |= 1; // Begin range mark
joachim99@8 1772 if ( melIt1==ml.mergeEditLineList.end() ) rangeMark |= 2; // End range mark
joachim99@8 1773
joachim99@8 1774 if ( mlIt == m_currentMergeLineIt ) rangeMark |= 4; // Mark of the current line
joachim99@8 1775
joachim99@68 1776 QString s;
joachim99@68 1777 s = mel.getString( this );
joachim99@69 1778 if ( convertToPosOnScreen(s,s.length(),m_pOptionDialog->m_tabSize) >nofColumns)
joachim99@68 1779 nofColumns = s.length();
joachim99@8 1780
joachim99@68 1781 writeLine( p, line, s, mel.src(), ml.mergeDetails, rangeMark,
joachim99@68 1782 mel.isModified(), mel.isRemoved(), ml.bWhiteSpaceConflict );
joachim99@8 1783 }
joachim99@8 1784 ++line;
joachim99@8 1785 }
joachim99@8 1786 }
joachim99@8 1787 }
joachim99@8 1788
joachim99@8 1789 if ( line != m_nofLines || nofColumns != m_nofColumns )
joachim99@8 1790 {
joachim99@8 1791 m_nofLines = line;
joachim99@58 1792 assert( m_nofLines == m_totalSize );
joachim99@8 1793
joachim99@8 1794 m_nofColumns = nofColumns;
joachim99@8 1795 emit resizeSignal();
joachim99@8 1796 }
joachim99@8 1797
joachim99@8 1798 p.end();
joachim99@8 1799 }
joachim99@8 1800
joachim99@8 1801 QPainter painter(this);
joachim99@8 1802
joachim99@8 1803 int topLineYOffset = fontHeight + 3;
joachim99@8 1804 int xOffset = fontWidth * leftInfoWidth;
joachim99@8 1805 int yOffset = ( m_cursorYPos - m_firstLine ) * fontHeight + topLineYOffset;
joachim99@8 1806 int xCursor = ( m_cursorXPos - m_firstColumn ) * fontWidth + xOffset;
joachim99@8 1807
joachim99@69 1808 if ( !m_bCursorUpdate )
joachim99@8 1809 painter.drawPixmap(0,0, m_pixmap);
joachim99@8 1810 else
joachim99@68 1811 {
joachim99@68 1812 if (!m_pOptionDialog->m_bRightToLeftLanguage)
joachim99@68 1813 painter.drawPixmap(xCursor-2, yOffset, m_pixmap,
joachim99@68 1814 xCursor-2, yOffset, 5, fontAscent+2 );
joachim99@68 1815 else
joachim99@68 1816 painter.drawPixmap(width()-1-4-(xCursor-2), yOffset, m_pixmap,
joachim99@68 1817 width()-1-4-(xCursor-2), yOffset, 5, fontAscent+2 );
joachim99@69 1818 m_bCursorUpdate = false;
joachim99@68 1819 }
joachim99@69 1820 painter.end();
joachim99@8 1821
joachim99@8 1822 if ( m_bCursorOn && hasFocus() && m_cursorYPos>=m_firstLine )
joachim99@8 1823 {
joachim99@68 1824 MyPainter painter(this, m_pOptionDialog->m_bRightToLeftLanguage, width(), fontWidth);
joachim99@8 1825 int topLineYOffset = fontHeight + 3;
joachim99@8 1826 int xOffset = fontWidth * leftInfoWidth;
joachim99@8 1827
joachim99@8 1828 int yOffset = ( m_cursorYPos-m_firstLine ) * fontHeight + topLineYOffset;
joachim99@8 1829
joachim99@8 1830 int xCursor = ( m_cursorXPos - m_firstColumn ) * fontWidth + xOffset;
joachim99@66 1831
joachim99@66 1832 painter.setPen( m_pOptionDialog->m_fgColor );
joachim99@66 1833
joachim99@8 1834 painter.drawLine( xCursor, yOffset, xCursor, yOffset+fontAscent );
joachim99@8 1835 painter.drawLine( xCursor-2, yOffset, xCursor+2, yOffset );
joachim99@8 1836 painter.drawLine( xCursor-2, yOffset+fontAscent+1, xCursor+2, yOffset+fontAscent+1 );
joachim99@8 1837 }
joachim99@8 1838
joachim99@8 1839 if( !bOldSelectionContainsData && m_selection.bSelectionContainsData )
joachim99@8 1840 emit newSelection();
joachim99@8 1841 }
joachim99@8 1842
joachim99@66 1843 void MergeResultWindow::updateSourceMask()
joachim99@66 1844 {
joachim99@66 1845 int srcMask=0;
joachim99@66 1846 int enabledMask = 0;
joachim99@66 1847 if( !hasFocus() || m_pDiff3LineList==0 || !m_bPaintingAllowed || m_currentMergeLineIt == m_mergeLineList.end() )
joachim99@66 1848 {
joachim99@66 1849 srcMask = 0;
joachim99@66 1850 enabledMask = 0;
joachim99@66 1851 }
joachim99@66 1852 else
joachim99@66 1853 {
joachim99@66 1854 enabledMask = m_pldC==0 ? 3 : 7;
joachim99@66 1855 MergeLine& ml = *m_currentMergeLineIt;
joachim99@66 1856
joachim99@66 1857 srcMask = 0;
joachim99@66 1858 bool bModified = false;
joachim99@66 1859 MergeEditLineList::iterator melIt;
joachim99@66 1860 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); ++melIt )
joachim99@66 1861 {
joachim99@66 1862 MergeEditLine& mel = *melIt;
joachim99@66 1863 if ( mel.src()==1 ) srcMask |= 1;
joachim99@66 1864 if ( mel.src()==2 ) srcMask |= 2;
joachim99@66 1865 if ( mel.src()==3 ) srcMask |= 4;
joachim99@66 1866 if ( mel.isModified() || !mel.isEditableText() ) bModified = true;
joachim99@66 1867 }
joachim99@68 1868
joachim99@66 1869 if ( ml.mergeDetails == eNoChange )
joachim99@69 1870 {
joachim99@66 1871 srcMask = 0;
joachim99@66 1872 enabledMask = bModified ? 1 : 0;
joachim99@66 1873 }
joachim99@66 1874 }
joachim99@68 1875
joachim99@66 1876 emit sourceMask( srcMask, enabledMask );
joachim99@66 1877 }
joachim99@66 1878
joachim99@66 1879 void MergeResultWindow::focusInEvent( QFocusEvent* e )
joachim99@66 1880 {
joachim99@66 1881 updateSourceMask();
joachim99@66 1882 QWidget::focusInEvent(e);
joachim99@66 1883 }
joachim99@66 1884
joachim99@8 1885 void MergeResultWindow::convertToLinePos( int x, int y, int& line, int& pos )
joachim99@8 1886 {
joachim99@8 1887 const QFontMetrics& fm = fontMetrics();
joachim99@8 1888 int fontHeight = fm.height();
joachim99@8 1889 int fontWidth = fm.width('W');
joachim99@8 1890 int xOffset = (leftInfoWidth-m_firstColumn)*fontWidth;
joachim99@8 1891 int topLineYOffset = fontHeight + 3;
joachim99@8 1892
joachim99@8 1893 int yOffset = topLineYOffset - m_firstLine * fontHeight;
joachim99@8 1894
joachim99@58 1895 line = min2( ( y - yOffset ) / fontHeight, m_totalSize-1 );
joachim99@68 1896 if ( ! m_pOptionDialog->m_bRightToLeftLanguage )
joachim99@68 1897 pos = ( x - xOffset ) / fontWidth;
joachim99@68 1898 else
joachim99@68 1899 pos = ( (width() - 1 - x) - xOffset ) / fontWidth;
joachim99@8 1900 }
joachim99@8 1901
joachim99@8 1902 void MergeResultWindow::mousePressEvent ( QMouseEvent* e )
joachim99@8 1903 {
joachim99@8 1904 m_bCursorOn = true;
joachim99@8 1905
joachim99@8 1906 int line;
joachim99@8 1907 int pos;
joachim99@8 1908 convertToLinePos( e->x(), e->y(), line, pos );
joachim99@8 1909
joachim99@69 1910 bool bLMB = e->button() == Qt::LeftButton;
joachim99@69 1911 bool bMMB = e->button() == Qt::MidButton;
joachim99@69 1912 bool bRMB = e->button() == Qt::RightButton;
joachim99@8 1913
joachim99@8 1914 if ( bLMB && pos < m_firstColumn || bRMB ) // Fast range selection
joachim99@8 1915 {
joachim99@8 1916 m_cursorXPos = 0;
joachim99@8 1917 m_cursorOldXPos = 0;
joachim99@8 1918 m_cursorYPos = max2(line,0);
joachim99@8 1919 int l = 0;
joachim99@8 1920 MergeLineList::iterator i = m_mergeLineList.begin();
joachim99@8 1921 for(i = m_mergeLineList.begin();i!=m_mergeLineList.end(); ++i)
joachim99@8 1922 {
joachim99@8 1923 if (l==line)
joachim99@8 1924 break;
joachim99@8 1925
joachim99@8 1926 l += i->mergeEditLineList.size();
joachim99@8 1927 if (l>line)
joachim99@8 1928 break;
joachim99@8 1929 }
joachim99@8 1930 m_selection.reset(); // Disable current selection
joachim99@8 1931
joachim99@8 1932 m_bCursorOn = true;
joachim99@8 1933 setFastSelector( i );
joachim99@8 1934
joachim99@8 1935 if (bRMB)
joachim99@8 1936 {
joachim99@8 1937 showPopupMenu( QCursor::pos() );
joachim99@8 1938 }
joachim99@8 1939 }
joachim99@8 1940 else if ( bLMB ) // Normal cursor placement
joachim99@8 1941 {
joachim99@8 1942 pos = max2(pos,0);
joachim99@8 1943 line = max2(line,0);
joachim99@69 1944 if ( e->state() & Qt::ShiftButton )
joachim99@8 1945 {
joachim99@8 1946 if (m_selection.firstLine==-1)
joachim99@8 1947 m_selection.start( line, pos );
joachim99@8 1948 m_selection.end( line, pos );
joachim99@8 1949 }
joachim99@8 1950 else
joachim99@8 1951 {
joachim99@8 1952 // Selection
joachim99@8 1953 m_selection.reset();
joachim99@8 1954 m_selection.start( line, pos );
joachim99@8 1955 m_selection.end( line, pos );
joachim99@8 1956 }
joachim99@8 1957 m_cursorXPos = pos;
joachim99@8 1958 m_cursorOldXPos = pos;
joachim99@8 1959 m_cursorYPos = line;
joachim99@8 1960
joachim99@8 1961 update();
joachim99@8 1962 //showStatusLine( line, m_winIdx, m_pFilename, m_pDiff3LineList, m_pStatusBar );
joachim99@8 1963 }
joachim99@8 1964 else if ( bMMB ) // Paste clipboard
joachim99@8 1965 {
joachim99@8 1966 pos = max2(pos,0);
joachim99@8 1967 line = max2(line,0);
joachim99@8 1968
joachim99@8 1969 m_selection.reset();
joachim99@8 1970 m_cursorXPos = pos;
joachim99@8 1971 m_cursorOldXPos = pos;
joachim99@8 1972 m_cursorYPos = line;
joachim99@8 1973
joachim99@66 1974 pasteClipboard( true );
joachim99@8 1975 }
joachim99@8 1976 }
joachim99@8 1977
joachim99@8 1978 void MergeResultWindow::mouseDoubleClickEvent( QMouseEvent* e )
joachim99@8 1979 {
joachim99@69 1980 if ( e->button() == Qt::LeftButton )
joachim99@8 1981 {
joachim99@8 1982 int line;
joachim99@8 1983 int pos;
joachim99@8 1984 convertToLinePos( e->x(), e->y(), line, pos );
joachim99@8 1985 m_cursorXPos = pos;
joachim99@8 1986 m_cursorOldXPos = pos;
joachim99@8 1987 m_cursorYPos = line;
joachim99@8 1988
joachim99@8 1989 // Get the string data of the current line
joachim99@8 1990
joachim99@8 1991 MergeLineList::iterator mlIt;
joachim99@8 1992 MergeEditLineList::iterator melIt;
joachim99@8 1993 calcIteratorFromLineNr( line, mlIt, melIt );
joachim99@68 1994 QString s = melIt->getString( this );
joachim99@8 1995
joachim99@68 1996 if ( !s.isEmpty() )
joachim99@8 1997 {
joachim99@8 1998 int pos1, pos2;
joachim99@8 1999
joachim99@69 2000 calcTokenPos( s, pos, pos1, pos2, m_pOptionDialog->m_tabSize );
joachim99@8 2001
joachim99@8 2002 resetSelection();
joachim99@69 2003 m_selection.start( line, convertToPosOnScreen( s, pos1, m_pOptionDialog->m_tabSize ) );
joachim99@69 2004 m_selection.end( line, convertToPosOnScreen( s, pos2, m_pOptionDialog->m_tabSize ) );
joachim99@8 2005
joachim99@8 2006 update();
joachim99@8 2007 // emit selectionEnd() happens in the mouseReleaseEvent.
joachim99@8 2008 }
joachim99@8 2009 }
joachim99@8 2010 }
joachim99@8 2011
joachim99@8 2012 void MergeResultWindow::mouseReleaseEvent ( QMouseEvent * e )
joachim99@8 2013 {
joachim99@69 2014 if ( e->button() == Qt::LeftButton )
joachim99@8 2015 {
joachim99@69 2016 killTimer(m_delayedDrawTimer);
joachim99@69 2017 m_delayedDrawTimer = 0;
joachim99@8 2018
joachim99@8 2019 if (m_selection.firstLine != -1 )
joachim99@8 2020 {
joachim99@8 2021 emit selectionEnd();
joachim99@8 2022 }
joachim99@8 2023 }
joachim99@8 2024 }
joachim99@8 2025
joachim99@8 2026 void MergeResultWindow::mouseMoveEvent ( QMouseEvent * e )
joachim99@8 2027 {
joachim99@8 2028 int line;
joachim99@8 2029 int pos;
joachim99@8 2030 convertToLinePos( e->x(), e->y(), line, pos );
joachim99@8 2031 m_cursorXPos = pos;
joachim99@8 2032 m_cursorOldXPos = pos;
joachim99@8 2033 m_cursorYPos = line;
joachim99@8 2034 if (m_selection.firstLine != -1 )
joachim99@8 2035 {
joachim99@8 2036 m_selection.end( line, pos );
joachim99@8 2037 myUpdate(0);
joachim99@8 2038
joachim99@8 2039 //showStatusLine( line, m_winIdx, m_pFilename, m_pDiff3LineList, m_pStatusBar );
joachim99@8 2040
joachim99@8 2041 // Scroll because mouse moved out of the window
joachim99@8 2042 const QFontMetrics& fm = fontMetrics();
joachim99@8 2043 int fontHeight = fm.height();
joachim99@8 2044 int fontWidth = fm.width('W');
joachim99@8 2045 int topLineYOffset = fontHeight + 3;
joachim99@8 2046 int deltaX=0;
joachim99@8 2047 int deltaY=0;
joachim99@68 2048 if ( ! m_pOptionDialog->m_bRightToLeftLanguage )
joachim99@68 2049 {
joachim99@68 2050 if ( e->x() < leftInfoWidth*fontWidth ) deltaX=-1;
joachim99@68 2051 if ( e->x() > width() ) deltaX=+1;
joachim99@68 2052 }
joachim99@68 2053 else
joachim99@68 2054 {
joachim99@68 2055 if ( e->x() > width()-1-leftInfoWidth*fontWidth ) deltaX=-1;
joachim99@68 2056 if ( e->x() < fontWidth ) deltaX=+1;
joachim99@68 2057 }
joachim99@8 2058 if ( e->y() < topLineYOffset ) deltaY=-1;
joachim99@8 2059 if ( e->y() > height() ) deltaY=+1;
joachim99@8 2060 m_scrollDeltaX = deltaX;
joachim99@8 2061 m_scrollDeltaY = deltaY;
joachim99@8 2062 if ( deltaX != 0 || deltaY!= 0)
joachim99@8 2063 {
joachim99@8 2064 emit scroll( deltaX, deltaY );
joachim99@8 2065 }
joachim99@8 2066 }
joachim99@8 2067 }
joachim99@8 2068
joachim99@8 2069
joachim99@8 2070 void MergeResultWindow::slotCursorUpdate()
joachim99@8 2071 {
joachim99@8 2072 m_cursorTimer.stop();
joachim99@8 2073 m_bCursorOn = !m_bCursorOn;
joachim99@8 2074
joachim99@8 2075 if ( isVisible() )
joachim99@69 2076 {
joachim99@69 2077 m_bCursorUpdate = true;
joachim99@69 2078
joachim99@69 2079 const QFontMetrics& fm = fontMetrics();
joachim99@69 2080 int fontWidth = fm.width("W");
joachim99@69 2081 int topLineYOffset = fm.height() + 3;
joachim99@69 2082 int xOffset = fontWidth * leftInfoWidth;
joachim99@69 2083 int yOffset = ( m_cursorYPos - m_firstLine ) * fm.height() + topLineYOffset;
joachim99@69 2084 int xCursor = ( m_cursorXPos - m_firstColumn ) * fontWidth + xOffset;
joachim99@69 2085
joachim99@69 2086 if (!m_pOptionDialog->m_bRightToLeftLanguage)
joachim99@69 2087 repaint( xCursor-2, yOffset, 5, fm.ascent()+2 );
joachim99@69 2088 else
joachim99@69 2089 repaint( width()-1-4-(xCursor-2), yOffset, 5, fm.ascent()+2 );
joachim99@69 2090
joachim99@69 2091 m_bCursorUpdate=false;
joachim99@69 2092 }
joachim99@8 2093
joachim99@8 2094 m_cursorTimer.start(500,true);
joachim99@8 2095 }
joachim99@8 2096
joachim99@8 2097
joachim99@8 2098 void MergeResultWindow::wheelEvent( QWheelEvent* e )
joachim99@8 2099 {
joachim99@8 2100 int d = -e->delta()*QApplication::wheelScrollLines()/120;
joachim99@8 2101 e->accept();
joachim99@8 2102 scroll( 0, min2(d, getNofVisibleLines()) );
joachim99@8 2103 }
joachim99@8 2104
joachim99@51 2105
joachim99@8 2106 void MergeResultWindow::keyPressEvent( QKeyEvent* e )
joachim99@8 2107 {
joachim99@8 2108 int y = m_cursorYPos;
joachim99@8 2109 MergeLineList::iterator mlIt;
joachim99@8 2110 MergeEditLineList::iterator melIt;
joachim99@8 2111 calcIteratorFromLineNr( y, mlIt, melIt );
joachim99@8 2112
joachim99@68 2113 QString str = melIt->getString( this );
joachim99@69 2114 int x = convertToPosInText( str, m_cursorXPos, m_pOptionDialog->m_tabSize );
joachim99@69 2115
joachim99@69 2116 bool bCtrl = ( e->state() & Qt::ControlButton ) != 0 ;
joachim99@69 2117 bool bShift = ( e->state() & Qt::ShiftButton ) != 0 ;
joachim99@8 2118 #ifdef _WIN32
joachim99@69 2119 bool bAlt = ( e->state() & Qt::AltButton ) != 0 ;
joachim99@8 2120 if ( bCtrl && bAlt ){ bCtrl=false; bAlt=false; } // AltGr-Key pressed.
joachim99@8 2121 #endif
joachim99@8 2122
joachim99@8 2123 bool bYMoveKey = false;
joachim99@8 2124 // Special keys
joachim99@8 2125 switch ( e->key() )
joachim99@8 2126 {
joachim99@69 2127 case Qt::Key_Escape: break;
joachim99@8 2128 //case Key_Tab: break;
joachim99@69 2129 case Qt::Key_Backtab: break;
joachim99@69 2130 case Qt::Key_Delete:
joachim99@8 2131 {
joachim99@68 2132 if ( deleteSelection2( str, x, y, mlIt, melIt )) break;
joachim99@8 2133 if( !melIt->isEditableText() ) break;
joachim99@68 2134 if (x>=(int)str.length())
joachim99@8 2135 {
joachim99@58 2136 if ( y<m_totalSize-1 )
joachim99@8 2137 {
joachim99@8 2138 setModified();
joachim99@8 2139 MergeLineList::iterator mlIt1;
joachim99@8 2140 MergeEditLineList::iterator melIt1;
joachim99@8 2141 calcIteratorFromLineNr( y+1, mlIt1, melIt1 );
joachim99@8 2142 if ( melIt1->isEditableText() )
joachim99@8 2143 {
joachim99@68 2144 QString s2 = melIt1->getString( this );
joachim99@68 2145 melIt->setString( str + s2 );
joachim99@8 2146
joachim99@8 2147 // Remove the line
joachim99@8 2148 if ( mlIt1->mergeEditLineList.size()>1 )
joachim99@8 2149 mlIt1->mergeEditLineList.erase( melIt1 );
joachim99@8 2150 else
joachim99@8 2151 melIt1->setRemoved();
joachim99@8 2152 }
joachim99@8 2153 }
joachim99@8 2154 }
joachim99@8 2155 else
joachim99@8 2156 {
joachim99@68 2157 QString s = str.left(x);
joachim99@68 2158 s += str.mid( x+1 );
joachim99@8 2159 melIt->setString( s );
joachim99@8 2160 setModified();
joachim99@8 2161 }
joachim99@8 2162 break;
joachim99@8 2163 }
joachim99@69 2164 case Qt::Key_Backspace:
joachim99@8 2165 {
joachim99@68 2166 if ( deleteSelection2( str, x, y, mlIt, melIt )) break;
joachim99@8 2167 if( !melIt->isEditableText() ) break;
joachim99@8 2168 if (x==0)
joachim99@8 2169 {
joachim99@8 2170 if ( y>0 )
joachim99@8 2171 {
joachim99@8 2172 setModified();
joachim99@8 2173 MergeLineList::iterator mlIt1;
joachim99@8 2174 MergeEditLineList::iterator melIt1;
joachim99@8 2175 calcIteratorFromLineNr( y-1, mlIt1, melIt1 );
joachim99@8 2176 if ( melIt1->isEditableText() )
joachim99@8 2177 {
joachim99@68 2178 QString s1 = melIt1->getString( this );
joachim99@68 2179 melIt1->setString( s1 + str );
joachim99@8 2180
joachim99@8 2181 // Remove the previous line
joachim99@8 2182 if ( mlIt->mergeEditLineList.size()>1 )
joachim99@8 2183 mlIt->mergeEditLineList.erase( melIt );
joachim99@8 2184 else
joachim99@8 2185 melIt->setRemoved();
joachim99@8 2186
joachim99@8 2187 --y;
joachim99@68 2188 x=str.length();
joachim99@8 2189 }
joachim99@8 2190 }
joachim99@8 2191 }
joachim99@8 2192 else
joachim99@8 2193 {
joachim99@68 2194 QString s = str.left( x-1 );
joachim99@68 2195 s += str.mid( x );
joachim99@8 2196 --x;
joachim99@8 2197 melIt->setString( s );
joachim99@8 2198 setModified();
joachim99@8 2199 }
joachim99@8 2200 break;
joachim99@8 2201 }
joachim99@69 2202 case Qt::Key_Return:
joachim99@69 2203 case Qt::Key_Enter:
joachim99@8 2204 {
joachim99@8 2205 if( !melIt->isEditableText() ) break;
joachim99@68 2206 deleteSelection2( str, x, y, mlIt, melIt );
joachim99@8 2207 setModified();
joachim99@68 2208 QString indentation;
joachim99@8 2209 if ( m_pOptionDialog->m_bAutoIndentation )
joachim99@8 2210 { // calc last indentation
joachim99@8 2211 MergeLineList::iterator mlIt1 = mlIt;
joachim99@8 2212 MergeEditLineList::iterator melIt1 = melIt;
joachim99@8 2213 for(;;) {
joachim99@68 2214 const QString s = melIt1->getString(this);
joachim99@68 2215 if ( !s.isEmpty() ) {
joachim99@68 2216 unsigned int i;
joachim99@68 2217 for( i=0; i<s.length(); ++i ){ if(s[i]!=' ' && s[i]!='\t') break; }
joachim99@68 2218 if (i<s.length()) {
joachim99@68 2219 indentation = s.left(i);
joachim99@8 2220 break;
joachim99@8 2221 }
joachim99@8 2222 }
joachim99@8 2223 --melIt1;
joachim99@8 2224 if ( melIt1 == mlIt1->mergeEditLineList.end() ) {
joachim99@8 2225 --mlIt1;
joachim99@8 2226 if ( mlIt1 == m_mergeLineList.end() ) break;
joachim99@8 2227 melIt1 = mlIt1->mergeEditLineList.end();
joachim99@8 2228 --melIt1;
joachim99@8 2229 }
joachim99@8 2230 }
joachim99@8 2231 }
joachim99@69 2232 MergeEditLine mel(mlIt->id3l); // Associate every mel with an id3l, even if not really valid.
joachim99@68 2233 mel.setString( indentation + str.mid(x) );
joachim99@8 2234
joachim99@68 2235 if ( x<(int)str.length() ) // Cut off the old line.
joachim99@8 2236 {
joachim99@8 2237 // Since ps possibly points into melIt->str, first copy it into a temporary.
joachim99@68 2238 QString temp = str.left(x);
joachim99@8 2239 melIt->setString( temp );
joachim99@8 2240 }
joachim99@8 2241
joachim99@8 2242 ++melIt;
joachim99@8 2243 mlIt->mergeEditLineList.insert( melIt, mel );
joachim99@68 2244 x = indentation.length();
joachim99@8 2245 ++y;
joachim99@8 2246 break;
joachim99@8 2247 }
joachim99@69 2248 case Qt::Key_Insert: m_bInsertMode = !m_bInsertMode; break;
joachim99@69 2249 case Qt::Key_Pause: break;
joachim99@69 2250 case Qt::Key_Print: break;
joachim99@69 2251 case Qt::Key_SysReq: break;
joachim99@69 2252 case Qt::Key_Home: x=0; if(bCtrl){y=0; } break; // cursor movement
joachim99@69 2253 case Qt::Key_End: x=INT_MAX; if(bCtrl){y=INT_MAX;} break;
joachim99@69 2254
joachim99@69 2255 case Qt::Key_Left:
joachim99@69 2256 case Qt::Key_Right:
joachim99@69 2257 if ( (e->key()==Qt::Key_Left) ^ m_pOptionDialog->m_bRightToLeftLanguage ) // operator^: XOR
joachim99@8 2258 {
joachim99@68 2259 if ( !bCtrl )
joachim99@68 2260 {
joachim99@68 2261 --x;
joachim99@68 2262 if(x<0 && y>0){--y; x=INT_MAX;}
joachim99@68 2263 }
joachim99@68 2264 else
joachim99@68 2265 {
joachim99@68 2266 while( x>0 && (str[x-1]==' ' || str[x-1]=='\t') ) --x;
joachim99@68 2267 while( x>0 && (str[x-1]!=' ' && str[x-1]!='\t') ) --x;
joachim99@68 2268 }
joachim99@8 2269 }
joachim99@8 2270 else
joachim99@8 2271 {
joachim99@68 2272 if ( !bCtrl )
joachim99@68 2273 {
joachim99@68 2274 ++x; if(x>(int)str.length() && y<m_totalSize-1){ ++y; x=0; }
joachim99@68 2275 }
joachim99@8 2276
joachim99@68 2277 else
joachim99@68 2278 {
joachim99@68 2279 while( x<(int)str.length() && (str[x]==' ' || str[x]=='\t') ) ++x;
joachim99@68 2280 while( x<(int)str.length() && (str[x]!=' ' && str[x]!='\t') ) ++x;
joachim99@68 2281 }
joachim99@8 2282 }
joachim99@8 2283 break;
joachim99@8 2284
joachim99@69 2285 case Qt::Key_Up: --y; bYMoveKey=true; break;
joachim99@69 2286 case Qt::Key_Down: ++y; bYMoveKey=true; break;
joachim99@69 2287 case Qt::Key_PageUp: y-=getNofVisibleLines(); bYMoveKey=true; break;
joachim99@69 2288 case Qt::Key_PageDown: y+=getNofVisibleLines(); bYMoveKey=true; break;
joachim99@8 2289 default:
joachim99@8 2290 {
joachim99@8 2291 QString t = e->text();
joachim99@8 2292 if( t.isEmpty() || bCtrl )
joachim99@8 2293 { e->ignore(); return; }
joachim99@8 2294 else
joachim99@8 2295 {
joachim99@8 2296 if( bCtrl )
joachim99@8 2297 {
joachim99@8 2298 e->ignore(); return;
joachim99@8 2299 }
joachim99@8 2300 else
joachim99@8 2301 {
joachim99@8 2302 if( !melIt->isEditableText() ) break;
joachim99@68 2303 deleteSelection2( str, x, y, mlIt, melIt );
joachim99@8 2304
joachim99@8 2305 setModified();
joachim99@8 2306 // Characters to insert
joachim99@68 2307 QString s=str;
joachim99@8 2308 if ( t[0]=='\t' && m_pOptionDialog->m_bReplaceTabs )
joachim99@8 2309 {
joachim99@69 2310 int spaces = (m_cursorXPos / m_pOptionDialog->m_tabSize + 1)*m_pOptionDialog->m_tabSize - m_cursorXPos;
joachim99@8 2311 t.fill( ' ', spaces );
joachim99@8 2312 }
joachim99@8 2313 if ( m_bInsertMode )
joachim99@68 2314 s.insert( x, t );
joachim99@8 2315 else
joachim99@68 2316 s.replace( x, t.length(), t );
joachim99@8 2317
joachim99@8 2318 melIt->setString( s );
joachim99@8 2319 x += t.length();
joachim99@8 2320 bShift = false;
joachim99@8 2321 }
joachim99@8 2322 }
joachim99@8 2323 }
joachim99@8 2324 }
joachim99@8 2325
joachim99@58 2326 y = minMaxLimiter( y, 0, m_totalSize-1 );
joachim99@8 2327
joachim99@8 2328 calcIteratorFromLineNr( y, mlIt, melIt );
joachim99@68 2329 str = melIt->getString( this );
joachim99@8 2330
joachim99@68 2331 x = minMaxLimiter( x, 0, (int)str.length() );
joachim99@8 2332
joachim99@8 2333 int newFirstLine = m_firstLine;
joachim99@8 2334 int newFirstColumn = m_firstColumn;
joachim99@8 2335
joachim99@8 2336 if ( y<m_firstLine )
joachim99@8 2337 newFirstLine = y;
joachim99@8 2338 else if ( y > m_firstLine + getNofVisibleLines() )
joachim99@8 2339 newFirstLine = y - getNofVisibleLines();
joachim99@8 2340
joachim99@8 2341 if (bYMoveKey)
joachim99@69 2342 x=convertToPosInText( str, m_cursorOldXPos, m_pOptionDialog->m_tabSize );
joachim99@69 2343
joachim99@69 2344 int xOnScreen = convertToPosOnScreen( str, x, m_pOptionDialog->m_tabSize );
joachim99@8 2345 if ( xOnScreen<m_firstColumn )
joachim99@8 2346 newFirstColumn = xOnScreen;
joachim99@8 2347 else if ( xOnScreen > m_firstColumn + getNofVisibleColumns() )
joachim99@8 2348 newFirstColumn = xOnScreen - getNofVisibleColumns();
joachim99@8 2349
joachim99@8 2350 if ( bShift )
joachim99@8 2351 {
joachim99@8 2352 if (m_selection.firstLine==-1)
joachim99@8 2353 m_selection.start( m_cursorYPos, m_cursorXPos );
joachim99@8 2354
joachim99@8 2355 m_selection.end( y, xOnScreen );
joachim99@8 2356 }
joachim99@8 2357 else
joachim99@8 2358 m_selection.reset();
joachim99@8 2359
joachim99@8 2360 m_cursorYPos = y;
joachim99@8 2361 m_cursorXPos = xOnScreen;
joachim99@8 2362 if ( ! bYMoveKey )
joachim99@8 2363 m_cursorOldXPos = m_cursorXPos;
joachim99@8 2364
joachim99@8 2365 m_bCursorOn = false;
joachim99@8 2366
joachim99@8 2367 if ( newFirstLine!=m_firstLine || newFirstColumn!=m_firstColumn )
joachim99@8 2368 {
joachim99@8 2369 m_bCursorOn = true;
joachim99@8 2370 scroll( newFirstColumn-m_firstColumn, newFirstLine-m_firstLine );
joachim99@8 2371 return;
joachim99@8 2372 }
joachim99@8 2373
joachim99@8 2374 m_bCursorOn = true;
joachim99@8 2375 update();
joachim99@8 2376 }
joachim99@8 2377
joachim99@8 2378 void MergeResultWindow::calcIteratorFromLineNr(
joachim99@8 2379 int line,
joachim99@8 2380 MergeResultWindow::MergeLineList::iterator& mlIt,
joachim99@8 2381 MergeResultWindow::MergeEditLineList::iterator& melIt
joachim99@8 2382 )
joachim99@8 2383 {
joachim99@8 2384 for( mlIt = m_mergeLineList.begin(); mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 2385 {
joachim99@8 2386 MergeLine& ml = *mlIt;
joachim99@8 2387 if ( line > ml.mergeEditLineList.size() )
joachim99@8 2388 {
joachim99@8 2389 line -= ml.mergeEditLineList.size();
joachim99@8 2390 }
joachim99@8 2391 else
joachim99@8 2392 {
joachim99@8 2393 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); ++melIt )
joachim99@8 2394 {
joachim99@8 2395 --line;
joachim99@8 2396 if (line<0) return;
joachim99@8 2397 }
joachim99@8 2398 }
joachim99@8 2399 }
joachim99@8 2400 assert(false);
joachim99@8 2401 }
joachim99@8 2402
joachim99@8 2403
joachim99@8 2404 QString MergeResultWindow::getSelection()
joachim99@8 2405 {
joachim99@8 2406 QString selectionString;
joachim99@8 2407
joachim99@8 2408 int line = 0;
joachim99@8 2409 MergeLineList::iterator mlIt = m_mergeLineList.begin();
joachim99@8 2410 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 2411 {
joachim99@8 2412 MergeLine& ml = *mlIt;
joachim99@8 2413 MergeEditLineList::iterator melIt;
joachim99@8 2414 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); ++melIt )
joachim99@8 2415 {
joachim99@8 2416 MergeEditLine& mel = *melIt;
joachim99@8 2417
joachim99@8 2418 if ( m_selection.lineWithin(line) )
joachim99@8 2419 {
joachim99@8 2420 int outPos = 0;
joachim99@8 2421 if (mel.isEditableText())
joachim99@8 2422 {
joachim99@68 2423 const QString str = mel.getString( this );
joachim99@8 2424
joachim99@8 2425 // Consider tabs
joachim99@8 2426
joachim99@68 2427 for( unsigned int i=0; i<str.length(); ++i )
joachim99@8 2428 {
joachim99@8 2429 int spaces = 1;
joachim99@68 2430 if ( str[i]=='\t' )
joachim99@8 2431 {
joachim99@69 2432 spaces = tabber( outPos, m_pOptionDialog->m_tabSize );
joachim99@8 2433 }
joachim99@8 2434
joachim99@8 2435 if( m_selection.within( line, outPos ) )
joachim99@8 2436 {
joachim99@68 2437 selectionString += str[i];
joachim99@8 2438 }
joachim99@8 2439
joachim99@8 2440 outPos += spaces;
joachim99@8 2441 }
joachim99@8 2442 }
joachim99@8 2443 else if ( mel.isConflict() )
joachim99@8 2444 {
joachim99@58 2445 selectionString += i18n("<Merge Conflict>");
joachim99@8 2446 }
joachim99@68 2447
joachim99@8 2448 if( m_selection.within( line, outPos ) )
joachim99@8 2449 {
joachim99@8 2450 #ifdef _WIN32
joachim99@8 2451 selectionString += '\r';
joachim99@8 2452 #endif
joachim99@8 2453 selectionString += '\n';
joachim99@8 2454 }
joachim99@8 2455 }
joachim99@8 2456
joachim99@8 2457 ++line;
joachim99@8 2458 }
joachim99@8 2459 }
joachim99@8 2460
joachim99@8 2461 return selectionString;
joachim99@8 2462 }
joachim99@8 2463
joachim99@68 2464 bool MergeResultWindow::deleteSelection2( QString& s, int& x, int& y,
joachim99@8 2465 MergeLineList::iterator& mlIt, MergeEditLineList::iterator& melIt )
joachim99@8 2466 {
joachim99@8 2467 if (m_selection.firstLine!=-1 && m_selection.bSelectionContainsData )
joachim99@8 2468 {
joachim99@8 2469 deleteSelection();
joachim99@8 2470 y = m_cursorYPos;
joachim99@8 2471 calcIteratorFromLineNr( y, mlIt, melIt );
joachim99@68 2472 s = melIt->getString( this );
joachim99@69 2473 x = convertToPosInText( s, m_cursorXPos, m_pOptionDialog->m_tabSize );
joachim99@8 2474 return true;
joachim99@8 2475 }
joachim99@8 2476 return false;
joachim99@8 2477 }
joachim99@8 2478
joachim99@8 2479 void MergeResultWindow::deleteSelection()
joachim99@8 2480 {
joachim99@8 2481 if ( m_selection.firstLine==-1 || !m_selection.bSelectionContainsData )
joachim99@8 2482 {
joachim99@8 2483 return;
joachim99@8 2484 }
joachim99@8 2485 setModified();
joachim99@8 2486
joachim99@8 2487 int line = 0;
joachim99@8 2488 MergeLineList::iterator mlItFirst;
joachim99@8 2489 MergeEditLineList::iterator melItFirst;
joachim99@68 2490 QString firstLineString;
joachim99@8 2491
joachim99@8 2492 int firstLine = -1;
joachim99@8 2493 int lastLine = -1;
joachim99@8 2494
joachim99@8 2495 MergeLineList::iterator mlIt;
joachim99@8 2496 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 2497 {
joachim99@8 2498 MergeLine& ml = *mlIt;
joachim99@8 2499 MergeEditLineList::iterator melIt;
joachim99@8 2500 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); ++melIt )
joachim99@8 2501 {
joachim99@8 2502 MergeEditLine& mel = *melIt;
joachim99@8 2503
joachim99@8 2504 if ( mel.isEditableText() && m_selection.lineWithin(line) )
joachim99@8 2505 {
joachim99@8 2506 if ( firstLine==-1 )
joachim99@8 2507 firstLine = line;
joachim99@8 2508 lastLine = line;
joachim99@8 2509 }
joachim99@8 2510
joachim99@8 2511 ++line;
joachim99@8 2512 }
joachim99@8 2513 }
joachim99@8 2514
joachim99@8 2515 if ( firstLine == -1 )
joachim99@8 2516 {
joachim99@8 2517 return; // Nothing to delete.
joachim99@8 2518 }
joachim99@8 2519
joachim99@8 2520 line = 0;
joachim99@8 2521 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 2522 {
joachim99@8 2523 MergeLine& ml = *mlIt;
joachim99@8 2524 MergeEditLineList::iterator melIt, melIt1;
joachim99@8 2525 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); )
joachim99@8 2526 {
joachim99@8 2527 MergeEditLine& mel = *melIt;
joachim99@8 2528 melIt1 = melIt;
joachim99@8 2529 ++melIt1;
joachim99@8 2530
joachim99@8 2531 if ( mel.isEditableText() && m_selection.lineWithin(line) )
joachim99@8 2532 {
joachim99@68 2533 QString lineString = mel.getString( this );
joachim99@8 2534
joachim99@8 2535 int firstPosInLine = m_selection.firstPosInLine(line);
joachim99@8 2536 int lastPosInLine = m_selection.lastPosInLine(line);
joachim99@8 2537
joachim99@8 2538 if ( line==firstLine )
joachim99@8 2539 {
joachim99@8 2540 mlItFirst = mlIt;
joachim99@8 2541 melItFirst = melIt;
joachim99@69 2542 int pos = convertToPosInText( lineString, firstPosInLine, m_pOptionDialog->m_tabSize );
joachim99@68 2543 firstLineString = lineString.left( pos );
joachim99@8 2544 }
joachim99@8 2545
joachim99@8 2546 if ( line==lastLine )
joachim99@8 2547 {
joachim99@8 2548 // This is the last line in the selection
joachim99@69 2549 int pos = convertToPosInText( lineString, lastPosInLine, m_pOptionDialog->m_tabSize );
joachim99@68 2550 firstLineString += lineString.mid( pos ); // rest of line
joachim99@8 2551 melItFirst->setString( firstLineString );
joachim99@8 2552 }
joachim99@8 2553
joachim99@8 2554 if ( line!=firstLine )
joachim99@8 2555 {
joachim99@8 2556 // Remove the line
joachim99@8 2557 if ( mlIt->mergeEditLineList.size()>1 )
joachim99@58 2558 mlIt->mergeEditLineList.erase( melIt );
joachim99@8 2559 else
joachim99@58 2560 melIt->setRemoved();
joachim99@8 2561 }
joachim99@8 2562 }
joachim99@8 2563
joachim99@8 2564 ++line;
joachim99@8 2565 melIt = melIt1;
joachim99@8 2566 }
joachim99@8 2567 }
joachim99@8 2568
joachim99@8 2569 m_cursorYPos = m_selection.beginLine();
joachim99@8 2570 m_cursorXPos = m_selection.beginPos();
joachim99@8 2571 m_cursorOldXPos = m_cursorXPos;
joachim99@8 2572
joachim99@8 2573 m_selection.reset();
joachim99@8 2574 }
joachim99@8 2575
joachim99@66 2576 void MergeResultWindow::pasteClipboard( bool bFromSelection )
joachim99@8 2577 {
joachim99@8 2578 if (m_selection.firstLine != -1 )
joachim99@8 2579 deleteSelection();
joachim99@8 2580
joachim99@8 2581 setModified();
joachim99@8 2582
joachim99@8 2583 int y = m_cursorYPos;
joachim99@8 2584 MergeLineList::iterator mlIt;
joachim99@8 2585 MergeEditLineList::iterator melIt, melItAfter;
joachim99@8 2586 calcIteratorFromLineNr( y, mlIt, melIt );
joachim99@8 2587 melItAfter = melIt;
joachim99@8 2588 ++melItAfter;
joachim99@68 2589 QString str = melIt->getString( this );
joachim99@69 2590 int x = convertToPosInText( str, m_cursorXPos, m_pOptionDialog->m_tabSize );
joachim99@8 2591
joachim99@66 2592 if ( !QApplication::clipboard()->supportsSelection() )
joachim99@66 2593 bFromSelection = false;
joachim99@8 2594
joachim99@68 2595 QString clipBoard = QApplication::clipboard()->text( bFromSelection ? QClipboard::Selection : QClipboard::Clipboard );
joachim99@68 2596
joachim99@68 2597 QString currentLine = str.left(x);
joachim99@68 2598 QString endOfLine = str.mid(x);
joachim99@8 2599 int i;
joachim99@68 2600 int len = clipBoard.length();
joachim99@68 2601 for( i=0; i<len; ++i )
joachim99@8 2602 {
joachim99@68 2603 QChar c = clipBoard[i];
joachim99@8 2604 if ( c == '\r' ) continue;
joachim99@8 2605 if ( c == '\n' )
joachim99@8 2606 {
joachim99@8 2607 melIt->setString( currentLine );
joachim99@69 2608 MergeEditLine mel(mlIt->id3l); // Associate every mel with an id3l, even if not really valid.
joachim99@69 2609 melIt = mlIt->mergeEditLineList.insert( melItAfter, mel );
joachim99@8 2610 currentLine = "";
joachim99@8 2611 x=0;
joachim99@8 2612 ++y;
joachim99@8 2613 }
joachim99@8 2614 else
joachim99@8 2615 {
joachim99@8 2616 currentLine += c;
joachim99@8 2617 ++x;
joachim99@8 2618 }
joachim99@8 2619 }
joachim99@8 2620
joachim99@8 2621 currentLine += endOfLine;
joachim99@8 2622 melIt->setString( currentLine );
joachim99@8 2623
joachim99@8 2624 m_cursorYPos = y;
joachim99@69 2625 m_cursorXPos = convertToPosOnScreen( currentLine, x, m_pOptionDialog->m_tabSize );
joachim99@8 2626 m_cursorOldXPos = m_cursorXPos;
joachim99@8 2627
joachim99@8 2628 update();
joachim99@8 2629 }
joachim99@8 2630
joachim99@8 2631 void MergeResultWindow::resetSelection()
joachim99@8 2632 {
joachim99@8 2633 m_selection.reset();
joachim99@8 2634 update();
joachim99@8 2635 }
joachim99@8 2636
joachim99@8 2637 void MergeResultWindow::setModified()
joachim99@8 2638 {
joachim99@8 2639 if (!m_bModified)
joachim99@8 2640 {
joachim99@8 2641 m_bModified = true;
joachim99@8 2642 emit modified();
joachim99@8 2643 }
joachim99@8 2644 }
joachim99@8 2645
joachim99@8 2646 /// Saves and returns true when successful.
joachim99@8 2647 bool MergeResultWindow::saveDocument( const QString& fileName )
joachim99@8 2648 {
joachim99@8 2649 m_fileName = fileName;
joachim99@8 2650
joachim99@8 2651 // Are still conflicts somewhere?
joachim99@8 2652 if ( getNrOfUnsolvedConflicts()>0 )
joachim99@8 2653 {
joachim99@8 2654 KMessageBox::error( this,
joachim99@8 2655 i18n("Not all conflicts are solved yet.\n"
joachim99@8 2656 "File not saved.\n"),
joachim99@51 2657 i18n("Conflicts Left"));
joachim99@8 2658 return false;
joachim99@8 2659 }
joachim99@8 2660
joachim99@8 2661 update();
joachim99@8 2662
joachim99@8 2663 FileAccess file( fileName, true /*bWantToWrite*/ );
joachim99@8 2664 if ( m_pOptionDialog->m_bDmCreateBakFiles && file.exists() )
joachim99@8 2665 {
joachim99@8 2666 bool bSuccess = file.createBackup(".orig");
joachim99@8 2667 if ( !bSuccess )
joachim99@8 2668 {
joachim99@68 2669 KMessageBox::error( this, file.getStatusText() + i18n("\n\nCreating backup failed. File not saved."), i18n("File Save Error") );
joachim99@8 2670 return false;
joachim99@8 2671 }
joachim99@8 2672 }
joachim99@8 2673
joachim99@8 2674 QByteArray dataArray;
joachim99@68 2675 QTextStream textOutStream(dataArray, IO_WriteOnly);
joachim99@68 2676 textOutStream.setCodec( m_pOptionDialog->m_pEncodingOut );
joachim99@68 2677
joachim99@68 2678 int line = 0;
joachim99@68 2679 MergeLineList::iterator mlIt = m_mergeLineList.begin();
joachim99@68 2680 for(mlIt = m_mergeLineList.begin();mlIt!=m_mergeLineList.end(); ++mlIt)
joachim99@8 2681 {
joachim99@68 2682 MergeLine& ml = *mlIt;
joachim99@68 2683 MergeEditLineList::iterator melIt;
joachim99@68 2684 for( melIt = ml.mergeEditLineList.begin(); melIt != ml.mergeEditLineList.end(); ++melIt )
joachim99@8 2685 {
joachim99@68 2686 MergeEditLine& mel = *melIt;
joachim99@68 2687
joachim99@68 2688 if ( mel.isEditableText() )
joachim99@8 2689 {
joachim99@68 2690 QString str = mel.getString( this );
joachim99@8 2691
joachim99@68 2692 if (line>0) // Prepend line feed, but not for first line
joachim99@8 2693 {
joachim99@68 2694 if ( m_pOptionDialog->m_lineEndStyle == eLineEndDos )
joachim99@68 2695 { str.prepend("\r\n"); }
joachim99@8 2696 else
joachim99@68 2697 { str.prepend("\n"); }
joachim99@8 2698 }
joachim99@8 2699
joachim99@68 2700 textOutStream << str;
joachim99@8 2701 }
joachim99@68 2702
joachim99@68 2703 ++line;
joachim99@8 2704 }
joachim99@8 2705 }
joachim99@68 2706 bool bSuccess = file.writeFile( dataArray.data(), dataArray.size() );
joachim99@8 2707 if ( ! bSuccess )
joachim99@8 2708 {
joachim99@51 2709 KMessageBox::error( this, i18n("Error while writing."), i18n("File Save Error") );
joachim99@8 2710 return false;
joachim99@8 2711 }
joachim99@8 2712
joachim99@8 2713 m_bModified = false;
joachim99@8 2714 update();
joachim99@8 2715
joachim99@8 2716 return true;
joachim99@8 2717 }
joachim99@8 2718
joachim99@68 2719 QString MergeResultWindow::getString( int lineIdx )
joachim99@8 2720 {
joachim99@8 2721 MergeResultWindow::MergeLineList::iterator mlIt;
joachim99@8 2722 MergeResultWindow::MergeEditLineList::iterator melIt;
joachim99@8 2723 calcIteratorFromLineNr( lineIdx, mlIt, melIt );
joachim99@68 2724 QString s = melIt->getString( this );
joachim99@68 2725 return s;
joachim99@8 2726 }
joachim99@8 2727
joachim99@68 2728 bool MergeResultWindow::findString( const QString& s, int& d3vLine, int& posInLine, bool bDirDown, bool bCaseSensitive )
joachim99@8 2729 {
joachim99@8 2730 int it = d3vLine;
joachim99@8 2731 int endIt = bDirDown ? getNofLines() : -1;
joachim99@8 2732 int step = bDirDown ? 1 : -1;
joachim99@8 2733 int startPos = posInLine;
joachim99@8 2734
joachim99@8 2735 for( ; it!=endIt; it+=step )
joachim99@8 2736 {
joachim99@68 2737 QString line = getString( it );
joachim99@8 2738 if ( !line.isEmpty() )
joachim99@8 2739 {
joachim99@8 2740 int pos = line.find( s, startPos, bCaseSensitive );
joachim99@8 2741 if ( pos != -1 )
joachim99@8 2742 {
joachim99@8 2743 d3vLine = it;
joachim99@8 2744 posInLine = pos;
joachim99@8 2745 return true;
joachim99@8 2746 }
joachim99@8 2747
joachim99@8 2748 startPos = 0;
joachim99@8 2749 }
joachim99@8 2750 }
joachim99@8 2751 return false;
joachim99@8 2752 }
joachim99@8 2753
joachim99@8 2754 void MergeResultWindow::setSelection( int firstLine, int startPos, int lastLine, int endPos )
joachim99@8 2755 {
joachim99@69 2756 if ( lastLine >= getNofLines() )
joachim99@69 2757 {
joachim99@69 2758 lastLine = getNofLines()-1;
joachim99@69 2759 QString s = getString( lastLine );
joachim99@69 2760 endPos = s.length();
joachim99@69 2761 }
joachim99@8 2762 m_selection.reset();
joachim99@69 2763 m_selection.start( firstLine, convertToPosOnScreen( getString(firstLine), startPos, m_pOptionDialog->m_tabSize ) );
joachim99@69 2764 m_selection.end( lastLine, convertToPosOnScreen( getString(lastLine), endPos, m_pOptionDialog->m_tabSize ) );
joachim99@8 2765 update();
joachim99@8 2766 }
joachim99@8 2767
joachim99@8 2768 Overview::Overview( QWidget* pParent, OptionDialog* pOptions )
joachim99@8 2769 : QWidget( pParent, 0, WRepaintNoErase )
joachim99@8 2770 {
joachim99@8 2771 m_pDiff3LineList = 0;
joachim99@8 2772 m_pOptions = pOptions;
joachim99@8 2773 m_bTripleDiff = false;
joachim99@66 2774 m_eOverviewMode = eOMNormal;
joachim99@66 2775 m_nofLines = 1;
joachim99@69 2776 m_bPaintingAllowed = false;
joachim99@8 2777 setFixedWidth(20);
joachim99@8 2778 }
joachim99@8 2779
joachim99@8 2780 void Overview::init( Diff3LineList* pDiff3LineList, bool bTripleDiff )
joachim99@8 2781 {
joachim99@8 2782 m_pDiff3LineList = pDiff3LineList;
joachim99@8 2783 m_bTripleDiff = bTripleDiff;
joachim99@8 2784 m_pixmap.resize( QSize(0,0) ); // make sure that a redraw happens
joachim99@8 2785 update();
joachim99@8 2786 }
joachim99@8 2787
joachim99@69 2788 void Overview::reset()
joachim99@69 2789 {
joachim99@69 2790 m_pDiff3LineList = 0;
joachim99@69 2791 }
joachim99@69 2792
joachim99@51 2793 void Overview::slotRedraw()
joachim99@51 2794 {
joachim99@51 2795 m_pixmap.resize( QSize(0,0) ); // make sure that a redraw happens
joachim99@51 2796 update();
joachim99@51 2797 }
joachim99@51 2798
joachim99@8 2799 void Overview::setRange( int firstLine, int pageHeight )
joachim99@8 2800 {
joachim99@8 2801 m_firstLine = firstLine;
joachim99@8 2802 m_pageHeight = pageHeight;
joachim99@8 2803 update();
joachim99@8 2804 }
joachim99@8 2805 void Overview::setFirstLine( int firstLine )
joachim99@8 2806 {
joachim99@8 2807 m_firstLine = firstLine;
joachim99@8 2808 update();
joachim99@8 2809 }
joachim99@8 2810
joachim99@66 2811 void Overview::setOverviewMode( e_OverviewMode eOverviewMode )
joachim99@66 2812 {
joachim99@66 2813 m_eOverviewMode = eOverviewMode;
joachim99@69 2814 slotRedraw();
joachim99@66 2815 }
joachim99@66 2816
joachim99@66 2817 Overview::e_OverviewMode Overview::getOverviewMode()
joachim99@66 2818 {
joachim99@66 2819 return m_eOverviewMode;
joachim99@66 2820 }
joachim99@66 2821
joachim99@8 2822 void Overview::mousePressEvent( QMouseEvent* e )
joachim99@8 2823 {
joachim99@8 2824 int h = height()-1;
joachim99@66 2825 int h1 = h * m_pageHeight / max2(1,m_nofLines)+3;
joachim99@8 2826 if ( h>0 )
joachim99@66 2827 emit setLine( ( e->y() - h1/2 )*m_nofLines/h );
joachim99@8 2828 }
joachim99@8 2829
joachim99@8 2830 void Overview::mouseMoveEvent( QMouseEvent* e )
joachim99@8 2831 {
joachim99@8 2832 mousePressEvent(e);
joachim99@8 2833 }
joachim99@8 2834
joachim99@8 2835 void Overview::setPaintingAllowed( bool bAllowPainting )
joachim99@8 2836 {
joachim99@8 2837 if (m_bPaintingAllowed != bAllowPainting)
joachim99@8 2838 {
joachim99@8 2839 m_bPaintingAllowed = bAllowPainting;
joachim99@8 2840 if ( m_bPaintingAllowed ) update();
joachim99@69 2841 else reset();
joachim99@8 2842 }
joachim99@8 2843 }
joachim99@8 2844
joachim99@66 2845 void Overview::drawColumn( QPainter& p, e_OverviewMode eOverviewMode, int x, int w, int h, int nofLines )
joachim99@66 2846 {
joachim99@69 2847 p.setPen(Qt::black);
joachim99@66 2848 p.drawLine( x, 0, x, h );
joachim99@66 2849
joachim99@66 2850 if (nofLines==0) return;
joachim99@66 2851
joachim99@66 2852 int line = 0;
joachim99@66 2853 int oldY = 0;
joachim99@66 2854 int oldConflictY = -1;
joachim99@66 2855 int wrapLineIdx=0;
joachim99@66 2856 Diff3LineList::const_iterator i;
joachim99@66 2857 for( i = m_pDiff3LineList->begin(); i!= m_pDiff3LineList->end(); )
joachim99@66 2858 {
joachim99@66 2859 const Diff3Line& d3l = *i;
joachim99@66 2860 int y = h * (line+1) / nofLines;
joachim99@66 2861 e_MergeDetails md;
joachim99@66 2862 bool bConflict;
joachim99@66 2863 bool bLineRemoved;
joachim99@66 2864 int src;
joachim99@66 2865 mergeOneLine( d3l, md, bConflict, bLineRemoved, src, !m_bTripleDiff );
joachim99@66 2866
joachim99@66 2867 QColor c = m_pOptions->m_bgColor;
joachim99@66 2868 bool bWhiteSpaceChange = false;
joachim99@66 2869 //if( bConflict ) c=m_pOptions->m_colorForConflict;
joachim99@66 2870 //else
joachim99@66 2871 if ( eOverviewMode==eOMNormal )
joachim99@66 2872 {
joachim99@66 2873 switch( md )
joachim99@66 2874 {
joachim99@66 2875 case eDefault:
joachim99@66 2876 case eNoChange:
joachim99@66 2877 c = m_pOptions->m_bgColor;
joachim99@66 2878 break;
joachim99@66 2879
joachim99@66 2880 case eBAdded:
joachim99@66 2881 case eBDeleted:
joachim99@66 2882 case eBChanged:
joachim99@66 2883 c = bConflict ? m_pOptions->m_colorForConflict : m_pOptions->m_colorB;
joachim99@66 2884 bWhiteSpaceChange = d3l.bAEqB || d3l.bWhiteLineA && d3l.bWhiteLineB;
joachim99@66 2885 break;
joachim99@66 2886
joachim99@66 2887 case eCAdded:
joachim99@66 2888 case eCDeleted:
joachim99@66 2889 case eCChanged:
joachim99@66 2890 bWhiteSpaceChange = d3l.bAEqC || d3l.bWhiteLineA && d3l.bWhiteLineC;
joachim99@66 2891 c = bConflict ? m_pOptions->m_colorForConflict : m_pOptions->m_colorC;
joachim99@66 2892 break;
joachim99@66 2893
joachim99@66 2894 case eBCChanged: // conflict
joachim99@66 2895 case eBCChangedAndEqual: // possible conflict
joachim99@66 2896 case eBCDeleted: // possible conflict
joachim99@66 2897 case eBChanged_CDeleted: // conflict
joachim99@66 2898 case eCChanged_BDeleted: // conflict
joachim99@66 2899 case eBCAdded: // conflict
joachim99@66 2900 case eBCAddedAndEqual: // possible conflict
joachim99@66 2901 c=m_pOptions->m_colorForConflict;
joachim99@66 2902 break;
joachim99@66 2903 default: assert(false); break;
joachim99@66 2904 }
joachim99@66 2905 }
joachim99@66 2906 else if ( eOverviewMode==eOMAvsB )
joachim99@66 2907 {
joachim99@66 2908 switch( md )
joachim99@66 2909 {
joachim99@66 2910 case eDefault:
joachim99@66 2911 case eNoChange:
joachim99@66 2912 case eCAdded:
joachim99@66 2913 case eCDeleted:
joachim99@66 2914 case eCChanged: break;
joachim99@66 2915 default: c = m_pOptions->m_colorForConflict;
joachim99@66 2916 bWhiteSpaceChange = d3l.bAEqB || d3l.bWhiteLineA && d3l.bWhiteLineB;
joachim99@66 2917 break;
joachim99@66 2918 }
joachim99@66 2919 }
joachim99@66 2920 else if ( eOverviewMode==eOMAvsC )
joachim99@66 2921 {
joachim99@66 2922 switch( md )
joachim99@66 2923 {
joachim99@66 2924 case eDefault:
joachim99@66 2925 case eNoChange:
joachim99@66 2926 case eBAdded:
joachim99@66 2927 case eBDeleted:
joachim99@66 2928 case eBChanged: break;
joachim99@66 2929 default: c = m_pOptions->m_colorForConflict;
joachim99@66 2930 bWhiteSpaceChange = d3l.bAEqC || d3l.bWhiteLineA && d3l.bWhiteLineC;
joachim99@69 2931 break;
joachim99@66 2932 }
joachim99@66 2933 }
joachim99@66 2934 else if ( eOverviewMode==eOMBvsC )
joachim99@66 2935 {
joachim99@66 2936 switch( md )
joachim99@66 2937 {
joachim99@66 2938 case eDefault:
joachim99@66 2939 case eNoChange:
joachim99@66 2940 case eBCChangedAndEqual:
joachim99@66 2941 case eBCDeleted:
joachim99@66 2942 case eBCAddedAndEqual: break;
joachim99@66 2943 default: c=m_pOptions->m_colorForConflict;
joachim99@66 2944 bWhiteSpaceChange = d3l.bBEqC || d3l.bWhiteLineB && d3l.bWhiteLineC;
joachim99@66 2945 break;
joachim99@66 2946 }
joachim99@66 2947 }
joachim99@66 2948
joachim99@66 2949 if (!bWhiteSpaceChange || m_pOptions->m_bShowWhiteSpace )
joachim99@66 2950 {
joachim99@66 2951 // Make sure that lines with conflict are not overwritten.
joachim99@66 2952 if ( c == m_pOptions->m_colorForConflict )
joachim99@66 2953 {
joachim99@69 2954 p.fillRect(x+1, oldY, w, max2(1,y-oldY), bWhiteSpaceChange ? QBrush(c,Qt::Dense4Pattern) : QBrush(c) );
joachim99@66 2955 oldConflictY = oldY;
joachim99@66 2956 }
joachim99@66 2957 else if ( c!=m_pOptions->m_bgColor && oldY>oldConflictY )
joachim99@66 2958 {
joachim99@69 2959 p.fillRect(x+1, oldY, w, max2(1,y-oldY), bWhiteSpaceChange ? QBrush(c,Qt::Dense4Pattern) : QBrush(c) );
joachim99@66 2960 }
joachim99@66 2961 }
joachim99@66 2962
joachim99@66 2963 oldY = y;
joachim99@66 2964
joachim99@66 2965 ++line;
joachim99@66 2966 if ( m_pOptions->m_bWordWrap )
joachim99@66 2967 {
joachim99@66 2968 ++wrapLineIdx;
joachim99@66 2969 if(wrapLineIdx>=d3l.linesNeededForDisplay)
joachim99@66 2970 {
joachim99@66 2971 wrapLineIdx=0;
joachim99@66 2972 ++i;
joachim99@66 2973 }
joachim99@66 2974 }
joachim99@66 2975 else
joachim99@66 2976 {
joachim99@66 2977 ++i;
joachim99@66 2978 }
joachim99@66 2979 }
joachim99@66 2980 }
joachim99@66 2981
joachim99@8 2982 void Overview::paintEvent( QPaintEvent* )
joachim99@8 2983 {
joachim99@8 2984 if (m_pDiff3LineList==0 || !m_bPaintingAllowed ) return;
joachim99@8 2985 int h = height()-1;
joachim99@8 2986 int w = width();
joachim99@66 2987
joachim99@8 2988
joachim99@8 2989 if ( m_pixmap.size() != size() )
joachim99@8 2990 {
joachim99@66 2991 if ( m_pOptions->m_bWordWrap )
joachim99@66 2992 {
joachim99@66 2993 m_nofLines = 0;
joachim99@66 2994 Diff3LineList::const_iterator i;
joachim99@66 2995 for( i = m_pDiff3LineList->begin(); i!= m_pDiff3LineList->end(); ++i )
joachim99@66 2996 {
joachim99@66 2997 m_nofLines += i->linesNeededForDisplay;
joachim99@69 2998 }
joachim99@66 2999 }
joachim99@66 3000 else
joachim99@66 3001 {
joachim99@66 3002 m_nofLines = m_pDiff3LineList->size();
joachim99@66 3003 }
joachim99@66 3004
joachim99@8 3005 m_pixmap.resize( size() );
joachim99@8 3006
joachim99@8 3007 QPainter p(&m_pixmap);
joachim99@66 3008 p.fillRect( rect(), m_pOptions->m_bgColor );
joachim99@8 3009
joachim99@66 3010 if ( !m_bTripleDiff || m_eOverviewMode == eOMNormal )
joachim99@8 3011 {
joachim99@66 3012 drawColumn( p, eOMNormal, 0, w, h, m_nofLines );
joachim99@8 3013 }
joachim99@66 3014 else
joachim99@66 3015 {
joachim99@66 3016 drawColumn( p, eOMNormal, 0, w/2, h, m_nofLines );
joachim99@66 3017 drawColumn( p, m_eOverviewMode, w/2, w/2, h, m_nofLines );
joachim99@69 3018 }
joachim99@8 3019 }
joachim99@8 3020
joachim99@8 3021 QPainter painter( this );
joachim99@8 3022 painter.drawPixmap( 0,0, m_pixmap );
joachim99@8 3023
joachim99@66 3024 int y1 = h * m_firstLine / m_nofLines-1;
joachim99@66 3025 int h1 = h * m_pageHeight / m_nofLines+3;
joachim99@69 3026 painter.setPen(Qt::black);
joachim99@8 3027 painter.drawRect( 1, y1, w-1, h1 );
joachim99@8 3028 }
joachim99@8 3029
joachim99@8 3030
joachim99@69 3031 #include "mergeresultwindow.moc"