2017-02-26 19:23:31 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
|
|
|
using System.Text;
|
|
|
|
using System.Threading;
|
2018-10-20 00:23:43 +00:00
|
|
|
using System.Windows.Forms;
|
2020-05-15 08:52:16 +00:00
|
|
|
using System.Diagnostics;
|
2017-02-26 19:23:31 +00:00
|
|
|
using WindowsDisplayAPI.DisplayConfig;
|
2020-04-23 08:16:16 +00:00
|
|
|
using HeliosPlus.Shared.Resources;
|
2018-10-20 00:13:40 +00:00
|
|
|
using Newtonsoft.Json;
|
2017-02-26 19:23:31 +00:00
|
|
|
using NvAPIWrapper.Mosaic;
|
|
|
|
using NvAPIWrapper.Native.Mosaic;
|
2020-05-09 13:02:07 +00:00
|
|
|
using HeliosPlus.Shared.Topology;
|
|
|
|
using System.Drawing;
|
|
|
|
using System.Drawing.Imaging;
|
2020-05-12 10:46:23 +00:00
|
|
|
using WindowsDisplayAPI;
|
2020-06-14 04:20:52 +00:00
|
|
|
using System.Text.RegularExpressions;
|
2017-02-26 19:23:31 +00:00
|
|
|
|
2020-04-23 08:16:16 +00:00
|
|
|
namespace HeliosPlus.Shared
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
2020-06-07 08:48:45 +00:00
|
|
|
public class ProfileItem
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
2020-06-07 08:48:45 +00:00
|
|
|
private static List<ProfileItem> _allSavedProfiles = new List<ProfileItem>();
|
2020-05-09 13:02:07 +00:00
|
|
|
private ProfileIcon _profileIcon;
|
2020-06-07 08:48:45 +00:00
|
|
|
private Bitmap _profileBitmap, _profileShortcutBitmap;
|
2020-05-12 10:46:23 +00:00
|
|
|
private static List<Display> _availableDisplays;
|
2020-05-13 11:04:18 +00:00
|
|
|
private static List<UnAttachedDisplay> _unavailableDisplays;
|
2017-02-26 19:23:31 +00:00
|
|
|
|
2020-05-09 13:02:07 +00:00
|
|
|
internal static string AppDataPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData), "HeliosPlus");
|
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
private string _uuid;
|
|
|
|
private Version _version;
|
|
|
|
private bool _isActive = false;
|
|
|
|
private bool _isPossible = false;
|
|
|
|
|
2020-05-09 13:02:07 +00:00
|
|
|
|
|
|
|
#region JsonConverterBitmap
|
|
|
|
internal class CustomBitmapConverter : JsonConverter
|
|
|
|
{
|
|
|
|
public override bool CanConvert(Type objectType)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//convert from byte to bitmap (deserialize)
|
|
|
|
|
|
|
|
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
|
|
|
|
{
|
|
|
|
string image = (string)reader.Value;
|
|
|
|
|
|
|
|
byte[] byteBuffer = Convert.FromBase64String(image);
|
|
|
|
MemoryStream memoryStream = new MemoryStream(byteBuffer);
|
|
|
|
memoryStream.Position = 0;
|
|
|
|
|
|
|
|
return (Bitmap)Bitmap.FromStream(memoryStream);
|
|
|
|
}
|
|
|
|
|
|
|
|
//convert bitmap to byte (serialize)
|
|
|
|
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
|
|
|
|
{
|
|
|
|
Bitmap bitmap = (Bitmap)value;
|
|
|
|
|
|
|
|
ImageConverter converter = new ImageConverter();
|
|
|
|
writer.WriteValue((byte[])converter.ConvertTo(bitmap, typeof(byte[])));
|
|
|
|
}
|
|
|
|
|
|
|
|
public static System.Drawing.Imaging.ImageFormat GetImageFormat(Bitmap bitmap)
|
|
|
|
{
|
|
|
|
ImageFormat img = bitmap.RawFormat;
|
|
|
|
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Jpeg))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Jpeg;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Bmp))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Bmp;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Png))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Png;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Emf))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Emf;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Exif))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Exif;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Gif))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Gif;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Icon))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Icon;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.MemoryBmp))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.MemoryBmp;
|
|
|
|
if (img.Equals(System.Drawing.Imaging.ImageFormat.Tiff))
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Tiff;
|
|
|
|
else
|
|
|
|
return System.Drawing.Imaging.ImageFormat.Wmf;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
2020-06-07 08:48:45 +00:00
|
|
|
static ProfileItem()
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
NvAPIWrapper.NVIDIA.Initialize();
|
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
// ignored
|
|
|
|
}
|
2020-05-09 13:02:07 +00:00
|
|
|
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
|
2020-05-09 13:02:07 +00:00
|
|
|
public static Version Version = new Version(2, 1);
|
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
#region Instance Properties
|
2018-10-23 23:34:49 +00:00
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
public string UUID
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2020-06-14 04:20:52 +00:00
|
|
|
if (String.IsNullOrWhiteSpace(_uuid))
|
|
|
|
_uuid = Guid.NewGuid().ToString("B");
|
|
|
|
return _uuid;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
|
|
|
string uuidV4Regex = @"/^[0-9A-F]{8}-[0-9A-F]{4}-4[0-9A-F]{3}-[89AB][0-9A-F]{3}-[0-9A-F]{12}$/i";
|
|
|
|
Match match = Regex.Match(value, uuidV4Regex, RegexOptions.IgnoreCase);
|
|
|
|
if (match.Success)
|
|
|
|
_uuid = value;
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-20 00:13:40 +00:00
|
|
|
[JsonIgnore]
|
2017-02-26 19:23:31 +00:00
|
|
|
public bool IsPossible
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2020-05-13 11:04:18 +00:00
|
|
|
List<string> unavailableDeviceNames = new List<string>();
|
|
|
|
// Then go through the displays in the profile and check they're live
|
|
|
|
foreach (ProfileViewport profileViewport in Viewports)
|
|
|
|
{
|
|
|
|
foreach (ProfileViewportTargetDisplay profileViewportTargetDisplay in profileViewport.TargetDisplays)
|
2020-05-12 10:46:23 +00:00
|
|
|
{
|
2020-05-13 11:04:18 +00:00
|
|
|
PathTargetInfo viewportTargetInfo = profileViewportTargetDisplay.ToPathTargetInfo();
|
2020-05-14 10:38:31 +00:00
|
|
|
// If viewportTargetInfo is null, then this viewportTargetDisplay isn't currently available
|
|
|
|
// so this makes the profile invalid
|
|
|
|
if (viewportTargetInfo == null)
|
2020-05-13 11:04:18 +00:00
|
|
|
unavailableDeviceNames.Add(profileViewportTargetDisplay.DisplayName);
|
2020-05-12 10:46:23 +00:00
|
|
|
}
|
2020-05-13 11:04:18 +00:00
|
|
|
}
|
2020-05-12 10:46:23 +00:00
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
if (unavailableDeviceNames.Count > 0)
|
|
|
|
{
|
|
|
|
// Darn - there was at least one unavilable displaytarget
|
|
|
|
return false;
|
2020-05-12 10:46:23 +00:00
|
|
|
}
|
2020-05-13 11:04:18 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// There were no unavailable DisplayTargets!
|
|
|
|
return true;
|
|
|
|
}
|
2018-10-20 00:27:25 +00:00
|
|
|
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public string Name { get; set; }
|
|
|
|
|
2020-05-12 10:46:23 +00:00
|
|
|
public ProfileViewport[] Viewports { get; set; } = new ProfileViewport[0];
|
2017-02-26 19:23:31 +00:00
|
|
|
|
2020-05-09 13:02:07 +00:00
|
|
|
public ProfileIcon ProfileIcon
|
|
|
|
{
|
2020-05-12 10:46:23 +00:00
|
|
|
get
|
|
|
|
{
|
|
|
|
if (_profileIcon != null)
|
|
|
|
return _profileIcon;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_profileIcon = new ProfileIcon(this);
|
|
|
|
return _profileIcon;
|
|
|
|
}
|
|
|
|
}
|
2020-05-09 13:02:07 +00:00
|
|
|
set
|
|
|
|
{
|
|
|
|
_profileIcon = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
public string SavedProfileIconCacheFilename { get; set; }
|
2020-05-10 10:47:18 +00:00
|
|
|
|
2020-05-09 13:02:07 +00:00
|
|
|
[JsonConverter(typeof(CustomBitmapConverter))]
|
|
|
|
public Bitmap ProfileBitmap
|
|
|
|
{
|
2020-05-12 10:46:23 +00:00
|
|
|
get
|
|
|
|
{
|
|
|
|
if (_profileBitmap != null)
|
|
|
|
return _profileBitmap;
|
|
|
|
else
|
|
|
|
{
|
2020-06-07 08:48:45 +00:00
|
|
|
_profileBitmap = this.ProfileIcon.ToBitmap(256, 256);
|
2020-05-12 10:46:23 +00:00
|
|
|
return _profileBitmap;
|
|
|
|
}
|
|
|
|
}
|
2020-05-09 13:02:07 +00:00
|
|
|
set
|
|
|
|
{
|
|
|
|
_profileBitmap = value;
|
|
|
|
}
|
|
|
|
|
2018-10-20 00:13:40 +00:00
|
|
|
}
|
2017-02-26 19:23:31 +00:00
|
|
|
|
2020-06-07 08:48:45 +00:00
|
|
|
[JsonConverter(typeof(CustomBitmapConverter))]
|
|
|
|
public Bitmap ProfileTightestBitmap
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
if (_profileShortcutBitmap != null)
|
|
|
|
return _profileShortcutBitmap;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_profileShortcutBitmap = this.ProfileIcon.ToTightestBitmap();
|
|
|
|
return _profileShortcutBitmap;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
|
|
|
_profileShortcutBitmap = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
#endregion
|
2017-02-26 19:23:31 +00:00
|
|
|
|
2020-05-09 13:02:07 +00:00
|
|
|
public static bool IsValidName(string testName)
|
|
|
|
{
|
2020-06-07 08:48:45 +00:00
|
|
|
foreach (ProfileItem loadedProfile in _allSavedProfiles)
|
2020-05-09 13:02:07 +00:00
|
|
|
{
|
|
|
|
if (loadedProfile.Name == testName)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static bool IsValidId(string testId)
|
|
|
|
{
|
2020-06-14 04:20:52 +00:00
|
|
|
string uuidV4Regex = @"/^[0-9A-F]{8}-[0-9A-F]{4}-4[0-9A-F]{3}-[89AB][0-9A-F]{3}-[0-9A-F]{12}$/i";
|
|
|
|
Match match = Regex.Match(testId, uuidV4Regex, RegexOptions.IgnoreCase);
|
|
|
|
if (match.Success)
|
2020-05-10 10:47:18 +00:00
|
|
|
return true;
|
2020-06-14 04:20:52 +00:00
|
|
|
else
|
|
|
|
return false;
|
2020-05-10 10:47:18 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 00:27:25 +00:00
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
public bool CopyTo(ProfileItem profile, bool overwriteId = false)
|
|
|
|
{
|
|
|
|
if (!(profile is ProfileItem))
|
|
|
|
return false;
|
2020-05-09 13:02:07 +00:00
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
if (overwriteId)
|
|
|
|
profile.UUID = UUID;
|
|
|
|
|
|
|
|
// Copy all our profile data over to the other profile
|
|
|
|
profile.Name = Name;
|
|
|
|
profile.UUID = UUID;
|
|
|
|
profile.Name = Name;
|
|
|
|
profile.Viewports = Viewports;
|
|
|
|
profile.ProfileIcon = ProfileIcon;
|
|
|
|
profile.SavedProfileIconCacheFilename = SavedProfileIconCacheFilename;
|
|
|
|
profile.ProfileBitmap = ProfileBitmap;
|
|
|
|
profile.ProfileTightestBitmap = ProfileTightestBitmap;
|
|
|
|
return true;
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
|
|
|
|
// The public override for the Object.Equals
|
2017-02-26 19:23:31 +00:00
|
|
|
public override bool Equals(object obj)
|
|
|
|
{
|
2020-06-07 08:48:45 +00:00
|
|
|
return this.Equals(obj as ProfileItem);
|
2020-05-13 11:04:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Profiles are equal if their contents (except name) are equal
|
2020-06-07 08:48:45 +00:00
|
|
|
public bool Equals(ProfileItem other)
|
2020-05-13 11:04:18 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
// If parameter is null, return false.
|
|
|
|
if (Object.ReferenceEquals(other, null))
|
2018-10-20 00:27:25 +00:00
|
|
|
return false;
|
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
// Optimization for a common success case.
|
|
|
|
if (Object.ReferenceEquals(this, other))
|
2018-10-20 00:27:25 +00:00
|
|
|
return true;
|
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
// If run-time types are not exactly the same, return false.
|
|
|
|
if (this.GetType() != other.GetType())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check whether the profiles' properties are equal
|
|
|
|
// We need to exclude the name as the name is solely for saving to disk
|
|
|
|
// and displaying to the user.
|
|
|
|
// Two profiles are equal only when they have the same viewport data
|
2020-05-14 10:38:31 +00:00
|
|
|
if (Viewports.SequenceEqual(other.Viewports))
|
2020-05-13 11:04:18 +00:00
|
|
|
return true;
|
|
|
|
else
|
2018-10-20 00:27:25 +00:00
|
|
|
return false;
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
// If Equals() returns true for this object compared to another
|
|
|
|
// then GetHashCode() must return the same value for these objects.
|
2017-02-26 19:23:31 +00:00
|
|
|
public override int GetHashCode()
|
|
|
|
{
|
2020-05-13 11:04:18 +00:00
|
|
|
|
|
|
|
// Get hash code for the Viewports field if it is not null.
|
|
|
|
int hashViewports = Viewports == null ? 0 : Viewports.GetHashCode();
|
|
|
|
|
|
|
|
//Calculate the hash code for the product.
|
|
|
|
return hashViewports;
|
|
|
|
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
|
2017-02-26 19:23:31 +00:00
|
|
|
public override string ToString()
|
|
|
|
{
|
2020-05-16 03:58:59 +00:00
|
|
|
return (Name ?? Language.UN_TITLED_PROFILE);
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
|
2020-05-10 10:47:18 +00:00
|
|
|
private static string GetValidFilename(string uncheckedFilename)
|
|
|
|
{
|
|
|
|
string invalid = new string(Path.GetInvalidFileNameChars()) + new string(Path.GetInvalidPathChars());
|
|
|
|
foreach (char c in invalid)
|
|
|
|
{
|
|
|
|
uncheckedFilename = uncheckedFilename.Replace(c.ToString(), "");
|
|
|
|
}
|
|
|
|
return uncheckedFilename;
|
|
|
|
}
|
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
public void ApplyTopos()
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
2020-05-15 08:52:16 +00:00
|
|
|
Debug.Print("_applyTopos()");
|
2017-02-26 19:23:31 +00:00
|
|
|
try
|
|
|
|
{
|
2020-05-15 08:52:16 +00:00
|
|
|
var surroundTopologies =
|
|
|
|
Viewports.SelectMany(viewport => viewport.TargetDisplays)
|
|
|
|
.Select(target => target.SurroundTopology)
|
|
|
|
.Where(topology => topology != null)
|
|
|
|
.Select(topology => topology.ToGridTopology())
|
|
|
|
.ToArray();
|
|
|
|
|
|
|
|
if (surroundTopologies.Length == 0)
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
2020-05-15 08:52:16 +00:00
|
|
|
var currentTopologies = GridTopology.GetGridTopologies();
|
2018-10-20 00:27:25 +00:00
|
|
|
|
2020-05-15 08:52:16 +00:00
|
|
|
if (currentTopologies.Any(topology => topology.Rows * topology.Columns > 1))
|
2018-10-20 00:27:25 +00:00
|
|
|
{
|
2020-05-15 08:52:16 +00:00
|
|
|
surroundTopologies =
|
|
|
|
GridTopology.GetGridTopologies()
|
|
|
|
.SelectMany(topology => topology.Displays)
|
|
|
|
.Select(displays => new GridTopology(1, 1, new[] { displays }))
|
|
|
|
.ToArray();
|
2018-10-20 00:27:25 +00:00
|
|
|
}
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
2020-05-15 08:52:16 +00:00
|
|
|
|
|
|
|
if (surroundTopologies.Length > 0)
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
2020-05-15 08:52:16 +00:00
|
|
|
GridTopology.SetGridTopologies(surroundTopologies, SetDisplayTopologyFlag.MaximizePerformance);
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
2020-05-15 08:52:16 +00:00
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
// ignored
|
|
|
|
}
|
|
|
|
}
|
2018-10-20 00:27:25 +00:00
|
|
|
|
2020-06-14 04:20:52 +00:00
|
|
|
public void ApplyPathInfos()
|
2020-05-15 08:52:16 +00:00
|
|
|
{
|
|
|
|
Debug.Print("_applyPathInfos()");
|
|
|
|
if (!IsPossible)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException(
|
|
|
|
$"Problem applying the '{Name}' Display Profile! The display configuration changed since this profile is created. Please re-create this profile.");
|
|
|
|
}
|
2018-10-20 00:27:25 +00:00
|
|
|
|
2020-05-15 08:52:16 +00:00
|
|
|
var pathInfos = Viewports.Select(viewport => viewport.ToPathInfo()).Where(info => info != null).ToArray();
|
|
|
|
PathInfo.ApplyPathInfos(pathInfos, true, true, true);
|
|
|
|
}
|
2020-05-13 11:04:18 +00:00
|
|
|
|
2020-05-15 08:52:16 +00:00
|
|
|
public IDictionary<string, Action> applyProfileActions()
|
|
|
|
{
|
|
|
|
var dict = new Dictionary<string, Action>()
|
|
|
|
{
|
2020-06-14 04:20:52 +00:00
|
|
|
{ "Applying_Topos", ApplyTopos },
|
|
|
|
{ "Applying_Paths", ApplyPathInfos }
|
2020-05-15 08:52:16 +00:00
|
|
|
};
|
|
|
|
return dict;
|
|
|
|
}
|
|
|
|
|
|
|
|
public IDictionary<string, string> applyProfileMsgs()
|
|
|
|
{
|
|
|
|
var dict = new Dictionary<string, string>()
|
|
|
|
{
|
2020-05-15 10:25:44 +00:00
|
|
|
{ "Applying_Topos", Language.Applying_First_Message },
|
|
|
|
{ "Applying_Paths", Language.Applying_Second_Message }
|
2020-05-15 08:52:16 +00:00
|
|
|
};
|
|
|
|
return dict;
|
|
|
|
}
|
|
|
|
|
|
|
|
public List<string> applyProfileSequence()
|
|
|
|
{
|
|
|
|
var list = new List<string>() { "Applying_Topos", "Applying_Paths" };
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Custom comparer for the Profile class
|
|
|
|
// Allows us to use 'Contains'
|
2020-06-07 08:48:45 +00:00
|
|
|
class ProfileComparer : IEqualityComparer<ProfileItem>
|
2020-05-13 11:04:18 +00:00
|
|
|
{
|
|
|
|
// Products are equal if their names and product numbers are equal.
|
2020-06-07 08:48:45 +00:00
|
|
|
public bool Equals(ProfileItem x, ProfileItem y)
|
2017-02-26 19:23:31 +00:00
|
|
|
{
|
2018-10-20 00:27:25 +00:00
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
//Check whether the compared objects reference the same data.
|
|
|
|
if (Object.ReferenceEquals(x, y)) return true;
|
2018-10-23 23:52:18 +00:00
|
|
|
|
2020-05-13 11:04:18 +00:00
|
|
|
//Check whether any of the compared objects is null.
|
|
|
|
if (Object.ReferenceEquals(x, null) || Object.ReferenceEquals(y, null))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check whether the profiles' properties are equal
|
|
|
|
// We need to exclude the name as the name is solely for saving to disk
|
|
|
|
// and displaying to the user.
|
|
|
|
// Two profiles are equal only when they have the same viewport data
|
|
|
|
if (x.Viewports.Equals(y.Viewports))
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
2020-05-13 11:04:18 +00:00
|
|
|
|
|
|
|
// If Equals() returns true for a pair of objects
|
|
|
|
// then GetHashCode() must return the same value for these objects.
|
2020-06-07 08:48:45 +00:00
|
|
|
public int GetHashCode(ProfileItem profile)
|
2020-05-13 11:04:18 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
// Check whether the object is null
|
|
|
|
if (Object.ReferenceEquals(profile, null)) return 0;
|
|
|
|
|
|
|
|
// Get hash code for the Viewports field if it is not null.
|
|
|
|
int hashViewports = profile.Viewports == null ? 0 : profile.Viewports.GetHashCode();
|
|
|
|
|
|
|
|
//Calculate the hash code for the product.
|
|
|
|
return hashViewports;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-02-26 19:23:31 +00:00
|
|
|
}
|
|
|
|
}
|