Browse Source

QVCopier: UI implemention, Log and filter, Dest choose from explorer and drag-drop

HOME 3 years ago
parent
commit
a11861cb4e

+ 7 - 68
QVCopier/Components/TextProgressBar/TextProgressBarControl.cs

@@ -3,30 +3,14 @@ using System.ComponentModel;
 using System.Drawing;
 using System.Windows.Forms;
 
-namespace QVCopier.Components.TextProgressBar
+namespace QVCopier.Components.ProgressBarLabel
 {
-    public class TextProgressBarControl : ProgressBar
+    public class ProgressBarLabelControl : ProgressBar
     {
-        private TextProgressBarDisplayMode _visualMode = TextProgressBarDisplayMode.CurrProgress;
-
         private readonly StringFormat _stringFormat = new() { Alignment = StringAlignment.Center, LineAlignment = StringAlignment.Center };
 
-        private string _text = string.Empty;
         private ContentAlignment _textAlign = ContentAlignment.MiddleCenter;
 
-
-
-        [Category("Additional Options"), Browsable(true)]
-        public TextProgressBarDisplayMode VisualMode
-        {
-            get => _visualMode;
-            set
-            {
-                _visualMode = value;
-                Invalidate();//redraw component after change value from VS Properties section
-            }
-        }
-
         [DefaultValue(ContentAlignment.MiddleCenter), Browsable(true)]
         public virtual ContentAlignment TextAlign
         {
@@ -90,51 +74,10 @@ namespace QVCopier.Components.TextProgressBar
             }
         }
 
-        [Description("If it's empty, % will be shown"), Category("Additional Options"), Browsable(true), EditorBrowsable(EditorBrowsableState.Always)]
-        public string CustomText
-        {
-            get => _text;
-            set
-            {
-                _text = value;
-                Invalidate();//redraw component after change value from VS Properties section
-            }
-        }
-
-        private string TextToDraw
-        {
-            get
-            {
-                var text = CustomText;
-
-                switch (VisualMode)
-                {
-                    case TextProgressBarDisplayMode.Percentage:
-                        text = PercentageStr;
-                        break;
-
-                    case TextProgressBarDisplayMode.CurrProgress:
-                        text = CurrProgressStr;
-                        break;
-
-                    case TextProgressBarDisplayMode.TextAndCurrProgress:
-                        text = $"{CustomText}: {CurrProgressStr}";
-                        break;
-
-                    case TextProgressBarDisplayMode.TextAndPercentage:
-                        text = $"{CustomText}: {PercentageStr}";
-                        break;
-                }
-
-                return text;
-            }
-        }
-
-        private string PercentageStr => $"{(int)((float)Value - Minimum) / ((float)Maximum - Minimum) * 100 } %";
-
-        private string CurrProgressStr => $"{Value}/{Maximum}";
+        [Browsable(true)]
+        public override string Text { get => base.Text; set => base.Text = value; }
 
-        public TextProgressBarControl()
+        public ProgressBarLabelControl()
         {
             Value = Minimum;
             FixComponentBlinking();
@@ -163,12 +106,12 @@ namespace QVCopier.Components.TextProgressBar
                 g.FillRectangle(brush, progressBarRect);
             }
 
-            if (VisualMode != TextProgressBarDisplayMode.NoText)
+            if (false == string.IsNullOrWhiteSpace(Text))
             {
                 void DrawText(Color c)
                 {
                     using var br = new SolidBrush(c);
-                    g.DrawString(TextToDraw, Font, br, rect, _stringFormat);
+                    g.DrawString(Text, Font, br, rect, _stringFormat);
                 }
 
                 g.SetClip(rect);
@@ -178,11 +121,7 @@ namespace QVCopier.Components.TextProgressBar
                 g.SetClip(progressBarRect);
                 DrawText(BackColor);
                 g.ResetClip();
-
-
             }
         }
-
-
     }
 }

+ 3 - 0
QVCopier/Components/ProgressBarLabel/README.md

@@ -0,0 +1,3 @@
+# ProgressBarLabel
+
+Progress bar with text

QVCopier/Components/TextProgressBar/README.md → QVCopier/Components/ProgressBarLabel/README.original.md


+ 0 - 12
QVCopier/Components/TextProgressBar/TextProgressBarDisplayMode.cs

@@ -1,12 +0,0 @@
-namespace QVCopier.Components.TextProgressBar
-{
-    public enum TextProgressBarDisplayMode
-    {
-        NoText,
-        Percentage,
-        CurrProgress,
-        CustomText,
-        TextAndPercentage,
-        TextAndCurrProgress
-    }
-}

+ 15 - 0
QVCopier/Models/LogEntry.cs

@@ -0,0 +1,15 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace QVCopier.Models
+{
+    internal class LogEntry
+    {
+        public DateTime Time { get; set; } = DateTime.Now;
+        public LogLevel Level { get; set; }
+        public string Log { get; set; }
+    }
+}

+ 11 - 0
QVCopier/Models/LogLevel.cs

@@ -0,0 +1,11 @@
+namespace QVCopier.Models
+{
+    internal enum LogLevel
+    {
+        Debug,
+        Info,
+        Warning,
+        Error,
+        Fatal
+    }
+}

+ 53 - 0
QVCopier/Models/WorkItem.cs

@@ -0,0 +1,53 @@
+using QVCopier.Utility;
+using System;
+using System.Windows.Forms;
+
+namespace QVCopier.Models
+{
+    public class WorkItem
+    {
+        public string SourcePath { get; set; }
+        public string DestPath { get; set; }
+        public long Size { get; set; }
+        public WorkItemStatus Status { get; set; }
+        public string SourceChecksum { get; set; }
+        public string DestChecksum { get; set; }
+        public string DestOps { get; set; }
+
+        public WorkItemDestPolicy DestPolicy { get; set; }
+
+        private ListViewItem _lvi;
+
+        public void UpdateUi()
+        {
+            if (_lvi == null) return;
+            if (_lvi.ListView?.InvokeRequired == true)
+            {
+                _lvi.ListView.Invoke(new Action(UpdateListViewItemContent));
+            }
+            else
+            {
+                UpdateListViewItemContent();
+            }
+        }
+
+        private void UpdateListViewItemContent()
+        {
+            _lvi.SubItems.Clear();
+            _lvi.Text = SourcePath;
+            _lvi.SubItems.Add(DestPath);
+            _lvi.SubItems.Add(Size.ToFriendFileSize());
+            _lvi.SubItems.Add(Status.ToString());
+            _lvi.SubItems.Add($"{SourceChecksum ?? "-"} / {DestChecksum ?? "-"}");
+            _lvi.SubItems.Add(DestOps);
+        }
+
+        public static implicit operator ListViewItem(WorkItem item)
+        {
+            if (item._lvi != null) return item._lvi;
+            item._lvi = new ListViewItem();
+            item.UpdateListViewItemContent();
+            return item._lvi;
+        }
+    }
+}

+ 8 - 0
QVCopier/Models/WorkItemDestPolicy.cs

@@ -0,0 +1,8 @@
+namespace QVCopier.Models
+{
+    public enum WorkItemDestPolicy
+    {
+        DefaultSkipExist,
+        OverwriteExist,
+    }
+}

+ 16 - 0
QVCopier/Models/WorkItemStatus.cs

@@ -0,0 +1,16 @@
+namespace QVCopier.Models
+{
+    public enum WorkItemStatus
+    {
+        Pending,
+        Error,
+        Reading,
+        Read,
+        Writing,
+        Copied,
+        Skipped,
+        Verified,
+        Mismatched,
+        SourceDeleted,
+    }
+}

+ 0 - 70
QVCopier/Properties/Resources.Designer.cs

@@ -1,70 +0,0 @@
-//------------------------------------------------------------------------------
-// <auto-generated>
-//     此代码由工具生成。
-//     运行时版本: 4.0.30319.42000
-//
-//     对此文件的更改可能导致不正确的行为,如果
-//     重新生成代码,则所做更改将丢失。
-// </auto-generated>
-//------------------------------------------------------------------------------
-
-
-namespace QVCopier.Properties
-{
-    /// <summary>
-    ///   强类型资源类,用于查找本地化字符串等。
-    /// </summary>
-    // 此类是由 StronglyTypedResourceBuilder
-    // 类通过类似于 ResGen 或 Visual Studio 的工具自动生成的。
-    // 若要添加或删除成员,请编辑 .ResX 文件,然后重新运行 ResGen
-    // (以 /str 作为命令选项),或重新生成 VS 项目。
-    [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")]
-    [global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
-    [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
-    internal class Resources
-    {
-
-        private static global::System.Resources.ResourceManager resourceMan;
-
-        private static global::System.Globalization.CultureInfo resourceCulture;
-
-        [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
-        internal Resources()
-        {
-        }
-
-        /// <summary>
-        ///   返回此类使用的缓存 ResourceManager 实例。
-        /// </summary>
-        [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
-        internal static global::System.Resources.ResourceManager ResourceManager
-        {
-            get
-            {
-                if ((resourceMan == null))
-                {
-                    global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("QVCopier.Properties.Resources", typeof(Resources).Assembly);
-                    resourceMan = temp;
-                }
-                return resourceMan;
-            }
-        }
-
-        /// <summary>
-        ///   重写当前线程的 CurrentUICulture 属性,对
-        ///   使用此强类型资源类的所有资源查找执行重写。
-        /// </summary>
-        [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
-        internal static global::System.Globalization.CultureInfo Culture
-        {
-            get
-            {
-                return resourceCulture;
-            }
-            set
-            {
-                resourceCulture = value;
-            }
-        }
-    }
-}

+ 0 - 117
QVCopier/Properties/Resources.resx

@@ -1,117 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<root>
-  <!-- 
-    Microsoft ResX Schema 
-    
-    Version 2.0
-    
-    The primary goals of this format is to allow a simple XML format 
-    that is mostly human readable. The generation and parsing of the 
-    various data types are done through the TypeConverter classes 
-    associated with the data types.
-    
-    Example:
-    
-    ... ado.net/XML headers & schema ...
-    <resheader name="resmimetype">text/microsoft-resx</resheader>
-    <resheader name="version">2.0</resheader>
-    <resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
-    <resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
-    <data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
-    <data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
-    <data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
-        <value>[base64 mime encoded serialized .NET Framework object]</value>
-    </data>
-    <data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
-        <value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
-        <comment>This is a comment</comment>
-    </data>
-                
-    There are any number of "resheader" rows that contain simple 
-    name/value pairs.
-    
-    Each data row contains a name, and value. The row also contains a 
-    type or mimetype. Type corresponds to a .NET class that support 
-    text/value conversion through the TypeConverter architecture. 
-    Classes that don't support this are serialized and stored with the 
-    mimetype set.
-    
-    The mimetype is used for serialized objects, and tells the 
-    ResXResourceReader how to depersist the object. This is currently not 
-    extensible. For a given mimetype the value must be set accordingly:
-    
-    Note - application/x-microsoft.net.object.binary.base64 is the format 
-    that the ResXResourceWriter will generate, however the reader can 
-    read any of the formats listed below.
-    
-    mimetype: application/x-microsoft.net.object.binary.base64
-    value   : The object must be serialized with 
-            : System.Serialization.Formatters.Binary.BinaryFormatter
-            : and then encoded with base64 encoding.
-    
-    mimetype: application/x-microsoft.net.object.soap.base64
-    value   : The object must be serialized with 
-            : System.Runtime.Serialization.Formatters.Soap.SoapFormatter
-            : and then encoded with base64 encoding.
-
-    mimetype: application/x-microsoft.net.object.bytearray.base64
-    value   : The object must be serialized into a byte array 
-            : using a System.ComponentModel.TypeConverter
-            : and then encoded with base64 encoding.
-    -->
-  <xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
-    <xsd:element name="root" msdata:IsDataSet="true">
-      <xsd:complexType>
-        <xsd:choice maxOccurs="unbounded">
-          <xsd:element name="metadata">
-            <xsd:complexType>
-              <xsd:sequence>
-                <xsd:element name="value" type="xsd:string" minOccurs="0" />
-              </xsd:sequence>
-              <xsd:attribute name="name" type="xsd:string" />
-              <xsd:attribute name="type" type="xsd:string" />
-              <xsd:attribute name="mimetype" type="xsd:string" />
-            </xsd:complexType>
-          </xsd:element>
-          <xsd:element name="assembly">
-            <xsd:complexType>
-              <xsd:attribute name="alias" type="xsd:string" />
-              <xsd:attribute name="name" type="xsd:string" />
-            </xsd:complexType>
-          </xsd:element>
-          <xsd:element name="data">
-            <xsd:complexType>
-              <xsd:sequence>
-                <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
-                <xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
-              </xsd:sequence>
-              <xsd:attribute name="name" type="xsd:string" msdata:Ordinal="1" />
-              <xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
-              <xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
-            </xsd:complexType>
-          </xsd:element>
-          <xsd:element name="resheader">
-            <xsd:complexType>
-              <xsd:sequence>
-                <xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
-              </xsd:sequence>
-              <xsd:attribute name="name" type="xsd:string" use="required" />
-            </xsd:complexType>
-          </xsd:element>
-        </xsd:choice>
-      </xsd:complexType>
-    </xsd:element>
-  </xsd:schema>
-  <resheader name="resmimetype">
-    <value>text/microsoft-resx</value>
-  </resheader>
-  <resheader name="version">
-    <value>2.0</value>
-  </resheader>
-  <resheader name="reader">
-    <value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
-  </resheader>
-  <resheader name="writer">
-    <value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
-  </resheader>
-</root>

+ 0 - 29
QVCopier/Properties/Settings.Designer.cs

@@ -1,29 +0,0 @@
-//------------------------------------------------------------------------------
-// <auto-generated>
-//     This code was generated by a tool.
-//     Runtime Version:4.0.30319.42000
-//
-//     Changes to this file may cause incorrect behavior and will be lost if
-//     the code is regenerated.
-// </auto-generated>
-//------------------------------------------------------------------------------
-
-
-namespace QVCopier.Properties
-{
-    [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
-    [global::System.CodeDom.Compiler.GeneratedCodeAttribute("Microsoft.VisualStudio.Editors.SettingsDesigner.SettingsSingleFileGenerator", "11.0.0.0")]
-    internal sealed partial class Settings : global::System.Configuration.ApplicationSettingsBase
-    {
-
-        private static Settings defaultInstance = ((Settings)(global::System.Configuration.ApplicationSettingsBase.Synchronized(new Settings())));
-
-        public static Settings Default
-        {
-            get
-            {
-                return defaultInstance;
-            }
-        }
-    }
-}

+ 0 - 7
QVCopier/Properties/Settings.settings

@@ -1,7 +0,0 @@
-<?xml version='1.0' encoding='utf-8'?>
-<SettingsFile xmlns="http://schemas.microsoft.com/VisualStudio/2004/01/settings" CurrentProfile="(Default)">
-  <Profiles>
-    <Profile Name="(Default)" />
-  </Profiles>
-  <Settings />
-</SettingsFile>

+ 26 - 22
QVCopier/QVCopier.csproj

@@ -35,6 +35,9 @@
     <LangVersion>latest</LangVersion>
   </PropertyGroup>
   <ItemGroup>
+    <Reference Include="Crc32.NET, Version=1.0.0.0, Culture=neutral, PublicKeyToken=dc0b95cf99bf4e99, processorArchitecture=MSIL">
+      <HintPath>C:\NuGetLocalRepo\Crc32.NET.1.2.0\lib\net20\Crc32.NET.dll</HintPath>
+    </Reference>
     <Reference Include="System" />
     <Reference Include="System.Core" />
     <Reference Include="System.Xml.Linq" />
@@ -48,7 +51,19 @@
     <Reference Include="System.Xml" />
   </ItemGroup>
   <ItemGroup>
-    <Compile Include="Components\TextProgressBar\TextProgressBarDisplayMode.cs" />
+    <Compile Include="QvcMainForm.MainLV.cs">
+      <DependentUpon>QvcMainForm.cs</DependentUpon>
+      <SubType>Form</SubType>
+    </Compile>
+    <Compile Include="Models\WorkItem.cs" />
+    <Compile Include="Models\WorkItemDestPolicy.cs" />
+    <Compile Include="Models\WorkItemStatus.cs" />
+    <Compile Include="QvcMainForm.Logs.cs">
+      <DependentUpon>QvcMainForm.cs</DependentUpon>
+      <SubType>Form</SubType>
+    </Compile>
+    <Compile Include="Models\LogEntry.cs" />
+    <Compile Include="Models\LogLevel.cs" />
     <Compile Include="QvcMainForm.cs">
       <SubType>Form</SubType>
     </Compile>
@@ -57,32 +72,19 @@
     </Compile>
     <Compile Include="Program.cs" />
     <Compile Include="Properties\AssemblyInfo.cs" />
-    <Compile Include="Components\TextProgressBar\TextProgressBarControl.cs">
+    <Compile Include="Components\ProgressBarLabel\ProgressBarLabelControl.cs">
       <SubType>Component</SubType>
     </Compile>
     <Compile Include="Utility\ExplorerAccessor.cs" />
-    <EmbeddedResource Include="Properties\Resources.resx">
-      <Generator>ResXFileCodeGenerator</Generator>
-      <LastGenOutput>Resources.Designer.cs</LastGenOutput>
-      <SubType>Designer</SubType>
-    </EmbeddedResource>
-    <Compile Include="Properties\Resources.Designer.cs">
-      <AutoGen>True</AutoGen>
-      <DependentUpon>Resources.resx</DependentUpon>
-    </Compile>
+    <Compile Include="Utility\Logger.cs" />
+    <Compile Include="Utility\RunningChecksum.cs" />
+    <Compile Include="Utility\TextFormatter.cs" />
     <EmbeddedResource Include="QvcMainForm.resx">
       <DependentUpon>QvcMainForm.cs</DependentUpon>
     </EmbeddedResource>
-    <None Include="Components\TextProgressBar\README.md" />
-    <None Include="Properties\Settings.settings">
-      <Generator>SettingsSingleFileGenerator</Generator>
-      <LastGenOutput>Settings.Designer.cs</LastGenOutput>
-    </None>
-    <Compile Include="Properties\Settings.Designer.cs">
-      <AutoGen>True</AutoGen>
-      <DependentUpon>Settings.settings</DependentUpon>
-      <DesignTimeSharedInput>True</DesignTimeSharedInput>
-    </Compile>
+    <None Include="Components\ProgressBarLabel\README.md" />
+    <None Include="Components\ProgressBarLabel\README.original.md" />
+    <None Include="packages.config" />
     <None Include="Readme.md" />
   </ItemGroup>
   <ItemGroup>
@@ -99,6 +101,8 @@
       <EmbedInteropTypes>True</EmbedInteropTypes>
     </COMReference>
   </ItemGroup>
-  <ItemGroup />
+  <ItemGroup>
+    <Folder Include="Core\" />
+  </ItemGroup>
   <Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
 </Project>

+ 196 - 137
QVCopier/QvcMainForm.Designer.cs

@@ -1,5 +1,5 @@
 
-using QVCopier.Components.TextProgressBar;
+using QVCopier.Components.ProgressBarLabel;
 
 namespace QVCopier
 {
@@ -31,29 +31,29 @@ namespace QVCopier
         /// </summary>
         private void InitializeComponent()
         {
-            System.Windows.Forms.ListViewItem listViewItem9 = new System.Windows.Forms.ListViewItem(new string[] {
+            System.Windows.Forms.ListViewItem listViewItem1 = new System.Windows.Forms.ListViewItem(new string[] {
             "c:\\from\\ep01.mp4",
             "d:\\dest\\ep01.mp4",
             "1.23 GB",
             "Readed",
             "AEDFCA / -",
             ""}, -1);
-            System.Windows.Forms.ListViewItem listViewItem10 = new System.Windows.Forms.ListViewItem(new string[] {
+            System.Windows.Forms.ListViewItem listViewItem2 = new System.Windows.Forms.ListViewItem(new string[] {
             "c:\\from\\file.ext",
             "d:\\dest\\file.ext",
             "123.45 MB",
             "Pending",
             "-/-",
-            "File already exist: Override"}, -1);
-            System.Windows.Forms.ListViewItem listViewItem11 = new System.Windows.Forms.ListViewItem(new string[] {
+            "Skip by exist"}, -1);
+            System.Windows.Forms.ListViewItem listViewItem3 = new System.Windows.Forms.ListViewItem(new string[] {
             "c:\\from\\asdf.exe",
             "d:\\dest\\asdf.exe",
             "45.678 GB",
             "Pending",
             "-/-",
-            "File already exist: Skip"}, -1);
-            System.Windows.Forms.ListViewItem listViewItem12 = new System.Windows.Forms.ListViewItem(new string[] {
-            "2022-02-22 22:22:22",
+            "Overwrite exist"}, -1);
+            System.Windows.Forms.ListViewItem listViewItem4 = new System.Windows.Forms.ListViewItem(new string[] {
+            "2022-02-22 22:22:22.22",
             "Info",
             "Application started"}, -1);
             this.MainSplitContainer = new System.Windows.Forms.SplitContainer();
@@ -63,7 +63,7 @@ namespace QVCopier
             this.SizeColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
             this.StatusColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
             this.ChecksumColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
-            this.MoreInfoColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
+            this.DestOpsColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
             this.LogListView = new System.Windows.Forms.ListView();
             this.TimeColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
             this.LevelColumnHeader = ((System.Windows.Forms.ColumnHeader)(new System.Windows.Forms.ColumnHeader()));
@@ -76,12 +76,15 @@ namespace QVCopier
             this.LogFilterLabel = new System.Windows.Forms.Label();
             this.FatalCheckBox = new System.Windows.Forms.CheckBox();
             this.MainFilterTableLayoutPanel = new System.Windows.Forms.TableLayoutPanel();
-            this.MismatchRadioButton = new System.Windows.Forms.RadioButton();
             this.FilterLabel = new System.Windows.Forms.Label();
-            this.VerifiedRadioButton = new System.Windows.Forms.RadioButton();
-            this.CopiedRadioButton = new System.Windows.Forms.RadioButton();
             this.PendingRadioButton = new System.Windows.Forms.RadioButton();
             this.AllRadioButton = new System.Windows.Forms.RadioButton();
+            this.WorkingRadioButton = new System.Windows.Forms.RadioButton();
+            this.SourceDeletedRadioButton = new System.Windows.Forms.RadioButton();
+            this.MismatchRadioButton = new System.Windows.Forms.RadioButton();
+            this.VerifiedRadioButton = new System.Windows.Forms.RadioButton();
+            this.DoneRadioButton = new System.Windows.Forms.RadioButton();
+            this.ErrorRadioButton = new System.Windows.Forms.RadioButton();
             this.TopTabControl = new System.Windows.Forms.TabControl();
             this.BeforeStartTabPage = new System.Windows.Forms.TabPage();
             this.StartButton = new System.Windows.Forms.Button();
@@ -95,9 +98,9 @@ namespace QVCopier
             this.BlockSizeUpDown = new System.Windows.Forms.NumericUpDown();
             this.BlockSizeMbLabel = new System.Windows.Forms.Label();
             this.RunningStatusTabPage = new System.Windows.Forms.TabPage();
-            this.WriteProgressBar = new QVCopier.Components.TextProgressBar.TextProgressBarControl();
-            this.BufferProgressBar = new QVCopier.Components.TextProgressBar.TextProgressBarControl();
-            this.ReadProgressBar = new QVCopier.Components.TextProgressBar.TextProgressBarControl();
+            this.SourceProgressBarLabel = new QVCopier.Components.ProgressBarLabel.ProgressBarLabelControl();
+            this.BufferProgressBarLabel = new QVCopier.Components.ProgressBarLabel.ProgressBarLabelControl();
+            this.DestProgressBarLabel = new QVCopier.Components.ProgressBarLabel.ProgressBarLabelControl();
             this.PauseButton = new System.Windows.Forms.Button();
             this.ResumeButton = new System.Windows.Forms.Button();
             this.AbortButton = new System.Windows.Forms.Button();
@@ -130,43 +133,46 @@ namespace QVCopier
             // 
             this.MainSplitContainer.Panel2.Controls.Add(this.LogListView);
             this.MainSplitContainer.Panel2.Controls.Add(this.LogFilterTableLayoutPanel);
-            this.MainSplitContainer.Size = new System.Drawing.Size(809, 402);
+            this.MainSplitContainer.Size = new System.Drawing.Size(1143, 402);
             this.MainSplitContainer.SplitterDistance = 281;
             this.MainSplitContainer.TabIndex = 2;
             // 
             // MainListView
             // 
+            this.MainListView.AllowDrop = true;
             this.MainListView.Columns.AddRange(new System.Windows.Forms.ColumnHeader[] {
             this.SourceColumnHeader,
             this.DestColumnHeader,
             this.SizeColumnHeader,
             this.StatusColumnHeader,
             this.ChecksumColumnHeader,
-            this.MoreInfoColumnHeader});
+            this.DestOpsColumnHeader});
             this.MainListView.Dock = System.Windows.Forms.DockStyle.Fill;
             this.MainListView.FullRowSelect = true;
             this.MainListView.GridLines = true;
             this.MainListView.HideSelection = false;
             this.MainListView.Items.AddRange(new System.Windows.Forms.ListViewItem[] {
-            listViewItem9,
-            listViewItem10,
-            listViewItem11});
+            listViewItem1,
+            listViewItem2,
+            listViewItem3});
             this.MainListView.Location = new System.Drawing.Point(0, 0);
             this.MainListView.Name = "MainListView";
-            this.MainListView.Size = new System.Drawing.Size(809, 281);
+            this.MainListView.Size = new System.Drawing.Size(1143, 281);
             this.MainListView.TabIndex = 0;
             this.MainListView.UseCompatibleStateImageBehavior = false;
             this.MainListView.View = System.Windows.Forms.View.Details;
+            this.MainListView.DragDrop += new System.Windows.Forms.DragEventHandler(this.MainListView_DragDrop);
+            this.MainListView.DragEnter += new System.Windows.Forms.DragEventHandler(this.MainListView_DragEnter);
             // 
             // SourceColumnHeader
             // 
             this.SourceColumnHeader.Text = "Source";
-            this.SourceColumnHeader.Width = 155;
+            this.SourceColumnHeader.Width = 200;
             // 
             // DestColumnHeader
             // 
             this.DestColumnHeader.Text = "Dest.";
-            this.DestColumnHeader.Width = 134;
+            this.DestColumnHeader.Width = 200;
             // 
             // SizeColumnHeader
             // 
@@ -183,10 +189,10 @@ namespace QVCopier
             this.ChecksumColumnHeader.Text = "Checksum";
             this.ChecksumColumnHeader.Width = 119;
             // 
-            // MoreInfoColumnHeader
+            // DestOpsColumnHeader
             // 
-            this.MoreInfoColumnHeader.Text = "MoreInfo";
-            this.MoreInfoColumnHeader.Width = 203;
+            this.DestOpsColumnHeader.Text = "Dest.Ops";
+            this.DestOpsColumnHeader.Width = 250;
             // 
             // LogListView
             // 
@@ -199,10 +205,10 @@ namespace QVCopier
             this.LogListView.GridLines = true;
             this.LogListView.HideSelection = false;
             this.LogListView.Items.AddRange(new System.Windows.Forms.ListViewItem[] {
-            listViewItem12});
+            listViewItem4});
             this.LogListView.Location = new System.Drawing.Point(0, 29);
             this.LogListView.Name = "LogListView";
-            this.LogListView.Size = new System.Drawing.Size(809, 88);
+            this.LogListView.Size = new System.Drawing.Size(1143, 88);
             this.LogListView.TabIndex = 0;
             this.LogListView.UseCompatibleStateImageBehavior = false;
             this.LogListView.View = System.Windows.Forms.View.Details;
@@ -210,7 +216,7 @@ namespace QVCopier
             // TimeColumnHeader
             // 
             this.TimeColumnHeader.Text = "Time";
-            this.TimeColumnHeader.Width = 130;
+            this.TimeColumnHeader.Width = 140;
             // 
             // LevelColumnHeader
             // 
@@ -242,7 +248,7 @@ namespace QVCopier
             this.LogFilterTableLayoutPanel.Name = "LogFilterTableLayoutPanel";
             this.LogFilterTableLayoutPanel.RowCount = 1;
             this.LogFilterTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Percent, 100F));
-            this.LogFilterTableLayoutPanel.Size = new System.Drawing.Size(809, 29);
+            this.LogFilterTableLayoutPanel.Size = new System.Drawing.Size(1143, 29);
             this.LogFilterTableLayoutPanel.TabIndex = 8;
             // 
             // DebugCheckBox
@@ -252,12 +258,13 @@ namespace QVCopier
             this.DebugCheckBox.AutoSize = true;
             this.DebugCheckBox.Checked = true;
             this.DebugCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
-            this.DebugCheckBox.Location = new System.Drawing.Point(666, 3);
+            this.DebugCheckBox.Location = new System.Drawing.Point(934, 3);
             this.DebugCheckBox.Name = "DebugCheckBox";
             this.DebugCheckBox.Size = new System.Drawing.Size(57, 22);
             this.DebugCheckBox.TabIndex = 5;
             this.DebugCheckBox.Text = "Debug()";
             this.DebugCheckBox.UseVisualStyleBackColor = true;
+            this.DebugCheckBox.CheckedChanged += new System.EventHandler(this.LogFilterCheckBox_CheckedChanged);
             // 
             // InfoCheckBox
             // 
@@ -266,12 +273,13 @@ namespace QVCopier
             this.InfoCheckBox.AutoSize = true;
             this.InfoCheckBox.Checked = true;
             this.InfoCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
-            this.InfoCheckBox.Location = new System.Drawing.Point(521, 3);
+            this.InfoCheckBox.Location = new System.Drawing.Point(722, 3);
             this.InfoCheckBox.Name = "InfoCheckBox";
             this.InfoCheckBox.Size = new System.Drawing.Size(51, 22);
             this.InfoCheckBox.TabIndex = 4;
             this.InfoCheckBox.Text = "Info()";
             this.InfoCheckBox.UseVisualStyleBackColor = true;
+            this.InfoCheckBox.CheckedChanged += new System.EventHandler(this.LogFilterCheckBox_CheckedChanged);
             // 
             // WarningCheckBox
             // 
@@ -280,12 +288,13 @@ namespace QVCopier
             this.WarningCheckBox.AutoSize = true;
             this.WarningCheckBox.Checked = true;
             this.WarningCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
-            this.WarningCheckBox.Location = new System.Drawing.Point(376, 3);
+            this.WarningCheckBox.Location = new System.Drawing.Point(510, 3);
             this.WarningCheckBox.Name = "WarningCheckBox";
             this.WarningCheckBox.Size = new System.Drawing.Size(69, 22);
             this.WarningCheckBox.TabIndex = 3;
             this.WarningCheckBox.Text = "Warning()";
             this.WarningCheckBox.UseVisualStyleBackColor = true;
+            this.WarningCheckBox.CheckedChanged += new System.EventHandler(this.LogFilterCheckBox_CheckedChanged);
             // 
             // ErrorCheckBox
             // 
@@ -294,12 +303,13 @@ namespace QVCopier
             this.ErrorCheckBox.AutoSize = true;
             this.ErrorCheckBox.Checked = true;
             this.ErrorCheckBox.CheckState = System.Windows.Forms.CheckState.Checked;
-            this.ErrorCheckBox.Location = new System.Drawing.Point(231, 3);
+            this.ErrorCheckBox.Location = new System.Drawing.Point(298, 3);
             this.ErrorCheckBox.Name = "ErrorCheckBox";
             this.ErrorCheckBox.Size = new System.Drawing.Size(57, 22);
             this.ErrorCheckBox.TabIndex = 2;
             this.ErrorCheckBox.Text = "Error()";
             this.ErrorCheckBox.UseVisualStyleBackColor = true;
+            this.ErrorCheckBox.CheckedChanged += new System.EventHandler(this.LogFilterCheckBox_CheckedChanged);
             // 
             // LogFilterLabel
             // 
@@ -324,42 +334,37 @@ namespace QVCopier
             this.FatalCheckBox.TabIndex = 1;
             this.FatalCheckBox.Text = "Fatal()";
             this.FatalCheckBox.UseVisualStyleBackColor = true;
+            this.FatalCheckBox.CheckedChanged += new System.EventHandler(this.LogFilterCheckBox_CheckedChanged);
             // 
             // MainFilterTableLayoutPanel
             // 
-            this.MainFilterTableLayoutPanel.ColumnCount = 6;
-            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Absolute, 100F));
-            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 20F));
-            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 20F));
-            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 20F));
-            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 20F));
-            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 20F));
-            this.MainFilterTableLayoutPanel.Controls.Add(this.MismatchRadioButton, 5, 0);
+            this.MainFilterTableLayoutPanel.ColumnCount = 9;
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Absolute, 60F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
+            this.MainFilterTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 12.5F));
             this.MainFilterTableLayoutPanel.Controls.Add(this.FilterLabel, 0, 0);
-            this.MainFilterTableLayoutPanel.Controls.Add(this.VerifiedRadioButton, 4, 0);
-            this.MainFilterTableLayoutPanel.Controls.Add(this.CopiedRadioButton, 3, 0);
             this.MainFilterTableLayoutPanel.Controls.Add(this.PendingRadioButton, 2, 0);
             this.MainFilterTableLayoutPanel.Controls.Add(this.AllRadioButton, 1, 0);
+            this.MainFilterTableLayoutPanel.Controls.Add(this.WorkingRadioButton, 3, 0);
+            this.MainFilterTableLayoutPanel.Controls.Add(this.SourceDeletedRadioButton, 8, 0);
+            this.MainFilterTableLayoutPanel.Controls.Add(this.MismatchRadioButton, 7, 0);
+            this.MainFilterTableLayoutPanel.Controls.Add(this.VerifiedRadioButton, 6, 0);
+            this.MainFilterTableLayoutPanel.Controls.Add(this.DoneRadioButton, 5, 0);
+            this.MainFilterTableLayoutPanel.Controls.Add(this.ErrorRadioButton, 4, 0);
             this.MainFilterTableLayoutPanel.Dock = System.Windows.Forms.DockStyle.Top;
             this.MainFilterTableLayoutPanel.Location = new System.Drawing.Point(0, 112);
             this.MainFilterTableLayoutPanel.Name = "MainFilterTableLayoutPanel";
             this.MainFilterTableLayoutPanel.RowCount = 1;
             this.MainFilterTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Percent, 100F));
-            this.MainFilterTableLayoutPanel.Size = new System.Drawing.Size(809, 29);
+            this.MainFilterTableLayoutPanel.Size = new System.Drawing.Size(1143, 29);
             this.MainFilterTableLayoutPanel.TabIndex = 7;
             // 
-            // MismatchRadioButton
-            // 
-            this.MismatchRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
-            this.MismatchRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
-            this.MismatchRadioButton.AutoSize = true;
-            this.MismatchRadioButton.Location = new System.Drawing.Point(667, 3);
-            this.MismatchRadioButton.Name = "MismatchRadioButton";
-            this.MismatchRadioButton.Size = new System.Drawing.Size(81, 22);
-            this.MismatchRadioButton.TabIndex = 4;
-            this.MismatchRadioButton.Text = "Mismatch( )";
-            this.MismatchRadioButton.UseVisualStyleBackColor = true;
-            // 
             // FilterLabel
             // 
             this.FilterLabel.Anchor = System.Windows.Forms.AnchorStyles.Left;
@@ -370,36 +375,12 @@ namespace QVCopier
             this.FilterLabel.TabIndex = 0;
             this.FilterLabel.Text = "Filter:";
             // 
-            // VerifiedRadioButton
-            // 
-            this.VerifiedRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
-            this.VerifiedRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
-            this.VerifiedRadioButton.AutoSize = true;
-            this.VerifiedRadioButton.Location = new System.Drawing.Point(526, 3);
-            this.VerifiedRadioButton.Name = "VerifiedRadioButton";
-            this.VerifiedRadioButton.Size = new System.Drawing.Size(81, 22);
-            this.VerifiedRadioButton.TabIndex = 3;
-            this.VerifiedRadioButton.Text = "Verified( )";
-            this.VerifiedRadioButton.UseVisualStyleBackColor = true;
-            // 
-            // CopiedRadioButton
-            // 
-            this.CopiedRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
-            this.CopiedRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
-            this.CopiedRadioButton.AutoSize = true;
-            this.CopiedRadioButton.Location = new System.Drawing.Point(385, 3);
-            this.CopiedRadioButton.Name = "CopiedRadioButton";
-            this.CopiedRadioButton.Size = new System.Drawing.Size(69, 22);
-            this.CopiedRadioButton.TabIndex = 2;
-            this.CopiedRadioButton.Text = "Copied( )";
-            this.CopiedRadioButton.UseVisualStyleBackColor = true;
-            // 
             // PendingRadioButton
             // 
             this.PendingRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
             this.PendingRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
             this.PendingRadioButton.AutoSize = true;
-            this.PendingRadioButton.Location = new System.Drawing.Point(244, 3);
+            this.PendingRadioButton.Location = new System.Drawing.Point(198, 3);
             this.PendingRadioButton.Name = "PendingRadioButton";
             this.PendingRadioButton.Size = new System.Drawing.Size(75, 22);
             this.PendingRadioButton.TabIndex = 1;
@@ -412,7 +393,7 @@ namespace QVCopier
             this.AllRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
             this.AllRadioButton.AutoSize = true;
             this.AllRadioButton.Checked = true;
-            this.AllRadioButton.Location = new System.Drawing.Point(103, 3);
+            this.AllRadioButton.Location = new System.Drawing.Point(63, 3);
             this.AllRadioButton.Name = "AllRadioButton";
             this.AllRadioButton.Size = new System.Drawing.Size(99, 22);
             this.AllRadioButton.TabIndex = 0;
@@ -420,6 +401,78 @@ namespace QVCopier
             this.AllRadioButton.Text = "All(3, 123 GB)";
             this.AllRadioButton.UseVisualStyleBackColor = true;
             // 
+            // WorkingRadioButton
+            // 
+            this.WorkingRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
+            this.WorkingRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
+            this.WorkingRadioButton.AutoSize = true;
+            this.WorkingRadioButton.Location = new System.Drawing.Point(333, 3);
+            this.WorkingRadioButton.Name = "WorkingRadioButton";
+            this.WorkingRadioButton.Size = new System.Drawing.Size(75, 22);
+            this.WorkingRadioButton.TabIndex = 1;
+            this.WorkingRadioButton.Text = "Working( )";
+            this.WorkingRadioButton.UseVisualStyleBackColor = true;
+            // 
+            // SourceDeletedRadioButton
+            // 
+            this.SourceDeletedRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
+            this.SourceDeletedRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
+            this.SourceDeletedRadioButton.AutoSize = true;
+            this.SourceDeletedRadioButton.Location = new System.Drawing.Point(1008, 3);
+            this.SourceDeletedRadioButton.Name = "SourceDeletedRadioButton";
+            this.SourceDeletedRadioButton.Size = new System.Drawing.Size(81, 22);
+            this.SourceDeletedRadioButton.TabIndex = 4;
+            this.SourceDeletedRadioButton.Text = "Src.Del.( )";
+            this.SourceDeletedRadioButton.UseVisualStyleBackColor = true;
+            // 
+            // MismatchRadioButton
+            // 
+            this.MismatchRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
+            this.MismatchRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
+            this.MismatchRadioButton.AutoSize = true;
+            this.MismatchRadioButton.Location = new System.Drawing.Point(873, 3);
+            this.MismatchRadioButton.Name = "MismatchRadioButton";
+            this.MismatchRadioButton.Size = new System.Drawing.Size(81, 22);
+            this.MismatchRadioButton.TabIndex = 4;
+            this.MismatchRadioButton.Text = "Mismatch( )";
+            this.MismatchRadioButton.UseVisualStyleBackColor = true;
+            // 
+            // VerifiedRadioButton
+            // 
+            this.VerifiedRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
+            this.VerifiedRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
+            this.VerifiedRadioButton.AutoSize = true;
+            this.VerifiedRadioButton.Location = new System.Drawing.Point(738, 3);
+            this.VerifiedRadioButton.Name = "VerifiedRadioButton";
+            this.VerifiedRadioButton.Size = new System.Drawing.Size(81, 22);
+            this.VerifiedRadioButton.TabIndex = 3;
+            this.VerifiedRadioButton.Text = "Verified( )";
+            this.VerifiedRadioButton.UseVisualStyleBackColor = true;
+            // 
+            // DoneRadioButton
+            // 
+            this.DoneRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
+            this.DoneRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
+            this.DoneRadioButton.AutoSize = true;
+            this.DoneRadioButton.Location = new System.Drawing.Point(603, 3);
+            this.DoneRadioButton.Name = "DoneRadioButton";
+            this.DoneRadioButton.Size = new System.Drawing.Size(57, 22);
+            this.DoneRadioButton.TabIndex = 2;
+            this.DoneRadioButton.Text = "Done( )";
+            this.DoneRadioButton.UseVisualStyleBackColor = true;
+            // 
+            // ErrorRadioButton
+            // 
+            this.ErrorRadioButton.Anchor = System.Windows.Forms.AnchorStyles.Left;
+            this.ErrorRadioButton.Appearance = System.Windows.Forms.Appearance.Button;
+            this.ErrorRadioButton.AutoSize = true;
+            this.ErrorRadioButton.Location = new System.Drawing.Point(468, 3);
+            this.ErrorRadioButton.Name = "ErrorRadioButton";
+            this.ErrorRadioButton.Size = new System.Drawing.Size(63, 22);
+            this.ErrorRadioButton.TabIndex = 1;
+            this.ErrorRadioButton.Text = "Error( )";
+            this.ErrorRadioButton.UseVisualStyleBackColor = true;
+            // 
             // TopTabControl
             // 
             this.TopTabControl.Appearance = System.Windows.Forms.TabAppearance.Buttons;
@@ -429,7 +482,7 @@ namespace QVCopier
             this.TopTabControl.Location = new System.Drawing.Point(0, 0);
             this.TopTabControl.Name = "TopTabControl";
             this.TopTabControl.SelectedIndex = 0;
-            this.TopTabControl.Size = new System.Drawing.Size(809, 112);
+            this.TopTabControl.Size = new System.Drawing.Size(1143, 112);
             this.TopTabControl.TabIndex = 0;
             // 
             // BeforeStartTabPage
@@ -447,7 +500,7 @@ namespace QVCopier
             this.BeforeStartTabPage.Location = new System.Drawing.Point(4, 25);
             this.BeforeStartTabPage.Name = "BeforeStartTabPage";
             this.BeforeStartTabPage.Padding = new System.Windows.Forms.Padding(3);
-            this.BeforeStartTabPage.Size = new System.Drawing.Size(801, 83);
+            this.BeforeStartTabPage.Size = new System.Drawing.Size(1135, 83);
             this.BeforeStartTabPage.TabIndex = 0;
             this.BeforeStartTabPage.Text = "Before Start";
             this.BeforeStartTabPage.UseVisualStyleBackColor = true;
@@ -455,7 +508,7 @@ namespace QVCopier
             // StartButton
             // 
             this.StartButton.Anchor = ((System.Windows.Forms.AnchorStyles)((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Right)));
-            this.StartButton.Location = new System.Drawing.Point(720, 6);
+            this.StartButton.Location = new System.Drawing.Point(1054, 6);
             this.StartButton.Name = "StartButton";
             this.StartButton.Size = new System.Drawing.Size(75, 75);
             this.StartButton.TabIndex = 4;
@@ -492,12 +545,13 @@ namespace QVCopier
             // ChooseDestButton
             // 
             this.ChooseDestButton.Anchor = System.Windows.Forms.AnchorStyles.Top;
-            this.ChooseDestButton.Location = new System.Drawing.Point(365, 33);
+            this.ChooseDestButton.Location = new System.Drawing.Point(532, 33);
             this.ChooseDestButton.Name = "ChooseDestButton";
             this.ChooseDestButton.Size = new System.Drawing.Size(234, 23);
             this.ChooseDestButton.TabIndex = 3;
             this.ChooseDestButton.Text = "Choose dest from opened explorer";
             this.ChooseDestButton.UseVisualStyleBackColor = true;
+            this.ChooseDestButton.Click += new System.EventHandler(this.ChooseDestButton_Click);
             // 
             // BufferSizeMbLabel
             // 
@@ -510,14 +564,17 @@ namespace QVCopier
             // 
             // DestFolderTextBox
             // 
+            this.DestFolderTextBox.AllowDrop = true;
             this.DestFolderTextBox.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
             | System.Windows.Forms.AnchorStyles.Right)));
             this.DestFolderTextBox.Location = new System.Drawing.Point(94, 6);
             this.DestFolderTextBox.Name = "DestFolderTextBox";
             this.DestFolderTextBox.ReadOnly = true;
-            this.DestFolderTextBox.Size = new System.Drawing.Size(620, 21);
+            this.DestFolderTextBox.Size = new System.Drawing.Size(954, 21);
             this.DestFolderTextBox.TabIndex = 0;
             this.DestFolderTextBox.Text = "(drag)";
+            this.DestFolderTextBox.DragDrop += new System.Windows.Forms.DragEventHandler(this.DestFolderTextBox_DragDrop);
+            this.DestFolderTextBox.DragEnter += new System.Windows.Forms.DragEventHandler(this.DestFolderTextBox_DragEnter);
             // 
             // BufferSizeUpDown
             // 
@@ -574,65 +631,62 @@ namespace QVCopier
             // 
             // RunningStatusTabPage
             // 
-            this.RunningStatusTabPage.Controls.Add(this.WriteProgressBar);
-            this.RunningStatusTabPage.Controls.Add(this.BufferProgressBar);
-            this.RunningStatusTabPage.Controls.Add(this.ReadProgressBar);
+            this.RunningStatusTabPage.Controls.Add(this.SourceProgressBarLabel);
+            this.RunningStatusTabPage.Controls.Add(this.BufferProgressBarLabel);
+            this.RunningStatusTabPage.Controls.Add(this.DestProgressBarLabel);
             this.RunningStatusTabPage.Controls.Add(this.PauseButton);
             this.RunningStatusTabPage.Controls.Add(this.ResumeButton);
             this.RunningStatusTabPage.Controls.Add(this.AbortButton);
             this.RunningStatusTabPage.Location = new System.Drawing.Point(4, 25);
             this.RunningStatusTabPage.Name = "RunningStatusTabPage";
             this.RunningStatusTabPage.Padding = new System.Windows.Forms.Padding(3);
-            this.RunningStatusTabPage.Size = new System.Drawing.Size(801, 83);
+            this.RunningStatusTabPage.Size = new System.Drawing.Size(1135, 83);
             this.RunningStatusTabPage.TabIndex = 1;
             this.RunningStatusTabPage.Text = "Running Status";
             this.RunningStatusTabPage.UseVisualStyleBackColor = true;
             // 
-            // WriteProgressBar
+            // SourceProgressBarLabel
             // 
-            this.WriteProgressBar.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
+            this.SourceProgressBarLabel.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
             | System.Windows.Forms.AnchorStyles.Right)));
-            this.WriteProgressBar.CustomText = "Write: 0 of 3 Files,    Total  0 Bytes / 172.5 GB ?? MB/s";
-            this.WriteProgressBar.Location = new System.Drawing.Point(6, 59);
-            this.WriteProgressBar.Maximum = 3;
-            this.WriteProgressBar.Name = "WriteProgressBar";
-            this.WriteProgressBar.Size = new System.Drawing.Size(544, 21);
-            this.WriteProgressBar.TabIndex = 7;
-            this.WriteProgressBar.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
-            this.WriteProgressBar.VisualMode = QVCopier.Components.TextProgressBar.TextProgressBarDisplayMode.CustomText;
-            // 
-            // BufferProgressBar
-            // 
-            this.BufferProgressBar.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
+            this.SourceProgressBarLabel.Location = new System.Drawing.Point(6, 59);
+            this.SourceProgressBarLabel.Maximum = 3;
+            this.SourceProgressBarLabel.Name = "SourceProgressBarLabel";
+            this.SourceProgressBarLabel.Size = new System.Drawing.Size(878, 21);
+            this.SourceProgressBarLabel.TabIndex = 7;
+            this.SourceProgressBarLabel.Text = "Dest: 0 of 3 Files,    Total  0 Bytes / 172.5 GB ?? MB/s";
+            this.SourceProgressBarLabel.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+            // 
+            // BufferProgressBarLabel
+            // 
+            this.BufferProgressBarLabel.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
             | System.Windows.Forms.AnchorStyles.Right)));
-            this.BufferProgressBar.CustomText = "Buffer: 1.23 GB / 2 GB";
-            this.BufferProgressBar.Location = new System.Drawing.Point(6, 33);
-            this.BufferProgressBar.Maximum = 2048;
-            this.BufferProgressBar.Name = "BufferProgressBar";
-            this.BufferProgressBar.Size = new System.Drawing.Size(544, 21);
-            this.BufferProgressBar.TabIndex = 7;
-            this.BufferProgressBar.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
-            this.BufferProgressBar.Value = 1234;
-            this.BufferProgressBar.VisualMode = QVCopier.Components.TextProgressBar.TextProgressBarDisplayMode.CustomText;
-            // 
-            // ReadProgressBar
-            // 
-            this.ReadProgressBar.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
+            this.BufferProgressBarLabel.Location = new System.Drawing.Point(6, 33);
+            this.BufferProgressBarLabel.Maximum = 2048;
+            this.BufferProgressBarLabel.Name = "BufferProgressBarLabel";
+            this.BufferProgressBarLabel.Size = new System.Drawing.Size(878, 21);
+            this.BufferProgressBarLabel.TabIndex = 7;
+            this.BufferProgressBarLabel.Text = "Buffer: 1.23 GB / 2 GB";
+            this.BufferProgressBarLabel.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+            this.BufferProgressBarLabel.Value = 1234;
+            // 
+            // DestProgressBarLabel
+            // 
+            this.DestProgressBarLabel.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Left) 
             | System.Windows.Forms.AnchorStyles.Right)));
-            this.ReadProgressBar.CustomText = "Read: 1 of 3 Files,     Total 1.23 GB of 172.5 GB.  ?? MB/s";
-            this.ReadProgressBar.Location = new System.Drawing.Point(6, 6);
-            this.ReadProgressBar.Maximum = 3;
-            this.ReadProgressBar.Name = "ReadProgressBar";
-            this.ReadProgressBar.Size = new System.Drawing.Size(544, 21);
-            this.ReadProgressBar.TabIndex = 7;
-            this.ReadProgressBar.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
-            this.ReadProgressBar.Value = 1;
-            this.ReadProgressBar.VisualMode = QVCopier.Components.TextProgressBar.TextProgressBarDisplayMode.CustomText;
+            this.DestProgressBarLabel.Location = new System.Drawing.Point(6, 6);
+            this.DestProgressBarLabel.Maximum = 3;
+            this.DestProgressBarLabel.Name = "DestProgressBarLabel";
+            this.DestProgressBarLabel.Size = new System.Drawing.Size(878, 21);
+            this.DestProgressBarLabel.TabIndex = 7;
+            this.DestProgressBarLabel.Text = "Source: 1 of 3 Files,     Total 1.23 GB of 172.5 GB.  ?? MB/s";
+            this.DestProgressBarLabel.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;
+            this.DestProgressBarLabel.Value = 1;
             // 
             // PauseButton
             // 
             this.PauseButton.Anchor = ((System.Windows.Forms.AnchorStyles)((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Right)));
-            this.PauseButton.Location = new System.Drawing.Point(720, 6);
+            this.PauseButton.Location = new System.Drawing.Point(1054, 6);
             this.PauseButton.Name = "PauseButton";
             this.PauseButton.Size = new System.Drawing.Size(75, 75);
             this.PauseButton.TabIndex = 2;
@@ -642,7 +696,7 @@ namespace QVCopier
             // ResumeButton
             // 
             this.ResumeButton.Anchor = ((System.Windows.Forms.AnchorStyles)((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Right)));
-            this.ResumeButton.Location = new System.Drawing.Point(639, 6);
+            this.ResumeButton.Location = new System.Drawing.Point(973, 6);
             this.ResumeButton.Name = "ResumeButton";
             this.ResumeButton.Size = new System.Drawing.Size(75, 75);
             this.ResumeButton.TabIndex = 1;
@@ -652,7 +706,7 @@ namespace QVCopier
             // AbortButton
             // 
             this.AbortButton.Anchor = ((System.Windows.Forms.AnchorStyles)((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Right)));
-            this.AbortButton.Location = new System.Drawing.Point(558, 6);
+            this.AbortButton.Location = new System.Drawing.Point(892, 6);
             this.AbortButton.Name = "AbortButton";
             this.AbortButton.Size = new System.Drawing.Size(75, 75);
             this.AbortButton.TabIndex = 0;
@@ -663,12 +717,14 @@ namespace QVCopier
             // 
             this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 12F);
             this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font;
-            this.ClientSize = new System.Drawing.Size(809, 543);
+            this.ClientSize = new System.Drawing.Size(1143, 543);
             this.Controls.Add(this.MainSplitContainer);
             this.Controls.Add(this.MainFilterTableLayoutPanel);
             this.Controls.Add(this.TopTabControl);
             this.Name = "QvcMainForm";
             this.Text = "QVCopier";
+            this.Load += new System.EventHandler(this.QvcMainForm_Load);
+            this.Shown += new System.EventHandler(this.QvcMainForm_Shown);
             this.MainSplitContainer.Panel1.ResumeLayout(false);
             this.MainSplitContainer.Panel2.ResumeLayout(false);
             ((System.ComponentModel.ISupportInitialize)(this.MainSplitContainer)).EndInit();
@@ -694,7 +750,7 @@ namespace QVCopier
         private System.Windows.Forms.ColumnHeader LevelColumnHeader;
         private System.Windows.Forms.RadioButton MismatchRadioButton;
         private System.Windows.Forms.RadioButton VerifiedRadioButton;
-        private System.Windows.Forms.RadioButton CopiedRadioButton;
+        private System.Windows.Forms.RadioButton DoneRadioButton;
         private System.Windows.Forms.RadioButton PendingRadioButton;
         private System.Windows.Forms.RadioButton AllRadioButton;
         private System.Windows.Forms.Label FilterLabel;
@@ -717,13 +773,13 @@ namespace QVCopier
         private System.Windows.Forms.ColumnHeader SizeColumnHeader;
         private System.Windows.Forms.ColumnHeader StatusColumnHeader;
         private System.Windows.Forms.ColumnHeader ChecksumColumnHeader;
-        private System.Windows.Forms.ColumnHeader MoreInfoColumnHeader;
+        private System.Windows.Forms.ColumnHeader DestOpsColumnHeader;
         private System.Windows.Forms.Button PauseButton;
         private System.Windows.Forms.Button ResumeButton;
         private System.Windows.Forms.Button AbortButton;
-        private TextProgressBarControl ReadProgressBar;
-        private TextProgressBarControl WriteProgressBar;
-        private TextProgressBarControl BufferProgressBar;
+        private ProgressBarLabelControl DestProgressBarLabel;
+        private ProgressBarLabelControl SourceProgressBarLabel;
+        private ProgressBarLabelControl BufferProgressBarLabel;
         private System.Windows.Forms.Button StartButton;
         private System.Windows.Forms.ColumnHeader LogColumnHeader;
         private System.Windows.Forms.TableLayoutPanel LogFilterTableLayoutPanel;
@@ -733,6 +789,9 @@ namespace QVCopier
         private System.Windows.Forms.CheckBox WarningCheckBox;
         private System.Windows.Forms.CheckBox ErrorCheckBox;
         private System.Windows.Forms.CheckBox FatalCheckBox;
+        private System.Windows.Forms.RadioButton WorkingRadioButton;
+        private System.Windows.Forms.RadioButton SourceDeletedRadioButton;
+        private System.Windows.Forms.RadioButton ErrorRadioButton;
     }
 }
 

+ 102 - 0
QVCopier/QvcMainForm.Logs.cs

@@ -0,0 +1,102 @@
+using QVCopier.Models;
+using System;
+using System.Linq;
+using System.Windows.Forms;
+using QVCopier.Utility;
+
+namespace QVCopier
+{
+    public partial class QvcMainForm : Form
+    {
+        private void InitLogger()
+        {
+            UpdateLogStat();
+            Logger.LogAdded += Logger_LogAdded;
+            Logger.Info("Application started.");
+        }
+
+
+        private void Logger_LogAdded(object sender, LogEntry e)
+        {
+            if (InvokeRequired)
+            {
+                Invoke(new Action<LogEntry>(AddLog), e);
+            }
+            else
+            {
+                AddLog(e);
+            }
+        }
+
+        private void AddLog(LogEntry e)
+        {
+            _logs.Add(e);
+            UpdateLogStat();
+            switch (e.Level)
+            {
+                case LogLevel.Debug:
+                    if (DebugCheckBox.Checked) LogListView.Items.Add(CreateLogListViewItem(e));
+                    break;
+
+                case LogLevel.Info:
+                    if (InfoCheckBox.Checked) LogListView.Items.Add(CreateLogListViewItem(e));
+                    break;
+
+                case LogLevel.Warning:
+                    if (WarningCheckBox.Checked) LogListView.Items.Add(CreateLogListViewItem(e));
+                    break;
+
+                case LogLevel.Error:
+                    if (ErrorCheckBox.Checked) LogListView.Items.Add(CreateLogListViewItem(e));
+                    break;
+
+                case LogLevel.Fatal:
+                    if (FatalCheckBox.Checked) LogListView.Items.Add(CreateLogListViewItem(e));
+                    break;
+
+                default:
+                    throw new ArgumentOutOfRangeException();
+            }
+        }
+
+        private void UpdateLogStat()
+        {
+            FatalCheckBox.Text = $"Fatal({_logs.Count(p => p.Level == LogLevel.Fatal)})";
+            ErrorCheckBox.Text = $"Error({_logs.Count(p => p.Level == LogLevel.Error)})";
+            WarningCheckBox.Text = $"Warning({_logs.Count(p => p.Level == LogLevel.Warning)})";
+            InfoCheckBox.Text = $"Info({_logs.Count(p => p.Level == LogLevel.Info)})";
+            DebugCheckBox.Text = $"Debug({_logs.Count(p => p.Level == LogLevel.Debug)})";
+        }
+
+        private ListViewItem CreateLogListViewItem(LogEntry logEntry)
+        {
+            return new()
+            {
+                Text = logEntry.Time.ToString("yyyy-MM-dd HH:mm:ss.ff"),
+                SubItems =
+                {
+                    logEntry.Level.ToString(),
+                    logEntry.Log
+                }
+            };
+        }
+
+        private void LogFilterCheckBox_CheckedChanged(object sender, EventArgs e)
+        {
+            LogListView.Items.Clear();
+            LogListView.Items.AddRange(_logs.Where(p =>
+            {
+                switch (p.Level)
+                {
+                    case LogLevel.Debug: return DebugCheckBox.Checked;
+                    case LogLevel.Info: return InfoCheckBox.Checked;
+                    case LogLevel.Warning: return WarningCheckBox.Checked;
+                    case LogLevel.Error: return ErrorCheckBox.Checked;
+                    case LogLevel.Fatal: return FatalCheckBox.Checked;
+                    default:
+                        throw new ArgumentOutOfRangeException();
+                }
+            }).Select(CreateLogListViewItem).ToArray());
+        }
+    }
+}

+ 20 - 0
QVCopier/QvcMainForm.MainLV.cs

@@ -0,0 +1,20 @@
+using QVCopier.Models;
+using System;
+using System.Collections.Generic;
+using System.Windows.Forms;
+
+namespace QVCopier
+{
+    public partial class QvcMainForm : Form
+    {
+        private void MainListView_DragEnter(object sender, DragEventArgs e)
+        {
+
+        }
+
+        private void MainListView_DragDrop(object sender, DragEventArgs e)
+        {
+
+        }
+    }
+}

+ 68 - 4
QVCopier/QvcMainForm.cs

@@ -1,20 +1,84 @@
-using System;
+using QVCopier.Models;
+using System;
 using System.Collections.Generic;
-using System.ComponentModel;
-using System.Data;
 using System.Drawing;
+using System.IO;
 using System.Linq;
-using System.Text;
 using System.Threading.Tasks;
 using System.Windows.Forms;
+using QVCopier.Utility;
 
 namespace QVCopier
 {
     public partial class QvcMainForm : Form
     {
+        private readonly List<LogEntry> _logs = new();
+        private readonly List<WorkItem> _items = new();
+
         public QvcMainForm()
         {
             InitializeComponent();
         }
+
+        private void QvcMainForm_Load(object sender, EventArgs e)
+        {
+            MainListView.Items.Clear();
+            LogListView.Items.Clear();
+
+            InitLogger();
+        }
+
+        private void QvcMainForm_Shown(object sender, EventArgs e)
+        {
+        }
+
+        private void DestFolderTextBox_DragEnter(object sender, DragEventArgs e)
+        {
+            string[] data;
+            e.Effect = e.Data.GetDataPresent(DataFormats.FileDrop)
+                       && (data = (string[])e.Data.GetData(DataFormats.FileDrop)).Length == 1
+                       && Directory.Exists(data[0])
+                ? DragDropEffects.Link
+                : DragDropEffects.None;
+        }
+
+        private void DestFolderTextBox_DragDrop(object sender, DragEventArgs e)
+        {
+            DestFolderTextBox.Text = ((string[])e.Data.GetData(DataFormats.FileDrop))[0];
+        }
+
+        private void ChooseDestButton_Click(object sender, EventArgs e)
+        {
+            var cm = new ContextMenuStrip();
+            var msgItem = cm.Items.Add("Looking up...");
+            msgItem.Enabled = false;
+            cm.Items.Add("-").Enabled = false; 
+
+            Task.Run(() =>
+            {
+                var paths = ExplorerAccessor.GetOpenedWindowPath();
+                if (paths.Length == 0)
+                {
+                    cm.Invoke(new Action(() => msgItem.Text = "No explorer window found"));
+                }
+                else
+                {
+                    cm.Invoke(new Action(() => msgItem.Text = $"{paths.Length} found"));
+
+                    foreach (var path in paths)
+                    {
+                        cm.Invoke(new Action(() =>
+                        {
+                            var item = new ToolStripMenuItem(path);
+                            item.Click += delegate { DestFolderTextBox.Text = path; };
+                            cm.Items.Add(item);
+                        }));
+
+                    }
+                }
+            });
+            cm.Show(ChooseDestButton, new Point(0, ChooseDestButton.Height));
+
+        }
     }
 }

+ 5 - 1
QVCopier/Utility/ExplorerAccessor.cs

@@ -9,7 +9,11 @@ namespace QVCopier.Utility
     {
         public static string[] GetOpenedWindowPath()
         {
-            var paths = new ShellWindows().Cast<InternetExplorer>().Select(p => Path.GetFullPath(new Uri(p.LocationURL).AbsolutePath)).ToArray();
+            var paths = new ShellWindows().Cast<InternetExplorer>()
+                .Where(p => p.FullName?.ToLower().EndsWith("explorer.exe") == true && false == string.IsNullOrWhiteSpace(p.LocationURL))
+                .Select(p => Path.GetFullPath(Uri.UnescapeDataString(new Uri(p.LocationURL).AbsolutePath)))
+                .Distinct()
+                .ToArray();
             return paths;
         }
     }

+ 25 - 0
QVCopier/Utility/Logger.cs

@@ -0,0 +1,25 @@
+using QVCopier.Models;
+using System;
+
+namespace QVCopier.Utility
+{
+    internal static class Logger
+    {
+        public static void Debug(string log) => OnLogAdded(new LogEntry { Level = LogLevel.Debug, Log = log });
+
+        public static void Info(string log) => OnLogAdded(new LogEntry { Level = LogLevel.Info, Log = log });
+
+        public static void Warning(string log) => OnLogAdded(new LogEntry { Level = LogLevel.Warning, Log = log });
+
+        public static void Error(string log) => OnLogAdded(new LogEntry { Level = LogLevel.Error, Log = log });
+
+        public static void Fatal(string log) => OnLogAdded(new LogEntry { Level = LogLevel.Fatal, Log = log });
+
+        public static event EventHandler<LogEntry> LogAdded;
+
+        private static void OnLogAdded(LogEntry e)
+        {
+            LogAdded?.Invoke(null, e);
+        }
+    }
+}

+ 11 - 0
QVCopier/Utility/RunningChecksum.cs

@@ -0,0 +1,11 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+using Force.Crc32;
+
+namespace QVCopier.Utility
+{
+   
+}

+ 30 - 0
QVCopier/Utility/TextFormatter.cs

@@ -0,0 +1,30 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace QVCopier.Utility
+{
+    public static class TextFormatter
+    {
+        public static string ToFriendFileSize(this long value)
+        {
+            string[] sizes = { "B", "KB", "MB", "GB", "TB" };
+            double len = value;
+            var order = 0;
+
+            while (len >= 1024 && order < sizes.Length - 1)
+            {
+                order++;
+                len = len / 1024;
+            }
+
+            // Adjust the format string to your preferences. For example "{0:0.#}{1}" would
+            // show a single decimal place, and no space.
+            var result = $"{len:0.##} {sizes[order]}";
+
+            return result;
+        }
+    }
+}

+ 4 - 0
QVCopier/packages.config

@@ -0,0 +1,4 @@
+<?xml version="1.0" encoding="utf-8"?>
+<packages>
+  <package id="Crc32.NET" version="1.2.0" targetFramework="net461" />
+</packages>