Switch to ExifDirectory from Property
This commit is contained in:
		
							
								
								
									
										151
									
								
								Shared/Models/Stateless/Methods/FilePair.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								Shared/Models/Stateless/Methods/FilePair.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,151 @@ | ||||
| using System.Collections.ObjectModel; | ||||
| using View_by_Distance.Shared.Models.Properties; | ||||
|  | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| internal abstract class FilePair | ||||
| { | ||||
|  | ||||
|     internal static ReadOnlyCollection<Models.FilePair> GetFilePairs(IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string extension, string jsonGroupDirectory, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection) | ||||
|     { | ||||
|         ReadOnlyCollection<Models.FilePair> results; | ||||
|         ReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles = FilePath.GetFilesKeyValuePairs(filePathsCollection); | ||||
|         results = GetFilePairs(propertyConfiguration, directorySearchFilter, extension, jsonGroupDirectory, filePathsCollection, fileNamesToFiles); | ||||
|         return results; | ||||
|     } | ||||
|  | ||||
|     internal static ReadOnlyCollection<Models.FilePair> GetFilePairs(IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string extension, string jsonGroupDirectory, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles) | ||||
|     { | ||||
|         List<Models.FilePair>? results = null; | ||||
|         int renamed; | ||||
|         const bool useCeilingAverage = true; | ||||
|         ReadOnlyCollection<string[]>? jsonFilesCollection = null; | ||||
|         IReadOnlyDictionary<int, List<FilePath>>? compareFileNamesToFiles = null; | ||||
|         for (int i = 0; i < fileNamesToFiles.Count; i++) | ||||
|         { | ||||
|             renamed = 0; | ||||
|             jsonFilesCollection = XDirectory.GetFilesCollection(jsonGroupDirectory, directorySearchFilter, extension, useCeilingAverage); | ||||
|             renamed += LookForAbandoned(propertyConfiguration, jsonFilesCollection, fileNamesToFiles, extension); | ||||
|             if (renamed > 0) | ||||
|                 continue; | ||||
|             compareFileNamesToFiles = GetKeyValuePairs(propertyConfiguration, jsonFilesCollection); | ||||
|             results = XDirectory.GetFiles(propertyConfiguration, filePathsCollection, fileNamesToFiles, compareFileNamesToFiles); | ||||
|             renamed += XDirectory.MaybeMove(propertyConfiguration, results, jsonGroupDirectory, extension); | ||||
|             if (renamed == 0) | ||||
|                 break; | ||||
|             if (i > 10) | ||||
|                 throw new NotImplementedException(); | ||||
|         } | ||||
|         if (results is null || jsonFilesCollection is null || compareFileNamesToFiles is null) | ||||
|             throw new NullReferenceException(nameof(results)); | ||||
|         return results.AsReadOnly(); | ||||
|     } | ||||
|  | ||||
|     private static int LookForAbandoned(IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> jsonFilesCollection, IReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles, string extension) | ||||
|     { | ||||
|         int result; | ||||
|         bool check; | ||||
|         bool moved = false; | ||||
|         List<string> renameCollection = []; | ||||
|         foreach (string[] files in jsonFilesCollection) | ||||
|         { | ||||
|             if (files.Length == 0) | ||||
|                 continue; | ||||
|             check = AnyMoved(propertyConfiguration, fileNamesToFiles, extension, renameCollection, files); | ||||
|             if (!moved && check) | ||||
|                 moved = true; | ||||
|         } | ||||
|         if (renameCollection.Count > 0) | ||||
|             XDirectory.MoveFiles(renameCollection, "{}", "{abd}"); | ||||
|         result = renameCollection.Count; | ||||
|         if (moved && result == 0) | ||||
|             result = 1; | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     private static bool AnyMoved(IPropertyConfiguration propertyConfiguration, IReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles, string extension, List<string> renameCollection, string[] files) | ||||
|     { | ||||
|         bool result = false; | ||||
|         string checkFile; | ||||
|         string directory; | ||||
|         FilePath filePath; | ||||
|         string fileNameWith; | ||||
|         string checkDirectory; | ||||
|         string directoryName; | ||||
|         List<FilePath>? collection; | ||||
|         Models.FileHolder fileHolder; | ||||
|         List<string> directoryNames = []; | ||||
|         foreach (string file in files) | ||||
|         { | ||||
|             if (!file.EndsWith(extension)) | ||||
|                 throw new Exception(); | ||||
|             fileHolder = IFileHolder.Get(file); | ||||
|             if (!fileHolder.Exists) | ||||
|                 continue; | ||||
|             filePath = FilePath.Get(propertyConfiguration, fileHolder, index: null); | ||||
|             if (filePath.Id is null) | ||||
|                 continue; | ||||
|             if (!fileNamesToFiles.TryGetValue(filePath.Id.Value, out collection)) | ||||
|                 renameCollection.Add(file); | ||||
|             else | ||||
|             { | ||||
|                 directoryNames.Clear(); | ||||
|                 directoryName = Path.GetFileName(filePath.DirectoryFullPath) ?? throw new Exception(); | ||||
|                 foreach (FilePath f in collection) | ||||
|                     directoryNames.Add(Path.GetFileName(f.DirectoryFullPath) ?? throw new Exception()); | ||||
|                 if (directoryNames.Count == 0 || directoryNames.Distinct().Count() != 1) | ||||
|                     continue; | ||||
|                 if (directoryName != directoryNames[0]) | ||||
|                 { | ||||
|                     directory = Path.GetDirectoryName(filePath.DirectoryFullPath) ?? throw new Exception(); | ||||
|                     checkDirectory = Path.Combine(directory, directoryNames[0]); | ||||
|                     if (!Directory.Exists(checkDirectory)) | ||||
|                         _ = Directory.CreateDirectory(checkDirectory); | ||||
|                     fileNameWith = collection.Count > 1 ? filePath.Name : $"{collection[0].Name}{extension}"; | ||||
|                     checkFile = Path.Combine(checkDirectory, fileNameWith); | ||||
|                     if (!result) | ||||
|                         result = true; | ||||
|                     if (!File.Exists(checkFile)) | ||||
|                         File.Move(file, checkFile); | ||||
|                     else | ||||
|                     { | ||||
|                         if (new FileInfo(file).LastWriteTime > new FileInfo(checkFile).LastWriteTime) | ||||
|                             File.Delete(file); | ||||
|                         else | ||||
|                             File.Move(file, checkFile, true); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     private static ReadOnlyDictionary<int, List<FilePath>> GetKeyValuePairs(IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection) | ||||
|     { | ||||
|         Dictionary<int, List<FilePath>> results = []; | ||||
|         List<FilePath>? collection; | ||||
|         FilePath filePath; | ||||
|         Models.FileHolder fileHolder; | ||||
|         foreach (string[] files in filesCollection) | ||||
|         { | ||||
|             foreach (string file in files) | ||||
|             { | ||||
|                 fileHolder = IFileHolder.Get(file); | ||||
|                 if (!fileHolder.Exists) | ||||
|                     continue; | ||||
|                 filePath = FilePath.Get(propertyConfiguration, fileHolder, index: null); | ||||
|                 if (filePath.Id is null) | ||||
|                     continue; | ||||
|                 if (!results.TryGetValue(filePath.Id.Value, out collection)) | ||||
|                 { | ||||
|                     results.Add(filePath.Id.Value, []); | ||||
|                     if (!results.TryGetValue(filePath.Id.Value, out collection)) | ||||
|                         throw new Exception(); | ||||
|                 } | ||||
|                 collection.Add(filePath); | ||||
|             } | ||||
|         } | ||||
|         return new(results); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										38
									
								
								Shared/Models/Stateless/Methods/IDate.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								Shared/Models/Stateless/Methods/IDate.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,38 @@ | ||||
| using System.Collections.ObjectModel; | ||||
|  | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| public interface IDate | ||||
| { | ||||
|  | ||||
|     public static DateTime GetMinimum(ExifDirectory exifDirectory) => | ||||
|         XDate.GetMinimum(exifDirectory); | ||||
|  | ||||
|     public static (int Season, string seasonName) GetSeason(int dayOfYear) => | ||||
|         XDate.GetSeason(dayOfYear); | ||||
|  | ||||
|     public static DateTime? GetDateTimeOriginal(ExifDirectory exifDirectory) => | ||||
|         XDate.GetDateTimeOriginal(exifDirectory); | ||||
|  | ||||
|     public static ReadOnlyCollection<DateTime> GetDateTimes(ExifDirectory exifDirectory) => | ||||
|         XDate.GetDateTimes(exifDirectory); | ||||
|  | ||||
|     public static (bool?, string[]) IsWrongYear(DirectoryInfo directoryInfo, FilePath filePath, ExifDirectory exifDirectory) => | ||||
|         XDate.IsWrongYear(directoryInfo, filePath, exifDirectory); | ||||
|  | ||||
|     internal DateTime TestStatic_GetMinimum(ExifDirectory exifDirectory) => | ||||
|         GetMinimum(exifDirectory); | ||||
|  | ||||
|     internal (int Season, string seasonName) TestStatic_GetSeason(int dayOfYear) => | ||||
|         GetSeason(dayOfYear); | ||||
|  | ||||
|     internal DateTime? TestStatic_GetDateTimeOriginal(ExifDirectory exifDirectory) => | ||||
|         GetDateTimeOriginal(exifDirectory); | ||||
|  | ||||
|     internal ReadOnlyCollection<DateTime> TestStatic_GetDateTimes(ExifDirectory exifDirectory) => | ||||
|         XDate.GetDateTimes(exifDirectory); | ||||
|  | ||||
|     internal (bool?, string[]) TestStatic_IsWrongYear(DirectoryInfo directoryInfo, FilePath filePath, ExifDirectory exifDirectory) => | ||||
|         IsWrongYear(directoryInfo, filePath, exifDirectory); | ||||
|  | ||||
| } | ||||
| @ -11,32 +11,32 @@ public interface IDirectory | ||||
|     public static char GetDirectory(string fileName) => | ||||
|         fileName.Split('-').Length > 2 ? '-' : fileName.Split('.')[0][^1]; | ||||
|  | ||||
|     public static (string[], List<(FilePath, string)>) GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, bool copyDuplicates, bool ifCanUseId, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, ReadOnlyDictionary<byte, ReadOnlyCollection<string>>> fileGroups, Action? tick) => | ||||
|         XDirectory.GetToDoCollection(propertyConfiguration, copyDuplicates, ifCanUseId, filePathsCollection, fileGroups, tick); | ||||
|  | ||||
|     public static void MoveFiles(List<string> files, string find, string replace) => | ||||
|         XDirectory.MoveFiles(files, find, replace); | ||||
|  | ||||
|     public static List<string> CopyOrMove(List<(FilePath, string)> toDoCollection, bool move, bool moveBack, Action? tick) => | ||||
|         XDirectory.CopyOrMove(toDoCollection, move, moveBack, tick); | ||||
|  | ||||
|     public static int MaybeMove(Properties.IPropertyConfiguration propertyConfiguration, List<FilePair> filePairs, string jsonGroupDirectory, string extension) => | ||||
|         XDirectory.MaybeMove(propertyConfiguration, filePairs, jsonGroupDirectory, extension); | ||||
|  | ||||
|     public static ReadOnlyCollection<string[]> GetFilesCollection(string directory, string directorySearchFilter, string fileSearchFilter, bool useCeilingAverage) => | ||||
|         XDirectory.GetFilesCollection(directory, directorySearchFilter, fileSearchFilter, useCeilingAverage); | ||||
|  | ||||
|     public static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection) => | ||||
|         XDirectory.GetFilePathCollections(propertyConfiguration, filesCollection); | ||||
|     public static int MaybeMove(Properties.IPropertyConfiguration propertyConfiguration, List<Models.FilePair> filePairs, string jsonGroupDirectory, string extension) => | ||||
|         XDirectory.MaybeMove(propertyConfiguration, filePairs, jsonGroupDirectory, extension); | ||||
|  | ||||
|     public static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string fileSearchFilter, string directory, bool useCeilingAverage) => | ||||
|         XDirectory.GetFilePathCollections(propertyConfiguration, directorySearchFilter, fileSearchFilter, directory, useCeilingAverage); | ||||
|     public static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection, bool useIgnoreExtensions) => | ||||
|         XDirectory.GetFilePathCollections(propertyConfiguration, filesCollection, useIgnoreExtensions); | ||||
|  | ||||
|     public static List<FilePair> GetFiles(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, List<string>> fileNamesToFiles, IReadOnlyDictionary<string, List<string>> compareFileNamesToFiles) => | ||||
|     public static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string fileSearchFilter, string directory, bool useIgnoreExtensions, bool useCeilingAverage) => | ||||
|         XDirectory.GetFilePathCollections(propertyConfiguration, directorySearchFilter, fileSearchFilter, directory, useIgnoreExtensions, useCeilingAverage); | ||||
|  | ||||
|     public static List<Models.FilePair> GetFiles(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles, IReadOnlyDictionary<int, List<FilePath>> compareFileNamesToFiles) => | ||||
|         XDirectory.GetFiles(propertyConfiguration, filePathsCollection, fileNamesToFiles, compareFileNamesToFiles); | ||||
|  | ||||
|     public static (string[], List<(FilePath, string)>) GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, ReadOnlyDictionary<byte, ReadOnlyCollection<string>>> fileGroups, Action? tick) => | ||||
|         XDirectory.GetToDoCollection(propertyConfiguration, copyDuplicates: false, ifCanUseId: true, filePathsCollection, fileGroups, tick); | ||||
|     public static (string[], List<(FilePath, string)>) GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, bool copyDuplicates, bool ifCanUseId, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<byte, ReadOnlyCollection<string>> fileGroups, Action? tick) => | ||||
|         XDirectory.GetToDoCollection(propertyConfiguration, copyDuplicates, ifCanUseId, filePathsCollection, fileGroups, null, tick); | ||||
|  | ||||
|     public static (string[], List<(FilePath, string)>) GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<byte, ReadOnlyCollection<string>> fileGroups, Dictionary<int, ExifDirectory> exifDirectoriesById, Action? tick) => | ||||
|         XDirectory.GetToDoCollection(propertyConfiguration, copyDuplicates: false, ifCanUseId: true, filePathsCollection, fileGroups, exifDirectoriesById, tick); | ||||
|  | ||||
|     internal int TestStatic_GetDirectory(char directory) => | ||||
|         GetDirectory(directory); | ||||
| @ -50,25 +50,25 @@ public interface IDirectory | ||||
|     internal List<string> TestStatic_CopyOrMove(List<(FilePath, string)> toDoCollection, bool move, bool moveBack, Action? tick) => | ||||
|         CopyOrMove(toDoCollection, move, moveBack, tick); | ||||
|  | ||||
|     internal int TestStatic_MaybeMove(Properties.IPropertyConfiguration propertyConfiguration, List<FilePair> filePairs, string jsonGroupDirectory, string extension) => | ||||
|         MaybeMove(propertyConfiguration, filePairs, jsonGroupDirectory, extension); | ||||
|  | ||||
|     internal ReadOnlyCollection<string[]> TestStatic_GetFilesCollection(string directory, string directorySearchFilter, string fileSearchFilter, bool useCeilingAverage) => | ||||
|         GetFilesCollection(directory, directorySearchFilter, fileSearchFilter, useCeilingAverage); | ||||
|  | ||||
|     internal ReadOnlyCollection<ReadOnlyCollection<FilePath>> TestStatic_GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection) => | ||||
|         GetFilePathCollections(propertyConfiguration, filesCollection); | ||||
|     internal int TestStatic_MaybeMove(Properties.IPropertyConfiguration propertyConfiguration, List<Models.FilePair> filePairs, string jsonGroupDirectory, string extension) => | ||||
|         MaybeMove(propertyConfiguration, filePairs, jsonGroupDirectory, extension); | ||||
|  | ||||
|     internal ReadOnlyCollection<ReadOnlyCollection<FilePath>> TestStatic_GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string fileSearchFilter, string directory, bool useCeilingAverage) => | ||||
|         GetFilePathCollections(propertyConfiguration, directorySearchFilter, fileSearchFilter, directory, useCeilingAverage); | ||||
|     internal ReadOnlyCollection<ReadOnlyCollection<FilePath>> TestStatic_GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection, bool useIgnoreExtensions) => | ||||
|         GetFilePathCollections(propertyConfiguration, filesCollection, useIgnoreExtensions); | ||||
|  | ||||
|     internal List<FilePair> TestStatic_GetFiles(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, List<string>> fileNamesToFiles, IReadOnlyDictionary<string, List<string>> compareFileNamesToFiles) => | ||||
|     internal ReadOnlyCollection<ReadOnlyCollection<FilePath>> TestStatic_GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string fileSearchFilter, string directory, bool useIgnoreExtensions, bool useCeilingAverage) => | ||||
|         GetFilePathCollections(propertyConfiguration, directorySearchFilter, fileSearchFilter, directory, useIgnoreExtensions, useCeilingAverage); | ||||
|  | ||||
|     internal List<Models.FilePair> TestStatic_GetFiles(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles, IReadOnlyDictionary<int, List<FilePath>> compareFileNamesToFiles) => | ||||
|         GetFiles(propertyConfiguration, filePathsCollection, fileNamesToFiles, compareFileNamesToFiles); | ||||
|  | ||||
|     internal (string[], List<(FilePath, string)>) TestStatic_GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, ReadOnlyDictionary<byte, ReadOnlyCollection<string>>> fileGroups, Action? tick) => | ||||
|         GetToDoCollection(propertyConfiguration, filePathsCollection, fileGroups, tick); | ||||
|  | ||||
|     internal (string[], List<(FilePath, string)>) TestStatic_GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, bool copyDuplicates, bool ifCanUseId, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, ReadOnlyDictionary<byte, ReadOnlyCollection<string>>> fileGroups, Action? tick) => | ||||
|     internal (string[], List<(FilePath, string)>) TestStatic_GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, bool copyDuplicates, bool ifCanUseId, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<byte, ReadOnlyCollection<string>> fileGroups, Action? tick) => | ||||
|         GetToDoCollection(propertyConfiguration, copyDuplicates, ifCanUseId, filePathsCollection, fileGroups, tick); | ||||
|  | ||||
|     internal (string[], List<(FilePath, string)>) TestStatic_GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<byte, ReadOnlyCollection<string>> fileGroups, Dictionary<int, ExifDirectory> exifDirectoriesById, Action? tick) => | ||||
|         GetToDoCollection(propertyConfiguration, filePathsCollection, fileGroups, exifDirectoriesById, tick); | ||||
|  | ||||
| } | ||||
| @ -5,6 +5,7 @@ public interface IDlibDotNet | ||||
|  | ||||
|     void Tick(); | ||||
|     (string, string) GetResultsFullGroupDirectories(); | ||||
|     void ConstructProgressBar(int maxTicks, string message); | ||||
|     (string, string, string, string) GetResultsFullGroupDirectories(string outputResolution); | ||||
|  | ||||
| } | ||||
							
								
								
									
										21
									
								
								Shared/Models/Stateless/Methods/IFilePair.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								Shared/Models/Stateless/Methods/IFilePair.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | ||||
| using System.Collections.ObjectModel; | ||||
| using View_by_Distance.Shared.Models.Properties; | ||||
|  | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| public interface IFilePair | ||||
| { | ||||
|  | ||||
|     public static ReadOnlyCollection<Models.FilePair> GetFilePairs(IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string extension, string jsonGroupDirectory, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection) => | ||||
|         FilePair.GetFilePairs(propertyConfiguration, directorySearchFilter, extension, jsonGroupDirectory, filePathsCollection); | ||||
|  | ||||
|     public static ReadOnlyCollection<Models.FilePair> GetFilePairs(IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string extension, string jsonGroupDirectory, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles) => | ||||
|         FilePair.GetFilePairs(propertyConfiguration, directorySearchFilter, extension, jsonGroupDirectory, filePathsCollection, fileNamesToFiles); | ||||
|  | ||||
|     internal ReadOnlyCollection<Models.FilePair> TestStatic_GetFilePairs(IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string extension, string jsonGroupDirectory, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection) => | ||||
|         GetFilePairs(propertyConfiguration, directorySearchFilter, extension, jsonGroupDirectory, filePathsCollection); | ||||
|  | ||||
|     internal ReadOnlyCollection<Models.FilePair> TestStatic_GetFilePairs(IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string extension, string jsonGroupDirectory, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles) => | ||||
|         GetFilePairs(propertyConfiguration, directorySearchFilter, extension, jsonGroupDirectory, filePathsCollection, fileNamesToFiles); | ||||
|  | ||||
| } | ||||
| @ -29,19 +29,19 @@ public interface IId | ||||
|         Id.NameWithoutExtensionIsIdFormat(propertyConfiguration, fileHolder.NameWithoutExtension.Split('.')[0]); | ||||
|  | ||||
|     public static bool NameWithoutExtensionIsIntelligentIdFormat(IPropertyConfiguration propertyConfiguration, string fileNameFirstSegment) => | ||||
|         fileNameFirstSegment.Length - 1 == propertyConfiguration.IntMinValueLength && fileNameFirstSegment[^1] is '1' or '2' or '3' or '4' or '5' or '6' or '7' or '8' or '9' && fileNameFirstSegment.All(char.IsNumber); | ||||
|         fileNameFirstSegment.Length - 1 == propertyConfiguration.IntMinValueLength && fileNameFirstSegment[^1] is '0' or '1' or '2' or '3' or '4' or '5' or '6' or '7' or '8' or '9' && fileNameFirstSegment.All(char.IsNumber); | ||||
|  | ||||
|     public static string GetIntelligentId(IPropertyConfiguration propertyConfiguration, long id, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal) => | ||||
|         Id.GetIntelligentId(propertyConfiguration, id, hasIgnoreKeyword, hasDateTimeOriginal); | ||||
|  | ||||
|     public static string GetPaddedId(IPropertyConfiguration propertyConfiguration, int id, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal, int? index) => | ||||
|         Id.GetPaddedId(propertyConfiguration, id, hasIgnoreKeyword, hasDateTimeOriginal, index); | ||||
|     public static string GetIntelligentId(IPropertyConfiguration propertyConfiguration, long id, string extensionLowered, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal) => | ||||
|         Id.GetIntelligentId(propertyConfiguration, id, extensionLowered, hasIgnoreKeyword, hasDateTimeOriginal); | ||||
|  | ||||
|     public static bool NameWithoutExtensionIsPaddedIntelligentIdFormat(IPropertyConfiguration propertyConfiguration, int sortOrderOnlyLengthIndex, string fileNameFirstSegment) => | ||||
|         fileNameFirstSegment.Length == propertyConfiguration.IntMinValueLength + sortOrderOnlyLengthIndex + 1 | ||||
|         && fileNameFirstSegment[^1] is '1' or '2' or '3' or '4' or '5' or '6' or '7' or '8' or '9' | ||||
|         && fileNameFirstSegment[^1] is '0' or '1' or '2' or '3' or '4' or '5' or '6' or '7' or '8' or '9' | ||||
|         && fileNameFirstSegment.All(char.IsNumber); | ||||
|  | ||||
|     public static string GetPaddedId(IPropertyConfiguration propertyConfiguration, int id, string extensionLowered, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal, int? index) => | ||||
|         Id.GetPaddedId(propertyConfiguration, id, extensionLowered, hasIgnoreKeyword, hasDateTimeOriginal, index); | ||||
|  | ||||
|     internal int TestStatic_GetDeterministicHashCode(byte[] value) => | ||||
|         GetDeterministicHashCode(value); | ||||
|  | ||||
| @ -63,13 +63,13 @@ public interface IId | ||||
|     internal bool TestStatic_NameWithoutExtensionIsIntelligentIdFormat(IPropertyConfiguration propertyConfiguration, string fileNameFirstSegment) => | ||||
|         NameWithoutExtensionIsIntelligentIdFormat(propertyConfiguration, fileNameFirstSegment); | ||||
|  | ||||
|     internal string TestStatic_GetIntelligentId(IPropertyConfiguration propertyConfiguration, long id, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal) => | ||||
|         GetIntelligentId(propertyConfiguration, id, hasIgnoreKeyword, hasDateTimeOriginal); | ||||
|  | ||||
|     internal string TestStatic_GetPaddedId(IPropertyConfiguration propertyConfiguration, int id, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal, int? index) => | ||||
|         GetPaddedId(propertyConfiguration, id, hasIgnoreKeyword, hasDateTimeOriginal, index); | ||||
|     internal string TestStatic_GetIntelligentId(IPropertyConfiguration propertyConfiguration, long id, string extensionLowered, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal) => | ||||
|         GetIntelligentId(propertyConfiguration, id, extensionLowered, hasIgnoreKeyword, hasDateTimeOriginal); | ||||
|  | ||||
|     internal bool TestStatic_NameWithoutExtensionIsPaddedIntelligentIdFormat(IPropertyConfiguration propertyConfiguration, int sortOrderOnlyLengthIndex, string fileNameFirstSegment) => | ||||
|         NameWithoutExtensionIsPaddedIntelligentIdFormat(propertyConfiguration, sortOrderOnlyLengthIndex, fileNameFirstSegment); | ||||
|  | ||||
|     internal string TestStatic_GetPaddedId(IPropertyConfiguration propertyConfiguration, int id, string extensionLowered, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal, int? index) => | ||||
|         GetPaddedId(propertyConfiguration, id, extensionLowered, hasIgnoreKeyword, hasDateTimeOriginal, index); | ||||
|  | ||||
| } | ||||
| @ -1,16 +1,18 @@ | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| public interface IMappingFromItem | ||||
| { // ... | ||||
| { | ||||
|  | ||||
|     MappingFromItem TestStatic_GetMappingFromItem(DateTime[] containerDateTimes, Models.Item item, Models.FileHolder? resizedFileHolder) | ||||
|         => GetMappingFromItem(containerDateTimes, item, resizedFileHolder); | ||||
|     static MappingFromItem GetMappingFromItem(DateTime[] containerDateTimes, Models.Item item, Models.FileHolder? resizedFileHolder) | ||||
|     public static MappingFromItem GetMappingFromItem(DateTime[] containerDateTimes, Models.Item item, Models.FileHolder? resizedFileHolder) | ||||
|         => MappingFromItem.GetMappingFromItem(containerDateTimes, item, resizedFileHolder); | ||||
|  | ||||
|     MappingFromItem TestStatic_GetMappingFromItem(Models.Item item) | ||||
|         => GetMappingFromItem(item); | ||||
|     static MappingFromItem GetMappingFromItem(Models.Item item) | ||||
|     public static MappingFromItem GetMappingFromItem(Models.Item item) | ||||
|         => GetMappingFromItem(containerDateTimes: [], item, item.ResizedFileHolder); | ||||
|  | ||||
|     internal MappingFromItem TestStatic_GetMappingFromItem(Models.Item item) | ||||
|         => GetMappingFromItem(item); | ||||
|  | ||||
|     internal MappingFromItem TestStatic_GetMappingFromItem(DateTime[] containerDateTimes, Models.Item item, Models.FileHolder? resizedFileHolder) | ||||
|             => GetMappingFromItem(containerDateTimes, item, resizedFileHolder); | ||||
|  | ||||
| } | ||||
							
								
								
									
										32
									
								
								Shared/Models/Stateless/Methods/IMetaBase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								Shared/Models/Stateless/Methods/IMetaBase.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | ||||
| using System.Collections.ObjectModel; | ||||
|  | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| public interface IMetaBase | ||||
| { | ||||
|  | ||||
|     public static string? GetMaker(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         MetaBase.GetMaker(exifBaseDirectories); | ||||
|  | ||||
|     public static string? GetModel(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         MetaBase.GetModel(exifBaseDirectories); | ||||
|  | ||||
|     public static int? GetOrientation(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         MetaBase.GetOrientation(exifBaseDirectories); | ||||
|  | ||||
|     public static ReadOnlyCollection<string> GetKeywords(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         MetaBase.GetKeywords(exifBaseDirectories); | ||||
|  | ||||
|     internal static string? TestStatic_GetMaker(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         GetMaker(exifBaseDirectories); | ||||
|  | ||||
|     internal static string? TestStatic_GetModel(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         GetModel(exifBaseDirectories); | ||||
|  | ||||
|     internal static int? TestStatic_GetOrientation(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         GetOrientation(exifBaseDirectories); | ||||
|  | ||||
|     internal static ReadOnlyCollection<string> TestStatic_GetKeywords(ExifDirectoryBase[]? exifBaseDirectories) => | ||||
|         GetKeywords(exifBaseDirectories); | ||||
|  | ||||
| } | ||||
| @ -1,8 +0,0 @@ | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| public interface IMetadataFile | ||||
| { | ||||
|  | ||||
|     // ... | ||||
|  | ||||
| } | ||||
| @ -24,6 +24,9 @@ public interface IPath | ||||
|     public static void MakeHiddenIfAllItemsAreHidden(string rootDirectory) => | ||||
|         XPath.MakeHiddenIfAllItemsAreHidden(rootDirectory); | ||||
|  | ||||
|     public static void CreateDirectories(ReadOnlyCollection<string> directories) => | ||||
|         XPath.CreateDirectories(directories); | ||||
|  | ||||
|     public static void ChangeDateForEmptyDirectories(string rootDirectory, long ticks) => | ||||
|         XPath.ChangeDateForEmptyDirectories(rootDirectory, ticks); | ||||
|  | ||||
| @ -65,6 +68,9 @@ public interface IPath | ||||
|     internal void TestStatic_MakeHiddenIfAllItemsAreHidden(string rootDirectory) => | ||||
|         MakeHiddenIfAllItemsAreHidden(rootDirectory); | ||||
|  | ||||
|     internal void TestStatic_CreateDirectories(ReadOnlyCollection<string> directories) => | ||||
|         CreateDirectories(directories); | ||||
|  | ||||
|     internal void TestStatic_ChangeDateForEmptyDirectories(string rootDirectory, long ticks) => | ||||
|         ChangeDateForEmptyDirectories(rootDirectory, ticks); | ||||
|  | ||||
|  | ||||
| @ -37,7 +37,7 @@ internal abstract class Id | ||||
|             _ = results.Append(intelligentId[i]); | ||||
|         _ = results.Append(intelligentId[^3]).Append(intelligentId[^2]); | ||||
|         result = int.Parse(results.ToString()); | ||||
|         if (intelligentId[^1] is '1' or '2' or '3' or '4') | ||||
|         if (intelligentId[^1] is '0' or '1' or '2' or '3' or '4') | ||||
|             result *= -1; | ||||
|         else if (intelligentId[^1] is not '9' and not '8' and not '7' and not '6' and not '5') | ||||
|             throw new NotSupportedException(); | ||||
| @ -48,7 +48,7 @@ internal abstract class Id | ||||
|         (byte)(!propertyConfiguration.ValidVideoFormatExtensions.Contains(filePath.ExtensionLowered) ? filePath.Id > -1 ? 9 : 1 : filePath.Id > -1 ? 6 : 4); | ||||
|  | ||||
|     internal static byte GetMissingDateTimeOriginal(Properties.IPropertyConfiguration propertyConfiguration, FilePath filePath) => | ||||
|         (byte)(!propertyConfiguration.ValidVideoFormatExtensions.Contains(filePath.ExtensionLowered) ? filePath.Id > -1 ? 7 : 3 : 5); | ||||
|         (byte)(!propertyConfiguration.ValidVideoFormatExtensions.Contains(filePath.ExtensionLowered) ? filePath.Id > -1 ? 7 : 3 : filePath.Id > -1 ? 5 : 0); | ||||
|  | ||||
|     internal static bool NameWithoutExtensionIsIdFormat(Properties.IPropertyConfiguration propertyConfiguration, string fileNameWithoutExtension) | ||||
|     { | ||||
| @ -63,25 +63,34 @@ internal abstract class Id | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static string GetIntelligentId(Properties.IPropertyConfiguration propertyConfiguration, long id, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal) | ||||
|     internal static string GetIntelligentId(Properties.IPropertyConfiguration propertyConfiguration, long id, string extensionLowered, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal) | ||||
|     { | ||||
|         string result; | ||||
|         StringBuilder stringBuilder = new(); | ||||
|         if (propertyConfiguration.IntMinValueLength < (propertyConfiguration.ResultAllInOneSubdirectoryLength + 2)) | ||||
|             throw new NotSupportedException(); | ||||
|         if (hasDateTimeOriginal is null) | ||||
|         { } | ||||
|         int key; | ||||
|         string value; | ||||
|         List<char> resultAllInOneSubdirectoryChars = []; | ||||
|         if (id > -1) | ||||
|         if (hasDateTimeOriginal is null) | ||||
|         { | ||||
|             key = hasIgnoreKeyword is not null && hasIgnoreKeyword.Value ? 8 : hasDateTimeOriginal is not null && hasDateTimeOriginal.Value ? 9 : 7; | ||||
|             key = 0; | ||||
|             value = id.ToString().PadLeft(propertyConfiguration.IntMinValueLength, '0'); | ||||
|         } | ||||
|         else if (id > -1) | ||||
|         { | ||||
|             if (!propertyConfiguration.ValidVideoFormatExtensions.Contains(extensionLowered)) | ||||
|                 key = hasIgnoreKeyword is not null && hasIgnoreKeyword.Value ? 8 : hasDateTimeOriginal.Value ? 9 : 7; | ||||
|             else | ||||
|                 key = hasIgnoreKeyword is not null && hasIgnoreKeyword.Value ? throw new NotImplementedException() : hasDateTimeOriginal.Value ? 6 : 5; | ||||
|             value = id.ToString().PadLeft(propertyConfiguration.IntMinValueLength, '0'); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             key = hasIgnoreKeyword is not null && hasIgnoreKeyword.Value ? 2 : hasDateTimeOriginal is not null && hasDateTimeOriginal.Value ? 1 : 3; | ||||
|             if (!propertyConfiguration.ValidVideoFormatExtensions.Contains(extensionLowered)) | ||||
|                 key = hasIgnoreKeyword is not null && hasIgnoreKeyword.Value ? 2 : hasDateTimeOriginal.Value ? 1 : 3; | ||||
|             else | ||||
|                 key = hasIgnoreKeyword is not null && hasIgnoreKeyword.Value ? throw new NotImplementedException() : hasDateTimeOriginal.Value ? 4 : 0; | ||||
|             value = id.ToString()[1..].PadLeft(propertyConfiguration.IntMinValueLength, '0'); | ||||
|         } | ||||
|         for (int i = value.Length - propertyConfiguration.ResultAllInOneSubdirectoryLength - 1; i > -1; i--) | ||||
| @ -92,14 +101,14 @@ internal abstract class Id | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static string GetPaddedId(Properties.IPropertyConfiguration propertyConfiguration, int id, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal, int? index) | ||||
|     internal static string GetPaddedId(Properties.IPropertyConfiguration propertyConfiguration, int id, string extensionLowered, bool? hasIgnoreKeyword, bool? hasDateTimeOriginal, int? index) | ||||
|     { | ||||
|         string result; | ||||
|         if (propertyConfiguration.Offset < 0) | ||||
|             result = Guid.NewGuid().ToString(); | ||||
|         else | ||||
|         { | ||||
|             string intelligentId = GetIntelligentId(propertyConfiguration, id, hasIgnoreKeyword, hasDateTimeOriginal); | ||||
|             string intelligentId = GetIntelligentId(propertyConfiguration, id, extensionLowered, hasIgnoreKeyword, hasDateTimeOriginal); | ||||
|             int check = GetId(propertyConfiguration, intelligentId); | ||||
|             if (check != id) | ||||
|                 throw new NotSupportedException(); | ||||
|  | ||||
							
								
								
									
										83
									
								
								Shared/Models/Stateless/Methods/MetaBase.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								Shared/Models/Stateless/Methods/MetaBase.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,83 @@ | ||||
| using System.Collections.ObjectModel; | ||||
|  | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| internal static class MetaBase | ||||
| { | ||||
|  | ||||
|     internal static string? GetMaker(ExifDirectoryBase[]? exifBaseDirectories) | ||||
|     { | ||||
|         string? result = null; | ||||
|         if (exifBaseDirectories is not null) | ||||
|         { | ||||
|             string value; | ||||
|             foreach (ExifDirectoryBase exifDirectoryBase in exifBaseDirectories) | ||||
|             { | ||||
|                 value = exifDirectoryBase?.Make is null ? string.Empty : exifDirectoryBase.Make.ToString().Trim(); | ||||
|                 if (string.IsNullOrEmpty(value)) | ||||
|                     result = null; | ||||
|                 else | ||||
|                 { | ||||
|                     result = $"{value[0].ToString().ToUpper()}{value[1..].ToLower()}"; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static string? GetModel(ExifDirectoryBase[]? exifBaseDirectories) | ||||
|     { | ||||
|         string? result = null; | ||||
|         if (exifBaseDirectories is not null) | ||||
|         { | ||||
|             string value; | ||||
|             foreach (ExifDirectoryBase exifDirectoryBase in exifBaseDirectories) | ||||
|             { | ||||
|                 value = exifDirectoryBase?.Model is null ? string.Empty : exifDirectoryBase.Model.ToString().Trim(); | ||||
|                 if (string.IsNullOrEmpty(value)) | ||||
|                     result = null; | ||||
|                 else | ||||
|                 { | ||||
|                     result = value; | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static int? GetOrientation(ExifDirectoryBase[]? exifBaseDirectories) | ||||
|     { | ||||
|         int? result = null; | ||||
|         // public const int TagOrientation = 274; | ||||
|         if (exifBaseDirectories is not null) | ||||
|         { | ||||
|             foreach (ExifDirectoryBase exifDirectoryBase in exifBaseDirectories) | ||||
|             { | ||||
|                 result = exifDirectoryBase?.OrientationValue; | ||||
|                 if (result is not null) | ||||
|                     break; | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static ReadOnlyCollection<string> GetKeywords(ExifDirectoryBase[]? exifBaseDirectories) | ||||
|     { | ||||
|         List<string> results = []; | ||||
|         if (exifBaseDirectories is not null) | ||||
|         { | ||||
|             string value; | ||||
|             foreach (ExifDirectoryBase exifDirectoryBase in exifBaseDirectories) | ||||
|             { | ||||
|                 value = exifDirectoryBase?.WinKeywords is null ? string.Empty : exifDirectoryBase.WinKeywords.ToString().Trim(); | ||||
|                 if (string.IsNullOrEmpty(value)) | ||||
|                     continue; | ||||
|                 results.Add(value); | ||||
|             } | ||||
|         } | ||||
|         return results.AsReadOnly(); | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										252
									
								
								Shared/Models/Stateless/Methods/XDate.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										252
									
								
								Shared/Models/Stateless/Methods/XDate.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,252 @@ | ||||
| using System.Collections.ObjectModel; | ||||
| using System.Globalization; | ||||
| using System.Text; | ||||
|  | ||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||
|  | ||||
| internal abstract class XDate | ||||
| { | ||||
|  | ||||
|     internal static ReadOnlyCollection<DateTime> GetDateTimes(ExifDirectory exifDirectory) | ||||
|     { | ||||
|         List<DateTime> results = []; | ||||
|         foreach (ExifDirectoryBase exifDirectoryBase in exifDirectory.ExifBaseDirectories) | ||||
|         { | ||||
|             if (exifDirectoryBase.DateTimeOriginal is not null) | ||||
|                 results.Add(exifDirectoryBase.DateTimeOriginal.Value); | ||||
|         } | ||||
|         foreach (AviDirectory aviDirectory in exifDirectory.AviDirectories) | ||||
|         { | ||||
|             if (aviDirectory.DateTimeOriginal is not null) | ||||
|                 results.Add(aviDirectory.DateTimeOriginal.Value); | ||||
|         } | ||||
|         foreach (QuickTimeMovieHeaderDirectory quickTimeMovieHeaderDirectory in exifDirectory.QuickTimeMovieHeaderDirectories) | ||||
|         { | ||||
|             if (quickTimeMovieHeaderDirectory.Created is not null) | ||||
|             { | ||||
|                 if (quickTimeMovieHeaderDirectory.Created.Value.Year == 1904 && quickTimeMovieHeaderDirectory.Created.Value.Month == 1 && quickTimeMovieHeaderDirectory.Created.Value.Day == 1) | ||||
|                     continue; | ||||
|                 results.Add(quickTimeMovieHeaderDirectory.Created.Value); | ||||
|             } | ||||
|         } | ||||
|         foreach (QuickTimeTrackHeaderDirectory quickTimeTrackHeaderDirectory in exifDirectory.QuickTimeTrackHeaderDirectories) | ||||
|         { | ||||
|             if (quickTimeTrackHeaderDirectory.Created is not null) | ||||
|             { | ||||
|                 if ((quickTimeTrackHeaderDirectory.Created.Value.Year is 1904 or 1970) && quickTimeTrackHeaderDirectory.Created.Value.Month == 1 && quickTimeTrackHeaderDirectory.Created.Value.Day == 1) | ||||
|                     continue; | ||||
|                 results.Add(quickTimeTrackHeaderDirectory.Created.Value); | ||||
|             } | ||||
|         } | ||||
|         if (results.Count == 0) | ||||
|         { | ||||
|             string fileNameWithoutExtension = Path.GetFileNameWithoutExtension(exifDirectory.FilePath.Name); | ||||
|             DateTime? dateTime = GetDateTimeFromName(fileNameWithoutExtension); | ||||
|             if (dateTime is not null) | ||||
|                 results.Add(dateTime.Value); | ||||
|             foreach (ExifDirectoryBase exifDirectoryBase in exifDirectory.ExifBaseDirectories) | ||||
|             { | ||||
|                 if (exifDirectoryBase.DateTime is not null) | ||||
|                     results.Add(exifDirectoryBase.DateTime.Value); | ||||
|                 if (exifDirectoryBase.DateTimeDigitized is not null) | ||||
|                     results.Add(exifDirectoryBase.DateTimeDigitized.Value); | ||||
|             } | ||||
|         } | ||||
|         if (results.Count == 0) | ||||
|         { | ||||
|             foreach (FileMetadataDirectory fileMetadataDirectory in exifDirectory.FileMetadataDirectories) | ||||
|             { | ||||
|                 if (fileMetadataDirectory.FileModifiedDate is not null) | ||||
|                     results.Add(fileMetadataDirectory.FileModifiedDate.Value); | ||||
|             } | ||||
|         } | ||||
|         return results.AsReadOnly(); | ||||
|     } | ||||
|  | ||||
|     private static DateTime? GetDateTimeFromName(string fileNameWithoutExtension) | ||||
|     { | ||||
|         DateTime? result = null; | ||||
|         int length; | ||||
|         string format; | ||||
|         string fullFormat; | ||||
|         StringBuilder value = new(); | ||||
|         const string ticksExample = "##################"; | ||||
|         string[][] dateFormats = | ||||
|         [ | ||||
|             [string.Empty, "yyyyMMdd_HHmmss", string.Empty], | ||||
|             [string.Empty, "yyyyMMddHHmmssfff", string.Empty], | ||||
|             [string.Empty, "yyyyMMdd_", ticksExample], | ||||
|             [string.Empty, "yyyy-MM-dd_", ticksExample], | ||||
|             [string.Empty, "yyyy-MM-dd.", ticksExample], | ||||
|             // [string.Empty, "yyyy-MM-dd.", $"{ticksExample}.{fileHolder.Length}"], | ||||
|             [string.Empty, "yyyy-MM-dd HH.mm.ss", string.Empty], | ||||
|             [string.Empty, "yyyyMMdd_HHmmss", "_LLS"], | ||||
|             [string.Empty, "yyyyMMdd_HHmmss", "_HDR"], | ||||
|             ["WIN_", "yyyyMMdd_HH_mm_ss", "_Pro"], | ||||
|             ["IMG_", "yyyyMMdd_HHmmss", string.Empty], | ||||
|             ["IMG#####-", "yyyyMMdd-HHmm", string.Empty], | ||||
|             ["CameraZOOM-", "yyyyMMddHHmmss", string.Empty], | ||||
|             ["VideoCapture_", "yyyyMMdd-HHmmss ", string.Empty] | ||||
|         ]; | ||||
|         foreach (string[] dateFormat in dateFormats) | ||||
|         { | ||||
|             _ = value.Clear(); | ||||
|             if (dateFormat.Length != 3) | ||||
|                 throw new Exception(); | ||||
|             fullFormat = string.Join(string.Empty, dateFormat); | ||||
|             if (fileNameWithoutExtension.Length != fullFormat.Length) | ||||
|                 continue; | ||||
|             format = dateFormat[1]; | ||||
|             length = dateFormat[0].Length + dateFormat[1].Length; | ||||
|             for (int i = dateFormat[0].Length; i < length; i++) | ||||
|                 _ = value.Append(fileNameWithoutExtension[i]); | ||||
|             if (value.Length != format.Length) | ||||
|                 continue; | ||||
|             if (DateTime.TryParseExact(value.ToString(), format, CultureInfo.InvariantCulture, DateTimeStyles.None, out DateTime checkDateTime)) | ||||
|             { | ||||
|                 if (fileNameWithoutExtension.Length < ticksExample.Length || !long.TryParse(fileNameWithoutExtension[^ticksExample.Length..], out long ticks)) | ||||
|                     result = checkDateTime; | ||||
|                 else | ||||
|                     result = new DateTime(ticks); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static DateTime GetMinimum(ExifDirectory exifDirectory) | ||||
|     { | ||||
|         DateTime result; | ||||
|         ReadOnlyCollection<DateTime> results = GetDateTimes(exifDirectory); | ||||
|         result = results.Count == 0 ? DateTime.MinValue : results.Min(); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static (int Season, string seasonName) GetSeason(int dayOfYear) | ||||
|     { | ||||
|         (int Season, string seasonName) result = dayOfYear switch | ||||
|         { | ||||
|             < 78 => new(0, "Winter"), | ||||
|             < 124 => new(1, "Spring"), | ||||
|             < 171 => new(2, "Spring"), | ||||
|             < 217 => new(3, "Summer"), | ||||
|             < 264 => new(4, "Summer"), | ||||
|             < 309 => new(5, "Fall"), | ||||
|             < 354 => new(6, "Fall"), | ||||
|             _ => new(7, "Winter") | ||||
|         }; | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static DateTime? GetDateTimeOriginal(ExifDirectory exifDirectory) | ||||
|     { | ||||
|         DateTime? result; | ||||
|         List<DateTime> results = []; | ||||
|         foreach (ExifDirectoryBase exifDirectoryBase in exifDirectory.ExifBaseDirectories) | ||||
|         { | ||||
|             if (exifDirectoryBase.DateTimeOriginal is not null) | ||||
|                 results.Add(exifDirectoryBase.DateTimeOriginal.Value); | ||||
|         } | ||||
|         foreach (AviDirectory aviDirectory in exifDirectory.AviDirectories) | ||||
|         { | ||||
|             if (aviDirectory.DateTimeOriginal is not null) | ||||
|                 results.Add(aviDirectory.DateTimeOriginal.Value); | ||||
|         } | ||||
|         foreach (QuickTimeMovieHeaderDirectory quickTimeMovieHeaderDirectory in exifDirectory.QuickTimeMovieHeaderDirectories) | ||||
|         { | ||||
|             if (quickTimeMovieHeaderDirectory.Created is not null) | ||||
|             { | ||||
|                 if (quickTimeMovieHeaderDirectory.Created.Value.Year == 1904 && quickTimeMovieHeaderDirectory.Created.Value.Month == 1 && quickTimeMovieHeaderDirectory.Created.Value.Day == 1) | ||||
|                     continue; | ||||
|                 results.Add(quickTimeMovieHeaderDirectory.Created.Value); | ||||
|             } | ||||
|         } | ||||
|         foreach (QuickTimeTrackHeaderDirectory quickTimeTrackHeaderDirectory in exifDirectory.QuickTimeTrackHeaderDirectories) | ||||
|         { | ||||
|             if (quickTimeTrackHeaderDirectory.Created is not null) | ||||
|             { | ||||
|                 if ((quickTimeTrackHeaderDirectory.Created.Value.Year is 1904 or 1970) && quickTimeTrackHeaderDirectory.Created.Value.Month == 1 && quickTimeTrackHeaderDirectory.Created.Value.Day == 1) | ||||
|                     continue; | ||||
|                 results.Add(quickTimeTrackHeaderDirectory.Created.Value); | ||||
|             } | ||||
|         } | ||||
|         result = results.Count == 0 ? null : results.Min(); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static (bool?, string[]) IsWrongYear(DirectoryInfo directoryInfo, FilePath filePath, ExifDirectory exifDirectory) | ||||
|     { | ||||
|         string[] results = []; | ||||
|         bool? result = null; | ||||
|         string year; | ||||
|         string directoryName; | ||||
|         string[] directorySegments; | ||||
|         List<DateTime> collection = []; | ||||
|         string? check = Path.GetFullPath(filePath.FullName); | ||||
|         DateTime? dateTimeOriginal = GetDateTimeOriginal(exifDirectory); | ||||
|         if (dateTimeOriginal is not null) | ||||
|             collection.Add(dateTimeOriginal.Value); | ||||
|         else | ||||
|         { | ||||
|             ReadOnlyCollection<DateTime> dateTimes = GetDateTimes(exifDirectory); | ||||
|             foreach (DateTime dateTime in dateTimes) | ||||
|                 collection.Add(dateTime); | ||||
|         } | ||||
|         foreach (DateTime dateTime in collection) | ||||
|         { | ||||
|             year = dateTime.ToString("yyyy"); | ||||
|             for (int i = 0; i < int.MaxValue; i++) | ||||
|             { | ||||
|                 check = Path.GetDirectoryName(check); | ||||
|                 if (string.IsNullOrEmpty(check)) | ||||
|                     break; | ||||
|                 directoryName = Path.GetFileName(check); | ||||
|                 directorySegments = directoryName.Split(' '); | ||||
|                 (result, results) = IsWrongYear(directorySegments, year); | ||||
|                 if (result is not null) | ||||
|                     break; | ||||
|                 if (check == directoryInfo.FullName) | ||||
|                     break; | ||||
|             } | ||||
|             if (result is not null && !result.Value) | ||||
|                 break; | ||||
|         } | ||||
|         return new(result, results); | ||||
|     } | ||||
|  | ||||
|     private static Record IsWrongYear(string[] segments, string year) | ||||
|     { | ||||
|         Record result; | ||||
|         bool? check; | ||||
|         string[] results = ( | ||||
|             from l | ||||
|             in segments | ||||
|             where l?.Length > 2 | ||||
|             && ( | ||||
|                 l[..2] is "18" or "19" or "20" | ||||
|             || (l.Length == 5 && l.Substring(1, 2) is "18" or "19" or "20" && (l[0] is '~' or '=' or '-' or '^' or '#')) | ||||
|             || (l.Length == 6 && l[..2] is "18" or "19" or "20" && l[4] == '.') | ||||
|             || (l.Length == 7 && l.Substring(1, 2) is "18" or "19" or "20" && l[5] == '.') | ||||
|             ) | ||||
|             select l | ||||
|         ).ToArray(); | ||||
|         string[] matches = ( | ||||
|             from l | ||||
|             in results | ||||
|             where l == year | ||||
|             || (l.Length == 5 && l.Substring(1, 4) == year && (l[0] is '~' or '=' or '-' or '^' or '#')) | ||||
|             || (l.Length == 6 && l[..4] == year && l[4] == '.') | ||||
|             || (l.Length == 7 && l.Substring(1, 4) == year && l[5] == '.') | ||||
|             select l | ||||
|         ).ToArray(); | ||||
|         if (results.Length == 0) | ||||
|             check = null; | ||||
|         else | ||||
|             check = matches.Length == 0; | ||||
|         result = new(check, results); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     private record Record(bool? IsWrongYear, string[] Years); | ||||
|  | ||||
| } | ||||
| @ -67,57 +67,6 @@ internal abstract partial class XDirectory | ||||
|         return results; | ||||
|     } | ||||
|  | ||||
|     internal static int MaybeMove(Properties.IPropertyConfiguration propertyConfiguration, List<FilePair> filePairs, string jsonGroupDirectory, string extension) | ||||
|     { | ||||
|         FileInfo? toFileInfo; | ||||
|         FileInfo fromFileInfo; | ||||
|         string checkDirectory; | ||||
|         List<(string, string)> rename = []; | ||||
|         foreach (FilePair filePair in filePairs) | ||||
|         { | ||||
|             if (filePair.IsUnique) | ||||
|                 continue; | ||||
|             IsNotUniqueLoop(propertyConfiguration, jsonGroupDirectory, extension, filePair, rename); | ||||
|         } | ||||
|         foreach ((string from, string to) in rename) | ||||
|         { | ||||
|             toFileInfo = null; | ||||
|             checkDirectory = to; | ||||
|             fromFileInfo = new(from); | ||||
|             if (!fromFileInfo.Exists) | ||||
|                 continue; | ||||
|             for (int i = 0; i < int.MaxValue; i++) | ||||
|             { | ||||
|                 toFileInfo = new(checkDirectory); | ||||
|                 if (toFileInfo.Directory is null) | ||||
|                     continue; | ||||
|                 if (!toFileInfo.Directory.Exists) | ||||
|                     _ = Directory.CreateDirectory(toFileInfo.Directory.FullName); | ||||
|                 if (checkDirectory.Length > 199) | ||||
|                     throw new Exception(); | ||||
|                 if (!toFileInfo.Exists) | ||||
|                     break; | ||||
|                 else if (fromFileInfo.Length == toFileInfo.Length && fromFileInfo.LastWriteTime == toFileInfo.LastWriteTime) | ||||
|                     checkDirectory = string.Concat(checkDirectory, ".del"); | ||||
|                 else | ||||
|                     checkDirectory = string.Concat(checkDirectory, ".j"); | ||||
|             } | ||||
|             File.Move(from, checkDirectory); | ||||
|         } | ||||
|         return rename.Count; | ||||
|     } | ||||
|  | ||||
|     private static void IsNotUniqueLoop(Properties.IPropertyConfiguration propertyConfiguration, string jsonGroupDirectory, string extension, FilePair filePair, List<(string, string)> rename) | ||||
|     { | ||||
|         int length = propertyConfiguration.RootDirectory.Length; | ||||
|         foreach (string path in filePair.Collection) | ||||
|         { | ||||
|             if (filePair.Match is null || path != filePair.Match) | ||||
|                 continue; | ||||
|             rename.Add(new(path, string.Concat(jsonGroupDirectory, filePair.Path[length..], extension))); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     internal static ReadOnlyCollection<string[]> GetFilesCollection(string directory, string directorySearchFilter, string fileSearchFilter, bool useCeilingAverage) | ||||
|     { | ||||
|         List<string[]> results = []; | ||||
| @ -173,7 +122,54 @@ internal abstract partial class XDirectory | ||||
|         return results; | ||||
|     } | ||||
|  | ||||
|     internal static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection) | ||||
|     internal static int MaybeMove(Properties.IPropertyConfiguration propertyConfiguration, List<Models.FilePair> filePairs, string jsonGroupDirectory, string extension) | ||||
|     { | ||||
|         FileInfo? toFileInfo; | ||||
|         string checkDirectory; | ||||
|         List<(FilePath, string)> rename = []; | ||||
|         foreach (Models.FilePair filePair in filePairs) | ||||
|         { | ||||
|             if (filePair.IsUnique) | ||||
|                 continue; | ||||
|             IsNotUniqueLoop(propertyConfiguration, jsonGroupDirectory, extension, filePair, rename); | ||||
|         } | ||||
|         foreach ((FilePath from, string to) in rename) | ||||
|         { | ||||
|             toFileInfo = null; | ||||
|             checkDirectory = to; | ||||
|             for (int i = 0; i < int.MaxValue; i++) | ||||
|             { | ||||
|                 toFileInfo = new(checkDirectory); | ||||
|                 if (toFileInfo.Directory is null) | ||||
|                     continue; | ||||
|                 if (!toFileInfo.Directory.Exists) | ||||
|                     _ = Directory.CreateDirectory(toFileInfo.Directory.FullName); | ||||
|                 if (checkDirectory.Length > 199) | ||||
|                     throw new Exception(); | ||||
|                 if (!toFileInfo.Exists) | ||||
|                     break; | ||||
|                 else if (from.Length == toFileInfo.Length && from.LastWriteTicks == toFileInfo.LastWriteTime.Ticks) | ||||
|                     checkDirectory = string.Concat(checkDirectory, ".del"); | ||||
|                 else | ||||
|                     checkDirectory = string.Concat(checkDirectory, ".j"); | ||||
|             } | ||||
|             File.Move(from.FullName, checkDirectory); | ||||
|         } | ||||
|         return rename.Count; | ||||
|     } | ||||
|  | ||||
|     private static void IsNotUniqueLoop(Properties.IPropertyConfiguration propertyConfiguration, string jsonGroupDirectory, string extension, Models.FilePair filePair, List<(FilePath, string)> rename) | ||||
|     { | ||||
|         int length = propertyConfiguration.RootDirectory.Length; | ||||
|         foreach (FilePath path in filePair.Collection) | ||||
|         { | ||||
|             if (filePair.Match is null || path != filePair.Match) | ||||
|                 continue; | ||||
|             rename.Add(new(path, string.Concat(jsonGroupDirectory, filePair.FilePath.FullName[length..], extension))); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     internal static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<string[]> filesCollection, bool useIgnoreExtensions) | ||||
|     { | ||||
|         List<ReadOnlyCollection<FilePath>> results = []; | ||||
|         FilePath filePath; | ||||
| @ -185,7 +181,9 @@ internal abstract partial class XDirectory | ||||
|             foreach (string file in files) | ||||
|             { | ||||
|                 fileHolder = IFileHolder.Get(file); | ||||
|                 if (propertyConfiguration.IgnoreExtensions.Contains(fileHolder.ExtensionLowered)) | ||||
|                 if (!fileHolder.Exists) | ||||
|                     continue; | ||||
|                 if (useIgnoreExtensions && propertyConfiguration.IgnoreExtensions.Contains(fileHolder.ExtensionLowered)) | ||||
|                     continue; | ||||
|                 filePath = FilePath.Get(propertyConfiguration, fileHolder, index: null); | ||||
|                 filePaths.Add(filePath); | ||||
| @ -195,38 +193,38 @@ internal abstract partial class XDirectory | ||||
|         return results.AsReadOnly(); | ||||
|     } | ||||
|  | ||||
|     internal static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string fileSearchFilter, string directory, bool useCeilingAverage) | ||||
|     internal static ReadOnlyCollection<ReadOnlyCollection<FilePath>> GetFilePathCollections(Properties.IPropertyConfiguration propertyConfiguration, string directorySearchFilter, string fileSearchFilter, string directory, bool useIgnoreExtensions, bool useCeilingAverage) | ||||
|     { | ||||
|         ReadOnlyCollection<ReadOnlyCollection<FilePath>> results; | ||||
|         ReadOnlyCollection<string[]> filesCollection = GetFilesCollection(directory, directorySearchFilter, fileSearchFilter, useCeilingAverage); | ||||
|         results = IDirectory.GetFilePathCollections(propertyConfiguration, filesCollection); | ||||
|         results = IDirectory.GetFilePathCollections(propertyConfiguration, filesCollection, useIgnoreExtensions); | ||||
|         return results; | ||||
|     } | ||||
|  | ||||
|     internal static List<FilePair> GetFiles(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, List<string>> fileNamesToFiles, IReadOnlyDictionary<string, List<string>> compareFileNamesToFiles) | ||||
|     internal static List<Models.FilePair> GetFiles(Properties.IPropertyConfiguration propertyConfiguration, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<int, List<FilePath>> fileNamesToFiles, IReadOnlyDictionary<int, List<FilePath>> compareFileNamesToFiles) | ||||
|     { | ||||
|         List<FilePair> results = []; | ||||
|         string? match; | ||||
|         FilePair filePair; | ||||
|         List<Models.FilePair> results = []; | ||||
|         FilePath? match; | ||||
|         bool uniqueFileName; | ||||
|         List<string>? collection; | ||||
|         List<FilePath>? collection; | ||||
|         Models.FilePair filePair; | ||||
|         bool? isNotUniqueAndNeedsReview; | ||||
|         string fileNameWithoutExtensionMinusOne; | ||||
|         foreach (ReadOnlyCollection<FilePath> filePaths in filePathsCollection) | ||||
|         { | ||||
|             foreach (FilePath filePath in filePaths) | ||||
|             { | ||||
|                 if (filePath.Id is null) | ||||
|                     continue; | ||||
|                 isNotUniqueAndNeedsReview = null; | ||||
|                 if (propertyConfiguration.IgnoreExtensions.Contains(filePath.ExtensionLowered)) | ||||
|                     continue; | ||||
|                 fileNameWithoutExtensionMinusOne = filePath.NameWithoutExtension[..^1]; | ||||
|                 if (!fileNamesToFiles.TryGetValue(fileNameWithoutExtensionMinusOne, out collection)) | ||||
|                 if (!fileNamesToFiles.TryGetValue(filePath.Id.Value, out collection)) | ||||
|                     throw new Exception(); | ||||
|                 uniqueFileName = collection.Count == 1; | ||||
|                 if (!uniqueFileName) | ||||
|                     isNotUniqueAndNeedsReview = GetIsNotUniqueAndNeedsReview(filePath.FullName, collection); | ||||
|                 if (!compareFileNamesToFiles.TryGetValue(fileNameWithoutExtensionMinusOne, out collection)) | ||||
|                     filePair = new(Path: filePath.FullName, | ||||
|                     isNotUniqueAndNeedsReview = GetIsNotUniqueAndNeedsReview(filePath, collection); | ||||
|                 if (!compareFileNamesToFiles.TryGetValue(filePath.Id.Value, out collection)) | ||||
|                     filePair = new(FilePath: filePath, | ||||
|                                    IsUnique: uniqueFileName, | ||||
|                                    IsNotUniqueAndNeedsReview: isNotUniqueAndNeedsReview, | ||||
|                                    Collection: [], | ||||
| @ -234,21 +232,21 @@ internal abstract partial class XDirectory | ||||
|                 else | ||||
|                 { | ||||
|                     if (collection.Count == 0) | ||||
|                         filePair = new(Path: filePath.FullName, | ||||
|                         filePair = new(FilePath: filePath, | ||||
|                                        IsUnique: uniqueFileName, | ||||
|                                        IsNotUniqueAndNeedsReview: isNotUniqueAndNeedsReview, | ||||
|                                        Collection: collection, | ||||
|                                        Match: null); | ||||
|                     else if (uniqueFileName && collection.Count == 1) | ||||
|                         filePair = new(Path: filePath.FullName, | ||||
|                         filePair = new(FilePath: filePath, | ||||
|                                        IsUnique: uniqueFileName, | ||||
|                                        IsNotUniqueAndNeedsReview: isNotUniqueAndNeedsReview, | ||||
|                                        Collection: collection, | ||||
|                                        Match: collection.First()); | ||||
|                     else | ||||
|                     { | ||||
|                         match = GetMatch(filePath.FullName, collection); | ||||
|                         filePair = new(Path: filePath.FullName, | ||||
|                         match = GetMatch(filePath, collection); | ||||
|                         filePair = new(FilePath: filePath, | ||||
|                                        IsUnique: uniqueFileName, | ||||
|                                        IsNotUniqueAndNeedsReview: isNotUniqueAndNeedsReview, | ||||
|                                        Collection: collection, | ||||
| @ -261,19 +259,20 @@ internal abstract partial class XDirectory | ||||
|         return results; | ||||
|     } | ||||
|  | ||||
|     private static bool GetIsNotUniqueAndNeedsReview(string file, List<string> collection) | ||||
|     private static bool GetIsNotUniqueAndNeedsReview(FilePath filePath, List<FilePath> collection) | ||||
|     { | ||||
|         bool result = false; | ||||
|         FileInfo possibleFileInfo; | ||||
|         FileInfo fileInfo = new(file); | ||||
|         foreach (string possible in collection) | ||||
|         long max; | ||||
|         foreach (FilePath possible in collection) | ||||
|         { | ||||
|             if (possible == file) | ||||
|             if (possible.FullName == filePath.FullName) | ||||
|                 continue; | ||||
|             possibleFileInfo = new(possible); | ||||
|             if (possibleFileInfo.LastWriteTime != fileInfo.LastWriteTime) | ||||
|                 File.SetLastWriteTime(file, new DateTime[] { possibleFileInfo.LastWriteTime, fileInfo.LastWriteTime }.Max()); | ||||
|             if (possibleFileInfo.LastWriteTime == fileInfo.LastWriteTime && possibleFileInfo.Length == fileInfo.Length) | ||||
|             if (possible.LastWriteTicks != filePath.LastWriteTicks) | ||||
|             { | ||||
|                 max = new long[] { possible.LastWriteTicks, filePath.LastWriteTicks }.Max(); | ||||
|                 File.SetLastWriteTime(filePath.FullName, new DateTime(max)); | ||||
|             } | ||||
|             if (possible.LastWriteTicks == filePath.LastWriteTicks && possible.Length == filePath.Length) | ||||
|                 continue; | ||||
|             if (!result) | ||||
|                 result = true; | ||||
| @ -281,50 +280,48 @@ internal abstract partial class XDirectory | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     private static string? GetMatch(string file, List<string> collection) | ||||
|     private static FilePath? GetMatch(FilePath filePath, List<FilePath> collection) | ||||
|     { | ||||
|         string? result = null; | ||||
|         FileInfo possibleFileInfo; | ||||
|         FilePath? result = null; | ||||
|         List<long> lengths = []; | ||||
|         List<string> matches = []; | ||||
|         FileInfo fileInfo = new(file); | ||||
|         List<DateTime> creationTimes = []; | ||||
|         foreach (string possible in collection) | ||||
|         List<FilePath> matches = []; | ||||
|         List<long> lastWriteTicks = []; | ||||
|         foreach (FilePath possible in collection) | ||||
|         { | ||||
|             possibleFileInfo = new(possible); | ||||
|             lengths.Add(possibleFileInfo.Length); | ||||
|             creationTimes.Add(possibleFileInfo.CreationTime); | ||||
|             if (possibleFileInfo.CreationTime != fileInfo.LastWriteTime) | ||||
|             lengths.Add(possible.Length); | ||||
|             lastWriteTicks.Add(possible.LastWriteTicks); | ||||
|             if (possible.LastWriteTicks != filePath.LastWriteTicks) | ||||
|                 continue; | ||||
|             matches.Add(possible); | ||||
|         } | ||||
|         if (matches.Count == 1 || (matches.Count > 0 && lengths.Distinct().Count() == 1 && creationTimes.Distinct().Count() == 1)) | ||||
|         if (matches.Count == 1 || (matches.Count > 0 && lengths.Distinct().Count() == 1 && lastWriteTicks.Distinct().Count() == 1)) | ||||
|             result = matches.First(); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     internal static (string[], List<(FilePath, string)>) GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, bool copyDuplicates, bool ifCanUseId, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, IReadOnlyDictionary<string, ReadOnlyDictionary<byte, ReadOnlyCollection<string>>> fileGroups, Action? tick) | ||||
|     internal static (string[], List<(FilePath, string)>) GetToDoCollection(Properties.IPropertyConfiguration propertyConfiguration, bool copyDuplicates, bool ifCanUseId, ReadOnlyCollection<ReadOnlyCollection<FilePath>> filePathsCollection, ReadOnlyDictionary<byte, ReadOnlyCollection<string>> fileGroups, Dictionary<int, ExifDirectory>? exifDirectoriesById, Action? tick) | ||||
|     { | ||||
|         List<(FilePath, string)> results = []; | ||||
|         string paddedId; | ||||
|         string checkFile; | ||||
|         string directory; | ||||
|         FileInfo fileInfo; | ||||
|         FilePath filePath; | ||||
|         DateTime? dateTime; | ||||
|         string paddedIdFile; | ||||
|         bool wrapped = false; | ||||
|         string intelligentId; | ||||
|         CombinedEnumAndIndex cei; | ||||
|         bool? hasIgnoreKeyword; | ||||
|         bool paddedCheck = false; | ||||
|         CombinedEnumAndIndex cei; | ||||
|         string fileDirectoryName; | ||||
|         bool? hasDateTimeOriginal; | ||||
|         List<int> distinctIds = []; | ||||
|         List<string> distinct = []; | ||||
|         ExifDirectory? exifDirectory; | ||||
|         Models.FileHolder fileHolder; | ||||
|         ReadOnlyCollection<string> keywords; | ||||
|         List<string> distinctDirectories = []; | ||||
|         FilePath[] sortedRecords = GetSortedRecords(filePathsCollection); | ||||
|         ReadOnlyDictionary<byte, ReadOnlyCollection<string>>? keyValuePairs; | ||||
|         if (!fileGroups.TryGetValue(propertyConfiguration.ResultContent, out keyValuePairs)) | ||||
|             throw new NotImplementedException(); | ||||
|         bool isOffsetDeterministicHashCode = IId.IsOffsetDeterministicHashCode(propertyConfiguration); | ||||
|         for (int i = 0; i < sortedRecords.Length; i++) | ||||
|         { | ||||
| @ -338,23 +335,45 @@ internal abstract partial class XDirectory | ||||
|             { | ||||
|                 if (wrapped) | ||||
|                     continue; | ||||
|                 directory = keyValuePairs[cei.Enum][cei.Index]; | ||||
|                 if (cei.Enum == 0) | ||||
|                     continue; | ||||
|                 directory = fileGroups[cei.Enum][cei.Index]; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 if (!wrapped) | ||||
|                     wrapped = true; | ||||
|                 directory = Path.Combine(keyValuePairs[cei.Enum][cei.Index], fileDirectoryName); | ||||
|                 directory = Path.Combine(fileGroups[cei.Enum][cei.Index], fileDirectoryName); | ||||
|             } | ||||
|             if (ifCanUseId && filePath.IsIntelligentIdFormat && filePath.Id is not null && filePath.DirectoryFullPath is not null) | ||||
|             { | ||||
|                 paddedId = IId.GetPaddedId(propertyConfiguration, filePath.Id.Value, filePath.HasIgnoreKeyword, filePath.HasDateTimeOriginal, i); | ||||
|                 if (filePath.Id == -748161839 || filePath.FileNameFirstSegment == "740318810015") | ||||
|                 { | ||||
|                     if (filePath.ExtensionLowered == ".mov") // -748161839) | ||||
|                     { } | ||||
|                 } | ||||
|                 if (exifDirectoriesById is null || !exifDirectoriesById.TryGetValue(filePath.Id.Value, out exifDirectory)) | ||||
|                 { | ||||
|                     hasIgnoreKeyword = filePath.HasIgnoreKeyword; | ||||
|                     hasDateTimeOriginal = filePath.HasDateTimeOriginal; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     dateTime = IDate.GetDateTimeOriginal(exifDirectory); | ||||
|                     hasDateTimeOriginal = dateTime is not null; | ||||
|                     if (dateTime is null && propertyConfiguration.ValidVideoFormatExtensions.Contains(filePath.ExtensionLowered)) | ||||
|                         continue; | ||||
|                     keywords = MetaBase.GetKeywords(exifDirectory?.ExifBaseDirectories); | ||||
|                     hasIgnoreKeyword = propertyConfiguration.IgnoreRulesKeyWords.Any(keywords.Contains); | ||||
|                 } | ||||
|                 paddedId = IId.GetPaddedId(propertyConfiguration, filePath.Id.Value, filePath.ExtensionLowered, hasIgnoreKeyword, hasDateTimeOriginal, i); | ||||
|                 paddedIdFile = Path.Combine(filePath.DirectoryFullPath, $"{paddedId}{filePath.ExtensionLowered}"); | ||||
|                 if (!File.Exists(paddedIdFile)) | ||||
|                 { | ||||
|                     File.Move(filePath.FullName, paddedIdFile); | ||||
|                     fileInfo = new(paddedIdFile); | ||||
|                     fileHolder = Models.FileHolder.Get(fileInfo); | ||||
|                     fileHolder = IFileHolder.Get(paddedIdFile); | ||||
|                     if (!fileHolder.Exists) | ||||
|                         continue; | ||||
|                     filePath = FilePath.Get(propertyConfiguration, fileHolder, index: null); | ||||
|                     if (!paddedCheck) | ||||
|                         paddedCheck = true; | ||||
| @ -366,14 +385,14 @@ internal abstract partial class XDirectory | ||||
|             { | ||||
|                 if (filePath.Id is null) | ||||
|                     throw new NullReferenceException(nameof(filePath.Id)); | ||||
|                 intelligentId = IId.GetIntelligentId(propertyConfiguration, filePath.Id.Value, filePath.HasIgnoreKeyword, filePath.HasDateTimeOriginal); | ||||
|                 intelligentId = IId.GetIntelligentId(propertyConfiguration, filePath.Id.Value, filePath.ExtensionLowered, filePath.HasIgnoreKeyword, filePath.HasDateTimeOriginal); | ||||
|                 if (!isOffsetDeterministicHashCode) | ||||
|                     checkFile = Path.Combine(directory, $"{intelligentId}{filePath.ExtensionLowered}"); | ||||
|                 else | ||||
|                 { | ||||
|                     if (filePath.DirectoryFullPath is null) | ||||
|                         continue; | ||||
|                     paddedId = IId.GetPaddedId(propertyConfiguration, filePath.Id.Value, filePath.HasIgnoreKeyword, filePath.HasDateTimeOriginal, i); | ||||
|                     paddedId = IId.GetPaddedId(propertyConfiguration, filePath.Id.Value, filePath.ExtensionLowered, filePath.HasIgnoreKeyword, filePath.HasDateTimeOriginal, i); | ||||
|                     paddedIdFile = Path.Combine(filePath.DirectoryFullPath, $"{paddedId}{filePath.ExtensionLowered}"); | ||||
|                     if (File.Exists(paddedIdFile)) | ||||
|                         continue; | ||||
| @ -391,8 +410,8 @@ internal abstract partial class XDirectory | ||||
|                     continue; | ||||
|                 for (int j = 1; j < int.MaxValue; j++) | ||||
|                 { | ||||
|                     fileInfo = new(checkFile); | ||||
|                     if (!fileInfo.Exists || filePath.Length == fileInfo.Length && filePath.LastWriteTicks == fileInfo.LastWriteTime.Ticks) | ||||
|                     fileHolder = IFileHolder.Get(checkFile); | ||||
|                     if (!fileHolder.Exists || fileHolder.LastWriteTime is null || filePath.Length == fileHolder.Length && filePath.LastWriteTicks == fileHolder.LastWriteTime.Value.Ticks) | ||||
|                         checkFile = Path.Combine(directory, $"{filePath.NameWithoutExtension}.{j}dup{filePath.ExtensionLowered}"); | ||||
|                     else | ||||
|                         checkFile = Path.Combine(directory, $"{filePath.NameWithoutExtension}.{j}why{filePath.ExtensionLowered}"); | ||||
|  | ||||
| @ -64,15 +64,20 @@ internal abstract class XPath | ||||
|         byte @enum; | ||||
|         int converted; | ||||
|         string combined; | ||||
|         if (filePath.HasIgnoreKeyword is null || filePath.HasDateTimeOriginal is null) | ||||
|             throw new NotImplementedException(); | ||||
|         byte missingDateTimeOriginal = IId.GetMissingDateTimeOriginal(propertyConfiguration, filePath); | ||||
|         if (filePath.HasIgnoreKeyword.Value) | ||||
|             @enum = IId.GetHasIgnoreKeyword(filePath); | ||||
|         else if (!filePath.HasDateTimeOriginal.Value) | ||||
|         if (!filePath.IsIntelligentIdFormat) | ||||
|             @enum = missingDateTimeOriginal; | ||||
|         else | ||||
|             @enum = IId.GetHasDateTimeOriginal(propertyConfiguration, filePath); | ||||
|         { | ||||
|             if (filePath.HasIgnoreKeyword is null || filePath.HasDateTimeOriginal is null) | ||||
|                 throw new NotImplementedException("Chicken and Egg!"); | ||||
|             if (filePath.HasIgnoreKeyword.Value) | ||||
|                 @enum = IId.GetHasIgnoreKeyword(filePath); | ||||
|             else if (!filePath.HasDateTimeOriginal.Value) | ||||
|                 @enum = missingDateTimeOriginal; | ||||
|             else | ||||
|                 @enum = IId.GetHasDateTimeOriginal(propertyConfiguration, filePath); | ||||
|         } | ||||
|         string fileNameBeforeFirst = fileNameWithoutExtension.Split('.')[0]; | ||||
|         string check = fileNameBeforeFirst.Length < propertyConfiguration.ResultAllInOneSubdirectoryLength ? | ||||
|             new('-', propertyConfiguration.ResultAllInOneSubdirectoryLength) : | ||||
| @ -228,6 +233,20 @@ internal abstract class XPath | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     internal static void CreateDirectories(ReadOnlyCollection<string> directories) | ||||
|     { | ||||
|         string checkDirectory; | ||||
|         foreach (string directory in directories) | ||||
|         { | ||||
|             for (int i = 0; i < 101; i++) | ||||
|             { | ||||
|                 checkDirectory = Path.Combine(directory, i.ToString("000")); | ||||
|                 if (!Directory.Exists(checkDirectory)) | ||||
|                     _ = Directory.CreateDirectory(checkDirectory); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     internal static void ChangeDateForEmptyDirectories(string rootDirectory, long ticks) | ||||
|     { | ||||
|         DateTime dateTime = new(ticks); | ||||
| @ -411,6 +430,7 @@ internal abstract class XPath | ||||
|  | ||||
|     private static byte[] GetBytes() => | ||||
|     [ | ||||
|         0, | ||||
|         1, | ||||
|         2, | ||||
|         3, | ||||
|  | ||||
		Reference in New Issue
	
	Block a user