using System;
using System.IO;
using ICSharpCode.SharpZipLib.Core;
namespace ICSharpCode.SharpZipLib.Zip
{
	/// 
	/// Basic implementation of 
	/// 
	public class ZipEntryFactory : IEntryFactory
	{
		#region Enumerations
		/// 
		/// Defines the possible values to be used for the .
		/// 
		public enum TimeSetting
		{
			/// 
			/// Use the recorded LastWriteTime value for the file.
			/// 
			LastWriteTime,
			/// 
			/// Use the recorded LastWriteTimeUtc value for the file
			/// 
			LastWriteTimeUtc,
			/// 
			/// Use the recorded CreateTime value for the file.
			/// 
			CreateTime,
			/// 
			/// Use the recorded CreateTimeUtc value for the file.
			/// 
			CreateTimeUtc,
			/// 
			/// Use the recorded LastAccessTime value for the file.
			/// 
			LastAccessTime,
			/// 
			/// Use the recorded LastAccessTimeUtc value for the file.
			/// 
			LastAccessTimeUtc,
			/// 
			/// Use a fixed value.
			/// 
			/// The actual  value used can be
			/// specified via the  constructor or
			/// using the  with the setting set
			/// to  which will use the  when this class was constructed.
			/// The  property can also be used to set this value.
			Fixed,
		}
		#endregion
		#region Constructors
		/// 
		/// Initialise a new instance of the  class.
		/// 
		/// A default , and the LastWriteTime for files is used.
		public ZipEntryFactory()
		{
			nameTransform_ = new ZipNameTransform();
		}
		/// 
		/// Initialise a new instance of  using the specified 
		/// 
		/// The time setting to use when creating Zip entries.
		public ZipEntryFactory(TimeSetting timeSetting)
		{
			timeSetting_ = timeSetting;
			nameTransform_ = new ZipNameTransform();
		}
		/// 
		/// Initialise a new instance of  using the specified 
		/// 
		/// The time to set all  values to.
		public ZipEntryFactory(DateTime time)
		{
			timeSetting_ = TimeSetting.Fixed;
			FixedDateTime = time;
			nameTransform_ = new ZipNameTransform();
		}
		#endregion
		#region Properties
		/// 
		/// Get / set the  to be used when creating new  values.
		/// 
		/// 
		/// Setting this property to null will cause a default name transform to be used.
		/// 
		public INameTransform NameTransform {
			get { return nameTransform_; }
			set {
				if (value == null) {
					nameTransform_ = new ZipNameTransform();
				} else {
					nameTransform_ = value;
				}
			}
		}
		/// 
		/// Get / set the  in use.
		/// 
		public TimeSetting Setting {
			get { return timeSetting_; }
			set { timeSetting_ = value; }
		}
		/// 
		/// Get / set the  value to use when  is set to 
		/// 
		public DateTime FixedDateTime {
			get { return fixedDateTime_; }
			set {
				if (value.Year < 1970) {
					throw new ArgumentException("Value is too old to be valid", "nameof(value)");
				}
				fixedDateTime_ = value;
			}
		}
		/// 
		/// A bitmask defining the attributes to be retrieved from the actual file.
		/// 
		/// The default is to get all possible attributes from the actual file.
		public int GetAttributes {
			get { return getAttributes_; }
			set { getAttributes_ = value; }
		}
		/// 
		/// A bitmask defining which attributes are to be set on.
		/// 
		/// By default no attributes are set on.
		public int SetAttributes {
			get { return setAttributes_; }
			set { setAttributes_ = value; }
		}
		/// 
		/// Get set a value indicating wether unidoce text should be set on.
		/// 
		public bool IsUnicodeText {
			get { return isUnicodeText_; }
			set { isUnicodeText_ = value; }
		}
		#endregion
		#region IEntryFactory Members
		/// 
		/// Make a new  for a file.
		/// 
		/// The name of the file to create a new entry for.
		/// Returns a new  based on the .
		public ZipEntry MakeFileEntry(string fileName)
		{
			return MakeFileEntry(fileName, null, true);
		}
		/// 
		/// Make a new  for a file.
		/// 
		/// The name of the file to create a new entry for.
		/// If true entry detail is retrieved from the file system if the file exists.
		/// Returns a new  based on the .
		public ZipEntry MakeFileEntry(string fileName, bool useFileSystem)
		{
			return MakeFileEntry(fileName, null, useFileSystem);
		}
		/// 
		/// Make a new  from a name.
		/// 
		/// The name of the file to create a new entry for.
		/// An alternative name to be used for the new entry. Null if not applicable.
		/// If true entry detail is retrieved from the file system if the file exists.
		/// Returns a new  based on the .
		public ZipEntry MakeFileEntry(string fileName, string entryName, bool useFileSystem)
		{
			var result = new ZipEntry(nameTransform_.TransformFile(!string.IsNullOrEmpty(entryName) ? entryName : fileName));
			result.IsUnicodeText = isUnicodeText_;
			int externalAttributes = 0;
			bool useAttributes = (setAttributes_ != 0);
			FileInfo fi = null;
			if (useFileSystem) {
				fi = new FileInfo(fileName);
			}
			if ((fi != null) && fi.Exists) {
				switch (timeSetting_) {
					case TimeSetting.CreateTime:
						result.DateTime = fi.CreationTime;
						break;
					case TimeSetting.CreateTimeUtc:
						result.DateTime = fi.CreationTimeUtc;
						break;
					case TimeSetting.LastAccessTime:
						result.DateTime = fi.LastAccessTime;
						break;
					case TimeSetting.LastAccessTimeUtc:
						result.DateTime = fi.LastAccessTimeUtc;
						break;
					case TimeSetting.LastWriteTime:
						result.DateTime = fi.LastWriteTime;
						break;
					case TimeSetting.LastWriteTimeUtc:
						result.DateTime = fi.LastWriteTimeUtc;
						break;
					case TimeSetting.Fixed:
						result.DateTime = fixedDateTime_;
						break;
					default:
						throw new ZipException("Unhandled time setting in MakeFileEntry");
				}
				result.Size = fi.Length;
				useAttributes = true;
				externalAttributes = ((int)fi.Attributes & getAttributes_);
			} else {
				if (timeSetting_ == TimeSetting.Fixed) {
					result.DateTime = fixedDateTime_;
				}
			}
			if (useAttributes) {
				externalAttributes |= setAttributes_;
				result.ExternalFileAttributes = externalAttributes;
			}
			return result;
		}
		/// 
		/// Make a new  for a directory.
		/// 
		/// The raw untransformed name for the new directory
		/// Returns a new  representing a directory.
		public ZipEntry MakeDirectoryEntry(string directoryName)
		{
			return MakeDirectoryEntry(directoryName, true);
		}
		/// 
		/// Make a new  for a directory.
		/// 
		/// The raw untransformed name for the new directory
		/// If true entry detail is retrieved from the file system if the file exists.
		/// Returns a new  representing a directory.
		public ZipEntry MakeDirectoryEntry(string directoryName, bool useFileSystem)
		{
			var result = new ZipEntry(nameTransform_.TransformDirectory(directoryName));
			result.IsUnicodeText = isUnicodeText_;
			result.Size = 0;
			int externalAttributes = 0;
			DirectoryInfo di = null;
			if (useFileSystem) {
				di = new DirectoryInfo(directoryName);
			}
			if ((di != null) && di.Exists) {
				switch (timeSetting_) {
					case TimeSetting.CreateTime:
						result.DateTime = di.CreationTime;
						break;
					case TimeSetting.CreateTimeUtc:
						result.DateTime = di.CreationTimeUtc;
						break;
					case TimeSetting.LastAccessTime:
						result.DateTime = di.LastAccessTime;
						break;
					case TimeSetting.LastAccessTimeUtc:
						result.DateTime = di.LastAccessTimeUtc;
						break;
					case TimeSetting.LastWriteTime:
						result.DateTime = di.LastWriteTime;
						break;
					case TimeSetting.LastWriteTimeUtc:
						result.DateTime = di.LastWriteTimeUtc;
						break;
					case TimeSetting.Fixed:
						result.DateTime = fixedDateTime_;
						break;
					default:
						throw new ZipException("Unhandled time setting in MakeDirectoryEntry");
				}
				externalAttributes = ((int)di.Attributes & getAttributes_);
			} else {
				if (timeSetting_ == TimeSetting.Fixed) {
					result.DateTime = fixedDateTime_;
				}
			}
			// Always set directory attribute on.
			externalAttributes |= (setAttributes_ | 16);
			result.ExternalFileAttributes = externalAttributes;
			return result;
		}
		#endregion
		#region Instance Fields
		INameTransform nameTransform_;
		DateTime fixedDateTime_ = DateTime.Now;
		TimeSetting timeSetting_;
		bool isUnicodeText_;
		int getAttributes_ = -1;
		int setAttributes_;
		#endregion
	}
}