B4J Question [BANano] [SOLVED] Putting a border on base64 images

Mashiane

Expert
Licensed User
Longtime User
Hi

B4X:
#if javascript
function addBorderToBase64Image(base64Image, borderSize) {
    return new Promise((resolve, reject) => {
        // Hard-coded border color
        const borderColor = 'black';

        // Create a new Image element
        const img = new Image();
        
        // Set the image source to the base64 string
        img.src = base64Image;

        // Once the image is loaded
        img.onload = function () {
            // Create a canvas element
            const canvas = document.createElement('canvas');
            const ctx = canvas.getContext('2d');

            // Set the canvas size (image size + borderSize on all sides)
            canvas.width = img.width + borderSize * 2;  // Add borderSize on both left and right sides
            canvas.height = img.height + borderSize * 2; // Add borderSize on both top and bottom sides

            // Draw the border
            ctx.fillStyle = borderColor;
            ctx.fillRect(0, 0, canvas.width, canvas.height);

            // Draw the image inside the canvas with the border size offset to account for the border
            ctx.drawImage(img, borderSize, borderSize);

            // Get the new base64 image string from the canvas
            const newBase64Image = canvas.toDataURL();

            // Resolve the promise with the new base64 image string
            resolve(newBase64Image);
        };

        // Handle image load error
        img.onerror = function (err) {
            reject(base64Image);
        };
    });
}
#End If

My translation..

I kindly request help making this a promise function with banano as demonstrated above.

B4X:
Sub AddBorderOnBase64Image(base64 As String, borderSize As Int) As String
    Dim base64WithBorder As String = ""
    'initialize the image
    Dim img As BANanoObject
    img.Initialize2("Image", Null)
    'set the onload event
    Dim event As BANanoEvent
    ' when loaded...
    img.AddEventListenerOpen("onload", Array(event))
    Dim imgW As Int = img.GetField("width")
    Dim imgH As Int = img.GetField("height")
    imgW = banano.parseInt(imgW) + (banano.parseInt(borderSize) * 2)
    imgH = banano.parseInt(imgH) + (banano.parseInt(borderSize) * 2)
    '
    Dim canvasE As BANanoElement = banano.CreateElement("canvas")
    Dim canvas As BANanoObject = canvasE.ToObject
    Dim ctx As BANanoObject = canvas.RunMethod("getContext", "2d")
    '
    canvas.SetField("width", imgW)
    canvas.SetField("height", imgH)
    Dim cW As Int = canvas.GetField("width")
    Dim cH As Int = canvas.GetField("height")
    '
    ctx.SetField("fillStyle", "black")
    ctx.RunMethod("fillRect", Array(0, 0, cW, cH))
    '
    ctx.RunMethod("drawImage", Array(img, borderSize, borderSize))
    '
    base64WithBorder = canvas.RunMethod("toDataURL", "image/jpeg").Result
    ' closing the 'onload' event listener function
    img.CloseEventListener
    img.SetField("src", base64)
    Return base64WithBorder
End Sub

Thanks in advance
 
Solution
I would write it something like this:

B4X:
' HERE STARTS YOUR APP
Sub BANano_Ready()
    Dim Base64Image As String = GetDemoImageData
    Dim result As String

    ' call and wait for the promise
    Try
        result = BANano.Await(AddBorderToBase64Image(Base64Image, 5))
        ' with border
        Log(result)
    Catch(result)
        ' the original as something went wrong
        Log(result)
    End Try   
   
    ' continue the rest of the code
    Log("Done")
End Sub

Sub AddBorderToBase64Image(Base64Image As String, borderSize As Int) As BANanoPromise
    ' make the promise and return the result
    Dim prom As BANanoPromise 'ignore
    prom.CallSub(Me, "AddBorderToBase64ImageWait", Array(Base64Image, borderSize))
    Return...

alwaysbusy

Expert
Licensed User
Longtime User
I would write it something like this:

B4X:
' HERE STARTS YOUR APP
Sub BANano_Ready()
    Dim Base64Image As String = GetDemoImageData
    Dim result As String

    ' call and wait for the promise
    Try
        result = BANano.Await(AddBorderToBase64Image(Base64Image, 5))
        ' with border
        Log(result)
    Catch(result)
        ' the original as something went wrong
        Log(result)
    End Try   
   
    ' continue the rest of the code
    Log("Done")
End Sub

Sub AddBorderToBase64Image(Base64Image As String, borderSize As Int) As BANanoPromise
    ' make the promise and return the result
    Dim prom As BANanoPromise 'ignore
    prom.CallSub(Me, "AddBorderToBase64ImageWait", Array(Base64Image, borderSize))
    Return prom
End Sub

Sub AddBorderToBase64ImageWait(base64Image As String, borderSize As Int) ' ...Wait to make it an async method   
    Dim borderColor As String = "black"   
   
    Dim img As BANanoObject
    img.Initialize2("Image", Null)
   
    img.SetField("src", base64Image)
   
    Dim LoadFunction As BANanoObject
    LoadFunction.Sub(Null)
        Dim Canvas As BANanoElement = BANano.CreateElement("canvas")
        Dim CanvasObj As BANanoObject = Canvas.ToObject
        Dim ctx As BANanoObject = CanvasObj.RunMethod("getContext", "2d")
       
        CanvasObj.SetField("width", BANano.parseInt(img.GetField("width")) + borderSize * 2)
        CanvasObj.SetField("height", BANano.parseInt(img.GetField("height")) + borderSize * 2)
       
        ctx.SetField("fillStyle", borderColor)
        ctx.RunMethod("fillRect", Array(0, 0, CanvasObj.GetField("width"), CanvasObj.GetField("height")))
       
        ctx.RunMethod("drawImage", Array(img, borderSize, borderSize))
       
        ' return the image with border
        BANano.ReturnThen(CanvasObj.RunMethod("toDataURL", "image/jpeg").Result)
    LoadFunction.EndSub
   
    Dim ErrorFunction As BANanoObject
    ErrorFunction.Sub(Null)
        ' return the original
        BANano.ReturnElse(base64Image)
    ErrorFunction.EndSub
   
    img.SetField("onload", LoadFunction)   
    img.SetField("onerror", ErrorFunction)
End Sub

public Sub GetDemoImageData() As String
    Return ""
End Sub

Alwaysbusy
 
Upvote 1
Solution

alwaysbusy

Expert
Licensed User
Longtime User
I just realize it can actually even be without the Wait method:

B4X:
Sub AddBorderToBase64Image(Base64Image As String, borderSize As Int) As BANanoPromise
    ' make the promise and return the result
    Dim prom As BANanoPromise 'ignore
    prom.NewStart
        Dim borderColor As String = "black"
       
        Dim img As BANanoObject
        img.Initialize2("Image", Null)
       
        img.SetField("src", Base64Image)
       
        Dim LoadFunction As BANanoObject
        LoadFunction.Sub(Null)
            Dim Canvas As BANanoElement = BANano.CreateElement("canvas")
            Dim CanvasObj As BANanoObject = Canvas.ToObject
            Dim ctx As BANanoObject = CanvasObj.RunMethod("getContext", "2d")
               
            CanvasObj.SetField("width", BANano.parseInt(img.GetField("width")) + borderSize * 2)
            CanvasObj.SetField("height", BANano.parseInt(img.GetField("height")) + borderSize * 2)
               
            ctx.SetField("fillStyle", borderColor)
            ctx.RunMethod("fillRect", Array(0, 0, CanvasObj.GetField("width"), CanvasObj.GetField("height")))
               
            ctx.RunMethod("drawImage", Array(img, borderSize, borderSize))
               
            ' return the image with border
            BANano.ReturnThen(CanvasObj.RunMethod("toDataURL", "image/jpeg").Result)
        LoadFunction.EndSub
       
        Dim ErrorFunction As BANanoObject
        ErrorFunction.Sub(Null)
            ' return the original
            BANano.ReturnElse(Base64Image)
        ErrorFunction.EndSub
       
        img.SetField("onload", LoadFunction)
        img.SetField("onerror", ErrorFunction)   
    prom.NewEnd
    Return prom
End Sub

Alwaysbusy
 
Upvote 0

Mashiane

Expert
Licensed User
Longtime User
I just realize it can actually even be without the Inner method:

B4X:
Sub AddBorderToBase64Image(Base64Image As String, borderSize As Int) As BANanoPromise
    ' make the promise and return the result
    Dim prom As BANanoPromise 'ignore
    prom.NewStart
        Dim borderColor As String = "black"
       
        Dim img As BANanoObject
        img.Initialize2("Image", Null)
       
        img.SetField("src", Base64Image)
       
        Dim LoadFunction As BANanoObject
        LoadFunction.Sub(Null)
            Dim Canvas As BANanoElement = BANano.CreateElement("canvas")
            Dim CanvasObj As BANanoObject = Canvas.ToObject
            Dim ctx As BANanoObject = CanvasObj.RunMethod("getContext", "2d")
               
            CanvasObj.SetField("width", BANano.parseInt(img.GetField("width")) + borderSize * 2)
            CanvasObj.SetField("height", BANano.parseInt(img.GetField("height")) + borderSize * 2)
               
            ctx.SetField("fillStyle", borderColor)
            ctx.RunMethod("fillRect", Array(0, 0, CanvasObj.GetField("width"), CanvasObj.GetField("height")))
               
            ctx.RunMethod("drawImage", Array(img, borderSize, borderSize))
               
            ' return the image with border
            BANano.ReturnThen(CanvasObj.RunMethod("toDataURL", "image/jpeg").Result)
        LoadFunction.EndSub
       
        Dim ErrorFunction As BANanoObject
        ErrorFunction.Sub(Null)
            ' return the original
            BANano.ReturnElse(Base64Image)
        ErrorFunction.EndSub
       
        img.SetField("onload", LoadFunction)
        img.SetField("onerror", ErrorFunction)   
    prom.NewEnd
    Return prom
End Sub

Alwaysbusy
Awesome, thank you so much. I like this 2nd implementation better. Sometimes the translation from JS to BANano can be very elusive, especially the BANanoPromise object, for me. Seeing this version makes it easier to grasp.
 
Upvote 0
Top