Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions CSF.Screenplay.Selenium/CSF.Screenplay.Selenium.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,12 @@
<Import Project="..\Tools\Version.props" />
<Import Project="..\Tools\PackageReadmes.props" />

<PropertyGroup>
<TargetFrameworks>$(TargetFrameworks);net6.0</TargetFrameworks>
<DefineConstants Condition="'$(TargetFramework)' == 'net6.0' or '$(TargetFramework)' == '$(DotNetLatestLts)'">$(DefineConstants);RECORD_STRUCT_SUPPORT</DefineConstants>
</PropertyGroup>


<PropertyGroup>
<RootNamespace>CSF.Screenplay.Selenium</RootNamespace>
<DocumentationFile>$(MSBuildProjectDirectory)\bin\$(Configuration)\$(TargetFramework)\$(AssemblyName).xml</DocumentationFile>
Expand Down
294 changes: 294 additions & 0 deletions CSF.Screenplay.Selenium/Color.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,294 @@
using System;
using SysColor = System.Drawing.Color;

namespace CSF.Screenplay.Selenium
{
/// <summary>
/// Immutable type represents a color on the web, in a manner which may be converted and compared between web-supported formats.
/// </summary>
/// <remarks>
/// <para>
/// An uninitialized instance of this type will represent "fully-transparent black".
/// Internally, this type stores color as three unsigned <see cref="byte"/> values (Red, Green, Blue) and a
/// <see cref="double"/> representing the transparency (Alpha). The Alpha value must be between 0 (fully transparent)
/// and 1 (fully opaque).
/// </para>
/// <para>
/// This struct contains functionality to compare equality with, and convert to/from the .NET built-in <see cref="System.Drawing.Color"/>.
/// It also contains parsing logic to parse/format an instance from/to string representations of color which are used by web browsers.
/// See <see href="https://developer.mozilla.org/en-US/docs/Web/CSS/Reference/Values/color_value">the MDN writeup of web color</see> for
/// more information about the valid formats.
/// </para>
/// </remarks>
#if RECORD_STRUCT_SUPPORT
public readonly record struct Color
#else
public struct Color
#endif
: IEquatable<SysColor>, IEquatable<Color>, IEquatable<string>
{
readonly double alpha;

/// <summary>
/// Gets the red component in the sRGB color space.
/// </summary>
#if RECORD_STRUCT_SUPPORT
public readonly byte Red { get; init; }
#else
public byte Red { get; private set; }
#endif

/// <summary>
/// Gets the green component in the sRGB color space.
/// </summary>
#if RECORD_STRUCT_SUPPORT
public readonly byte Green { get; init; }
#else
public byte Green { get; private set; }
#endif

/// <summary>
/// Gets the blue component in the sRGB color space.
/// </summary>
#if RECORD_STRUCT_SUPPORT
public readonly byte Blue { get; init; }
#else
public byte Blue { get; private set; }
#endif

/// <summary>
/// Gets the alpha (transparency) component in the sRGB color space.
/// </summary>
/// <remarks>
/// <para>The value of this property will always be between zero (fully transparent) and one (fully opaque).</para>
/// </remarks>
#if RECORD_STRUCT_SUPPORT
public readonly double Alpha
{
get => alpha;
init => this.alpha = SanitizeAlpha(value);
}
#else
public double Alpha { get => alpha; }
#endif

/// <summary>
/// Gets a representation of <see cref="Alpha"/>, as a byte.
/// </summary>
public byte AlphaAsByte => (byte) Math.Floor(Math.Round(Alpha * 255, 0, MidpointRounding.AwayFromZero));

/// <summary>
/// Converts this color to a <see cref="System.Drawing.Color"/>.
/// </summary>
/// <remarks>
/// <para>
/// In some situations, it might be useful to convert to/from the .NET built-in <see cref="System.Drawing.Color"/>
/// type. Use this method and <see cref="FromSystemDrawingColour"/> to do so.
/// </para>
/// </remarks>
/// <returns>A <see cref="System.Drawing.Color"/> representation of this color.</returns>
public SysColor ToSystemDrawingColor() => SysColor.FromArgb(AlphaAsByte, Red, Green, Blue);

/// <inheritdoc/>
public bool Equals(SysColor other) => ToSystemDrawingColor().Equals(other);

/// <inheritdoc/>
public bool Equals(Color other) => other.Red == Red
&& other.Green == Green
&& other.Blue == Blue
&& other.AlphaAsByte == AlphaAsByte;

/// <inheritdoc/>
public bool Equals(string other) => TryParse(other, out var color) && Equals(color);

#if !RECORD_STRUCT_SUPPORT
/// <inheritdoc/>
public override bool Equals(object obj)
{
if(obj is Color color) return Equals(color);
if(obj is SysColor sysColor) return Equals(sysColor);
if(obj is string strColor) return Equals(strColor);
return false;
}
#endif

/// <inheritdoc/>
public override int GetHashCode() => HashCode.Combine(Red, Green, Blue, Alpha);

/// <summary>
/// Creates a new instance of <see cref="Color"/> from the specified RGBA values.
/// </summary>
/// <remarks>
/// <para>
/// The byte data type for the R, G and B components of the color ensure that it is impossible to specify
/// values outside the permitted range. As for the <paramref name="alpha"/>, if a value of less than zero is specified
/// here then it is treated as zero. Likewise if a value of greater than one is specified then it will be treated as one.
/// </para>
/// </remarks>
/// <param name="red">The red component in the sRGB color space</param>
/// <param name="green">The green component in the sRGB color space</param>
/// <param name="blue">The blue component in the sRGB color space</param>
/// <param name="alpha">The alpha (transparency) component, which must be between zero and one.</param>
public Color(byte red, byte green, byte blue, double alpha)
{
Red = red;
Green = green;
Blue = blue;
this.alpha = SanitizeAlpha(alpha);
}

/// <summary>
/// Sanitizes a value intended for <see cref="Alpha"/>.
/// </summary>
/// <remarks>
/// <para>If <paramref name="alpha"/> is less than zero then this method returns zero.
/// If <paramref name="alpha"/> is more than one then this method returns one.
/// Otherwise, this method returns <paramref name="alpha"/>.</para>
/// </remarks>
/// <param name="alpha">The alpha value.</param>
/// <returns>A sanitized alpha value.</returns>
static double SanitizeAlpha(double alpha) => alpha < 0 ? 0 : alpha > 1 ? 1 : alpha;

Check warning on line 150 in CSF.Screenplay.Selenium/Color.cs

View workflow job for this annotation

GitHub Actions / Build, test & package

Extract this nested ternary operation into an independent statement. (https://rules.sonarsource.com/csharp/RSPEC-3358)

Check warning on line 150 in CSF.Screenplay.Selenium/Color.cs

View workflow job for this annotation

GitHub Actions / Build, test & package

Extract this nested ternary operation into an independent statement. (https://rules.sonarsource.com/csharp/RSPEC-3358)

Check warning on line 150 in CSF.Screenplay.Selenium/Color.cs

View workflow job for this annotation

GitHub Actions / Build, test & package

Extract this nested ternary operation into an independent statement. (https://rules.sonarsource.com/csharp/RSPEC-3358)

Check warning on line 150 in CSF.Screenplay.Selenium/Color.cs

View workflow job for this annotation

GitHub Actions / Build, test & package

Extract this nested ternary operation into an independent statement. (https://rules.sonarsource.com/csharp/RSPEC-3358)

Check warning on line 150 in CSF.Screenplay.Selenium/Color.cs

View workflow job for this annotation

GitHub Actions / Build, test & package

Extract this nested ternary operation into an independent statement. (https://rules.sonarsource.com/csharp/RSPEC-3358)

Check warning on line 150 in CSF.Screenplay.Selenium/Color.cs

View workflow job for this annotation

GitHub Actions / Build, test & package

Extract this nested ternary operation into an independent statement. (https://rules.sonarsource.com/csharp/RSPEC-3358)

/// <summary>
/// Converts a <see cref="System.Drawing.Color"/> into a <see cref="Color"/>.
/// </summary>
/// <remarks>
/// <para>
/// In some situations, it might be useful to convert to/from the .NET built-in <see cref="System.Drawing.Color"/>
/// type. Use this method and <see cref="ToSystemDrawingColor"/> to do so.
/// </para>
/// </remarks>
/// <returns>A <see cref="Color"/> representation of the specified color.</returns>
public static Color FromSystemDrawingColour(SysColor color)
=> new Color(color.R, color.G, color.B, color.A / 255);

/// <summary>
/// Gets a value indicating whether the specified <paramref name="webColorValue"/> is a valid string representation
/// for a <see cref="Color"/>. If this returns <see langword="true"/> then exposes the parsed <see cref="Color"/>
/// value.
/// </summary>
/// <remarks>
/// <para>
/// If this method returns <see langword="true"/> then the output parameter <paramref name="color"/> will contain
/// the color parsed from the specified string. If this method returns <see langword="false"/> then the output parameter
/// has an undefined value and must not be used.
/// </para>
/// <para>This method will not raise an exception and it is safe to use with unknown strings, including those which
/// might be <see langword="null"/>.</para>
/// </remarks>
/// <param name="webColorValue">A string containing a color value which is compatible with the web.
/// See <see href="https://developer.mozilla.org/en-US/docs/Web/CSS/Reference/Values/color_value">the MDN writeup on
/// web colors</see> for more information.</param>
/// <param name="color">If this method returns <see langword="true"/>, then this parameter exposes the parsed color value.
/// If this method returns <see langword="false"/> then the value of this parameter must be ignored.</param>
/// <returns><see langword="true"/> if parsing is successful (the <paramref name="webColorValue"/> contains a valid &amp;
/// recognized color representation); <see langword="false"/> if not</returns>
public static bool TryParse(string webColorValue, out Color color)
{
throw new NotImplementedException("To be written!");
}

/// <summary>
/// Parses a <see cref="Color"/> from a string representation of a color.
/// </summary>
/// <remarks>
/// <para>
/// If <paramref name="webColorValue"/> is a valid representation of a web-compatible color value then this method will
/// return that color as an instance of <see cref="Color"/>. If not, this method will throw an exception.
/// </para>
/// <para>If you are not certain that the <paramref name="webColorValue"/> represents a valid colour then use
/// <see cref="TryParse(string, out Color)"/> instead.</para>
/// </remarks>
/// <param name="webColorValue">A string containing a color value which is compatible with the web.
/// See <see href="https://developer.mozilla.org/en-US/docs/Web/CSS/Reference/Values/color_value">the MDN writeup on
/// web colors</see> for more information.</param>
/// <returns>The parsed color.</returns>
/// <exception cref="ArgumentNullException">If <paramref name="webColorValue"/> is <see langword="null"/>.</exception>
/// <exception cref="FormatException">If <paramref name="webColorValue"/> is not a recognized representation of a color.</exception>
public static Color Parse(string webColorValue)
{
if (webColorValue is null)
throw new ArgumentNullException(nameof(webColorValue));

return TryParse(webColorValue, out var color)
? color
: throw new FormatException("The string color representation must be a in a web-compatible recognized format.");
}

/// <summary>
/// Gets a value indicating whether the specified <see cref="Color"/> and <see cref="System.Drawing.Color"/>
/// are equal.
/// </summary>
/// <remarks>
/// <para>Despite the type differences, two colors are equal if they represent the same sRGB color.</para>
/// </remarks>
/// <param name="color">The <see cref="Color"/></param>
/// <param name="sysColor">The <see cref="System.Drawing.Color"/></param>
/// <returns><see langword="true"/> if the two different colors represent the same sRGB color; <see langword="false"/> if not.</returns>
public static bool operator ==(Color color, SysColor sysColor) => color.Equals(sysColor);

/// <summary>
/// Gets a value indicating whether the specified <see cref="Color"/> and <see cref="System.Drawing.Color"/>
/// are not equal.
/// </summary>
/// <remarks>
/// <para>Despite the type differences, two colors are equal if they represent the same sRGB color.</para>
/// </remarks>
/// <param name="color">The <see cref="Color"/></param>
/// <param name="sysColor">The <see cref="System.Drawing.Color"/></param>
/// <returns><see langword="false"/> if the two different colors represent the same sRGB color; <see langword="true"/> if not.</returns>
public static bool operator !=(Color color, SysColor sysColor) => !(color == sysColor);

/// <summary>
/// Gets a value indicating whether the specified <see cref="Color"/> and <see cref="string"/> representation of a color
/// are equal.
/// </summary>
/// <remarks>
/// <para>A color is equal to a string if the result of <see cref="TryParse(string, out Color)"/> is <see langword="true"/>
/// and if the resulting parsed <see cref="Color"/> instance is equal to the current color.
/// Note that this means that a color is never equal to a string which is not a valid color representation, or which is <see langword="null"/>.</para>
/// </remarks>
/// <param name="color">The <see cref="Color"/></param>
/// <param name="stringColor">The string which represents a color</param>
/// <returns><see langword="true"/> if the current color and the string color representation indicate the same sRGB color; <see langword="false"/> if not.</returns>
public static bool operator ==(Color color, string stringColor) => color.Equals(stringColor);

/// <summary>
/// Gets a value indicating whether the specified <see cref="Color"/> and <see cref="string"/> representation of a color
/// are not equal.
/// </summary>
/// <remarks>
/// <para>A color is equal to a string if the result of <see cref="TryParse(string, out Color)"/> is <see langword="true"/>
/// and if the resulting parsed <see cref="Color"/> instance is equal to the current color.
/// Note that this means that a color is never equal to a string which is not a valid color representation, or which is <see langword="null"/>.</para>
/// </remarks>
/// <param name="color">The <see cref="Color"/></param>
/// <param name="stringColor">The string which represents a color</param>
/// <returns><see langword="false"/> if the current color and the string color representation indicate the same sRGB color; <see langword="true"/> if not.</returns>
public static bool operator !=(Color color, string stringColor) => !(color == stringColor);

#if !RECORD_STRUCT_SUPPORT
/// <summary>
/// Gets a value indicating whether the two specified <see cref="Color"/> instances are equal.
/// </summary>
/// <remarks>
/// <para>Two colors are equal if they represent the same sRGB color.</para>
/// </remarks>
/// <param name="first">The first <see cref="Color"/></param>
/// <param name="second">The second <see cref="Color"/></param>
/// <returns><see langword="true"/> if the two colors represent the same sRGB color; <see langword="false"/> if not.</returns>
public static bool operator ==(Color first, Color second) => first.Equals(second);

/// <summary>
/// Gets a value indicating whether the two specified <see cref="Color"/> instances are not equal.
/// </summary>
/// <remarks>
/// <para>Two colors are equal if they represent the same sRGB color.</para>
/// </remarks>
/// <param name="first">The first <see cref="Color"/></param>
/// <param name="second">The second <see cref="Color"/></param>
/// <returns><see langword="false"/> if the two colors represent the same sRGB color; <see langword="true"/> if not.</returns>
public static bool operator !=(Color first, Color second) => !(first == second);
#endif
}
}
Loading
Loading