Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[WIP/QRCoder2] Fluent API experimentation PR #558

Open
wants to merge 2 commits into
base: master
Choose a base branch
from

Conversation

Shane32
Copy link
Contributor

@Shane32 Shane32 commented Jun 1, 2024

Summary

Experimentation with fluent API syntax

Samples

var code = QRCodeBuilder.CreateEmail("[email protected]")
    .WithSubject("Testing")
    .WithBody("Hello World!")
    .WithErrorCorrection(QRCodeGenerator.ECCLevel.H)
    .RenderWith<AsciiRenderer>()
    .WithQuietZone(false)
    .ToString();

var image = QRCodeBuilder.CreatePhoneNumber("1234567890")
    .WithErrorCorrection(QRCodeGenerator.ECCLevel.H)
    .RenderWith<SystemDrawingRenderer>(20)
    .WithQuietZone(false)
    .ToBitmap();

var base64 = QRCodeBuilder.CreateMMS("1234567890", "Hello")
    .RenderWith<PngRenderer>()
    .ToBase64String();

Intellisense samples

image

image

Preliminary Findings

  1. Could start with new QRCodeBuilder.Email("[email protected]") rather than QRCodeBuilder.CreateEmail("[email protected]") - which is pretty similar to the existing new PayloadGenerator.Wifi() syntax actually, with the difference being the fluent syntax for configuration. One benefit of using new is that additional payloads can be added by simply adding a new class into the correct namespace. Whereas methods cannot be added to QRCodeBuilder outside of QRCoder.
  2. Using RenderWith<T> breaks the intellisense pattern because T does not give a list of the specific renderers available. Probably better to create an extension method for each renderer, like RenderAsAscii() and RenderAsPng(20) instead
  3. RenderWith<T> does not allow for render-specific constructor parameters. I've compensated with support for only two patterns: (1) no arguments (2) pixels per module argument. Having RenderAs...() dedicated methods would allow each renderer to require specific arguments.
  4. The builder syntax could likely be added to v1.x as a new layer, as shown in this PR, with any implementation changes we wish to make within the builder methods. Then v2 just removes (or makes internal) all the old methods.
  5. The new builders, when layered on the old code, are quite easy to write. Once the old code is removed, additional optimizations can be added to enhance trimming support. This trimming capability may consolidate the QRCode and ArtQRCode classes.
  6. All of the supporting code can be nested as deep within namespaces as desired, since intellisense always provides the correct context-sensitive methods. Except for RenderWith<T>, as noted in item 2 above. Another reason to use RenderAsAscii() or similar.
  7. The extension methods on the renderers (ToArray, ToStream, ToFile, etc) are really cool -- each renderer need only implement ToStream and all the other methods are available via extension methods. Similar functionality for text renderers; even with only implementing ToString they can also provide ToFile, ToStream, ToBase64, and so on via extension methods.


namespace QRCoder.Builders.Payloads.Implementations
{
public class EmailPayload : PayloadBase
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sample where the ctor only has the email address, and the subject/body are provided by extension methods. Realistically, I'm thinking that the required or common parameters always appear in the constructor (or as an overload), but all of the optional/rare options (like MailEncoding) are extension methods.

@@ -0,0 +1,14 @@
namespace QRCoder.Builders.Payloads.Implementations
{
public class StringPayload : PayloadBase
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can use this class for any payload that immediately serializes to a string value. At least if we design it like CreateUrl(...). Of course it would be nearly as easy to have additional classes for each payload type, similar to how it is now, if we want new QRCodeBuilder.Url(...) or whatnot.

@@ -0,0 +1,21 @@
namespace QRCoder.Builders.Payloads
{
public abstract class PayloadBase : IPayload, IConfigurableEccLevel, IConfigurableEciMode, IConfigurableVersion
Copy link
Contributor Author

@Shane32 Shane32 Jun 1, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The idea here is that while most all payloads can inherit from PayloadBase and inherit some configurable settings, perhaps some payloads, like the swiss banking payload, does not support configuring the Eci mode or whatever. Or, we can override some of these properties to add validation, for instance ensuring that a certain minimum ECC level is maintained. There's a number of ways to actually implement this, all with similar end results.

public abstract class PayloadBase : IPayload, IConfigurableEccLevel, IConfigurableEciMode, IConfigurableVersion
{
protected virtual QRCodeGenerator.ECCLevel EccLevel { get; set; } = QRCodeGenerator.ECCLevel.Default;
QRCodeGenerator.ECCLevel IConfigurableEccLevel.EccLevel { get => EccLevel; set => EccLevel = value; }
Copy link
Contributor Author

@Shane32 Shane32 Jun 1, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

These interfaces are always explicitly implemented so that they do not appear on the intellisense list of accessible members for the class. Rather, the extension method will appear, which uses generics to return the same type.

If that's confusing, for example, this code would break the fluent syntax:

public PayloadBase WithEciMode(EciMode value)
{
    _eciMode = value;
}

Because it returns the base class type (PayloadBase), rather than the type of the derived type (e.g. WiFiPayload). Extension methods allow the intended behavior, while eliminating duplicate code across each payload class.

Comment on lines +30 to +36
public static T RenderWith<T>(this IPayload payload)
where T : IRenderer, new()
{
var renderer = new T();
renderer.Payload = payload;
return renderer;
}
Copy link
Contributor Author

@Shane32 Shane32 Jun 1, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This functions, but I think explicit extension methods such as RenderWithAsciiRenderer() or RenderAsPng() would be better than generics here. I specifically don't like that intellisense does not show the compatible renderers for T. Sorta breaks the goal.

image

Comment on lines +5 to +8
public interface IStreamRenderer
{
MemoryStream ToStream();
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Concept is that all byte-array-type renderers (or ones that can serialize to a byte array) implement this interface. If they use a MemoryStream under the hood, it's a direct pass-through. If they generate a byte array directly, then wrapping it in a MemoryStream is quite inexpensive, as it does not duplicate the entire byte array.

Comment on lines +30 to +38
public static byte[] ToArray(this IStreamRenderer streamRenderer)
{
var memoryStream = streamRenderer.ToStream();
#if NETSTANDARD || NETCOREAPP // todo: target .NET Framework 4.6 or newer so this code path is supported
// by using TryGetBuffer, there is extremely small consequence to wrapping a byte[] in a MemoryStream temporarily
if (memoryStream.TryGetBuffer(out var buffer) && buffer.Count == buffer.Array.Length)
{
return buffer.Array;
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

And here we can extract the original array without duplicating it.

Comment on lines +5 to +8
public interface ITextRenderer
{
string ToString();
}
Copy link
Contributor Author

@Shane32 Shane32 Jun 1, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Concept is that all string-based renderers (Ascii, Svg) implement this interface. Another option would be to have it implement ToStringBuilder instead, more similar to IStreamRenderer above. However, we can't create an extension method called ToString because object.ToString() would always take precedence over an extension method.

QuietZone, _iconBackgroundColor);
}

public MemoryStream ToStream()
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Renderers which return a specialized type may optionally also implement IStreamRenderer as is shown here. Then ToFile and other extension methods for byte-array type renderers are available.


protected bool QuietZone { get; set; } = true;
bool IConfigurableQuietZones.QuietZone { get => QuietZone; set => QuietZone = value; }
IPayload IRenderer.Payload { set => QrCodeData = value.ToMatrix(); }
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Another option would be to store the IPayload object itself. I opted to call ToMatrix immediately, so if there were any exceptions while converting the payload to a module matrix, it would be caught at this point, enhancing the ability to diagnose faulty fluent syntax.

return Convert.ToBase64String(data.Array, data.Offset, data.Count);
}

public static void ToFile(this IStreamRenderer streamRenderer, string fileName)
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note that if any renderer wanted to optimize something like ToFile, they could implement the method directly, which takes precedence over extension methods.

@csturm83
Copy link
Contributor

csturm83 commented Jun 2, 2024

The builder syntax could likely be added to v1.x as a new layer, as shown in this PR

QRCoder2.FluentExtensions maybe? 😉

@codebude
Copy link
Owner

codebude commented Jun 2, 2024

Hi Shane, thanks for the PR. That looks pretty interesting. However, I don't want to merge this into v1.x, but would like to include it in v2 first, as I don't want to make/document/communicate any more major API changes to v1. I hope that's okay with you.

@Shane32
Copy link
Contributor Author

Shane32 commented Jun 2, 2024

Certainly. I wrote this more of a discussion/preview/review of a potential approach.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants