annotate kdiff3/src-QT4/fileaccess.cpp @ 77:1184fc843210

KDiff3-0.9.92
author joachim99
date Mon, 16 Apr 2007 21:26:08 +0000
parents 08ea9b86c12c
children fcd146072e0c
rev   line source
joachim99@8 1 /***************************************************************************
joachim99@77 2 * Copyright (C) 2003-2007 by Joachim Eibl *
joachim99@75 3 * joachim.eibl at gmx.de *
joachim99@8 4 * *
joachim99@8 5 * This program is free software; you can redistribute it and/or modify *
joachim99@8 6 * it under the terms of the GNU General Public License as published by *
joachim99@8 7 * the Free Software Foundation; either version 2 of the License, or *
joachim99@8 8 * (at your option) any later version. *
joachim99@8 9 ***************************************************************************/
joachim99@8 10
joachim99@8 11 #include "fileaccess.h"
joachim99@8 12 #include "optiondialog.h"
joachim99@75 13 #include "common.h"
joachim99@75 14
joachim99@8 15 #include <qlayout.h>
joachim99@8 16 #include <qlabel.h>
joachim99@8 17 #include <qapplication.h>
joachim99@51 18 #include <qpushbutton.h>
joachim99@8 19 #include <qdir.h>
joachim99@8 20 #include <qregexp.h>
joachim99@8 21 #include <qtextstream.h>
joachim99@75 22 #include <qeventloop.h>
joachim99@75 23 #include <QProcess>
joachim99@75 24
joachim99@8 25 #include <vector>
joachim99@77 26 #include <cstdlib>
joachim99@75 27 #include <iostream>
joachim99@8 28 #include <klocale.h>
joachim99@75 29 #include <ktempfile.h>
joachim99@75 30 #include <kio/global.h>
joachim99@75 31 #include <kmessagebox.h>
joachim99@8 32
joachim99@8 33 #include <sys/types.h>
joachim99@8 34 #include <sys/stat.h>
joachim99@8 35
joachim99@8 36 #ifdef _WIN32
joachim99@8 37 #include <sys/utime.h>
joachim99@8 38 #include <io.h>
joachim99@58 39 #include <windows.h>
joachim99@77 40 #include <process.h>
joachim99@8 41 #else
joachim99@8 42 #include <unistd.h> // Needed for creating symbolic links via symlink().
joachim99@8 43 #include <utime.h>
joachim99@8 44 #endif
joachim99@8 45
joachim99@8 46
joachim99@69 47 ProgressDialog* g_pProgressDialog=0;
joachim99@8 48
joachim99@8 49
joachim99@8 50 FileAccess::FileAccess( const QString& name, bool bWantToWrite )
joachim99@8 51 {
joachim99@8 52 setFile( name, bWantToWrite );
joachim99@8 53 }
joachim99@8 54
joachim99@8 55 FileAccess::FileAccess()
joachim99@8 56 {
joachim99@8 57 m_bValidData = false;
joachim99@8 58 m_size = 0;
joachim99@8 59 m_creationTime = QDateTime();
joachim99@8 60 m_accessTime = QDateTime();
joachim99@8 61 m_modificationTime = QDateTime();
joachim99@8 62 m_bReadable = false;
joachim99@8 63 m_bWritable = false;
joachim99@8 64 m_bExecutable = false;
joachim99@8 65 m_bLocal = false;
joachim99@8 66 m_bHidden = false;
joachim99@8 67 m_bExists = false;
joachim99@8 68 m_bFile = false;
joachim99@8 69 m_bDir = false;
joachim99@8 70 m_bSymLink = false;
joachim99@8 71 }
joachim99@8 72
joachim99@8 73 FileAccess::~FileAccess()
joachim99@8 74 {
joachim99@8 75 if( !m_localCopy.isEmpty() )
joachim99@8 76 {
joachim99@73 77 removeTempFile( m_localCopy );
joachim99@8 78 }
joachim99@8 79 }
joachim99@8 80
joachim99@8 81 void FileAccess::setFile( const QString& name, bool bWantToWrite )
joachim99@8 82 {
joachim99@8 83 m_url = KURL::fromPathOrURL( name );
joachim99@8 84 m_bValidData = false;
joachim99@8 85
joachim99@8 86 m_size = 0;
joachim99@8 87 m_creationTime = QDateTime();
joachim99@8 88 m_accessTime = QDateTime();
joachim99@8 89 m_modificationTime = QDateTime();
joachim99@8 90 m_bReadable = false;
joachim99@8 91 m_bWritable = false;
joachim99@8 92 m_bExecutable = false;
joachim99@8 93 m_bHidden = false;
joachim99@8 94 m_bExists = false;
joachim99@8 95 m_bFile = false;
joachim99@8 96 m_bDir = false;
joachim99@8 97 m_bSymLink = false;
joachim99@8 98 m_linkTarget = "";
joachim99@8 99 m_fileType = -1;
joachim99@69 100 m_bLocal = true;
joachim99@8 101
joachim99@8 102 // Note: Checking if the filename-string is empty is necessary for Win95/98/ME.
joachim99@8 103 // The isFile() / isDir() queries would cause the program to crash.
joachim99@8 104 // (This is a Win95-bug which has been corrected only in WinNT/2000/XP.)
joachim99@8 105 if ( !name.isEmpty() )
joachim99@8 106 {
joachim99@51 107 // FileAccess tries to detect if the given name is an URL or a local file.
joachim99@51 108 // This is a problem if the filename looks like an URL (i.e. contains a colon ':').
joachim99@51 109 // e.g. "file:f.txt" is a valid filename.
joachim99@51 110 // Most of the time it is sufficient to check if the file exists locally.
joachim99@51 111 // 2 Problems remain:
joachim99@51 112 // 1. When the local file exists and the remote location is wanted nevertheless. (unlikely)
joachim99@51 113 // 2. When the local file doesn't exist and should be written to.
joachim99@51 114
joachim99@51 115 bool bExistsLocal = QDir().exists(name);
joachim99@51 116 if ( m_url.isLocalFile() || !m_url.isValid() || bExistsLocal ) // assuming that invalid means relative
joachim99@8 117 {
joachim99@8 118 QString localName = name;
joachim99@75 119 if ( !bExistsLocal && m_url.isLocalFile() && name.left(5).toLower()=="file:" )
joachim99@8 120 {
joachim99@8 121 localName = m_url.path(); // I want the path without preceding "file:"
joachim99@8 122 }
joachim99@8 123 QFileInfo fi( localName );
joachim99@75 124 #if defined(Q_WS_WIN)
joachim99@75 125 // On some windows machines in a network this takes very long.
joachim99@75 126 // and it's not so important anyway.
joachim99@75 127 m_bReadable = true;
joachim99@75 128 m_bWritable = true; // in certain situations this might become a problem though
joachim99@75 129 m_bExecutable = false;
joachim99@75 130 #else
joachim99@8 131 m_bReadable = fi.isReadable();
joachim99@8 132 m_bWritable = fi.isWritable();
joachim99@8 133 m_bExecutable = fi.isExecutable();
joachim99@75 134 #endif
joachim99@8 135 m_creationTime = fi.created();
joachim99@8 136 m_bHidden = fi.isHidden();
joachim99@8 137 m_modificationTime = fi.lastModified();
joachim99@8 138 m_accessTime = fi.lastRead();
joachim99@8 139 m_size = fi.size();
joachim99@8 140 m_bSymLink = fi.isSymLink();
joachim99@8 141 m_bFile = fi.isFile();
joachim99@8 142 m_bDir = fi.isDir();
joachim99@8 143 m_bExists = fi.exists();
joachim99@8 144 m_name = fi.fileName();
joachim99@8 145 m_path = fi.filePath();
joachim99@75 146 m_absFilePath= fi.absoluteFilePath();
joachim99@8 147 if ( m_bSymLink ) m_linkTarget = fi.readLink();
joachim99@8 148 m_bLocal = true;
joachim99@8 149 m_bValidData = true;
joachim99@8 150 if ( ! m_url.isValid() )
joachim99@8 151 {
joachim99@8 152 m_url.setPath( m_absFilePath );
joachim99@8 153 }
joachim99@69 154
joachim99@69 155 if ( !m_bExists && m_absFilePath.contains("@@") )
joachim99@69 156 {
joachim99@69 157 // Try reading a clearcase file
joachim99@69 158 m_localCopy = FileAccess::tempFileName();
joachim99@73 159 QString cmd = "cleartool get -to \"" + m_localCopy + "\" \"" + m_absFilePath + "\"";
joachim99@75 160 QProcess process;
joachim99@75 161 process.start( cmd );
joachim99@75 162 process.waitForFinished(-1);
joachim99@75 163 //::system( cmd.local8Bit() );
joachim99@69 164
joachim99@69 165 QFileInfo fi( m_localCopy );
joachim99@75 166 #if defined(Q_WS_WIN)
joachim99@75 167 m_bReadable = true;//fi.isReadable();
joachim99@75 168 m_bWritable = true;//fi.isWritable();
joachim99@75 169 m_bExecutable = false;//fi.isExecutable();
joachim99@75 170 #else
joachim99@69 171 m_bReadable = fi.isReadable();
joachim99@69 172 m_bWritable = fi.isWritable();
joachim99@69 173 m_bExecutable = fi.isExecutable();
joachim99@75 174 #endif
joachim99@69 175 m_creationTime = fi.created();
joachim99@69 176 m_bHidden = fi.isHidden();
joachim99@69 177 m_modificationTime = fi.lastModified();
joachim99@69 178 m_accessTime = fi.lastRead();
joachim99@69 179 m_size = fi.size();
joachim99@69 180 m_bSymLink = fi.isSymLink();
joachim99@69 181 m_bFile = fi.isFile();
joachim99@69 182 m_bDir = fi.isDir();
joachim99@69 183 m_bExists = fi.exists();
joachim99@69 184 }
joachim99@8 185 }
joachim99@8 186 else
joachim99@8 187 {
joachim99@8 188 m_absFilePath = name;
joachim99@8 189 m_name = m_url.fileName();
joachim99@8 190 m_bLocal = false;
joachim99@8 191
joachim99@8 192 FileAccessJobHandler jh( this ); // A friend, which writes to the parameters of this class!
joachim99@8 193 jh.stat(2/*all details*/, bWantToWrite); // returns bSuccess, ignored
joachim99@8 194
joachim99@8 195 m_path = name;
joachim99@51 196 m_bValidData = true; // After running stat() the variables are initialised
joachim99@51 197 // and valid even if the file doesn't exist and the stat
joachim99@51 198 // query failed.
joachim99@8 199 }
joachim99@8 200 }
joachim99@8 201 }
joachim99@8 202
joachim99@8 203 void FileAccess::addPath( const QString& txt )
joachim99@8 204 {
joachim99@51 205 if ( m_url.isValid() )
joachim99@51 206 {
joachim99@51 207 m_url.addPath( txt );
joachim99@51 208 setFile( m_url.url() ); // reinitialise
joachim99@51 209 }
joachim99@51 210 else
joachim99@51 211 {
joachim99@51 212 QString slash = (txt.isEmpty() || txt[0]=='/') ? "" : "/";
joachim99@51 213 setFile( absFilePath() + slash + txt );
joachim99@51 214 }
joachim99@8 215 }
joachim99@8 216
joachim99@8 217 /* Filetype:
joachim99@8 218 S_IFMT 0170000 bitmask for the file type bitfields
joachim99@8 219 S_IFSOCK 0140000 socket
joachim99@8 220 S_IFLNK 0120000 symbolic link
joachim99@8 221 S_IFREG 0100000 regular file
joachim99@8 222 S_IFBLK 0060000 block device
joachim99@8 223 S_IFDIR 0040000 directory
joachim99@8 224 S_IFCHR 0020000 character device
joachim99@8 225 S_IFIFO 0010000 fifo
joachim99@8 226 S_ISUID 0004000 set UID bit
joachim99@8 227 S_ISGID 0002000 set GID bit (see below)
joachim99@8 228 S_ISVTX 0001000 sticky bit (see below)
joachim99@8 229
joachim99@8 230 Access:
joachim99@8 231 S_IRWXU 00700 mask for file owner permissions
joachim99@8 232 S_IRUSR 00400 owner has read permission
joachim99@8 233 S_IWUSR 00200 owner has write permission
joachim99@8 234 S_IXUSR 00100 owner has execute permission
joachim99@8 235 S_IRWXG 00070 mask for group permissions
joachim99@8 236 S_IRGRP 00040 group has read permission
joachim99@8 237 S_IWGRP 00020 group has write permission
joachim99@8 238 S_IXGRP 00010 group has execute permission
joachim99@8 239 S_IRWXO 00007 mask for permissions for others (not in group)
joachim99@8 240 S_IROTH 00004 others have read permission
joachim99@8 241 S_IWOTH 00002 others have write permisson
joachim99@8 242 S_IXOTH 00001 others have execute permission
joachim99@8 243 */
joachim99@8 244
joachim99@58 245 #ifdef KREPLACEMENTS_H
joachim99@58 246 void FileAccess::setUdsEntry( const KIO::UDSEntry& ){} // not needed if KDE is not available
joachim99@58 247 #else
joachim99@8 248 void FileAccess::setUdsEntry( const KIO::UDSEntry& e )
joachim99@8 249 {
joachim99@8 250 KIO::UDSEntry::const_iterator ei;
joachim99@8 251 long acc = 0;
joachim99@8 252 long fileType = 0;
joachim99@8 253 for( ei=e.begin(); ei!=e.end(); ++ei )
joachim99@8 254 {
joachim99@8 255 const KIO::UDSAtom& a = *ei;
joachim99@8 256 switch( a.m_uds )
joachim99@8 257 {
joachim99@8 258 case KIO::UDS_SIZE : m_size = a.m_long; break;
joachim99@8 259 case KIO::UDS_USER : m_user = a.m_str; break;
joachim99@8 260 case KIO::UDS_GROUP : m_group = a.m_str; break;
joachim99@8 261 case KIO::UDS_NAME : m_path = a.m_str; break; // During listDir the relative path is given here.
joachim99@8 262 case KIO::UDS_MODIFICATION_TIME : m_modificationTime.setTime_t( a.m_long ); break;
joachim99@8 263 case KIO::UDS_ACCESS_TIME : m_accessTime.setTime_t( a.m_long ); break;
joachim99@8 264 case KIO::UDS_CREATION_TIME : m_creationTime.setTime_t( a.m_long ); break;
joachim99@8 265 case KIO::UDS_LINK_DEST : m_linkTarget = a.m_str; break;
joachim99@8 266 case KIO::UDS_ACCESS :
joachim99@8 267 {
joachim99@8 268 acc = a.m_long;
joachim99@8 269 m_bReadable = (acc & S_IRUSR)!=0;
joachim99@8 270 m_bWritable = (acc & S_IWUSR)!=0;
joachim99@8 271 m_bExecutable = (acc & S_IXUSR)!=0;
joachim99@8 272 break;
joachim99@8 273 }
joachim99@8 274 case KIO::UDS_FILE_TYPE :
joachim99@8 275 {
joachim99@8 276 fileType = a.m_long;
joachim99@8 277 m_bDir = ( fileType & S_IFMT ) == S_IFDIR;
joachim99@8 278 m_bFile = ( fileType & S_IFMT ) == S_IFREG;
joachim99@8 279 m_bSymLink = ( fileType & S_IFMT ) == S_IFLNK;
joachim99@8 280 m_bExists = fileType != 0;
joachim99@8 281 m_fileType = fileType;
joachim99@8 282 break;
joachim99@8 283 }
joachim99@8 284
joachim99@8 285 case KIO::UDS_URL : // m_url = KURL( a.str );
joachim99@8 286 break;
joachim99@8 287 case KIO::UDS_MIME_TYPE : break;
joachim99@8 288 case KIO::UDS_GUESSED_MIME_TYPE : break;
joachim99@8 289 case KIO::UDS_XML_PROPERTIES : break;
joachim99@8 290 default: break;
joachim99@8 291 }
joachim99@8 292 }
joachim99@8 293
joachim99@8 294 m_bExists = acc!=0 || fileType!=0;
joachim99@8 295
joachim99@8 296 m_bLocal = false;
joachim99@8 297 m_bValidData = true;
joachim99@8 298 m_bSymLink = !m_linkTarget.isEmpty();
joachim99@8 299 if ( m_name.isEmpty() )
joachim99@8 300 {
joachim99@8 301 int pos = m_path.findRev('/') + 1;
joachim99@8 302 m_name = m_path.mid( pos );
joachim99@8 303 }
joachim99@8 304 m_bHidden = m_name[0]=='.';
joachim99@58 305 }
joachim99@8 306 #endif
joachim99@8 307
joachim99@8 308
joachim99@8 309 bool FileAccess::isValid() const { return m_bValidData; }
joachim99@8 310 bool FileAccess::isFile() const { return m_bFile; }
joachim99@8 311 bool FileAccess::isDir() const { return m_bDir; }
joachim99@8 312 bool FileAccess::isSymLink() const { return m_bSymLink; }
joachim99@8 313 bool FileAccess::exists() const { return m_bExists; }
joachim99@8 314 long FileAccess::size() const { return m_size; }
joachim99@8 315 KURL FileAccess::url() const { return m_url; }
joachim99@8 316 bool FileAccess::isLocal() const { return m_bLocal; }
joachim99@8 317 bool FileAccess::isReadable() const { return m_bReadable; }
joachim99@8 318 bool FileAccess::isWritable() const { return m_bWritable; }
joachim99@8 319 bool FileAccess::isExecutable() const { return m_bExecutable; }
joachim99@8 320 bool FileAccess::isHidden() const { return m_bHidden; }
joachim99@8 321 QString FileAccess::readLink() const { return m_linkTarget; }
joachim99@8 322 QString FileAccess::absFilePath() const{ return m_absFilePath; } // Full abs path
joachim99@8 323 QString FileAccess::fileName() const { return m_name; } // Just the name-part of the path, without parent directories
joachim99@8 324 QString FileAccess::filePath() const { return m_path; } // The path-string that was used during construction
joachim99@8 325 QString FileAccess::prettyAbsPath() const { return isLocal() ? m_absFilePath : m_url.prettyURL(); }
joachim99@8 326
joachim99@8 327 QDateTime FileAccess::created() const
joachim99@8 328 {
joachim99@8 329 return ( m_creationTime.isValid() ? m_creationTime : m_modificationTime );
joachim99@8 330 }
joachim99@8 331
joachim99@8 332 QDateTime FileAccess::lastModified() const
joachim99@8 333 {
joachim99@8 334 return m_modificationTime;
joachim99@8 335 }
joachim99@8 336
joachim99@8 337 QDateTime FileAccess::lastRead() const
joachim99@8 338 {
joachim99@8 339 return ( m_accessTime.isValid() ? m_accessTime : m_modificationTime );
joachim99@8 340 }
joachim99@8 341
joachim99@8 342 static bool interruptableReadFile( QFile& f, void* pDestBuffer, unsigned long maxLength )
joachim99@8 343 {
joachim99@66 344 ProgressProxy pp;
joachim99@8 345 const unsigned long maxChunkSize = 100000;
joachim99@8 346 unsigned long i=0;
joachim99@8 347 while( i<maxLength )
joachim99@8 348 {
joachim99@8 349 unsigned long nextLength = min2( maxLength-i, maxChunkSize );
joachim99@75 350 unsigned long reallyRead = f.read( (char*)pDestBuffer+i, nextLength );
joachim99@8 351 if ( reallyRead != nextLength )
joachim99@8 352 {
joachim99@8 353 return false;
joachim99@8 354 }
joachim99@8 355 i+=reallyRead;
joachim99@8 356
joachim99@66 357 pp.setCurrent( double(i)/maxLength );
joachim99@66 358 if ( pp.wasCancelled() ) return false;
joachim99@8 359 }
joachim99@8 360 return true;
joachim99@8 361 }
joachim99@8 362
joachim99@8 363 bool FileAccess::readFile( void* pDestBuffer, unsigned long maxLength )
joachim99@8 364 {
joachim99@8 365 if ( !m_localCopy.isEmpty() )
joachim99@8 366 {
joachim99@8 367 QFile f( m_localCopy );
joachim99@70 368 if ( f.open( QIODevice::ReadOnly ) )
joachim99@8 369 return interruptableReadFile(f, pDestBuffer, maxLength);// maxLength == f.readBlock( (char*)pDestBuffer, maxLength );
joachim99@8 370 }
joachim99@8 371 else if (m_bLocal)
joachim99@8 372 {
joachim99@8 373 QFile f( filePath() );
joachim99@8 374
joachim99@70 375 if ( f.open( QIODevice::ReadOnly ) )
joachim99@8 376 return interruptableReadFile(f, pDestBuffer, maxLength); //maxLength == f.readBlock( (char*)pDestBuffer, maxLength );
joachim99@8 377 }
joachim99@8 378 else
joachim99@8 379 {
joachim99@8 380 FileAccessJobHandler jh( this );
joachim99@8 381 return jh.get( pDestBuffer, maxLength );
joachim99@8 382 }
joachim99@8 383 return false;
joachim99@8 384 }
joachim99@8 385
joachim99@58 386 bool FileAccess::writeFile( const void* pSrcBuffer, unsigned long length )
joachim99@8 387 {
joachim99@66 388 ProgressProxy pp;
joachim99@8 389 if (m_bLocal)
joachim99@8 390 {
joachim99@8 391 QFile f( filePath() );
joachim99@70 392 if ( f.open( QIODevice::WriteOnly ) )
joachim99@8 393 {
joachim99@8 394 const unsigned long maxChunkSize = 100000;
joachim99@8 395 unsigned long i=0;
joachim99@8 396 while( i<length )
joachim99@8 397 {
joachim99@8 398 unsigned long nextLength = min2( length-i, maxChunkSize );
joachim99@75 399 unsigned long reallyWritten = f.write( (char*)pSrcBuffer+i, nextLength );
joachim99@8 400 if ( reallyWritten != nextLength )
joachim99@8 401 {
joachim99@8 402 return false;
joachim99@8 403 }
joachim99@8 404 i+=reallyWritten;
joachim99@8 405
joachim99@66 406 pp.setCurrent( double(i)/length );
joachim99@66 407 if ( pp.wasCancelled() ) return false;
joachim99@8 408 }
joachim99@73 409 f.close();
joachim99@73 410 #ifndef _WIN32
joachim99@73 411 if ( isExecutable() ) // value is true if the old file was executable
joachim99@73 412 {
joachim99@73 413 // Preserve attributes
joachim99@75 414 f.setPermissions(QFile::ExeUser);
joachim99@75 415 //struct stat srcFileStatus;
joachim99@75 416 //int statResult = ::stat( filePath().toLocal8Bit().constData(), &srcFileStatus );
joachim99@75 417 //if (statResult==0)
joachim99@75 418 //{
joachim99@75 419 // ::chmod ( filePath().toLocal8Bit().constData(), srcFileStatus.st_mode | S_IXUSR );
joachim99@75 420 //}
joachim99@73 421 }
joachim99@73 422 #endif
joachim99@73 423
joachim99@8 424 return true;
joachim99@8 425 }
joachim99@8 426 }
joachim99@8 427 else
joachim99@8 428 {
joachim99@8 429 FileAccessJobHandler jh( this );
joachim99@8 430 return jh.put( pSrcBuffer, length, true /*overwrite*/ );
joachim99@8 431 }
joachim99@8 432 return false;
joachim99@8 433 }
joachim99@8 434
joachim99@8 435 bool FileAccess::copyFile( const QString& dest )
joachim99@8 436 {
joachim99@8 437 FileAccessJobHandler jh( this );
joachim99@8 438 return jh.copyFile( dest ); // Handles local and remote copying.
joachim99@8 439 }
joachim99@8 440
joachim99@8 441 bool FileAccess::rename( const QString& dest )
joachim99@8 442 {
joachim99@8 443 FileAccessJobHandler jh( this );
joachim99@8 444 return jh.rename( dest );
joachim99@8 445 }
joachim99@8 446
joachim99@8 447 bool FileAccess::removeFile()
joachim99@8 448 {
joachim99@8 449 if ( isLocal() )
joachim99@8 450 {
joachim99@8 451 return QDir().remove( absFilePath() );
joachim99@8 452 }
joachim99@8 453 else
joachim99@8 454 {
joachim99@8 455 FileAccessJobHandler jh( this );
joachim99@8 456 return jh.removeFile( absFilePath() );
joachim99@8 457 }
joachim99@8 458 }
joachim99@8 459
joachim99@8 460 bool FileAccess::removeFile( const QString& name ) // static
joachim99@8 461 {
joachim99@8 462 return FileAccess(name).removeFile();
joachim99@8 463 }
joachim99@8 464
joachim99@8 465 bool FileAccess::listDir( t_DirectoryList* pDirList, bool bRecursive, bool bFindHidden,
joachim99@8 466 const QString& filePattern, const QString& fileAntiPattern, const QString& dirAntiPattern,
joachim99@8 467 bool bFollowDirLinks, bool bUseCvsIgnore )
joachim99@8 468 {
joachim99@8 469 FileAccessJobHandler jh( this );
joachim99@8 470 return jh.listDir( pDirList, bRecursive, bFindHidden, filePattern, fileAntiPattern,
joachim99@8 471 dirAntiPattern, bFollowDirLinks, bUseCvsIgnore );
joachim99@8 472 }
joachim99@8 473
joachim99@8 474 QString FileAccess::tempFileName()
joachim99@8 475 {
joachim99@8 476 #ifdef KREPLACEMENTS_H
joachim99@8 477
joachim99@8 478 QString fileName;
joachim99@8 479 #ifdef _WIN32
joachim99@8 480 QString tmpDir = getenv("TEMP");
joachim99@8 481 #else
joachim99@8 482 QString tmpDir = "/tmp";
joachim99@69 483 #endif
joachim99@69 484 for(int i=0; ;++i)
joachim99@69 485 {
joachim99@69 486 // short filenames for WIN98 because for system() the command must not exceed 120 characters.
joachim99@69 487 #ifdef _WIN32
joachim99@75 488 if ( QSysInfo::WindowsVersion & QSysInfo::WV_DOS_based ) // Win95, 98, ME
joachim99@69 489 fileName = tmpDir + "\\" + QString::number(i);
joachim99@69 490 else
joachim99@77 491 fileName = tmpDir + "/kdiff3_" + QString::number(_getpid()) + "_" + QString::number(i) +".tmp";
joachim99@77 492 #else
joachim99@77 493 fileName = tmpDir + "/kdiff3_" + QString::number(getpid()) + "_" + QString::number(i) +".tmp";
joachim99@69 494 #endif
joachim99@69 495 if ( ! FileAccess::exists(fileName) &&
joachim99@70 496 QFile(fileName).open(QIODevice::WriteOnly) ) // open, truncate and close the file, true if successful
joachim99@73 497 {
joachim99@69 498 break;
joachim99@73 499 }
joachim99@69 500 }
joachim99@73 501 return QDir::convertSeparators(fileName+".2");
joachim99@8 502
joachim99@8 503 #else // using KDE
joachim99@8 504
joachim99@8 505 KTempFile tmpFile;
joachim99@69 506 //tmpFile.setAutoDelete( true ); // We only want the name. Delete the precreated file immediately.
joachim99@69 507 tmpFile.close();
joachim99@73 508 return tmpFile.name()+".2";
joachim99@8 509
joachim99@8 510 #endif
joachim99@8 511 }
joachim99@8 512
joachim99@73 513 bool FileAccess::removeTempFile( const QString& name ) // static
joachim99@73 514 {
joachim99@73 515 if (name.endsWith(".2"))
joachim99@73 516 FileAccess(name.left(name.length()-2)).removeFile();
joachim99@73 517 return FileAccess(name).removeFile();
joachim99@73 518 }
joachim99@73 519
joachim99@73 520
joachim99@8 521 bool FileAccess::makeDir( const QString& dirName )
joachim99@8 522 {
joachim99@8 523 FileAccessJobHandler fh(0);
joachim99@8 524 return fh.mkDir( dirName );
joachim99@8 525 }
joachim99@8 526
joachim99@8 527 bool FileAccess::removeDir( const QString& dirName )
joachim99@8 528 {
joachim99@8 529 FileAccessJobHandler fh(0);
joachim99@8 530 return fh.rmDir( dirName );
joachim99@8 531 }
joachim99@8 532
joachim99@75 533 #ifdef _WIN32
joachim99@75 534 bool FileAccess::symLink( const QString& /*linkTarget*/, const QString& /*linkLocation*/ )
joachim99@75 535 {
joachim99@75 536 return false;
joachim99@75 537 }
joachim99@75 538 #else
joachim99@8 539 bool FileAccess::symLink( const QString& linkTarget, const QString& linkLocation )
joachim99@8 540 {
joachim99@75 541 return 0==::symlink( linkTarget.toLocal8Bit().constData(), linkLocation.toLocal8Bit().constData() );
joachim99@8 542 //FileAccessJobHandler fh(0);
joachim99@8 543 //return fh.symLink( linkTarget, linkLocation );
joachim99@75 544 }
friseb123@19 545 #endif
joachim99@8 546
joachim99@8 547 bool FileAccess::exists( const QString& name )
joachim99@8 548 {
joachim99@8 549 FileAccess fa( name );
joachim99@8 550 return fa.exists();
joachim99@8 551 }
joachim99@8 552
joachim99@8 553 // If the size couldn't be determined by stat() then the file is copied to a local temp file.
joachim99@8 554 long FileAccess::sizeForReading()
joachim99@8 555 {
joachim99@8 556 if ( m_size == 0 && !isLocal() )
joachim99@8 557 {
joachim99@8 558 // Size couldn't be determined. Copy the file to a local temp place.
joachim99@8 559 QString localCopy = tempFileName();
joachim99@8 560 bool bSuccess = copyFile( localCopy );
joachim99@8 561 if ( bSuccess )
joachim99@8 562 {
joachim99@8 563 QFileInfo fi( localCopy );
joachim99@8 564 m_size = fi.size();
joachim99@8 565 m_localCopy = localCopy;
joachim99@8 566 return m_size;
joachim99@8 567 }
joachim99@8 568 else
joachim99@8 569 {
joachim99@8 570 return 0;
joachim99@8 571 }
joachim99@8 572 }
joachim99@8 573 else
joachim99@8 574 return m_size;
joachim99@8 575 }
joachim99@8 576
joachim99@8 577 QString FileAccess::getStatusText()
joachim99@8 578 {
joachim99@8 579 return m_statusText;
joachim99@8 580 }
joachim99@8 581
joachim99@8 582 QString FileAccess::cleanDirPath( const QString& path ) // static
joachim99@8 583 {
joachim99@8 584 KURL url(path);
joachim99@8 585 if ( url.isLocalFile() || ! url.isValid() )
joachim99@8 586 {
joachim99@75 587 return QDir().cleanPath( path );
joachim99@8 588 }
joachim99@8 589 else
joachim99@8 590 {
joachim99@8 591 return path;
joachim99@8 592 }
joachim99@8 593 }
joachim99@8 594
joachim99@8 595 bool FileAccess::createBackup( const QString& bakExtension )
joachim99@8 596 {
joachim99@8 597 if ( exists() )
joachim99@8 598 {
joachim99@8 599 // First rename the existing file to the bak-file. If a bak-file file exists, delete that.
joachim99@8 600 QString bakName = absFilePath() + bakExtension;
joachim99@8 601 FileAccess bakFile( bakName, true /*bWantToWrite*/ );
joachim99@8 602 if ( bakFile.exists() )
joachim99@8 603 {
joachim99@8 604 bool bSuccess = bakFile.removeFile();
joachim99@8 605 if ( !bSuccess )
joachim99@8 606 {
joachim99@8 607 m_statusText = i18n("While trying to make a backup, deleting an older backup failed. \nFilename: ") + bakName;
joachim99@8 608 return false;
joachim99@8 609 }
joachim99@8 610 }
joachim99@8 611 bool bSuccess = rename( bakName );
joachim99@8 612 if (!bSuccess)
joachim99@8 613 {
joachim99@8 614 m_statusText = i18n("While trying to make a backup, renaming failed. \nFilenames: ") +
joachim99@8 615 absFilePath() + " -> " + bakName;
joachim99@8 616 return false;
joachim99@8 617 }
joachim99@8 618 }
joachim99@8 619 return true;
joachim99@8 620 }
joachim99@8 621
joachim99@8 622 FileAccessJobHandler::FileAccessJobHandler( FileAccess* pFileAccess )
joachim99@8 623 {
joachim99@8 624 m_pFileAccess = pFileAccess;
joachim99@8 625 m_bSuccess = false;
joachim99@8 626 }
joachim99@8 627
joachim99@8 628 bool FileAccessJobHandler::stat( int detail, bool bWantToWrite )
joachim99@8 629 {
joachim99@8 630 m_bSuccess = false;
joachim99@8 631 m_pFileAccess->m_statusText = QString();
joachim99@8 632 KIO::StatJob* pStatJob = KIO::stat( m_pFileAccess->m_url, ! bWantToWrite, detail, false );
joachim99@8 633
joachim99@8 634 connect( pStatJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotStatResult(KIO::Job*)));
joachim99@8 635
joachim99@51 636 g_pProgressDialog->enterEventLoop( pStatJob, i18n("Getting file status: %1").arg(m_pFileAccess->prettyAbsPath()) );
joachim99@8 637
joachim99@8 638 return m_bSuccess;
joachim99@8 639 }
joachim99@8 640
joachim99@8 641 void FileAccessJobHandler::slotStatResult(KIO::Job* pJob)
joachim99@8 642 {
joachim99@8 643 if ( pJob->error() )
joachim99@8 644 {
joachim99@8 645 //pJob->showErrorDialog(g_pProgressDialog);
joachim99@8 646 m_pFileAccess->m_bExists = false;
joachim99@8 647 m_bSuccess = true;
joachim99@8 648 }
joachim99@8 649 else
joachim99@8 650 {
joachim99@8 651 m_bSuccess = true;
joachim99@8 652
joachim99@8 653 m_pFileAccess->m_bValidData = true;
joachim99@8 654 const KIO::UDSEntry e = static_cast<KIO::StatJob*>(pJob)->statResult();
joachim99@8 655
joachim99@8 656 m_pFileAccess->setUdsEntry( e );
joachim99@8 657 }
joachim99@8 658
joachim99@8 659 g_pProgressDialog->exitEventLoop();
joachim99@8 660 }
joachim99@8 661
joachim99@8 662
joachim99@8 663 bool FileAccessJobHandler::get(void* pDestBuffer, long maxLength )
joachim99@8 664 {
joachim99@66 665 ProgressProxy pp; // Implicitly used in slotPercent()
joachim99@66 666 if ( maxLength>0 && !pp.wasCancelled() )
joachim99@8 667 {
joachim99@8 668 KIO::TransferJob* pJob = KIO::get( m_pFileAccess->m_url, false /*reload*/, false );
joachim99@8 669 m_transferredBytes = 0;
joachim99@8 670 m_pTransferBuffer = (char*)pDestBuffer;
joachim99@8 671 m_maxLength = maxLength;
joachim99@8 672 m_bSuccess = false;
joachim99@8 673 m_pFileAccess->m_statusText = QString();
joachim99@8 674
joachim99@8 675 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 676 connect( pJob, SIGNAL(data(KIO::Job*,const QByteArray &)), this, SLOT(slotGetData(KIO::Job*, const QByteArray&)));
joachim99@8 677 connect( pJob, SIGNAL(percent(KIO::Job*,unsigned long)), this, SLOT(slotPercent(KIO::Job*, unsigned long)));
joachim99@8 678
joachim99@51 679 g_pProgressDialog->enterEventLoop( pJob, i18n("Reading file: %1").arg(m_pFileAccess->prettyAbsPath()) );
joachim99@8 680 return m_bSuccess;
joachim99@8 681 }
joachim99@8 682 else
joachim99@8 683 return true;
joachim99@8 684 }
joachim99@8 685
joachim99@8 686 void FileAccessJobHandler::slotGetData( KIO::Job* pJob, const QByteArray& newData )
joachim99@8 687 {
joachim99@8 688 if ( pJob->error() )
joachim99@8 689 {
joachim99@8 690 pJob->showErrorDialog(g_pProgressDialog);
joachim99@8 691 }
joachim99@8 692 else
joachim99@8 693 {
joachim99@8 694 long length = min2( long(newData.size()), m_maxLength - m_transferredBytes );
joachim99@8 695 ::memcpy( m_pTransferBuffer + m_transferredBytes, newData.data(), newData.size() );
joachim99@8 696 m_transferredBytes += length;
joachim99@8 697 }
joachim99@8 698 }
joachim99@8 699
joachim99@58 700 bool FileAccessJobHandler::put(const void* pSrcBuffer, long maxLength, bool bOverwrite, bool bResume, int permissions )
joachim99@8 701 {
joachim99@8 702 if ( maxLength>0 )
joachim99@8 703 {
joachim99@8 704 KIO::TransferJob* pJob = KIO::put( m_pFileAccess->m_url, permissions, bOverwrite, bResume, false );
joachim99@8 705 m_transferredBytes = 0;
joachim99@8 706 m_pTransferBuffer = (char*)pSrcBuffer;
joachim99@8 707 m_maxLength = maxLength;
joachim99@8 708 m_bSuccess = false;
joachim99@8 709 m_pFileAccess->m_statusText = QString();
joachim99@8 710
joachim99@8 711 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotPutJobResult(KIO::Job*)));
joachim99@8 712 connect( pJob, SIGNAL(dataReq(KIO::Job*, QByteArray&)), this, SLOT(slotPutData(KIO::Job*, QByteArray&)));
joachim99@8 713 connect( pJob, SIGNAL(percent(KIO::Job*,unsigned long)), this, SLOT(slotPercent(KIO::Job*, unsigned long)));
joachim99@8 714
joachim99@51 715 g_pProgressDialog->enterEventLoop( pJob, i18n("Writing file: %1").arg(m_pFileAccess->prettyAbsPath()) );
joachim99@8 716 return m_bSuccess;
joachim99@8 717 }
joachim99@8 718 else
joachim99@8 719 return true;
joachim99@8 720 }
joachim99@8 721
joachim99@8 722 void FileAccessJobHandler::slotPutData( KIO::Job* pJob, QByteArray& data )
joachim99@8 723 {
joachim99@8 724 if ( pJob->error() )
joachim99@8 725 {
joachim99@8 726 pJob->showErrorDialog(g_pProgressDialog);
joachim99@8 727 }
joachim99@8 728 else
joachim99@8 729 {
joachim99@8 730 long maxChunkSize = 100000;
joachim99@8 731 long length = min2( maxChunkSize, m_maxLength - m_transferredBytes );
joachim99@70 732 data.resize( length );
joachim99@70 733 if ( data.size()==length )
joachim99@8 734 {
joachim99@8 735 if ( length>0 )
joachim99@8 736 {
joachim99@8 737 ::memcpy( data.data(), m_pTransferBuffer + m_transferredBytes, data.size() );
joachim99@8 738 m_transferredBytes += length;
joachim99@8 739 }
joachim99@8 740 }
joachim99@8 741 else
joachim99@8 742 {
joachim99@8 743 KMessageBox::error( g_pProgressDialog, i18n("Out of memory") );
joachim99@8 744 data.resize(0);
joachim99@8 745 m_bSuccess = false;
joachim99@8 746 }
joachim99@8 747 }
joachim99@8 748 }
joachim99@8 749
joachim99@8 750 void FileAccessJobHandler::slotPutJobResult(KIO::Job* pJob)
joachim99@8 751 {
joachim99@8 752 if ( pJob->error() )
joachim99@8 753 {
joachim99@8 754 pJob->showErrorDialog(g_pProgressDialog);
joachim99@8 755 }
joachim99@8 756 else
joachim99@8 757 {
joachim99@8 758 m_bSuccess = (m_transferredBytes == m_maxLength); // Special success condition
joachim99@8 759 }
joachim99@8 760 g_pProgressDialog->exitEventLoop(); // Close the dialog, return from exec()
joachim99@8 761 }
joachim99@8 762
joachim99@8 763 bool FileAccessJobHandler::mkDir( const QString& dirName )
joachim99@8 764 {
joachim99@8 765 KURL dirURL = KURL::fromPathOrURL( dirName );
joachim99@8 766 if ( dirName.isEmpty() )
joachim99@8 767 return false;
joachim99@8 768 else if ( dirURL.isLocalFile() )
joachim99@8 769 {
joachim99@8 770 return QDir().mkdir( dirURL.path() );
joachim99@8 771 }
joachim99@8 772 else
joachim99@8 773 {
joachim99@8 774 m_bSuccess = false;
joachim99@8 775 KIO::SimpleJob* pJob = KIO::mkdir( dirURL );
joachim99@8 776 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 777
joachim99@51 778 g_pProgressDialog->enterEventLoop( pJob, i18n("Making directory: %1").arg(dirName) );
joachim99@8 779 return m_bSuccess;
joachim99@8 780 }
joachim99@8 781 }
joachim99@8 782
joachim99@8 783 bool FileAccessJobHandler::rmDir( const QString& dirName )
joachim99@8 784 {
joachim99@8 785 KURL dirURL = KURL::fromPathOrURL( dirName );
joachim99@8 786 if ( dirName.isEmpty() )
joachim99@8 787 return false;
joachim99@8 788 else if ( dirURL.isLocalFile() )
joachim99@8 789 {
joachim99@8 790 return QDir().rmdir( dirURL.path() );
joachim99@8 791 }
joachim99@8 792 else
joachim99@8 793 {
joachim99@8 794 m_bSuccess = false;
joachim99@8 795 KIO::SimpleJob* pJob = KIO::rmdir( dirURL );
joachim99@8 796 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 797
joachim99@51 798 g_pProgressDialog->enterEventLoop(pJob, i18n("Removing directory: %1").arg(dirName));
joachim99@8 799 return m_bSuccess;
joachim99@8 800 }
joachim99@8 801 }
joachim99@8 802
joachim99@8 803 bool FileAccessJobHandler::removeFile( const QString& fileName )
joachim99@8 804 {
joachim99@8 805 if ( fileName.isEmpty() )
joachim99@8 806 return false;
joachim99@8 807 else
joachim99@8 808 {
joachim99@8 809 m_bSuccess = false;
joachim99@68 810 KIO::SimpleJob* pJob = KIO::file_delete( KURL::fromPathOrURL(fileName), false );
joachim99@8 811 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 812
joachim99@51 813 g_pProgressDialog->enterEventLoop( pJob, i18n("Removing file: %1").arg(fileName) );
joachim99@8 814 return m_bSuccess;
joachim99@8 815 }
joachim99@8 816 }
joachim99@8 817
joachim99@8 818 bool FileAccessJobHandler::symLink( const QString& linkTarget, const QString& linkLocation )
joachim99@8 819 {
joachim99@8 820 if ( linkTarget.isEmpty() || linkLocation.isEmpty() )
joachim99@8 821 return false;
joachim99@8 822 else
joachim99@8 823 {
joachim99@8 824 m_bSuccess = false;
joachim99@68 825 KIO::CopyJob* pJob = KIO::link( KURL::fromPathOrURL(linkTarget), KURL::fromPathOrURL(linkLocation), false );
joachim99@8 826 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 827
joachim99@51 828 g_pProgressDialog->enterEventLoop( pJob,
joachim99@51 829 i18n("Creating symbolic link: %1 -> %2").arg(linkLocation).arg(linkTarget) );
joachim99@8 830 return m_bSuccess;
joachim99@8 831 }
joachim99@8 832 }
joachim99@8 833
joachim99@8 834 bool FileAccessJobHandler::rename( const QString& dest )
joachim99@8 835 {
joachim99@8 836 if ( dest.isEmpty() )
joachim99@8 837 return false;
joachim99@77 838
joachim99@77 839 KURL kurl = KURL::fromPathOrURL( dest );
joachim99@77 840 if ( !kurl.isValid() )
joachim99@77 841 kurl = KURL::fromPathOrURL( QDir().absoluteFilePath(dest) ); // assuming that invalid means relative
joachim99@77 842
joachim99@77 843 if ( m_pFileAccess->isLocal() && kurl.isLocalFile() )
joachim99@8 844 {
joachim99@8 845 return QDir().rename( m_pFileAccess->absFilePath(), kurl.path() );
joachim99@8 846 }
joachim99@8 847 else
joachim99@8 848 {
joachim99@8 849 bool bOverwrite = false;
joachim99@8 850 bool bResume = false;
joachim99@8 851 bool bShowProgress = false;
joachim99@8 852 int permissions=-1;
joachim99@8 853 m_bSuccess = false;
joachim99@68 854 KIO::FileCopyJob* pJob = KIO::file_move( m_pFileAccess->m_url, kurl, permissions, bOverwrite, bResume, bShowProgress );
joachim99@8 855 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 856 connect( pJob, SIGNAL(percent(KIO::Job*,unsigned long)), this, SLOT(slotPercent(KIO::Job*, unsigned long)));
joachim99@8 857
joachim99@51 858 g_pProgressDialog->enterEventLoop( pJob,
joachim99@51 859 i18n("Renaming file: %1 -> %2").arg(m_pFileAccess->prettyAbsPath()).arg(dest) );
joachim99@8 860 return m_bSuccess;
joachim99@8 861 }
joachim99@8 862 }
joachim99@8 863
joachim99@8 864 void FileAccessJobHandler::slotSimpleJobResult(KIO::Job* pJob)
joachim99@8 865 {
joachim99@8 866 if ( pJob->error() )
joachim99@8 867 {
joachim99@8 868 pJob->showErrorDialog(g_pProgressDialog);
joachim99@8 869 }
joachim99@8 870 else
joachim99@8 871 {
joachim99@8 872 m_bSuccess = true;
joachim99@8 873 }
joachim99@8 874 g_pProgressDialog->exitEventLoop(); // Close the dialog, return from exec()
joachim99@8 875 }
joachim99@8 876
joachim99@8 877
joachim99@8 878 // Copy local or remote files.
joachim99@8 879 bool FileAccessJobHandler::copyFile( const QString& dest )
joachim99@8 880 {
joachim99@66 881 ProgressProxy pp;
joachim99@8 882 KURL destUrl = KURL::fromPathOrURL( dest );
joachim99@8 883 m_pFileAccess->m_statusText = QString();
joachim99@8 884 if ( ! m_pFileAccess->isLocal() || ! destUrl.isLocalFile() ) // if either url is nonlocal
joachim99@8 885 {
joachim99@8 886 bool bOverwrite = false;
joachim99@8 887 bool bResume = false;
joachim99@8 888 bool bShowProgress = false;
joachim99@8 889 int permissions = (m_pFileAccess->isExecutable()?0111:0)+(m_pFileAccess->isWritable()?0222:0)+(m_pFileAccess->isReadable()?0444:0);
joachim99@8 890 m_bSuccess = false;
joachim99@68 891 KIO::FileCopyJob* pJob = KIO::file_copy ( m_pFileAccess->m_url, destUrl, permissions, bOverwrite, bResume, bShowProgress );
joachim99@8 892 connect( pJob, SIGNAL(result(KIO::Job*)), this, SLOT(slotSimpleJobResult(KIO::Job*)));
joachim99@8 893 connect( pJob, SIGNAL(percent(KIO::Job*,unsigned long)), this, SLOT(slotPercent(KIO::Job*, unsigned long)));
joachim99@51 894 g_pProgressDialog->enterEventLoop( pJob,
joachim99@51 895 i18n("Copying file: %1 -> %2").arg(m_pFileAccess->prettyAbsPath()).arg(dest) );
joachim99@8 896
joachim99@8 897 return m_bSuccess;
joachim99@8 898 // Note that the KIO-slave preserves the original date, if this is supported.
joachim99@8 899 }
joachim99@8 900
joachim99@8 901 // Both files are local:
joachim99@8 902 QString srcName = m_pFileAccess->absFilePath();
joachim99@8 903 QString destName = dest;
joachim99@8 904 QFile srcFile( srcName );
joachim99@8 905 QFile destFile( destName );
joachim99@70 906 bool bReadSuccess = srcFile.open( QIODevice::ReadOnly );
joachim99@8 907 if ( bReadSuccess == false )
joachim99@8 908 {
joachim99@51 909 m_pFileAccess->m_statusText = i18n("Error during file copy operation: Opening file for reading failed. Filename: %1").arg(srcName);
joachim99@8 910 return false;
joachim99@8 911 }
joachim99@70 912 bool bWriteSuccess = destFile.open( QIODevice::WriteOnly );
joachim99@8 913 if ( bWriteSuccess == false )
joachim99@8 914 {
joachim99@51 915 m_pFileAccess->m_statusText = i18n("Error during file copy operation: Opening file for writing failed. Filename: %1").arg(destName);
joachim99@8 916 return false;
joachim99@8 917 }
joachim99@8 918
joachim99@8 919 #if QT_VERSION==230
joachim99@8 920 typedef long Q_LONG;
joachim99@8 921 #endif
joachim99@8 922 std::vector<char> buffer(100000);
joachim99@75 923 qint64 bufSize = buffer.size();
joachim99@75 924 qint64 srcSize = srcFile.size();
joachim99@66 925 while ( srcSize > 0 && !pp.wasCancelled() )
joachim99@8 926 {
joachim99@75 927 qint64 readSize = srcFile.read( &buffer[0], min2( srcSize, bufSize ) );
joachim99@53 928 if ( readSize==-1 || readSize==0 )
joachim99@8 929 {
joachim99@51 930 m_pFileAccess->m_statusText = i18n("Error during file copy operation: Reading failed. Filename: %1").arg(srcName);
joachim99@8 931 return false;
joachim99@8 932 }
joachim99@8 933 srcSize -= readSize;
joachim99@8 934 while ( readSize > 0 )
joachim99@8 935 {
joachim99@75 936 qint64 writeSize = destFile.write( &buffer[0], readSize );
joachim99@53 937 if ( writeSize==-1 || writeSize==0 )
joachim99@8 938 {
joachim99@51 939 m_pFileAccess->m_statusText = i18n("Error during file copy operation: Writing failed. Filename: %1").arg(destName);
joachim99@8 940 return false;
joachim99@8 941 }
joachim99@8 942 readSize -= writeSize;
joachim99@8 943 }
joachim99@8 944 destFile.flush();
joachim99@66 945 pp.setCurrent( (double)(srcFile.size()-srcSize)/srcFile.size(), false );
joachim99@8 946 }
joachim99@8 947 srcFile.close();
joachim99@8 948 destFile.close();
joachim99@8 949
joachim99@8 950 // Update the times of the destFile
joachim99@8 951 #ifdef _WIN32
joachim99@8 952 struct _stat srcFileStatus;
joachim99@75 953 int statResult = ::_stat( srcName.toLocal8Bit().constData(), &srcFileStatus );
joachim99@8 954 if (statResult==0)
joachim99@8 955 {
joachim99@8 956 _utimbuf destTimes;
joachim99@8 957 destTimes.actime = srcFileStatus.st_atime;/* time of last access */
joachim99@8 958 destTimes.modtime = srcFileStatus.st_mtime;/* time of last modification */
joachim99@8 959
joachim99@75 960 _utime ( destName.toLocal8Bit().constData(), &destTimes );
joachim99@75 961 _chmod ( destName.toLocal8Bit().constData(), srcFileStatus.st_mode );
joachim99@8 962 }
joachim99@8 963 #else
joachim99@8 964 struct stat srcFileStatus;
joachim99@75 965 int statResult = ::stat( srcName.toLocal8Bit().constData(), &srcFileStatus );
joachim99@8 966 if (statResult==0)
joachim99@8 967 {
joachim99@8 968 utimbuf destTimes;
joachim99@8 969 destTimes.actime = srcFileStatus.st_atime;/* time of last access */
joachim99@8 970 destTimes.modtime = srcFileStatus.st_mtime;/* time of last modification */
joachim99@8 971
joachim99@75 972 utime ( destName.toLocal8Bit().constData(), &destTimes );
joachim99@75 973 chmod ( destName.toLocal8Bit().constData(), srcFileStatus.st_mode );
joachim99@8 974 }
joachim99@8 975 #endif
joachim99@8 976 return true;
joachim99@8 977 }
joachim99@8 978
joachim99@69 979 bool wildcardMultiMatch( const QString& wildcard, const QString& testString, bool bCaseSensitive )
joachim99@8 980 {
joachim99@75 981 QStringList sl = wildcard.split( ";" );
joachim99@8 982
joachim99@8 983 for ( QStringList::Iterator it = sl.begin(); it != sl.end(); ++it )
joachim99@8 984 {
joachim99@75 985 QRegExp pattern( *it, bCaseSensitive ? Qt::CaseSensitive : Qt::CaseInsensitive, QRegExp::Wildcard );
joachim99@8 986 if ( pattern.exactMatch( testString ) )
joachim99@8 987 return true;
joachim99@8 988 }
joachim99@8 989
joachim99@8 990 return false;
joachim99@8 991 }
joachim99@8 992
joachim99@8 993
joachim99@8 994 // class CvsIgnoreList from Cervisia cvsdir.cpp
joachim99@69 995 // Copyright (C) 1999-2002 Bernd Gehrmann <bernd at mail.berlios.de>
joachim99@8 996 // with elements from class StringMatcher
joachim99@69 997 // Copyright (c) 2003 Andr�Woebeking <Woebbeking at web.de>
joachim99@8 998 // Modifications for KDiff3 by Joachim Eibl
joachim99@8 999 class CvsIgnoreList
joachim99@8 1000 {
joachim99@8 1001 public:
joachim99@8 1002 CvsIgnoreList(){}
joachim99@8 1003 void init(FileAccess& dir, bool bUseLocalCvsIgnore );
joachim99@66 1004 bool matches(const QString& fileName, bool bCaseSensitive ) const;
joachim99@8 1005
joachim99@8 1006 private:
joachim99@8 1007 void addEntriesFromString(const QString& str);
joachim99@8 1008 void addEntriesFromFile(const QString& name);
joachim99@8 1009 void addEntry(const QString& entry);
joachim99@8 1010
joachim99@8 1011 QStringList m_exactPatterns;
joachim99@8 1012 QStringList m_startPatterns;
joachim99@8 1013 QStringList m_endPatterns;
joachim99@8 1014 QStringList m_generalPatterns;
joachim99@8 1015 };
joachim99@8 1016
joachim99@8 1017
joachim99@8 1018 void CvsIgnoreList::init( FileAccess& dir, bool bUseLocalCvsIgnore )
joachim99@8 1019 {
joachim99@8 1020 static const char *ignorestr = ". .. core RCSLOG tags TAGS RCS SCCS .make.state "
joachim99@8 1021 ".nse_depinfo #* .#* cvslog.* ,* CVS CVS.adm .del-* *.a *.olb *.o *.obj "
joachim99@8 1022 "*.so *.Z *~ *.old *.elc *.ln *.bak *.BAK *.orig *.rej *.exe _$* *$";
joachim99@8 1023
joachim99@8 1024 addEntriesFromString(QString::fromLatin1(ignorestr));
joachim99@75 1025 addEntriesFromFile(QDir::homePath() + "/.cvsignore");
joachim99@8 1026 addEntriesFromString(QString::fromLocal8Bit(::getenv("CVSIGNORE")));
joachim99@8 1027
joachim99@8 1028 if (bUseLocalCvsIgnore)
joachim99@8 1029 {
joachim99@8 1030 FileAccess file(dir);
joachim99@8 1031 file.addPath( ".cvsignore" );
joachim99@8 1032 int size = file.exists() ? file.sizeForReading() : 0;
joachim99@8 1033 if ( size>0 )
joachim99@8 1034 {
joachim99@8 1035 char* buf=new char[size];
joachim99@8 1036 if (buf!=0)
joachim99@8 1037 {
joachim99@8 1038 file.readFile( buf, size );
joachim99@8 1039 int pos1 = 0;
joachim99@8 1040 for ( int pos = 0; pos<=size; ++pos )
joachim99@8 1041 {
joachim99@8 1042 if( pos==size || buf[pos]==' ' || buf[pos]=='\t' || buf[pos]=='\n' || buf[pos]=='\r' )
joachim99@8 1043 {
joachim99@8 1044 if (pos>pos1)
joachim99@8 1045 {
joachim99@69 1046 addEntry( QString::fromLatin1( &buf[pos1], pos-pos1 ) );
joachim99@8 1047 }
joachim99@69 1048 ++pos1;
joachim99@8 1049 }
joachim99@8 1050 }
joachim99@8 1051 delete buf;
joachim99@8 1052 }
joachim99@8 1053 }
joachim99@8 1054 }
joachim99@8 1055 }
joachim99@8 1056
joachim99@8 1057
joachim99@8 1058 void CvsIgnoreList::addEntriesFromString(const QString& str)
joachim99@8 1059 {
joachim99@8 1060 int posLast(0);
joachim99@8 1061 int pos;
joachim99@75 1062 while ((pos = str.indexOf(' ', posLast)) >= 0)
joachim99@8 1063 {
joachim99@8 1064 if (pos > posLast)
joachim99@8 1065 addEntry(str.mid(posLast, pos - posLast));
joachim99@8 1066 posLast = pos + 1;
joachim99@8 1067 }
joachim99@8 1068
joachim99@8 1069 if (posLast < static_cast<int>(str.length()))
joachim99@8 1070 addEntry(str.mid(posLast));
joachim99@8 1071 }
joachim99@8 1072
joachim99@8 1073
joachim99@8 1074 void CvsIgnoreList::addEntriesFromFile(const QString &name)
joachim99@8 1075 {
joachim99@8 1076 QFile file(name);
joachim99@8 1077
joachim99@70 1078 if( file.open(QIODevice::ReadOnly) )
joachim99@8 1079 {
joachim99@8 1080 QTextStream stream(&file);
joachim99@70 1081 while( !stream.atEnd() )
joachim99@8 1082 {
joachim99@8 1083 addEntriesFromString(stream.readLine());
joachim99@8 1084 }
joachim99@8 1085 }
joachim99@8 1086 }
joachim99@8 1087
joachim99@8 1088 void CvsIgnoreList::addEntry(const QString& pattern)
joachim99@8 1089 {
joachim99@69 1090 if (pattern != QString("!"))
joachim99@8 1091 {
joachim99@8 1092 if (pattern.isEmpty()) return;
joachim99@8 1093
joachim99@8 1094 // The general match is general but slow.
joachim99@8 1095 // Special tests for '*' and '?' at the beginning or end of a pattern
joachim99@8 1096 // allow fast checks.
joachim99@8 1097
joachim99@8 1098 // Count number of '*' and '?'
joachim99@8 1099 unsigned int nofMetaCharacters = 0;
joachim99@8 1100
joachim99@8 1101 const QChar* pos;
joachim99@8 1102 pos = pattern.unicode();
joachim99@8 1103 const QChar* posEnd;
joachim99@8 1104 posEnd=pos + pattern.length();
joachim99@8 1105 while (pos < posEnd)
joachim99@8 1106 {
joachim99@8 1107 if( *pos==QChar('*') || *pos==QChar('?') ) ++nofMetaCharacters;
joachim99@8 1108 ++pos;
joachim99@8 1109 }
joachim99@8 1110
joachim99@8 1111 if ( nofMetaCharacters==0 )
joachim99@8 1112 {
joachim99@8 1113 m_exactPatterns.append(pattern);
joachim99@8 1114 }
joachim99@8 1115 else if ( nofMetaCharacters==1 )
joachim99@8 1116 {
joachim99@75 1117 if ( pattern.at(0) == QChar('*') )
joachim99@8 1118 {
joachim99@8 1119 m_endPatterns.append( pattern.right( pattern.length() - 1) );
joachim99@8 1120 }
joachim99@75 1121 else if (pattern.at(pattern.length() - 1) == QChar('*'))
joachim99@8 1122 {
joachim99@8 1123 m_startPatterns.append( pattern.left( pattern.length() - 1) );
joachim99@8 1124 }
joachim99@8 1125 else
joachim99@8 1126 {
joachim99@75 1127 m_generalPatterns.append(pattern.toLocal8Bit());
joachim99@8 1128 }
joachim99@8 1129 }
joachim99@8 1130 else
joachim99@8 1131 {
joachim99@75 1132 m_generalPatterns.append(pattern.toLocal8Bit());
joachim99@8 1133 }
joachim99@8 1134 }
joachim99@8 1135 else
joachim99@8 1136 {
joachim99@8 1137 m_exactPatterns.clear();
joachim99@8 1138 m_startPatterns.clear();
joachim99@8 1139 m_endPatterns.clear();
joachim99@8 1140 m_generalPatterns.clear();
joachim99@8 1141 }
joachim99@8 1142 }
joachim99@8 1143
joachim99@66 1144 bool CvsIgnoreList::matches(const QString& text, bool bCaseSensitive ) const
joachim99@8 1145 {
joachim99@75 1146 if ( m_exactPatterns.indexOf(text) >=0 )
joachim99@8 1147 {
joachim99@8 1148 return true;
joachim99@8 1149 }
joachim99@8 1150
joachim99@8 1151 QStringList::ConstIterator it;
joachim99@8 1152 QStringList::ConstIterator itEnd;
joachim99@8 1153 for ( it=m_startPatterns.begin(), itEnd=m_startPatterns.end(); it != itEnd; ++it)
joachim99@8 1154 {
joachim99@8 1155 if (text.startsWith(*it))
joachim99@8 1156 {
joachim99@8 1157 return true;
joachim99@8 1158 }
joachim99@8 1159 }
joachim99@8 1160
joachim99@8 1161 for ( it = m_endPatterns.begin(), itEnd=m_endPatterns.end(); it != itEnd; ++it)
joachim99@8 1162 {
joachim99@8 1163 if (text.mid( text.length() - (*it).length() )==*it) //(text.endsWith(*it))
joachim99@8 1164 {
joachim99@8 1165 return true;
joachim99@8 1166 }
joachim99@8 1167 }
joachim99@8 1168
joachim99@8 1169 /*
joachim99@8 1170 for (QValueList<QCString>::const_iterator it(m_generalPatterns.begin()),
joachim99@8 1171 itEnd(m_generalPatterns.end());
joachim99@8 1172 it != itEnd; ++it)
joachim99@8 1173 {
joachim99@8 1174 if (::fnmatch(*it, text.local8Bit(), FNM_PATHNAME) == 0)
joachim99@8 1175 {
joachim99@8 1176 return true;
joachim99@8 1177 }
joachim99@8 1178 }
joachim99@8 1179 */
joachim99@8 1180
joachim99@8 1181
joachim99@8 1182 for ( it = m_generalPatterns.begin(); it != m_generalPatterns.end(); ++it )
joachim99@8 1183 {
joachim99@75 1184 QRegExp pattern( *it, bCaseSensitive ? Qt::CaseSensitive : Qt::CaseInsensitive, QRegExp::Wildcard );
joachim99@8 1185 #if QT_VERSION==230
joachim99@8 1186 int len=0;
joachim99@8 1187 if ( pattern.match( text, 0, &len )!=-1 && len==text.length())
joachim99@8 1188 return true;
joachim99@8 1189 #else
joachim99@8 1190 if ( pattern.exactMatch( text ) )
joachim99@8 1191 return true;
joachim99@8 1192 #endif
joachim99@8 1193 }
joachim99@8 1194
joachim99@8 1195 return false;
joachim99@8 1196 }
joachim99@8 1197
joachim99@8 1198 static QString nicePath( const QFileInfo& fi )
joachim99@8 1199 {
joachim99@8 1200 QString fp = fi.filePath();
joachim99@8 1201 if ( fp.length()>2 && fp[0] == '.' && fp[1] == '/' )
joachim99@8 1202 {
joachim99@8 1203 return fp.mid(2);
joachim99@8 1204 }
joachim99@8 1205 return fp;
joachim99@8 1206 }
joachim99@8 1207
joachim99@8 1208 static bool cvsIgnoreExists( t_DirectoryList* pDirList )
joachim99@8 1209 {
joachim99@8 1210 t_DirectoryList::iterator i;
joachim99@8 1211 for( i = pDirList->begin(); i!=pDirList->end(); ++i )
joachim99@8 1212 {
joachim99@8 1213 if ( i->fileName()==".cvsignore" )
joachim99@8 1214 return true;
joachim99@8 1215 }
joachim99@8 1216 return false;
joachim99@8 1217 }
joachim99@8 1218
joachim99@8 1219 bool FileAccessJobHandler::listDir( t_DirectoryList* pDirList, bool bRecursive, bool bFindHidden, const QString& filePattern,
joachim99@8 1220 const QString& fileAntiPattern, const QString& dirAntiPattern, bool bFollowDirLinks, bool bUseCvsIgnore )
joachim99@8 1221 {
joachim99@66 1222 ProgressProxy pp;
joachim99@8 1223 m_pDirList = pDirList;
joachim99@8 1224 m_pDirList->clear();
joachim99@8 1225 m_bFindHidden = bFindHidden;
joachim99@8 1226 m_bRecursive = bRecursive;
joachim99@8 1227 m_bFollowDirLinks = bFollowDirLinks; // Only relevant if bRecursive==true.
joachim99@8 1228 m_fileAntiPattern = fileAntiPattern;
joachim99@8 1229 m_filePattern = filePattern;
joachim99@8 1230 m_dirAntiPattern = dirAntiPattern;
joachim99@8 1231
joachim99@66 1232 if ( pp.wasCancelled() )
joachim99@8 1233 return true; // Cancelled is not an error.
joachim99@8 1234
joachim99@66 1235 pp.setInformation( i18n("Reading directory: ") + m_pFileAccess->absFilePath(), 0, false );
joachim99@8 1236
joachim99@8 1237 if( m_pFileAccess->isLocal() )
joachim99@8 1238 {
joachim99@75 1239 QString currentPath = QDir::currentPath();
joachim99@8 1240 m_bSuccess = QDir::setCurrent( m_pFileAccess->absFilePath() );
joachim99@8 1241 if ( m_bSuccess )
joachim99@8 1242 {
joachim99@58 1243 #ifndef _WIN32
joachim99@8 1244 m_bSuccess = true;
joachim99@8 1245 QDir dir( "." );
joachim99@8 1246
joachim99@8 1247 dir.setSorting( QDir::Name | QDir::DirsFirst );
joachim99@75 1248 dir.setFilter( QDir::Files | QDir::Dirs | QDir::AllDirs | QDir::Hidden );
joachim99@8 1249
joachim99@70 1250 QFileInfoList fiList = dir.entryInfoList();
joachim99@70 1251 if ( fiList.isEmpty() )
joachim99@8 1252 {
joachim99@8 1253 // No Permission to read directory or other error.
joachim99@8 1254 m_bSuccess = false;
joachim99@8 1255 }
joachim99@8 1256 else
joachim99@8 1257 {
joachim99@70 1258 foreach ( QFileInfo fi, fiList ) // for each file...
joachim99@8 1259 {
joachim99@70 1260 if ( fi.fileName() == "." || fi.fileName()==".." )
joachim99@8 1261 continue;
joachim99@8 1262
joachim99@70 1263 pDirList->push_back( FileAccess( nicePath(fi) ) );
joachim99@8 1264 }
joachim99@8 1265 }
joachim99@58 1266 #else
joachim99@58 1267 QString pattern ="*.*";
joachim99@58 1268 WIN32_FIND_DATA findData;
joachim99@58 1269 WIN32_FIND_DATAA& findDataA=*(WIN32_FIND_DATAA*)&findData; // Needed for Win95
joachim99@58 1270
joachim99@70 1271 Qt::HANDLE searchHandle = QT_WA_INLINE(
joachim99@75 1272 FindFirstFile( (TCHAR*)pattern.utf16(), &findData ),
joachim99@75 1273 FindFirstFileA( pattern.toLocal8Bit(), &findDataA )
joachim99@58 1274 );
joachim99@58 1275
joachim99@58 1276 if ( searchHandle != INVALID_HANDLE_VALUE )
joachim99@58 1277 {
joachim99@58 1278 QString absPath = m_pFileAccess->absFilePath();
joachim99@58 1279 QString relPath = m_pFileAccess->filePath();
joachim99@58 1280 bool bFirst=true;
joachim99@66 1281 while( ! pp.wasCancelled() )
joachim99@58 1282 {
joachim99@58 1283 if (!bFirst)
joachim99@58 1284 {
joachim99@58 1285 if ( ! QT_WA_INLINE(
joachim99@58 1286 FindNextFile(searchHandle,&findData),
joachim99@58 1287 FindNextFileA(searchHandle,&findDataA)) )
joachim99@58 1288 break;
joachim99@58 1289 }
joachim99@58 1290 bFirst = false;
joachim99@58 1291 FileAccess fa;
joachim99@58 1292 fa.m_size = findData.nFileSizeLow ;//+ findData.nFileSizeHigh;
joachim99@58 1293
joachim99@58 1294 FILETIME ft;
joachim99@58 1295 SYSTEMTIME t;
joachim99@58 1296 FileTimeToLocalFileTime( &findData.ftLastWriteTime, &ft ); FileTimeToSystemTime(&ft,&t);
joachim99@58 1297 fa.m_modificationTime = QDateTime( QDate(t.wYear, t.wMonth, t.wDay), QTime(t.wHour, t.wMinute, t.wSecond) );
joachim99@58 1298 FileTimeToLocalFileTime( &findData.ftLastAccessTime, &ft ); FileTimeToSystemTime(&ft,&t);
joachim99@58 1299 fa.m_accessTime = QDateTime( QDate(t.wYear, t.wMonth, t.wDay), QTime(t.wHour, t.wMinute, t.wSecond) );
joachim99@58 1300 FileTimeToLocalFileTime( &findData.ftCreationTime, &ft ); FileTimeToSystemTime(&ft,&t);
joachim99@58 1301 fa.m_creationTime = QDateTime( QDate(t.wYear, t.wMonth, t.wDay), QTime(t.wHour, t.wMinute, t.wSecond) );
joachim99@58 1302
joachim99@58 1303 int a = findData.dwFileAttributes;
joachim99@68 1304 fa.m_bWritable = ( a & FILE_ATTRIBUTE_READONLY) == 0;
joachim99@58 1305 fa.m_bDir = ( a & FILE_ATTRIBUTE_DIRECTORY ) != 0;
joachim99@58 1306 fa.m_bFile = !fa.m_bDir;
joachim99@58 1307 fa.m_bHidden = ( a & FILE_ATTRIBUTE_HIDDEN) != 0;
joachim99@58 1308
joachim99@58 1309 fa.m_bExecutable = false; // Useless on windows
joachim99@58 1310 fa.m_bExists = true;
joachim99@58 1311 fa.m_bReadable = true;
joachim99@58 1312 fa.m_bLocal = true;
joachim99@58 1313 fa.m_bValidData = true;
joachim99@58 1314 fa.m_bSymLink = false;
joachim99@58 1315 fa.m_fileType = 0;
joachim99@58 1316
joachim99@68 1317 fa.m_name = QT_WA_INLINE(
joachim99@75 1318 QString::fromUtf16((const ushort*)findData.cFileName),
joachim99@68 1319 QString::fromLocal8Bit(findDataA.cFileName)
joachim99@58 1320 );
joachim99@58 1321
joachim99@58 1322 fa.m_path = fa.m_name;
joachim99@58 1323 fa.m_absFilePath = absPath + "/" + fa.m_name;
joachim99@58 1324 fa.m_url.setPath( fa.m_absFilePath );
joachim99@58 1325 if ( fa.m_name!="." && fa.m_name!=".." )
joachim99@58 1326 pDirList->push_back( fa );
joachim99@58 1327 }
joachim99@58 1328 FindClose( searchHandle );
joachim99@58 1329 }
joachim99@58 1330 else
joachim99@58 1331 {
joachim99@66 1332 QDir::setCurrent( currentPath ); // restore current path
joachim99@58 1333 return false;
joachim99@58 1334 }
joachim99@58 1335 #endif
joachim99@8 1336 }
joachim99@66 1337 QDir::setCurrent( currentPath ); // restore current path
joachim99@8 1338 }
joachim99@8 1339 else
joachim99@8 1340 {
joachim99@8 1341 bool bShowProgress = false;
joachim99@8 1342
joachim99@8 1343 KIO::ListJob* pListJob=0;
joachim99@8 1344 pListJob = KIO::listDir( m_pFileAccess->m_url, bShowProgress, true /*bFindHidden*/ );
joachim99@8 1345
joachim99@8 1346 m_bSuccess = false;
joachim99@8 1347 if ( pListJob!=0 )
joachim99@8 1348 {
joachim99@8 1349 connect( pListJob, SIGNAL( entries( KIO::Job *, const KIO::UDSEntryList& ) ),
joachim99@8 1350 this, SLOT( slotListDirProcessNewEntries( KIO::Job *, const KIO::UDSEntryList& )) );
joachim99@8 1351 connect( pListJob, SIGNAL( result( KIO::Job* )),
joachim99@8 1352 this, SLOT( slotSimpleJobResult(KIO::Job*) ) );
joachim99@8 1353
joachim99@8 1354 connect( pListJob, SIGNAL( infoMessage(KIO::Job*, const QString&)),
joachim99@8 1355 this, SLOT( slotListDirInfoMessage(KIO::Job*, const QString&) ));
joachim99@8 1356
joachim99@8 1357 // This line makes the transfer via fish unreliable.:-(
joachim99@8 1358 //connect( pListJob, SIGNAL(percent(KIO::Job*,unsigned long)), this, SLOT(slotPercent(KIO::Job*, unsigned long)));
joachim99@8 1359
joachim99@51 1360 g_pProgressDialog->enterEventLoop( pListJob,
joachim99@51 1361 i18n("Listing directory: %1").arg(m_pFileAccess->prettyAbsPath()) );
joachim99@8 1362 }
joachim99@8 1363 }
joachim99@8 1364
joachim99@8 1365 CvsIgnoreList cvsIgnoreList;
joachim99@8 1366 if ( bUseCvsIgnore )
joachim99@8 1367 {
joachim99@8 1368 cvsIgnoreList.init( *m_pFileAccess, cvsIgnoreExists(pDirList) );
joachim99@8 1369 }
joachim99@66 1370 #ifdef _WIN32
joachim99@66 1371 bool bCaseSensitive = false;
joachim99@66 1372 #else
joachim99@66 1373 bool bCaseSensitive = true;
joachim99@66 1374 #endif
joachim99@8 1375
joachim99@8 1376 // Now remove all entries that don't match:
joachim99@8 1377 t_DirectoryList::iterator i;
joachim99@8 1378 for( i = pDirList->begin(); i!=pDirList->end(); )
joachim99@8 1379 {
joachim99@8 1380 t_DirectoryList::iterator i2=i;
joachim99@8 1381 ++i2;
joachim99@8 1382 QString fn = i->fileName();
joachim99@8 1383 if ( (!bFindHidden && i->isHidden() )
joachim99@8 1384 ||
joachim99@8 1385 (i->isFile() &&
joachim99@66 1386 ( !wildcardMultiMatch( filePattern, i->fileName(), bCaseSensitive ) ||
joachim99@66 1387 wildcardMultiMatch( fileAntiPattern, i->fileName(), bCaseSensitive ) ) )
joachim99@8 1388 ||
joachim99@66 1389 (i->isDir() && wildcardMultiMatch( dirAntiPattern, i->fileName(), bCaseSensitive ) )
joachim99@8 1390 ||
joachim99@66 1391 cvsIgnoreList.matches( i->fileName(), bCaseSensitive )
joachim99@8 1392 )
joachim99@8 1393 {
joachim99@8 1394 // Remove it
joachim99@8 1395 pDirList->erase( i );
joachim99@8 1396 i = i2;
joachim99@8 1397 }
joachim99@8 1398 else
joachim99@8 1399 {
joachim99@8 1400 ++i;
joachim99@8 1401 }
joachim99@8 1402 }
joachim99@8 1403
joachim99@8 1404 if ( bRecursive )
joachim99@8 1405 {
joachim99@8 1406 t_DirectoryList subDirsList;
joachim99@8 1407
joachim99@8 1408 t_DirectoryList::iterator i;
joachim99@8 1409 for( i = m_pDirList->begin(); i!=m_pDirList->end(); ++i )
joachim99@8 1410 {
joachim99@8 1411 if ( i->isDir() && (!i->isSymLink() || m_bFollowDirLinks))
joachim99@8 1412 {
joachim99@8 1413 t_DirectoryList dirList;
joachim99@8 1414 i->listDir( &dirList, bRecursive, bFindHidden,
joachim99@8 1415 filePattern, fileAntiPattern, dirAntiPattern, bFollowDirLinks, bUseCvsIgnore );
joachim99@8 1416
joachim99@8 1417 t_DirectoryList::iterator j;
joachim99@8 1418 for( j = dirList.begin(); j!=dirList.end(); ++j )
joachim99@8 1419 {
joachim99@8 1420 j->m_path = i->fileName() + "/" + j->m_path;
joachim99@8 1421 }
joachim99@8 1422
joachim99@8 1423 // append data onto the main list
joachim99@8 1424 subDirsList.splice( subDirsList.end(), dirList );
joachim99@8 1425 }
joachim99@8 1426 }
joachim99@8 1427
joachim99@8 1428 m_pDirList->splice( m_pDirList->end(), subDirsList );
joachim99@8 1429 }
joachim99@8 1430
joachim99@8 1431 return m_bSuccess;
joachim99@8 1432 }
joachim99@8 1433
joachim99@8 1434
joachim99@8 1435 void FileAccessJobHandler::slotListDirProcessNewEntries( KIO::Job *, const KIO::UDSEntryList& l )
joachim99@8 1436 {
joachim99@8 1437 KURL parentUrl( m_pFileAccess->m_absFilePath );
joachim99@8 1438
joachim99@8 1439 KIO::UDSEntryList::ConstIterator i;
joachim99@8 1440 for ( i=l.begin(); i!=l.end(); ++i )
joachim99@8 1441 {
joachim99@8 1442 const KIO::UDSEntry& e = *i;
joachim99@8 1443 FileAccess fa;
joachim99@8 1444 fa.setUdsEntry( e );
joachim99@8 1445
joachim99@8 1446 if ( fa.filePath() != "." && fa.filePath() != ".." )
joachim99@8 1447 {
joachim99@8 1448 fa.m_url = parentUrl;
joachim99@8 1449 fa.m_url.addPath( fa.filePath() );
joachim99@8 1450 fa.m_absFilePath = fa.m_url.url();
joachim99@8 1451 m_pDirList->push_back( fa );
joachim99@8 1452 }
joachim99@8 1453 }
joachim99@8 1454 }
joachim99@8 1455
joachim99@8 1456 void FileAccessJobHandler::slotListDirInfoMessage( KIO::Job*, const QString& msg )
joachim99@8 1457 {
joachim99@66 1458 g_pProgressDialog->setInformation( msg, 0.0 );
joachim99@8 1459 }
joachim99@8 1460
joachim99@8 1461 void FileAccessJobHandler::slotPercent( KIO::Job*, unsigned long percent )
joachim99@8 1462 {
joachim99@66 1463 g_pProgressDialog->setCurrent( percent/100.0 );
joachim99@8 1464 }
joachim99@8 1465
joachim99@8 1466
joachim99@8 1467 ProgressDialog::ProgressDialog( QWidget* pParent )
joachim99@75 1468 : QDialog( pParent )
joachim99@8 1469 {
joachim99@75 1470 setObjectName("ProgressDialog");
joachim99@77 1471 m_bStayHidden = false;
joachim99@75 1472 setModal(true);
joachim99@75 1473 QVBoxLayout* layout = new QVBoxLayout(this);
joachim99@8 1474
joachim99@8 1475 m_pInformation = new QLabel( " ", this );
joachim99@8 1476 layout->addWidget( m_pInformation );
joachim99@8 1477
joachim99@75 1478 m_pProgressBar = new KProgress();
joachim99@75 1479 m_pProgressBar->setRange(0,1000);
joachim99@8 1480 layout->addWidget( m_pProgressBar );
joachim99@8 1481
joachim99@8 1482 m_pSubInformation = new QLabel( " ", this);
joachim99@8 1483 layout->addWidget( m_pSubInformation );
joachim99@8 1484
joachim99@75 1485 m_pSubProgressBar = new KProgress();
joachim99@75 1486 m_pSubProgressBar->setRange(0,1000);
joachim99@8 1487 layout->addWidget( m_pSubProgressBar );
joachim99@8 1488
joachim99@69 1489 m_pSlowJobInfo = new QLabel( " ", this);
joachim99@69 1490 layout->addWidget( m_pSlowJobInfo );
joachim99@69 1491
joachim99@75 1492 QHBoxLayout* hlayout = new QHBoxLayout();
joachim99@75 1493 layout->addLayout(hlayout);
joachim99@69 1494 hlayout->addStretch(1);
joachim99@68 1495 m_pAbortButton = new QPushButton( i18n("&Cancel"), this);
joachim99@51 1496 hlayout->addWidget( m_pAbortButton );
joachim99@51 1497 connect( m_pAbortButton, SIGNAL(clicked()), this, SLOT(slotAbort()) );
joachim99@51 1498
joachim99@69 1499 m_progressDelayTimer = 0;
joachim99@8 1500 resize( 400, 100 );
joachim99@8 1501 m_t1.start();
joachim99@8 1502 m_t2.start();
joachim99@8 1503 m_bWasCancelled = false;
joachim99@51 1504 m_pJob = 0;
joachim99@8 1505 }
joachim99@8 1506
joachim99@77 1507 void ProgressDialog::setStayHidden( bool bStayHidden )
joachim99@77 1508 {
joachim99@77 1509 m_bStayHidden = bStayHidden;
joachim99@77 1510 }
joachim99@77 1511
joachim99@66 1512 void ProgressDialog::push()
joachim99@66 1513 {
joachim99@66 1514 ProgressLevelData pld;
joachim99@66 1515 if ( !m_progressStack.empty() )
joachim99@66 1516 {
joachim99@66 1517 pld.m_dRangeMax = m_progressStack.back().m_dSubRangeMax;
joachim99@66 1518 pld.m_dRangeMin = m_progressStack.back().m_dSubRangeMin;
joachim99@66 1519 }
joachim99@66 1520 else
joachim99@66 1521 {
joachim99@66 1522 m_bWasCancelled = false;
joachim99@66 1523 m_t1.restart();
joachim99@66 1524 m_t2.restart();
joachim99@77 1525 if ( !m_bStayHidden )
joachim99@77 1526 show();
joachim99@66 1527 }
joachim99@66 1528
joachim99@66 1529 m_progressStack.push_back( pld );
joachim99@66 1530 }
joachim99@66 1531
joachim99@66 1532 void ProgressDialog::pop( bool bRedrawUpdate )
joachim99@66 1533 {
joachim99@66 1534 if ( !m_progressStack.empty() )
joachim99@68 1535 {
joachim99@66 1536 m_progressStack.pop_back();
joachim99@66 1537 if ( m_progressStack.empty() )
joachim99@66 1538 hide();
joachim99@68 1539 else
joachim99@66 1540 recalc(bRedrawUpdate);
joachim99@66 1541 }
joachim99@66 1542 }
joachim99@8 1543
joachim99@8 1544 void ProgressDialog::setInformation(const QString& info, double dCurrent, bool bRedrawUpdate )
joachim99@8 1545 {
joachim99@68 1546 if ( m_progressStack.empty() )
joachim99@66 1547 return;
joachim99@66 1548 ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1549 pld.m_dCurrent = dCurrent;
joachim99@66 1550 int level = m_progressStack.size();
joachim99@66 1551 if ( level==1 )
joachim99@66 1552 {
joachim99@66 1553 m_pInformation->setText( info );
joachim99@66 1554 m_pSubInformation->setText("");
joachim99@66 1555 }
joachim99@66 1556 else if ( level==2 )
joachim99@66 1557 {
joachim99@66 1558 m_pSubInformation->setText( info );
joachim99@66 1559 }
joachim99@8 1560 recalc(bRedrawUpdate);
joachim99@8 1561 }
joachim99@8 1562
joachim99@8 1563 void ProgressDialog::setInformation(const QString& info, bool bRedrawUpdate )
joachim99@8 1564 {
joachim99@68 1565 if ( m_progressStack.empty() )
joachim99@66 1566 return;
joachim99@66 1567 //ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1568 int level = m_progressStack.size();
joachim99@66 1569 if ( level==1 )
joachim99@66 1570 {
joachim99@66 1571 m_pInformation->setText( info );
joachim99@66 1572 m_pSubInformation->setText( "" );
joachim99@66 1573 }
joachim99@66 1574 else if ( level==2 )
joachim99@66 1575 {
joachim99@66 1576 m_pSubInformation->setText( info );
joachim99@66 1577 }
joachim99@8 1578 recalc(bRedrawUpdate);
joachim99@8 1579 }
joachim99@8 1580
joachim99@66 1581 void ProgressDialog::setMaxNofSteps( int maxNofSteps )
joachim99@8 1582 {
joachim99@68 1583 if ( m_progressStack.empty() )
joachim99@66 1584 return;
joachim99@66 1585 ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1586 pld.m_maxNofSteps = maxNofSteps;
joachim99@66 1587 pld.m_dCurrent = 0;
joachim99@8 1588 }
joachim99@8 1589
joachim99@8 1590 void ProgressDialog::step( bool bRedrawUpdate )
joachim99@8 1591 {
joachim99@68 1592 if ( m_progressStack.empty() )
joachim99@66 1593 return;
joachim99@66 1594 ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1595 pld.m_dCurrent += 1.0/pld.m_maxNofSteps;
joachim99@8 1596 recalc(bRedrawUpdate);
joachim99@8 1597 }
joachim99@8 1598
joachim99@66 1599 void ProgressDialog::setCurrent( double dSubCurrent, bool bRedrawUpdate )
joachim99@8 1600 {
joachim99@68 1601 if ( m_progressStack.empty() )
joachim99@66 1602 return;
joachim99@66 1603 ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1604 pld.m_dCurrent = dSubCurrent;
joachim99@8 1605 recalc( bRedrawUpdate );
joachim99@8 1606 }
joachim99@8 1607
joachim99@66 1608 // The progressbar goes from 0 to 1 usually.
joachim99@66 1609 // By supplying a subrange transformation the subCurrent-values
joachim99@66 1610 // 0 to 1 will be transformed to dMin to dMax instead.
joachim99@66 1611 // Requirement: 0 < dMin < dMax < 1
joachim99@66 1612 void ProgressDialog::setRangeTransformation( double dMin, double dMax )
joachim99@66 1613 {
joachim99@68 1614 if ( m_progressStack.empty() )
joachim99@66 1615 return;
joachim99@66 1616 ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1617 pld.m_dRangeMin = dMin;
joachim99@66 1618 pld.m_dRangeMax = dMax;
joachim99@66 1619 pld.m_dCurrent = 0;
joachim99@66 1620 }
joachim99@66 1621
joachim99@66 1622 void ProgressDialog::setSubRangeTransformation( double dMin, double dMax )
joachim99@66 1623 {
joachim99@68 1624 if ( m_progressStack.empty() )
joachim99@66 1625 return;
joachim99@66 1626 ProgressLevelData& pld = m_progressStack.back();
joachim99@66 1627 pld.m_dSubRangeMin = dMin;
joachim99@66 1628 pld.m_dSubRangeMax = dMax;
joachim99@66 1629 }
joachim99@8 1630
joachim99@8 1631 void qt_enter_modal(QWidget*);
joachim99@8 1632 void qt_leave_modal(QWidget*);
joachim99@8 1633
joachim99@51 1634 void ProgressDialog::enterEventLoop( KIO::Job* pJob, const QString& jobInfo )
joachim99@8 1635 {
joachim99@51 1636 m_pJob = pJob;
joachim99@69 1637 m_pSlowJobInfo->setText("");
joachim99@51 1638 m_currentJobInfo = jobInfo;
joachim99@75 1639 if ( m_progressDelayTimer )
joachim99@75 1640 killTimer( m_progressDelayTimer );
joachim99@69 1641 m_progressDelayTimer = startTimer( 3000 ); /* 3 s delay */
joachim99@51 1642
joachim99@8 1643 // instead of using exec() the eventloop is entered and exited often without hiding/showing the window.
joachim99@70 1644 //qt_enter_modal(this);
joachim99@70 1645 QEventLoop* pEventLoop = new QEventLoop(this);
joachim99@70 1646 m_eventLoopStack.push_back( pEventLoop );
joachim99@70 1647 pEventLoop->exec(); // this function only returns after ProgressDialog::exitEventLoop() is called.
joachim99@70 1648 delete pEventLoop;
joachim99@70 1649 m_eventLoopStack.pop_back();
joachim99@70 1650 //qt_leave_modal(this);
joachim99@8 1651 }
joachim99@8 1652
joachim99@8 1653 void ProgressDialog::exitEventLoop()
joachim99@8 1654 {
joachim99@75 1655 if ( m_progressDelayTimer )
joachim99@75 1656 killTimer( m_progressDelayTimer );
joachim99@69 1657 m_progressDelayTimer = 0;
joachim99@51 1658 m_pJob = 0;
joachim99@70 1659 if (!m_eventLoopStack.empty())
joachim99@70 1660 m_eventLoopStack.back()->exit();
joachim99@8 1661 }
joachim99@8 1662
joachim99@8 1663 void ProgressDialog::recalc( bool bUpdate )
joachim99@8 1664 {
joachim99@75 1665 if ( m_progressDelayTimer )
joachim99@75 1666 killTimer( m_progressDelayTimer );
joachim99@69 1667 m_progressDelayTimer = startTimer( 3000 ); /* 3 s delay */
joachim99@69 1668
joachim99@66 1669 int level = m_progressStack.size();
joachim99@66 1670 if( ( bUpdate && level==1) || m_t1.elapsed()>200 )
joachim99@8 1671 {
joachim99@66 1672 if (m_progressStack.empty() )
joachim99@66 1673 {
joachim99@75 1674 m_pProgressBar->setValue( 0 );
joachim99@75 1675 m_pSubProgressBar->setValue( 0 );
joachim99@66 1676 }
joachim99@66 1677 else
joachim99@66 1678 {
joachim99@66 1679 std::list<ProgressLevelData>::iterator i = m_progressStack.begin();
joachim99@75 1680 m_pProgressBar->setValue( int( 1000.0 * ( i->m_dCurrent * (i->m_dRangeMax - i->m_dRangeMin) + i->m_dRangeMin ) ) );
joachim99@66 1681 ++i;
joachim99@66 1682 if ( i!=m_progressStack.end() )
joachim99@75 1683 m_pSubProgressBar->setValue( int( 1000.0 * ( i->m_dCurrent * (i->m_dRangeMax - i->m_dRangeMin) + i->m_dRangeMin ) ) );
joachim99@66 1684 else
joachim99@75 1685 m_pSubProgressBar->setValue( int( 1000.0 * m_progressStack.front().m_dSubRangeMin ) );
joachim99@66 1686 }
joachim99@68 1687
joachim99@77 1688 if ( !m_bStayHidden && !isVisible() )
joachim99@77 1689 show();
joachim99@8 1690 qApp->processEvents();
joachim99@8 1691 m_t1.restart();
joachim99@8 1692 }
joachim99@8 1693 }
joachim99@8 1694
joachim99@8 1695
joachim99@8 1696 #include <qtimer.h>
joachim99@8 1697 void ProgressDialog::show()
joachim99@8 1698 {
joachim99@75 1699 if ( m_progressDelayTimer )
joachim99@75 1700 killTimer( m_progressDelayTimer );
joachim99@69 1701 m_progressDelayTimer = 0;
joachim99@69 1702 if ( !isVisible() && (parentWidget()==0 || parentWidget()->isVisible()) )
joachim99@24 1703 {
joachim99@24 1704 QDialog::show();
joachim99@24 1705 }
joachim99@8 1706 }
joachim99@8 1707
joachim99@8 1708 void ProgressDialog::hide()
joachim99@8 1709 {
joachim99@75 1710 if ( m_progressDelayTimer )
joachim99@75 1711 killTimer( m_progressDelayTimer );
joachim99@69 1712 m_progressDelayTimer = 0;
joachim99@8 1713 // Calling QDialog::hide() directly doesn't always work. (?)
joachim99@8 1714 QTimer::singleShot( 100, this, SLOT(delayedHide()) );
joachim99@8 1715 }
joachim99@8 1716
joachim99@8 1717 void ProgressDialog::delayedHide()
joachim99@8 1718 {
joachim99@51 1719 if (m_pJob!=0)
joachim99@51 1720 {
joachim99@51 1721 m_pJob->kill(false);
joachim99@51 1722 m_pJob = 0;
joachim99@51 1723 }
joachim99@8 1724 QDialog::hide();
joachim99@51 1725 m_pInformation->setText( "" );
joachim99@66 1726
joachim99@69 1727 //m_progressStack.clear();
joachim99@68 1728
joachim99@75 1729 m_pProgressBar->setValue( 0 );
joachim99@75 1730 m_pSubProgressBar->setValue( 0 );
joachim99@51 1731 m_pSubInformation->setText("");
joachim99@51 1732 m_pSlowJobInfo->setText("");
joachim99@8 1733 }
joachim99@8 1734
joachim99@8 1735 void ProgressDialog::reject()
joachim99@8 1736 {
joachim99@8 1737 m_bWasCancelled = true;
joachim99@8 1738 QDialog::reject();
joachim99@8 1739 }
joachim99@8 1740
joachim99@51 1741 void ProgressDialog::slotAbort()
joachim99@51 1742 {
joachim99@51 1743 reject();
joachim99@51 1744 }
joachim99@51 1745
joachim99@8 1746 bool ProgressDialog::wasCancelled()
joachim99@8 1747 {
joachim99@8 1748 if( m_t2.elapsed()>100 )
joachim99@8 1749 {
joachim99@8 1750 qApp->processEvents();
joachim99@8 1751 m_t2.restart();
joachim99@8 1752 }
joachim99@8 1753 return m_bWasCancelled;
joachim99@8 1754 }
joachim99@8 1755
joachim99@8 1756
joachim99@51 1757 void ProgressDialog::timerEvent(QTimerEvent*)
joachim99@51 1758 {
joachim99@51 1759 if( !isVisible() )
joachim99@51 1760 {
joachim99@51 1761 show();
joachim99@51 1762 }
joachim99@51 1763 m_pSlowJobInfo->setText( m_currentJobInfo );
joachim99@51 1764 }
joachim99@51 1765
joachim99@8 1766
joachim99@66 1767 ProgressProxy::ProgressProxy()
joachim99@66 1768 {
joachim99@66 1769 g_pProgressDialog->push();
joachim99@66 1770 }
joachim99@66 1771
joachim99@66 1772 ProgressProxy::~ProgressProxy()
joachim99@66 1773 {
joachim99@66 1774 g_pProgressDialog->pop(false);
joachim99@66 1775 }
joachim99@66 1776
joachim99@66 1777 void ProgressProxy::setInformation( const QString& info, bool bRedrawUpdate )
joachim99@66 1778 {
joachim99@66 1779 g_pProgressDialog->setInformation( info, bRedrawUpdate );
joachim99@66 1780 }
joachim99@66 1781
joachim99@66 1782 void ProgressProxy::setInformation( const QString& info, double dCurrent, bool bRedrawUpdate )
joachim99@66 1783 {
joachim99@66 1784 g_pProgressDialog->setInformation( info, dCurrent, bRedrawUpdate );
joachim99@66 1785 }
joachim99@66 1786
joachim99@66 1787 void ProgressProxy::setCurrent( double dCurrent, bool bRedrawUpdate )
joachim99@66 1788 {
joachim99@66 1789 g_pProgressDialog->setCurrent( dCurrent, bRedrawUpdate );
joachim99@66 1790 }
joachim99@66 1791
joachim99@66 1792 void ProgressProxy::step( bool bRedrawUpdate )
joachim99@66 1793 {
joachim99@66 1794 g_pProgressDialog->step( bRedrawUpdate );
joachim99@66 1795 }
joachim99@66 1796
joachim99@66 1797 void ProgressProxy::setMaxNofSteps( int maxNofSteps )
joachim99@66 1798 {
joachim99@66 1799 g_pProgressDialog->setMaxNofSteps( maxNofSteps );
joachim99@66 1800 }
joachim99@66 1801
joachim99@66 1802 bool ProgressProxy::wasCancelled()
joachim99@66 1803 {
joachim99@66 1804 return g_pProgressDialog->wasCancelled();
joachim99@66 1805 }
joachim99@66 1806
joachim99@66 1807 void ProgressProxy::setRangeTransformation( double dMin, double dMax )
joachim99@66 1808 {
joachim99@66 1809 g_pProgressDialog->setRangeTransformation( dMin, dMax );
joachim99@66 1810 }
joachim99@66 1811
joachim99@66 1812 void ProgressProxy::setSubRangeTransformation( double dMin, double dMax )
joachim99@66 1813 {
joachim99@66 1814 g_pProgressDialog->setSubRangeTransformation( dMin, dMax );
joachim99@66 1815 }
joachim99@66 1816
joachim99@66 1817
joachim99@66 1818
joachim99@66 1819
joachim99@66 1820
joachim99@70 1821 //#include "fileaccess.moc"