Click here to Skip to main content
16,001,882 members
Articles / Programming Languages / Visual Basic
Tip/Trick

Create QR Codes with Google Web APIs

Rate me:
Please Sign up or sign in to vote.
4.69/5 (10 votes)
3 Jan 2015CPOL4 min read 24.1K   17   3
How to create a UserControl to show QR Codes created with Google APIs

Introduction

In this tip, I'll cover a simple method to create a QR Code inside a standard control, wrapping up everything inside a UserControl for future reference. There are many libraries which could help us in operations like that, but in this post, I will use Google Charts (https://developers.google.com/chart/), together with System.Net and System.IO namespaces.

Background

Google Charts can be queried though POST request (see here for details: https://developers.google.com/chart/image/docs/post_requests?csw=1), so we must:

  1. query remote server, with particular POST parameters (more on this later)
  2. retrieve the server's response (a PNG image)
  3. use it for our means, namely paint it on our control

    So, open a new project in Visual Studio, then add a new User Control. I've set BorderStyle property to Fixed3D, and DoubleBuffered to True (in order to avoid flickering when the control refreshes itself).

Using the Code

The standard URL we will query is the following: http://chart.googleapis.com/chart?chs={WIDTH}x{HEIGHT}&cht=qr&chl={DATA} (parameters in brackets will be replaced with real parameters). chs will specify QR Code resolution (width x height), while chl will contain the data to be represented through barcode. The parameter related to the barcode size could be easily deduced from our control's property (since a standard control naturally have a width and a height), but we will create a new property to store a certain amount of text, representing the data our QR Code will show.

In our UserControl, we start declaring the standard URI as a constant, our Data Property, and an internal variable to store data in the local context:

VB.NET
Const _GOOGLE_URL As String = "http://chart.googleapis.com/chart?chs=_ 
[This link is external to TechNet Wiki. It will open in a new window.] _
{WIDTH}x{HEIGHT}&cht=qr&chl={DATA}"
Dim _DATA As String = String.Empty
 
Property Data As String
  Get
    Return _DATA
  End Get
  Set(value As String)
    _DATA = value
  End Set
End Property

When we will use our control, Data Property will be available in both Code View and Design Mode:

Now that we can compose a URI with all the requested parameters, we must forge Data Property in order to encode its content before web request. This way, we can be sure no special character will come to break our query. I've implemented a private function for this. Calling on it return an URI with replaced parameters, the least of which will be encoded (thanks to the WebUtility.UrlEncode function)

VB.NET
Private Function getQRURI() As String
    Dim _qrAddr As String = _GOOGLE_URL.Replace("{WIDTH}", Me.Width.ToString).Replace("{HEIGHT}", _
    Me.Height.ToString)
    _qrAddr = _qrAddr.Replace("{DATA}", WebUtility.UrlEncode(_DATA))
 
    Return _qrAddr
End Function

We'll replace the first two tag parameters, {WIDTH} and {HEIGHT}, with our control's size, while the data's parameter will be the content of our Data Property, encoded (for details on WebUtility.UrlEncode, please refer here).

We're now ready to fetch our image from remote servers, and use the returning buffer to paint our QR Code on our control. Since I want QR Code to be painted at the standard OnPaint Event (taking advantage of the PaintEventArgs argument it exposes), I will override it, adding my code:

VB.NET
Protected Overrides Sub OnPaint(e As PaintEventArgs)
    MyBase.OnPaint(e)
    If _DATA Is Nothing Then Exit Sub
 
    Dim client As New WebClient()
    Dim bytes() As Byte = client.DownloadData(getQRURI())
    client.Dispose()
 
    Dim memStream As New IO.MemoryStream(bytes)
    Dim bmp As Bitmap = Bitmap.FromStream(memStream)
    memStream.Dispose()
 
    e.Graphics.DrawImage(bmp, 0, 0)
End Sub

MyBase.OnPaints calls on standard paint operations. Next, we check if there is data to query (exiting method otherwise), and proceeding querying remote server with a new instance of WebClient. Through the call at DownloadData, to which we pass the result of our URI-formatting function, we'll fill an array of bytes, which is the server response, i.e. the PNG image representing our QR Code.

An image-type variable can be initialized through reading a Stream (like when we wish to open an image existing on our hard-drive, a stream to our local copy of it). Since we'll have our bytes in memory, we can declare a MemoryStream based on our array, and use it as a Bitmap source. At this point, having a perfectly working bitmap, we can exploit the variable e, which the OnPaint event give access to us, to draw the image on our control, at [0;0] location.

After compiling our project, the QRBox will be available in the ToolBox, ready to be used on our Forms.

Using it is simple, it's sufficient to set its Data Property, and call for a control's refresh.
The following example Form shows how it works: I've added to my Form a QrBox control, together with a standard TextBox ad Button.

When the user presses the "Make" Button, we'll read the TextBox content, passing it to the QrBox Data Property, and invoking the Refresh() method, in order to start the remote query towards Google Charts. The code for the Button's click will be simply like that:

VB.NET
Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click
    QrBox1.Data = TextBox1.Text
    QrBox1.Refresh()
End Sub

UserControl's Complete Source Code

The complete code for QrBox UserControl is as follows:

VB.NET
Imports System.Net
 
Public Class QRBox
    Const _GOOGLE_URL As String = _
    "http://chart.googleapis.com/chart?chs={WIDTH}x{HEIGHT}&cht=qr&chl={DATA}"
    Dim _DATA As String = String.Empty
 
    Property Data As String
        Get
            Return _DATA
        End Get
        Set(value As String)
            _DATA = value
        End Set
    End Property
 
    Private Function getQRURI() As String
        Dim _qrAddr As String = _GOOGLE_URL.Replace("{WIDTH}", _
        Me.Width.ToString).Replace("{HEIGHT}", Me.Height.ToString)
        _qrAddr = _qrAddr.Replace("{DATA}", WebUtility.UrlEncode(_DATA))
 
        Return _qrAddr
    End Function
 
    Protected Overrides Sub OnPaint(e As PaintEventArgs)
        MyBase.OnPaint(e)
        If _DATA Is Nothing Then Exit Sub
 
        Dim client As New WebClient()
        Dim bytes() As Byte = client.DownloadData(getQRURI())
        client.Dispose()
 
        Dim memStream As New IO.MemoryStream(bytes)
        Dim bmp As Bitmap = Bitmap.FromStream(memStream)
        memStream.Dispose()
 
        e.Graphics.DrawImage(bmp, 0, 0)
    End Sub
 
    Public Sub New()
        InitializeComponent()
    End Sub
End Class

I hope this could be useful for your projects. Happy coding, and good luck with your work!

History

  • 2015-01-03: First release for CodeProject

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer
Italy Italy
Working in IT since 2003 as Software Developer for Essetre Srl, a company in Northern Italy.
I was awarded in 2014, 2015 and 2016 with Microsoft MVP, for Visual Studio and Development Technologies expertise. My technology interests and main skills are in .NET Framework, Visual Basic, Visual C# and SQL Server, but i'm proficient in PHP and MySQL also.

Comments and Discussions

 
QuestionText limit Pin
BertrandLQ4-Jun-15 10:52
BertrandLQ4-Jun-15 10:52 
For the author: I just quickly tested your code. It works fine: 5 stars because the subject is interesting, your code is really simple and it help me to understand few thinks Smile | :) Thanks.

For other people, it's not necessary to build User Control to evaluate the code.
To test it, you simply add next code to a PictureBox Paint event. You add TextBox1 and Button.

So I tested it with VCard ascii content and it doesn't work. I think there is a Google limit...

VB
Private Sub OK_Button_Click(ByVal sender As System.Object, ByVal e As EventArgs) Handles OK_Button.Click
        Try
            Call PictureBox1.Refresh()

        Catch Err As Exception
            Call AddError(InfoLabel & "- validating selection operation failed." & errSeparator & Err.Message, True)
        End Try
End Sub


VB
Private Sub PictureBox1_Paint(sender As Object, e As PaintEventArgs) Handles PictureBox1.Paint
        Dim memStream As IO.MemoryStream = Nothing

        Try
            If Len(TextBox1.Text) <= 0 Then Exit Sub

            Call MyBase.OnPaint(e)

            Dim _qrAddr As String = _GOOGLE_URL.Replace("{WIDTH}", PictureBox1.Width.ToString).Replace("{HEIGHT}", PictureBox1.Height.ToString)
            Dim client As New WebClient()
            Dim bytes() As Byte = client.DownloadData(_qrAddr.Replace("{DATA}", WebUtility.UrlEncode(TextBox1.Text)))

            Call client.Dispose()

            memStream = New IO.MemoryStream(bytes)
            Dim bmp As Bitmap = CType(Bitmap.FromStream(memStream), Bitmap)

            Call e.Graphics.DrawImage(bmp, 0, 0)

        Catch ex As Exception

        Finally
            If memStream IsNot Nothing Then Call memStream.Dispose()
        End Try
    End Sub

QuestionNice, practical and easy to understand! Pin
JohannQ5-Jan-15 18:30
JohannQ5-Jan-15 18:30 
AnswerRe: Nice, practical and easy to understand! Pin
Emiliano Musso5-Jan-15 22:04
professionalEmiliano Musso5-Jan-15 22:04 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.