Compare commits
22 Commits
ad1e92362d
...
master
Author | SHA1 | Date | |
---|---|---|---|
a6129a73c0 | |||
3e40ab9eec | |||
52f07cf547 | |||
43a2d756dc | |||
182813bd21 | |||
e705d0d286 | |||
99ab384ade | |||
4157993abd | |||
232846f4d1 | |||
47822a4693 | |||
63d4f49bdd | |||
a8e89b841e | |||
64a14f39c1 | |||
01b2c0e5e7 | |||
02d757f35e | |||
29c8fe06c6 | |||
e62c996c6e | |||
eb11d7d2db | |||
4455254952 | |||
1d316fbe27 | |||
aad679b3f1 | |||
96e90f8693 |
File diff suppressed because it is too large
Load Diff
@ -1,41 +0,0 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using ExifLibrary;
|
||||
using JetBrains.Annotations;
|
||||
using PhotoRenamer.Base.Types;
|
||||
|
||||
namespace PhotoRenamer.Base
|
||||
{
|
||||
public static class FilesHelper
|
||||
{
|
||||
private static readonly string[] SupportedFileExtensions = {".jpg", ".cr2", ".mp4"};
|
||||
|
||||
public static IEnumerable<string> FindSupportedFilesRecursively(string path)
|
||||
{
|
||||
var files = Directory.EnumerateFiles(path, "*", SearchOption.AllDirectories);
|
||||
foreach (var file in files)
|
||||
{
|
||||
var fileExt = Path.GetExtension(file);
|
||||
if(fileExt == null) continue;
|
||||
foreach (var supportedFileExtension in SupportedFileExtensions)
|
||||
{
|
||||
if (fileExt.Equals(supportedFileExtension, StringComparison.InvariantCultureIgnoreCase)) yield return file;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public static IEnumerable<MediaFile> GetMediaFiles(IEnumerable<string> files)
|
||||
{
|
||||
if (files == null) throw new ArgumentNullException(nameof(files));
|
||||
|
||||
foreach (var file in files)
|
||||
{
|
||||
var imageFile = ImageFile.FromFile(file);
|
||||
var dateTime = imageFile.Properties.Get<ExifDateTime>(ExifTag.DateTimeOriginal);
|
||||
var mediaFile = new MediaFile(file, dateTime.Value);
|
||||
yield return mediaFile;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -1,17 +0,0 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>netstandard2.0</TargetFramework>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<Nullable>enable</Nullable>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="ExifLibNet" Version="2.1.1" />
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.FxCopAnalyzers" Version="2.9.8">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
@ -1,8 +0,0 @@
|
||||
namespace PhotoRenamer.Base
|
||||
{
|
||||
public static class RegionNames
|
||||
{
|
||||
public static string ContentRegion { get; } = "ContentRegion";
|
||||
public static string LoggingRegion { get; } = "LoggingRegion";
|
||||
}
|
||||
}
|
@ -1,17 +0,0 @@
|
||||
using System;
|
||||
|
||||
namespace PhotoRenamer.Base.Types
|
||||
{
|
||||
public class MediaFile
|
||||
{
|
||||
public MediaFile(string path, DateTime creationDate)
|
||||
{
|
||||
Path = path;
|
||||
CreationDate = creationDate;
|
||||
}
|
||||
|
||||
public string Path { get; }
|
||||
public DateTime CreationDate { get; }
|
||||
|
||||
}
|
||||
}
|
@ -1,30 +0,0 @@
|
||||
using System;
|
||||
using PhotoRenamer.Base;
|
||||
using PhotoRenamer.File.Views;
|
||||
using Prism.Ioc;
|
||||
using Prism.Modularity;
|
||||
using Prism.Regions;
|
||||
|
||||
namespace PhotoRenamer.File
|
||||
{
|
||||
public class FileModule : IModule
|
||||
{
|
||||
private readonly IRegionManager _regionManager;
|
||||
|
||||
public FileModule(IRegionManager regionManager)
|
||||
{
|
||||
_regionManager = regionManager;
|
||||
}
|
||||
|
||||
public void OnInitialized(IContainerProvider containerProvider)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
public void RegisterTypes(IContainerRegistry containerRegistry)
|
||||
{
|
||||
containerRegistry.RegisterForNavigation<ViewA>();
|
||||
_regionManager.RequestNavigate(RegionNames.ContentRegion, new Uri(nameof(ViewA), UriKind.Relative));
|
||||
}
|
||||
}
|
||||
}
|
@ -1,19 +0,0 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk.WindowsDesktop">
|
||||
<PropertyGroup>
|
||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<UseWPF>true</UseWPF>
|
||||
<AssemblyName>PhotoRenamer.File</AssemblyName>
|
||||
<Nullable>enable</Nullable>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.FxCopAnalyzers" Version="2.9.8">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Prism.Wpf" Version="7.2.0.1422" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\PhotoRenamer.Base\PhotoRenamer.Base.csproj" />
|
||||
</ItemGroup>
|
||||
</Project>
|
@ -1,14 +0,0 @@
|
||||
using Prism.Mvvm;
|
||||
using System.Windows.Input;
|
||||
using JetBrains.Annotations;
|
||||
|
||||
namespace PhotoRenamer.File.ViewModels
|
||||
{
|
||||
[UsedImplicitly]
|
||||
public class ViewAViewModel : BindableBase
|
||||
{
|
||||
public ICommand SelectFilesCommand { get; }
|
||||
|
||||
public ICommand ClearCommand { get; }
|
||||
}
|
||||
}
|
@ -1,41 +0,0 @@
|
||||
<UserControl
|
||||
x:Class="PhotoRenamer.File.Views.ViewA"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:prism="http://prismlibrary.com/"
|
||||
xmlns:viewModels="clr-namespace:PhotoRenamer.File.ViewModels"
|
||||
d:DataContext="{d:DesignInstance viewModels:ViewAViewModel}"
|
||||
d:DesignHeight="300"
|
||||
d:DesignWidth="300"
|
||||
prism:ViewModelLocator.AutoWireViewModel="True"
|
||||
mc:Ignorable="d">
|
||||
<DockPanel>
|
||||
<ToolBarTray DockPanel.Dock="Top" Orientation="Horizontal">
|
||||
<ToolBar ToolBarTray.IsLocked="True">
|
||||
<Button Command="{Binding SelectFilesCommand}" Content="Select Files" />
|
||||
<Button Command="{Binding ClearCommand}" Content="Clear" />
|
||||
</ToolBar>
|
||||
</ToolBarTray>
|
||||
<Grid>
|
||||
<Grid.RowDefinitions>
|
||||
<RowDefinition />
|
||||
<RowDefinition Height="Auto" />
|
||||
<RowDefinition />
|
||||
</Grid.RowDefinitions>
|
||||
<GroupBox
|
||||
Grid.Row="0"
|
||||
Margin="5"
|
||||
Header="Source">
|
||||
<ListView />
|
||||
</GroupBox>
|
||||
<GroupBox
|
||||
Grid.Row="2"
|
||||
Margin="5"
|
||||
Header="Target">
|
||||
<ListView />
|
||||
</GroupBox>
|
||||
</Grid>
|
||||
</DockPanel>
|
||||
</UserControl>
|
@ -1,13 +0,0 @@
|
||||
namespace PhotoRenamer.File.Views
|
||||
{
|
||||
/// <summary>
|
||||
/// Interaction logic for ViewA.xaml
|
||||
/// </summary>
|
||||
public partial class ViewA
|
||||
{
|
||||
public ViewA()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
}
|
||||
}
|
@ -1,30 +0,0 @@
|
||||
using System;
|
||||
using PhotoRenamer.Base;
|
||||
using PhotoRenamer.Logging.Views;
|
||||
using Prism.Ioc;
|
||||
using Prism.Modularity;
|
||||
using Prism.Regions;
|
||||
|
||||
namespace PhotoRenamer.Logging
|
||||
{
|
||||
public class LoggingModule : IModule
|
||||
{
|
||||
private readonly IRegionManager _regionManager;
|
||||
|
||||
public LoggingModule(IRegionManager regionManager)
|
||||
{
|
||||
_regionManager = regionManager;
|
||||
}
|
||||
|
||||
public void OnInitialized(IContainerProvider containerProvider)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
public void RegisterTypes(IContainerRegistry containerRegistry)
|
||||
{
|
||||
containerRegistry.RegisterForNavigation<ViewA>();
|
||||
_regionManager.RequestNavigate(RegionNames.LoggingRegion, new Uri(nameof(ViewA), UriKind.Relative));
|
||||
}
|
||||
}
|
||||
}
|
@ -1,19 +0,0 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk.WindowsDesktop">
|
||||
<PropertyGroup>
|
||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<UseWPF>true</UseWPF>
|
||||
<AssemblyName>PhotoRenamer.Logging</AssemblyName>
|
||||
<Nullable>enable</Nullable>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.FxCopAnalyzers" Version="2.9.8">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Prism.Wpf" Version="7.2.0.1422" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\PhotoRenamer.Base\PhotoRenamer.Base.csproj" />
|
||||
</ItemGroup>
|
||||
</Project>
|
@ -1,14 +0,0 @@
|
||||
using System.Windows.Input;
|
||||
using JetBrains.Annotations;
|
||||
using Prism.Mvvm;
|
||||
|
||||
namespace PhotoRenamer.Logging.ViewModels
|
||||
{
|
||||
[UsedImplicitly]
|
||||
public class ViewAViewModel : BindableBase
|
||||
{
|
||||
public ICommand SelectFilesCommand { get; }
|
||||
|
||||
public ICommand ClearCommand { get; }
|
||||
}
|
||||
}
|
@ -1,17 +0,0 @@
|
||||
<UserControl
|
||||
x:Class="PhotoRenamer.Logging.Views.ViewA"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:prism="http://prismlibrary.com/"
|
||||
xmlns:viewModels="clr-namespace:PhotoRenamer.Logging.ViewModels"
|
||||
d:DataContext="{d:DesignInstance viewModels:ViewAViewModel}"
|
||||
d:DesignHeight="300"
|
||||
d:DesignWidth="300"
|
||||
prism:ViewModelLocator.AutoWireViewModel="True"
|
||||
mc:Ignorable="d">
|
||||
<GroupBox Margin="5" Header="Log">
|
||||
<ListView />
|
||||
</GroupBox>
|
||||
</UserControl>
|
@ -1,13 +0,0 @@
|
||||
namespace PhotoRenamer.Logging.Views
|
||||
{
|
||||
/// <summary>
|
||||
/// Interaction logic for ViewA.xaml
|
||||
/// </summary>
|
||||
public partial class ViewA
|
||||
{
|
||||
public ViewA()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
}
|
||||
}
|
@ -2,35 +2,33 @@ using System;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
||||
using PhotoRenamer.Base;
|
||||
using PhotoRenamer.Base.Types;
|
||||
using PhotoRenamer.Types;
|
||||
|
||||
namespace PhotoRenamer.Test
|
||||
namespace PhotoRenamer.Test;
|
||||
|
||||
[TestClass]
|
||||
public class FilesTest
|
||||
{
|
||||
[TestClass]
|
||||
public class FilesTest
|
||||
private readonly string[] _files;
|
||||
|
||||
public FilesTest()
|
||||
{
|
||||
private readonly string[] _files;
|
||||
var path = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Assets");
|
||||
_files = FilesHelper.FindSupportedFilesRecursively(path).ToArray();
|
||||
}
|
||||
|
||||
public FilesTest()
|
||||
{
|
||||
var path = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Assets");
|
||||
_files = FilesHelper.FindSupportedFilesRecursively(path).ToArray();
|
||||
}
|
||||
[TestMethod, Priority(0)]
|
||||
public void FindSupportedFiles()
|
||||
{
|
||||
Assert.AreEqual(2, _files.Length);
|
||||
Assert.AreEqual("IMG_20120528_125912.jpg", Path.GetFileName(_files[0]));
|
||||
Assert.AreEqual("IMG_20120526_170007.jpg", Path.GetFileName(_files[1]));
|
||||
}
|
||||
|
||||
[TestMethod, Priority(0)]
|
||||
public void FindSupportedFiles()
|
||||
{
|
||||
Assert.AreEqual(2, _files.Length);
|
||||
Assert.AreEqual("IMG_20120528_125912.jpg", Path.GetFileName(_files[0]));
|
||||
Assert.AreEqual("IMG_20120526_170007.jpg", Path.GetFileName(_files[1]));
|
||||
}
|
||||
|
||||
[TestMethod, Priority(1)]
|
||||
public void GetMetaData()
|
||||
{
|
||||
var expected = new[] {new MediaFile("r", DateTime.Now), new MediaFile("r", DateTime.Now)};
|
||||
var actual = FilesHelper.GetMediaFiles(_files);
|
||||
}
|
||||
[TestMethod, Priority(1)]
|
||||
public void GetMetaData()
|
||||
{
|
||||
var expected = new[] { new MediaFile("r", DateTime.Now), new MediaFile("r", DateTime.Now) };
|
||||
// var actual = FilesHelper.GetMediaFiles(_files);
|
||||
}
|
||||
}
|
@ -1,20 +1,15 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<IsPackable>false</IsPackable>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.FxCopAnalyzers" Version="2.9.8">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.4.0" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="2.0.0" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="2.0.0" />
|
||||
<PackageReference Include="coverlet.collector" Version="1.2.0">
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.8.0" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="3.1.1" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="3.1.1" />
|
||||
<PackageReference Include="coverlet.collector" Version="6.0.0">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
@ -39,7 +34,7 @@
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\PhotoRenamer.Base\PhotoRenamer.Base.csproj" />
|
||||
<ProjectReference Include="..\PhotoRenamer\PhotoRenamer.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
|
@ -5,13 +5,9 @@ VisualStudioVersion = 16.0.29609.76
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PhotoRenamer", "PhotoRenamer\PhotoRenamer.csproj", "{C2C8C238-CB3D-4DDA-96FC-297D029F6022}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PhotoRenamer.File", "PhotoRenamer.File\PhotoRenamer.File.csproj", "{185069C2-C1EB-4116-8C33-952A2C0BA1F9}"
|
||||
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "PhotoRenamer.Test", "PhotoRenamer.Test\PhotoRenamer.Test.csproj", "{07F827BC-CF99-4E81-A5C7-54085C97FC10}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PhotoRenamer.Test", "PhotoRenamer.Test\PhotoRenamer.Test.csproj", "{07F827BC-CF99-4E81-A5C7-54085C97FC10}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PhotoRenamer.Base", "PhotoRenamer.Base\PhotoRenamer.Base.csproj", "{DE85C395-3D50-4C12-9B7A-021180E26CE1}"
|
||||
EndProject
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PhotoRenamer.Logging", "PhotoRenamer.Logging\PhotoRenamer.Logging.csproj", "{1AB0E84A-BEEA-4AAA-96BC-D0D66AC63E11}"
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "PhotoRenamerGui", "PhotoRenamerGui\PhotoRenamerGui.csproj", "{733313FE-599D-49A4-A909-BBDBFB15E27D}"
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
@ -23,22 +19,14 @@ Global
|
||||
{C2C8C238-CB3D-4DDA-96FC-297D029F6022}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{C2C8C238-CB3D-4DDA-96FC-297D029F6022}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{C2C8C238-CB3D-4DDA-96FC-297D029F6022}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{185069C2-C1EB-4116-8C33-952A2C0BA1F9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{185069C2-C1EB-4116-8C33-952A2C0BA1F9}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{185069C2-C1EB-4116-8C33-952A2C0BA1F9}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{185069C2-C1EB-4116-8C33-952A2C0BA1F9}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{07F827BC-CF99-4E81-A5C7-54085C97FC10}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{07F827BC-CF99-4E81-A5C7-54085C97FC10}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{07F827BC-CF99-4E81-A5C7-54085C97FC10}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{07F827BC-CF99-4E81-A5C7-54085C97FC10}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{DE85C395-3D50-4C12-9B7A-021180E26CE1}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{DE85C395-3D50-4C12-9B7A-021180E26CE1}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{DE85C395-3D50-4C12-9B7A-021180E26CE1}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{DE85C395-3D50-4C12-9B7A-021180E26CE1}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{1AB0E84A-BEEA-4AAA-96BC-D0D66AC63E11}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{1AB0E84A-BEEA-4AAA-96BC-D0D66AC63E11}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{1AB0E84A-BEEA-4AAA-96BC-D0D66AC63E11}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{1AB0E84A-BEEA-4AAA-96BC-D0D66AC63E11}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{733313FE-599D-49A4-A909-BBDBFB15E27D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{733313FE-599D-49A4-A909-BBDBFB15E27D}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{733313FE-599D-49A4-A909-BBDBFB15E27D}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{733313FE-599D-49A4-A909-BBDBFB15E27D}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
|
27
PhotoRenamer/.vscode/launch.json
vendored
Normal file
27
PhotoRenamer/.vscode/launch.json
vendored
Normal file
@ -0,0 +1,27 @@
|
||||
{
|
||||
// Use IntelliSense to find out which attributes exist for C# debugging
|
||||
// Use hover for the description of the existing attributes
|
||||
// For further information visit https://github.com/OmniSharp/omnisharp-vscode/blob/master/debugger-launchjson.md
|
||||
"version": "0.2.0",
|
||||
"configurations": [
|
||||
{
|
||||
"name": ".NET Core Launch (console)",
|
||||
"type": "coreclr",
|
||||
"request": "launch",
|
||||
"preLaunchTask": "build",
|
||||
// If you have changed target frameworks, make sure to update the program path.
|
||||
"program": "${workspaceFolder}/bin/Debug/netcoreapp3.1/PhotoRenamer.dll",
|
||||
"args": [],
|
||||
"cwd": "${workspaceFolder}",
|
||||
// For more information about the 'console' field, see https://aka.ms/VSCode-CS-LaunchJson-Console
|
||||
"console": "internalConsole",
|
||||
"stopAtEntry": false
|
||||
},
|
||||
{
|
||||
"name": ".NET Core Attach",
|
||||
"type": "coreclr",
|
||||
"request": "attach",
|
||||
"processId": "${command:pickProcess}"
|
||||
}
|
||||
]
|
||||
}
|
42
PhotoRenamer/.vscode/tasks.json
vendored
Normal file
42
PhotoRenamer/.vscode/tasks.json
vendored
Normal file
@ -0,0 +1,42 @@
|
||||
{
|
||||
"version": "2.0.0",
|
||||
"tasks": [
|
||||
{
|
||||
"label": "build",
|
||||
"command": "dotnet",
|
||||
"type": "process",
|
||||
"args": [
|
||||
"build",
|
||||
"${workspaceFolder}/PhotoRenamer.csproj",
|
||||
"/property:GenerateFullPaths=true",
|
||||
"/consoleloggerparameters:NoSummary"
|
||||
],
|
||||
"problemMatcher": "$msCompile"
|
||||
},
|
||||
{
|
||||
"label": "publish",
|
||||
"command": "dotnet",
|
||||
"type": "process",
|
||||
"args": [
|
||||
"publish",
|
||||
"${workspaceFolder}/PhotoRenamer.csproj",
|
||||
"/property:GenerateFullPaths=true",
|
||||
"/consoleloggerparameters:NoSummary"
|
||||
],
|
||||
"problemMatcher": "$msCompile"
|
||||
},
|
||||
{
|
||||
"label": "watch",
|
||||
"command": "dotnet",
|
||||
"type": "process",
|
||||
"args": [
|
||||
"watch",
|
||||
"run",
|
||||
"${workspaceFolder}/PhotoRenamer.csproj",
|
||||
"/property:GenerateFullPaths=true",
|
||||
"/consoleloggerparameters:NoSummary"
|
||||
],
|
||||
"problemMatcher": "$msCompile"
|
||||
}
|
||||
]
|
||||
}
|
@ -1,7 +0,0 @@
|
||||
<prism:PrismApplication
|
||||
x:Class="PhotoRenamer.App"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:prism="http://prismlibrary.com/">
|
||||
<Application.Resources />
|
||||
</prism:PrismApplication>
|
@ -1,32 +0,0 @@
|
||||
using Prism.Ioc;
|
||||
using PhotoRenamer.Views;
|
||||
using System.Windows;
|
||||
using PhotoRenamer.File;
|
||||
using PhotoRenamer.Logging;
|
||||
using Prism.Modularity;
|
||||
|
||||
namespace PhotoRenamer
|
||||
{
|
||||
/// <summary>
|
||||
/// Interaction logic for App.xaml
|
||||
/// </summary>
|
||||
public partial class App
|
||||
{
|
||||
protected override Window CreateShell()
|
||||
{
|
||||
return Container.Resolve<MainWindow>();
|
||||
}
|
||||
|
||||
protected override void RegisterTypes(IContainerRegistry containerRegistry)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
protected override void ConfigureModuleCatalog(IModuleCatalog moduleCatalog)
|
||||
{
|
||||
moduleCatalog.AddModule<FileModule>();
|
||||
moduleCatalog.AddModule<LoggingModule>();
|
||||
base.ConfigureModuleCatalog(moduleCatalog);
|
||||
}
|
||||
}
|
||||
}
|
25
PhotoRenamer/FilesHelper.cs
Normal file
25
PhotoRenamer/FilesHelper.cs
Normal file
@ -0,0 +1,25 @@
|
||||
namespace PhotoRenamer;
|
||||
|
||||
public static class FilesHelper
|
||||
{
|
||||
private static readonly string[] SupportedFileExtensions = { ".jpg", ".cr2", ".mp4", ".mov" };
|
||||
|
||||
public static IEnumerable<string> FindSupportedFilesRecursively(string path)
|
||||
{
|
||||
var files = Directory.EnumerateFiles(path, "*", SearchOption.AllDirectories);
|
||||
foreach (var file in files)
|
||||
{
|
||||
var fileExt = Path.GetExtension(file);
|
||||
foreach (var supportedFileExtension in SupportedFileExtensions)
|
||||
{
|
||||
if (
|
||||
fileExt.Equals(
|
||||
supportedFileExtension,
|
||||
StringComparison.InvariantCultureIgnoreCase
|
||||
)
|
||||
)
|
||||
yield return file;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
25
PhotoRenamer/FolderExtension.cs
Normal file
25
PhotoRenamer/FolderExtension.cs
Normal file
@ -0,0 +1,25 @@
|
||||
namespace PhotoRenamer;
|
||||
|
||||
public static class FolderExtension
|
||||
{
|
||||
public static string CreateFolderStructureByDate(string targetRoot, DateOnly dateTime)
|
||||
{
|
||||
var folder = BuildTargetDirectoryByDate(targetRoot, dateTime);
|
||||
if (!Directory.Exists(folder))
|
||||
{
|
||||
Directory.CreateDirectory(folder);
|
||||
}
|
||||
|
||||
return folder;
|
||||
}
|
||||
|
||||
public static string BuildTargetDirectoryByDate(string targetRoot, DateOnly dateTime)
|
||||
{
|
||||
return Path.Combine(targetRoot, dateTime.Year.ToString(), dateTime.Month.ToString("D2"));
|
||||
}
|
||||
|
||||
public static string[] GetAllFiles(string sourceRoot)
|
||||
{
|
||||
return Directory.GetFiles(sourceRoot, "*", SearchOption.AllDirectories);
|
||||
}
|
||||
}
|
61
PhotoRenamer/MediaFileExtension.cs
Normal file
61
PhotoRenamer/MediaFileExtension.cs
Normal file
@ -0,0 +1,61 @@
|
||||
using MetadataExtractor;
|
||||
using MetadataExtractor.Formats.Exif;
|
||||
using MetadataExtractor.Formats.QuickTime;
|
||||
using Serilog;
|
||||
|
||||
namespace PhotoRenamer;
|
||||
|
||||
public static class MediaFileExtension
|
||||
{
|
||||
public static DateTime GetDateTimeFromSourceFile(string file)
|
||||
{
|
||||
DateTime dateTime;
|
||||
try
|
||||
{
|
||||
var metadata = ImageMetadataReader.ReadMetadata(file);
|
||||
dateTime =
|
||||
GetDateTimeFromExif(metadata)
|
||||
?? GetDateTimeFromMp4(metadata)
|
||||
?? GetDateTimeFromLastWrite(file);
|
||||
}
|
||||
catch (ImageProcessingException e)
|
||||
{
|
||||
Log.Error(e, $"Error reading file information from {file}");
|
||||
dateTime = GetDateTimeFromLastWrite(file);
|
||||
}
|
||||
|
||||
return dateTime;
|
||||
}
|
||||
|
||||
private static DateTime GetDateTimeFromLastWrite(string file)
|
||||
{
|
||||
var creationTime = File.GetLastWriteTime(file);
|
||||
return creationTime;
|
||||
}
|
||||
|
||||
private static DateTime? GetDateTimeFromExif(
|
||||
IEnumerable<MetadataExtractor.Directory> directories
|
||||
)
|
||||
{
|
||||
return
|
||||
directories
|
||||
.OfType<ExifIfd0Directory>()
|
||||
.FirstOrDefault()
|
||||
?.TryGetDateTime(ExifDirectoryBase.TagDateTime, out var dateTime) == true
|
||||
? dateTime
|
||||
: null;
|
||||
}
|
||||
|
||||
private static DateTime? GetDateTimeFromMp4(
|
||||
IEnumerable<MetadataExtractor.Directory> directories
|
||||
)
|
||||
{
|
||||
return
|
||||
directories
|
||||
.OfType<QuickTimeMovieHeaderDirectory>()
|
||||
.FirstOrDefault()
|
||||
?.TryGetDateTime(QuickTimeMovieHeaderDirectory.TagCreated, out var dateTime) == true
|
||||
? dateTime
|
||||
: null;
|
||||
}
|
||||
}
|
@ -1,28 +1,28 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk.WindowsDesktop">
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<OutputType>WinExe</OutputType>
|
||||
<TargetFramework>netcoreapp3.1</TargetFramework>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<UseWPF>true</UseWPF>
|
||||
<AssemblyName>PhotoRenamer</AssemblyName>
|
||||
<StartupObject>PhotoRenamer.App</StartupObject>
|
||||
<PublishTrimmed>true</PublishTrimmed>
|
||||
<PublishReadyToRun>true</PublishReadyToRun>
|
||||
<PublishSingleFile>false</PublishSingleFile>
|
||||
<RuntimeIdentifier>win-x64</RuntimeIdentifier>
|
||||
<RunAnalyzersDuringBuild>true</RunAnalyzersDuringBuild>
|
||||
<OutputType>Exe</OutputType>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.FxCopAnalyzers" Version="2.9.8">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Prism.DryIoc" Version="7.2.0.1422" />
|
||||
<PackageReference Include="PublishAotCompressed" Version="1.0.2" />
|
||||
<PackageReference Include="MetadataExtractor" Version="2.8.1" />
|
||||
<PackageReference Include="Microsoft.Extensions.Configuration" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.Extensions.Configuration.CommandLine" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.Extensions.Configuration.EnvironmentVariables" Version="8.0.0" />
|
||||
<PackageReference Include="Microsoft.Extensions.Configuration.Json" Version="8.0.0" />
|
||||
<PackageReference Include="Serilog" Version="3.1.1" />
|
||||
<PackageReference Include="Serilog.Sinks.Console" Version="5.0.1" />
|
||||
<PackageReference Include="Spectre.Console" Version="0.48.0" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\PhotoRenamer.Base\PhotoRenamer.Base.csproj" />
|
||||
<ProjectReference Include="..\PhotoRenamer.File\PhotoRenamer.File.csproj" />
|
||||
<ProjectReference Include="..\PhotoRenamer.Logging\PhotoRenamer.Logging.csproj" />
|
||||
<None Update="appsettings.json">
|
||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||
</None>
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
37
PhotoRenamer/Program.cs
Normal file
37
PhotoRenamer/Program.cs
Normal file
@ -0,0 +1,37 @@
|
||||
using Microsoft.Extensions.Configuration;
|
||||
using Serilog;
|
||||
|
||||
namespace PhotoRenamer;
|
||||
|
||||
internal static class Program
|
||||
{
|
||||
private static int Main(string[] args)
|
||||
{
|
||||
Log.Logger = new LoggerConfiguration().WriteTo.Console().CreateLogger();
|
||||
var configuration = CreateHostBuilder(args).Build();
|
||||
foreach (var (key, value) in configuration.AsEnumerable())
|
||||
{
|
||||
Log.Information($"{{{key}: {value}}}");
|
||||
}
|
||||
try
|
||||
{
|
||||
var renamer = new Renamer(configuration);
|
||||
return renamer.RunAsync().GetAwaiter().GetResult();
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
Log.Error(e, "Error executing program");
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
public static IConfigurationBuilder CreateHostBuilder(string[] args) =>
|
||||
new ConfigurationBuilder()
|
||||
.SetBasePath(Directory.GetCurrentDirectory())
|
||||
.AddJsonFile(
|
||||
AppDomain.CurrentDomain.BaseDirectory + "\\appsettings.json",
|
||||
optional: true,
|
||||
reloadOnChange: true
|
||||
)
|
||||
.AddCommandLine(args);
|
||||
}
|
111
PhotoRenamer/Renamer.cs
Normal file
111
PhotoRenamer/Renamer.cs
Normal file
@ -0,0 +1,111 @@
|
||||
using Microsoft.Extensions.Configuration;
|
||||
using Serilog;
|
||||
using Spectre.Console;
|
||||
|
||||
namespace PhotoRenamer;
|
||||
|
||||
internal class Renamer
|
||||
{
|
||||
private readonly string _sourcePath;
|
||||
private readonly string _targetPath;
|
||||
private int _currentCount;
|
||||
|
||||
public Renamer(IConfiguration configuration)
|
||||
{
|
||||
_sourcePath = Path.GetFullPath(
|
||||
configuration["Source"]
|
||||
?? throw new NotSupportedException("Source directory needs to be set.")
|
||||
);
|
||||
_targetPath = Path.GetFullPath(
|
||||
configuration["Target"]
|
||||
?? throw new NotSupportedException("Target directory needs to be set.")
|
||||
);
|
||||
Log.Information($"Source path: {_sourcePath}");
|
||||
Log.Information($"Target path: {_targetPath}");
|
||||
}
|
||||
|
||||
public async Task<int> RunAsync()
|
||||
{
|
||||
var files = FolderExtension.GetAllFiles(_sourcePath);
|
||||
_currentCount = 0;
|
||||
var po = new ParallelOptions { MaxDegreeOfParallelism = 3 };
|
||||
await AnsiConsole
|
||||
.Progress()
|
||||
.StartAsync(async ctx =>
|
||||
{
|
||||
await Parallel.ForEachAsync(files, po, (s, token) => Body(s, ctx, token));
|
||||
});
|
||||
return 0;
|
||||
}
|
||||
|
||||
private async ValueTask Body(
|
||||
string file,
|
||||
ProgressContext progressContext,
|
||||
CancellationToken token
|
||||
)
|
||||
{
|
||||
if (token.IsCancellationRequested)
|
||||
return;
|
||||
var task = progressContext.AddTask($"[green]{file}[/]");
|
||||
try
|
||||
{
|
||||
var dateTime = MediaFileExtension.GetDateTimeFromSourceFile(file);
|
||||
var folder = FolderExtension.CreateFolderStructureByDate(
|
||||
_targetPath,
|
||||
DateOnly.FromDateTime(dateTime)
|
||||
);
|
||||
var progress = new Progress<double>(v => task.Value = v * 100);
|
||||
|
||||
await CopyFileAsync(folder, file, progress);
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
Log.Error(ex, "Error reading file information");
|
||||
}
|
||||
finally
|
||||
{
|
||||
Interlocked.Increment(ref _currentCount);
|
||||
task.StopTask();
|
||||
}
|
||||
}
|
||||
|
||||
private static void ResetTimes(FileSystemInfo destination, FileSystemInfo source)
|
||||
{
|
||||
destination.LastWriteTime = source.LastWriteTime;
|
||||
destination.LastWriteTimeUtc = source.LastWriteTimeUtc;
|
||||
destination.CreationTime = source.CreationTime;
|
||||
destination.CreationTimeUtc = source.CreationTimeUtc;
|
||||
}
|
||||
|
||||
private static async Task CopyFileAsync(
|
||||
string folder,
|
||||
string file,
|
||||
IProgress<double>? progress = null
|
||||
)
|
||||
{
|
||||
var destination = new FileInfo(Path.Combine(folder, Path.GetFileName(file)));
|
||||
var source = new FileInfo(file);
|
||||
if (destination.Exists && destination.Length == source.Length)
|
||||
{
|
||||
progress?.Report(1);
|
||||
return;
|
||||
}
|
||||
|
||||
await using var sourceStream = File.Open(
|
||||
source.FullName,
|
||||
FileMode.Open,
|
||||
FileAccess.Read,
|
||||
FileShare.Read
|
||||
);
|
||||
await using var targetStream = File.Open(
|
||||
destination.FullName,
|
||||
FileMode.Create,
|
||||
FileAccess.ReadWrite,
|
||||
FileShare.ReadWrite
|
||||
);
|
||||
|
||||
await sourceStream.CopyToAsync(targetStream, 81920, progress: progress);
|
||||
|
||||
ResetTimes(destination, source);
|
||||
}
|
||||
}
|
39
PhotoRenamer/StreamExtensions.cs
Normal file
39
PhotoRenamer/StreamExtensions.cs
Normal file
@ -0,0 +1,39 @@
|
||||
namespace PhotoRenamer;
|
||||
|
||||
public static class StreamExtensions
|
||||
{
|
||||
public static async Task CopyToAsync(
|
||||
this Stream source,
|
||||
Stream destination,
|
||||
int bufferSize,
|
||||
IProgress<double>? progress = null,
|
||||
CancellationToken cancellationToken = default
|
||||
)
|
||||
{
|
||||
if (source == null)
|
||||
throw new ArgumentNullException(nameof(source));
|
||||
if (!source.CanRead)
|
||||
throw new ArgumentException("Has to be readable", nameof(source));
|
||||
if (destination == null)
|
||||
throw new ArgumentNullException(nameof(destination));
|
||||
if (!destination.CanWrite)
|
||||
throw new ArgumentException("Has to be writable", nameof(destination));
|
||||
if (bufferSize < 0)
|
||||
throw new ArgumentOutOfRangeException(nameof(bufferSize));
|
||||
|
||||
var buffer = new byte[bufferSize];
|
||||
long totalBytesRead = 0;
|
||||
int bytesRead;
|
||||
while (
|
||||
(bytesRead = await source.ReadAsync(buffer, cancellationToken).ConfigureAwait(false))
|
||||
!= 0
|
||||
)
|
||||
{
|
||||
await destination
|
||||
.WriteAsync(buffer.AsMemory(0, bytesRead), cancellationToken)
|
||||
.ConfigureAwait(false);
|
||||
totalBytesRead += bytesRead;
|
||||
progress?.Report(totalBytesRead / (double)source.Length);
|
||||
}
|
||||
}
|
||||
}
|
3
PhotoRenamer/Types/MediaFile.cs
Normal file
3
PhotoRenamer/Types/MediaFile.cs
Normal file
@ -0,0 +1,3 @@
|
||||
namespace PhotoRenamer.Types;
|
||||
|
||||
public record MediaFile(string Path, DateTime CreationDate);
|
@ -1,16 +0,0 @@
|
||||
using JetBrains.Annotations;
|
||||
using Prism.Mvvm;
|
||||
|
||||
namespace PhotoRenamer.ViewModels
|
||||
{
|
||||
[UsedImplicitly]
|
||||
public class MainWindowViewModel : BindableBase
|
||||
{
|
||||
private string _title = "Prism Application";
|
||||
public string Title
|
||||
{
|
||||
get => _title;
|
||||
set => SetProperty(ref _title, value);
|
||||
}
|
||||
}
|
||||
}
|
@ -1,24 +0,0 @@
|
||||
<Window
|
||||
x:Class="PhotoRenamer.Views.MainWindow"
|
||||
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
||||
xmlns:base="clr-namespace:PhotoRenamer.Base;assembly=PhotoRenamer.Base"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:prism="http://prismlibrary.com/"
|
||||
xmlns:viewModels="clr-namespace:PhotoRenamer.ViewModels"
|
||||
Title="{Binding Title}"
|
||||
Width="1000"
|
||||
Height="800"
|
||||
d:DataContext="{d:DesignInstance viewModels:MainWindowViewModel}"
|
||||
prism:ViewModelLocator.AutoWireViewModel="True"
|
||||
mc:Ignorable="d">
|
||||
<Grid>
|
||||
<Grid.ColumnDefinitions>
|
||||
<ColumnDefinition />
|
||||
<ColumnDefinition />
|
||||
</Grid.ColumnDefinitions>
|
||||
<ContentControl prism:RegionManager.RegionName="{x:Static base:RegionNames.ContentRegion}" />
|
||||
<ContentControl Grid.Column="1" prism:RegionManager.RegionName="{x:Static base:RegionNames.LoggingRegion}" />
|
||||
</Grid>
|
||||
</Window>
|
@ -1,13 +0,0 @@
|
||||
namespace PhotoRenamer.Views
|
||||
{
|
||||
/// <summary>
|
||||
/// Interaction logic for MainWindow.xaml
|
||||
/// </summary>
|
||||
public partial class MainWindow
|
||||
{
|
||||
public MainWindow()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
}
|
||||
}
|
4
PhotoRenamer/appsettings.json
Normal file
4
PhotoRenamer/appsettings.json
Normal file
@ -0,0 +1,4 @@
|
||||
{
|
||||
"Source": "/Volumes/EOS_DIGITAL/DCIM/100CANON",
|
||||
"Target": "/Users/holger/Nextcloud/SofortUploadCamera"
|
||||
}
|
12
PhotoRenamerGui/App.axaml
Normal file
12
PhotoRenamerGui/App.axaml
Normal file
@ -0,0 +1,12 @@
|
||||
<Application
|
||||
RequestedThemeVariant="Default"
|
||||
x:Class="PhotoRenamerGui.App"
|
||||
xmlns="https://github.com/avaloniaui"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
|
||||
<!-- "Default" ThemeVariant follows system theme variant. "Dark" or "Light" are other available options. -->
|
||||
|
||||
<Application.Styles>
|
||||
<FluentTheme />
|
||||
<StyleInclude Source="avares://Avalonia.Controls.DataGrid/Themes/Fluent.xaml" />
|
||||
</Application.Styles>
|
||||
</Application>
|
23
PhotoRenamerGui/App.axaml.cs
Normal file
23
PhotoRenamerGui/App.axaml.cs
Normal file
@ -0,0 +1,23 @@
|
||||
using Avalonia;
|
||||
using Avalonia.Controls.ApplicationLifetimes;
|
||||
using Avalonia.Markup.Xaml;
|
||||
|
||||
namespace PhotoRenamerGui;
|
||||
|
||||
public partial class App : Application
|
||||
{
|
||||
public override void Initialize()
|
||||
{
|
||||
AvaloniaXamlLoader.Load(this);
|
||||
}
|
||||
|
||||
public override void OnFrameworkInitializationCompleted()
|
||||
{
|
||||
if (ApplicationLifetime is IClassicDesktopStyleApplicationLifetime desktop)
|
||||
{
|
||||
desktop.MainWindow = new MainWindow();
|
||||
}
|
||||
|
||||
base.OnFrameworkInitializationCompleted();
|
||||
}
|
||||
}
|
79
PhotoRenamerGui/MainWindow.axaml
Normal file
79
PhotoRenamerGui/MainWindow.axaml
Normal file
@ -0,0 +1,79 @@
|
||||
<Window
|
||||
Title="PhotoRenamerGui"
|
||||
d:DesignHeight="450"
|
||||
d:DesignWidth="800"
|
||||
mc:Ignorable="d"
|
||||
x:Class="PhotoRenamerGui.MainWindow"
|
||||
x:DataType="photoRenamerGui:MainWindowViewModel"
|
||||
xmlns="https://github.com/avaloniaui"
|
||||
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
|
||||
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
|
||||
xmlns:photoRenamerGui="clr-namespace:PhotoRenamerGui"
|
||||
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
|
||||
<Window.DataContext>
|
||||
<photoRenamerGui:MainWindowViewModel />
|
||||
</Window.DataContext>
|
||||
<Grid ColumnDefinitions="Auto,*,Auto" RowDefinitions="Auto,Auto,*">
|
||||
<Grid.Styles>
|
||||
<Style Selector=":is(TemplatedControl)">
|
||||
<Setter Property="Margin" Value="0,5" />
|
||||
</Style>
|
||||
</Grid.Styles>
|
||||
<Label
|
||||
Content="Input Folder"
|
||||
Grid.Column="0"
|
||||
Grid.Row="0"
|
||||
VerticalAlignment="Center"
|
||||
VerticalContentAlignment="Center" />
|
||||
<Label
|
||||
Content="Output Folder"
|
||||
Grid.Column="0"
|
||||
Grid.Row="1"
|
||||
VerticalAlignment="Center"
|
||||
VerticalContentAlignment="Center" />
|
||||
<TextBox
|
||||
Grid.Column="1"
|
||||
Grid.Row="0"
|
||||
Text="{Binding InputFolder}"
|
||||
Watermark="Input">
|
||||
<TextBox.InnerRightContent>
|
||||
<Button
|
||||
Command="{Binding SelectInputFolderCommand}"
|
||||
Content="..."
|
||||
Margin="0,5,5,5"
|
||||
VerticalAlignment="Stretch" />
|
||||
</TextBox.InnerRightContent>
|
||||
</TextBox>
|
||||
<TextBox
|
||||
Grid.Column="1"
|
||||
Grid.Row="1"
|
||||
Text="{Binding OutputFolder}"
|
||||
Watermark="Output">
|
||||
<TextBox.InnerRightContent>
|
||||
<Button
|
||||
Command="{Binding SelectOutputFolderCommand}"
|
||||
Content="..."
|
||||
Margin="0,5,5,5"
|
||||
VerticalAlignment="Stretch" />
|
||||
</TextBox.InnerRightContent>
|
||||
</TextBox>
|
||||
|
||||
|
||||
<DataGrid
|
||||
AutoGenerateColumns="False"
|
||||
BorderBrush="Gray"
|
||||
BorderThickness="1"
|
||||
Grid.Column="0"
|
||||
Grid.ColumnSpan="2"
|
||||
Grid.Row="2"
|
||||
GridLinesVisibility="All"
|
||||
IsReadOnly="True"
|
||||
ItemsSource="{Binding MediaFiles}">
|
||||
<DataGrid.Columns>
|
||||
<DataGridCheckBoxColumn Binding="{Binding IsSelected}" />
|
||||
<DataGridTextColumn Binding="{Binding SourceDirectory}" Header="Source" />
|
||||
<DataGridTextColumn Binding="{Binding TargetDirectory}" Header="Target" />
|
||||
</DataGrid.Columns>
|
||||
</DataGrid>
|
||||
</Grid>
|
||||
</Window>
|
11
PhotoRenamerGui/MainWindow.axaml.cs
Normal file
11
PhotoRenamerGui/MainWindow.axaml.cs
Normal file
@ -0,0 +1,11 @@
|
||||
using Avalonia.Controls;
|
||||
|
||||
namespace PhotoRenamerGui;
|
||||
|
||||
public partial class MainWindow : Window
|
||||
{
|
||||
public MainWindow()
|
||||
{
|
||||
InitializeComponent();
|
||||
}
|
||||
}
|
78
PhotoRenamerGui/MainWindowViewModel.cs
Normal file
78
PhotoRenamerGui/MainWindowViewModel.cs
Normal file
@ -0,0 +1,78 @@
|
||||
using System;
|
||||
using System.Collections.ObjectModel;
|
||||
using System.IO;
|
||||
using System.Threading.Tasks;
|
||||
using Avalonia;
|
||||
using Avalonia.Controls.ApplicationLifetimes;
|
||||
using Avalonia.Platform.Storage;
|
||||
using CommunityToolkit.Mvvm.ComponentModel;
|
||||
using CommunityToolkit.Mvvm.Input;
|
||||
using PhotoRenamer;
|
||||
|
||||
namespace PhotoRenamerGui;
|
||||
|
||||
public partial class MainWindowViewModel : ObservableObject
|
||||
{
|
||||
[ObservableProperty] private string? _inputFolder;
|
||||
[ObservableProperty] private string? _outputFolder;
|
||||
public ObservableCollection<MediaFile> MediaFiles { get; } = new();
|
||||
|
||||
[RelayCommand]
|
||||
private async Task SelectInputFolderAsync()
|
||||
{
|
||||
InputFolder = await SearchFolderAsync();
|
||||
}
|
||||
|
||||
[RelayCommand]
|
||||
private async Task SelectOutputFolderAsync()
|
||||
{
|
||||
OutputFolder = await SearchFolderAsync();
|
||||
}
|
||||
|
||||
private static async Task<string?> SearchFolderAsync()
|
||||
{
|
||||
var storageProvider = (Application.Current?.ApplicationLifetime as IClassicDesktopStyleApplicationLifetime)
|
||||
?.MainWindow?
|
||||
.StorageProvider;
|
||||
if (storageProvider is null) return "<not accessible>";
|
||||
var result = await storageProvider.OpenFolderPickerAsync(new FolderPickerOpenOptions());
|
||||
return result.Count > 0 ? result[0].TryGetLocalPath() : null;
|
||||
}
|
||||
|
||||
partial void OnInputFolderChanged(string? value)
|
||||
{
|
||||
UpdatePreviewList();
|
||||
}
|
||||
|
||||
partial void OnOutputFolderChanged(string? value)
|
||||
{
|
||||
UpdatePreviewList();
|
||||
}
|
||||
|
||||
private void UpdatePreviewList()
|
||||
{
|
||||
MediaFiles.Clear();
|
||||
if (string.IsNullOrWhiteSpace(InputFolder)) return;
|
||||
if (string.IsNullOrWhiteSpace(OutputFolder)) return;
|
||||
var allFiles = FolderExtension.GetAllFiles(InputFolder);
|
||||
foreach (var file in allFiles)
|
||||
{
|
||||
var source = Path.GetRelativePath(InputFolder, file);
|
||||
var date = MediaFileExtension.GetDateTimeFromSourceFile(file);
|
||||
var target = FolderExtension.BuildTargetDirectoryByDate(OutputFolder, DateOnly.FromDateTime(date));
|
||||
|
||||
MediaFiles.Add(new MediaFile(true, source, Path.GetRelativePath(OutputFolder, target)));
|
||||
}
|
||||
|
||||
//TODO
|
||||
}
|
||||
}
|
||||
|
||||
public class MediaFile(bool isSelected, string? sourceDirectory, string? targetDirectory) : ObservableObject
|
||||
{
|
||||
public bool IsSelected { get=> isSelected;
|
||||
set => SetProperty(ref isSelected, value);
|
||||
}
|
||||
public string? SourceDirectory { get; } = sourceDirectory;
|
||||
public string? TargetDirectory { get; } = targetDirectory;
|
||||
}
|
27
PhotoRenamerGui/PhotoRenamerGui.csproj
Normal file
27
PhotoRenamerGui/PhotoRenamerGui.csproj
Normal file
@ -0,0 +1,27 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<OutputType>WinExe</OutputType>
|
||||
<TargetFramework>net8.0</TargetFramework>
|
||||
<Nullable>enable</Nullable>
|
||||
<BuiltInComInteropSupport>true</BuiltInComInteropSupport>
|
||||
<ApplicationManifest>app.manifest</ApplicationManifest>
|
||||
<AvaloniaUseCompiledBindingsByDefault>true</AvaloniaUseCompiledBindingsByDefault>
|
||||
</PropertyGroup>
|
||||
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Avalonia" Version="11.0.6" />
|
||||
<PackageReference Include="Avalonia.Controls.DataGrid" Version="11.0.6" />
|
||||
<PackageReference Include="Avalonia.Desktop" Version="11.0.6" />
|
||||
<PackageReference Include="Avalonia.Themes.Fluent" Version="11.0.6" />
|
||||
<PackageReference Include="Avalonia.Fonts.Inter" Version="11.0.6" />
|
||||
<!--Condition below is needed to remove Avalonia.Diagnostics package from build output in Release configuration.-->
|
||||
<PackageReference Condition="'$(Configuration)' == 'Debug'" Include="Avalonia.Diagnostics" Version="11.0.6" />
|
||||
<PackageReference Include="CommunityToolkit.Mvvm" Version="8.2.2" />
|
||||
</ItemGroup>
|
||||
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\PhotoRenamer\PhotoRenamer.csproj" />
|
||||
</ItemGroup>
|
||||
</Project>
|
21
PhotoRenamerGui/Program.cs
Normal file
21
PhotoRenamerGui/Program.cs
Normal file
@ -0,0 +1,21 @@
|
||||
using Avalonia;
|
||||
using System;
|
||||
|
||||
namespace PhotoRenamerGui;
|
||||
|
||||
class Program
|
||||
{
|
||||
// Initialization code. Don't use any Avalonia, third-party APIs or any
|
||||
// SynchronizationContext-reliant code before AppMain is called: things aren't initialized
|
||||
// yet and stuff might break.
|
||||
[STAThread]
|
||||
public static void Main(string[] args) => BuildAvaloniaApp()
|
||||
.StartWithClassicDesktopLifetime(args);
|
||||
|
||||
// Avalonia configuration, don't remove; also used by visual designer.
|
||||
public static AppBuilder BuildAvaloniaApp()
|
||||
=> AppBuilder.Configure<App>()
|
||||
.UsePlatformDetect()
|
||||
.WithInterFont()
|
||||
.LogToTrace();
|
||||
}
|
18
PhotoRenamerGui/app.manifest
Normal file
18
PhotoRenamerGui/app.manifest
Normal file
@ -0,0 +1,18 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<assembly manifestVersion="1.0" xmlns="urn:schemas-microsoft-com:asm.v1">
|
||||
<!-- This manifest is used on Windows only.
|
||||
Don't remove it as it might cause problems with window transparency and embeded controls.
|
||||
For more details visit https://learn.microsoft.com/en-us/windows/win32/sbscs/application-manifests -->
|
||||
<assemblyIdentity version="1.0.0.0" name="PhotoRenamerGui.Desktop"/>
|
||||
|
||||
<compatibility xmlns="urn:schemas-microsoft-com:compatibility.v1">
|
||||
<application>
|
||||
<!-- A list of the Windows versions that this application has been tested on
|
||||
and is designed to work with. Uncomment the appropriate elements
|
||||
and Windows will automatically select the most compatible environment. -->
|
||||
|
||||
<!-- Windows 10 -->
|
||||
<supportedOS Id="{8e0f7a12-bfb3-4fe8-b9a5-48fd50a15a9a}" />
|
||||
</application>
|
||||
</compatibility>
|
||||
</assembly>
|
Reference in New Issue
Block a user