File indexing completed on 2024-09-15 13:11:14
0001 /*************************************************************************** 0002 batchrenamer.h - description 0003 ------------------- 0004 begin : Sat Aug 18 2001 0005 copyright : (C) 2001 by Dominik Seichter 0006 email : domseichter@web.de 0007 ***************************************************************************/ 0008 0009 /*************************************************************************** 0010 * * 0011 * This program is free software; you can redistribute it and/or modify * 0012 * it under the terms of the GNU General Public License as published by * 0013 * the Free Software Foundation; either version 2 of the License, or * 0014 * (at your option) any later version. * 0015 * * 0016 ***************************************************************************/ 0017 0018 #ifndef BATCHRENAMER_H 0019 #define BATCHRENAMER_H 0020 0021 #include <QList> 0022 #include <QObject> 0023 0024 #include "krenamefile.h" 0025 0026 class QFile; 0027 class QProgressDialog; 0028 class QString; 0029 class QTextStream; 0030 0031 typedef struct __tag_tCounterValues { 0032 int value; // current value of this counter 0033 int start; // start value of this counter (for findResetCounter) 0034 int step; // stepping value of this counter; 0035 } tCounterValues; 0036 0037 /** A structure describing a string or regular 0038 * expression that is to be found in the final/resulting 0039 * filename and to be replaced with another string. 0040 */ 0041 typedef struct __tag_TReplaceItem { 0042 QString find; ///< Text to replace 0043 QString replace; ///< Replace with 0044 bool reg; ///< is it a reg expression ? 0045 bool doProcessTokens; ///< Process tokens in result 0046 } TReplaceItem; 0047 0048 /** An enum to set the renaming mode of KRename 0049 */ 0050 enum ERenameMode { 0051 eRenameMode_Rename, ///< All files are renamed in place 0052 eRenameMode_Move, ///< All files are moved to a new directory and renamed 0053 eRenameMode_Copy, ///< Only copies of the files are renamed 0054 eRenameMode_Link ///< Symbolic links are created and renamed 0055 }; 0056 0057 class QObject; 0058 class ProgressDialog; 0059 0060 /** This is the core class for renaming. 0061 * 0062 * It transforms the filenames according to 0063 * the users settings using custom functions 0064 * and external plugins. 0065 * Also the actual renaming, copying or moving 0066 * of the files is done by this class 0067 * 0068 * @author Dominik Seichter 0069 */ 0070 class BatchRenamer : public QObject 0071 { 0072 Q_OBJECT 0073 0074 public: 0075 BatchRenamer(); 0076 ~BatchRenamer(); 0077 0078 /** Sets the list of files 0079 * which contains all filenames which should be transformed. 0080 * 0081 * The renamed filenames are stored in this list, too,. 0082 * 0083 * @param list a list of KRenameFile objects 0084 */ 0085 inline void setFiles(KRenameFile::List *list); 0086 0087 /** Get access to the list of files used for renaming. 0088 * @returns the list of files or NULL if none was set. 0089 */ 0090 inline const KRenameFile::List *files() const; 0091 0092 /** 0093 * @returns the current renaming mode 0094 */ 0095 inline ERenameMode renameMode() const; 0096 0097 /** 0098 * @returns the current start index for counters 0099 */ 0100 inline int numberStartIndex() const; 0101 0102 /** 0103 * @returns the stepping for counters 0104 */ 0105 inline int numberStepping() const; 0106 0107 /** 0108 * @returns if counters are reset on new directories 0109 */ 0110 inline int numberReset() const; 0111 0112 /** 0113 * @returns the list of numbers that are skipped during renaming 0114 */ 0115 inline QList<int> numberSkipList() const; 0116 0117 /** Does the actuall renaming, 0118 * transforming all source filenames in the file list 0119 * to the new destination filenames. 0120 * 0121 * No actuall renaming is done, only the new filenames 0122 * are calculated. 0123 */ 0124 void processFilenames(); 0125 0126 /** Process all files 0127 * 0128 * processFilenames has to be called before 0129 * 0130 * All files will be renamed according to the users settings. 0131 * 0132 * Output will be reported through a progress dialog 0133 * 0134 * @param p use this dialog to report progress to the user 0135 */ 0136 void processFiles(ProgressDialog *p); 0137 0138 /** undo all files 0139 * 0140 * processFiles has to be called before 0141 * 0142 * A previous renaming operation will be undone. 0143 * 0144 * Output will be reported through a progress dialog 0145 * 0146 * @param p use this dialog to report progress to the user 0147 */ 0148 void undoFiles(ProgressDialog *p); 0149 0150 /** Build the destination url from a KRenameFile 0151 * 0152 * @param file a KRenameFile 0153 * @returns a valid QUrl 0154 */ 0155 const QUrl buildDestinationUrl(const KRenameFile &file) const; 0156 0157 inline void setUndoScript(const QString &t) 0158 { 0159 m_undoScript = t; 0160 } 0161 inline void setUndo(bool b) 0162 { 0163 undo = b; 0164 } 0165 0166 inline void setReplaceList(const QList<TReplaceItem> &r) 0167 { 0168 m_replace = r; 0169 } 0170 inline const QList<TReplaceItem> &replaceList() const 0171 { 0172 return m_replace; 0173 } 0174 0175 inline void setMode(int m) 0176 { 0177 m_mode = m; 0178 } 0179 inline int mode() const 0180 { 0181 return m_mode; 0182 } 0183 0184 QString findToken(const QString &oldname, QString token, int i); 0185 QString findPartStrings(QString oldname, QString token); 0186 static QString findDirName(QString token, QString path); 0187 0188 /** 0189 * Replace the token [dirsep] with a slash "/". 0190 * This token is useful to create directories from within regular expressions. 0191 */ 0192 static QString findDirSep(const QString &token, const QString &path); 0193 0194 static QString &doEscape(QString &text); 0195 static QString &unEscape(QString &text); 0196 static void escape(QString &text, const QString &token, const QString &sequence); 0197 0198 /** Capitalize a string. 0199 * 0200 * Used to implement the * token. 0201 * 0202 * @param a text string 0203 * @return a capitalized version of this string (every first letter is a capital letter now) 0204 */ 0205 QString capitalize(const QString &text) const; 0206 0207 /** Handle the [length] tokens 0208 * 0209 * @param token a token found in square brackets 0210 * @param name the filename of the current file 0211 * 0212 * @return QString::null if no length token was found or the a new string 0213 */ 0214 QString findLength(const QString &token, const QString &name); 0215 0216 /** Handle the [trimmed] token 0217 * 0218 * @param token a token found in square brackets 0219 * @param name the filename of the current file 0220 * @param index index of the current file 0221 * 0222 * @return QString::null if no length token was found or the a new string 0223 */ 0224 QString findTrimmed(const QString &token, const QString &name, int index); 0225 0226 QString processString(QString text, const QString &originalName, int i, bool doFindReplace = true); 0227 QString processBrackets(QString text, int *length, const QString &oldname, int index); 0228 QString processNumber(int length, const QString &appendix); 0229 QString processToken(QString token, QString oldname, int i); 0230 0231 public Q_SLOTS: 0232 0233 /** Sets the current mode of renaming. 0234 * KRename can rename files, move them while 0235 * renaming to another directory, rename copies 0236 * or create renamed sym-links. 0237 * 0238 * This mode specifies what should be done with the files. 0239 * 0240 * @param mode the renaming mode. 0241 */ 0242 inline void setRenameMode(ERenameMode mode); 0243 0244 /** Sets the template for the filename that is used 0245 * to transform the filename to its final representation. 0246 * 0247 * @param t the new template 0248 */ 0249 inline void setFilenameTemplate(const QString &t); 0250 0251 /** Sets the template for the filename that is used 0252 * to transform the filename to its final representation. 0253 * 0254 * @param t the new template 0255 */ 0256 inline void setExtensionTemplate(const QString &t); 0257 0258 /** Set the start index for the basic counters 0259 * which do not specify an own start index in their 0260 * appendiy (e.g. ###{7}, means 7 is the start index 0261 * 0262 * @param i start index 0263 */ 0264 inline void setNumberStartIndex(int i) 0265 { 0266 m_index = i; 0267 } 0268 0269 /** Set the stepping for the basic counters 0270 * which do not specify an own stepping in their 0271 * appendiy (e.g. ###{1,2}, means 2 is the stepping 0272 * 0273 * @param s stepping 0274 */ 0275 inline void setNumberStepping(int s) 0276 { 0277 m_step = s; 0278 } 0279 0280 /** Sets whether all counters should be reset for new 0281 * directories 0282 * 0283 * @param r if true counters will be reset 0284 */ 0285 inline void setNumberReset(bool r) 0286 { 0287 m_reset = r; 0288 } 0289 0290 /** Sets the list of numbers that are skipped by counters 0291 * 0292 * @param s a list of numbers that is skipped 0293 */ 0294 inline void setNumberSkipList(const QList<int> &s) 0295 { 0296 m_skip = s; 0297 } 0298 0299 /** Sets if existing files maybe overwritten during renaming 0300 * 0301 * @param overwrite if true existing files will be overwritten 0302 */ 0303 inline void setOverwriteExistingFiles(bool overwrite) 0304 { 0305 m_overwrite = overwrite; 0306 } 0307 0308 /** Sets the destination url (a directory) for copy, move and link operations 0309 * 0310 * @param url destination directory 0311 */ 0312 inline void setDestinationDir(const QUrl &url) 0313 { 0314 m_destination = url; 0315 } 0316 0317 private: 0318 /** Do find and replace on the final resulting filename. 0319 * 0320 * \param text the new final filename with all other changes applied. 0321 * 0322 * \returns the new filename with all find and replace being done. 0323 * 0324 * \see m_replace 0325 */ 0326 QString findReplace(const QString &text, const QString &origFilename, int index); 0327 0328 /** 0329 * Replace one string (which might be a regular expression) in the final filename 0330 * with another string and return a new filename. 0331 * 0332 * \param text the new final filename with all other changes applied. 0333 * \param find the string or regular expression to find 0334 * \param replace replace a matched string with this value 0335 * \param reg if true treat find as regular expression 0336 * \param doProcessTokens process tokens in replaced results 0337 * \param originalName original filename for replacing tokens 0338 * \param index current index 0339 * 0340 * \returns the new filename with find and replace being done. 0341 */ 0342 QString doReplace(const QString &text, const QString &find, const QString &replace, 0343 bool reg, bool doProcessTokens, const QString &origFilename, int index); 0344 0345 private: 0346 /** Execute all plugins of a certain type 0347 * 0348 * @param index the current index 0349 * @param filenameOrPath the current filename or path 0350 * @param type the type of the plugins to run 0351 * @param errorCount the number of errors will be written to this value 0352 * @param p dialog for error reporting (maybe NULL) 0353 * 0354 * @returns either a new filename or an error message (depends on plugin type) 0355 */ 0356 QString executePlugin(int index, const QString &filenameOrPath, int type, int &errorCount, ProgressDialog *p); 0357 0358 void work(ProgressDialog *p); 0359 void writeUndoScript(QTextStream *t); 0360 0361 /** 0362 * Parse a new filename and create missing subdirectories. 0363 * 0364 * This will look for '/' and create all not existing directories 0365 * 0366 * @param destUrl to check for not existing directories 0367 * @param dialog ProgressDialog for error reporting 0368 */ 0369 void createMissingSubDirs(const QUrl &destUrl, ProgressDialog *dialog); 0370 0371 /** resets all counters to there start value if the directory name at @p i 0372 * in m_files changes. 0373 * The caller has to check m_reset before calling this function. 0374 */ 0375 void findCounterReset(int i); 0376 0377 QString text; // template 0378 QString extext; // Extension template 0379 QString m_undoScript; // Filename of undoscript 0380 bool undo; // create an undo script 0381 bool m_reset; // reset counter on new directories 0382 int m_mode; // renaming mode 0383 QList<int> m_skip; // Numbers to skip 0384 0385 // a is used in find number and 0386 // required for skipping. 0387 int m_counter_index; 0388 int m_index; // index for numbers 0389 int m_step; // step for numbers 0390 0391 QList<tCounterValues> m_counters; 0392 0393 private: 0394 KRenameFile::List *m_files; ///< The list of files to rename and the resulting renamed filenames 0395 ERenameMode m_renameMode; ///< The rename mode specifies if files are renamed, copied or moved (or linked) 0396 bool m_overwrite; ///< Overwrite existing files 0397 QUrl m_destination; ///< Destination directory for copy, move and link 0398 0399 QList<TReplaceItem> m_replace; ///< List of strings for find and replace 0400 0401 protected: 0402 QFile *f; 0403 QTime t; 0404 QProgressDialog *progress; 0405 }; 0406 0407 void BatchRenamer::setFiles(KRenameFile::List *list) 0408 { 0409 m_files = list; 0410 } 0411 0412 const KRenameFile::List *BatchRenamer::files() const 0413 { 0414 return m_files; 0415 } 0416 0417 void BatchRenamer::setRenameMode(ERenameMode mode) 0418 { 0419 m_renameMode = mode; 0420 } 0421 0422 ERenameMode BatchRenamer::renameMode() const 0423 { 0424 return m_renameMode; 0425 } 0426 0427 void BatchRenamer::setFilenameTemplate(const QString &t) 0428 { 0429 text = t; 0430 } 0431 0432 void BatchRenamer::setExtensionTemplate(const QString &t) 0433 { 0434 extext = t; 0435 } 0436 0437 int BatchRenamer::numberStartIndex() const 0438 { 0439 return m_index; 0440 } 0441 0442 int BatchRenamer::numberStepping() const 0443 { 0444 return m_step; 0445 } 0446 0447 int BatchRenamer::numberReset() const 0448 { 0449 return m_reset; 0450 } 0451 0452 QList<int> BatchRenamer::numberSkipList() const 0453 { 0454 return m_skip; 0455 } 0456 0457 #endif