Skip to Content
Technical Articles
Author's profile photo Ferry Djaja

SAP Analytics Cloud Barcode Scanner Custom Widget

In this blog, I will demonstrate how to create a barcode scanner custom widget in SAP Analytics Cloud.

What component we need to have:

  • SAPUI5 components: sap.m.Label, sap.m.Input, sap.m.Button, sap.ui.layout.form.SimpleForm.
  • For barcode scanner, we will be using library from quaggaJS.

Custom Widget JSON

Let start with creating a  custom widget JSON file, barcodescanner.json. In this file, I added the valid properties for Root object, Webcomponent object and Properties object. I am not using the Methods and Events object.

{
  "id": "com.fd.djaja.sap.sac.scanner",
  "version": "0.0.2",
  "name": "barcodeScan",
  "description": "Barcode Scanner",
  "newInstancePrefix": "barcodeScan",
  "icon": "",
  "vendor": "Ferry Djaja",
  "eula": "",
  "license": "",
  "imports": [ "input-controls" ],
  "webcomponents": [
    {
      "kind": "main",
      "tag": "com-fd-djaja-sap-sac-scanner",
      "url": "http://localhost/SAC/sacbarcodescanner/barcodescanner.js",
      "integrity": "",
      "ignoreIntegrity": true
    }
  ],
  "properties": {
    "metadata": {
      "type": "string",
      "description": "For internal use",
      "default": ""
    }
  },
  "methods": {
  },
  "events": {
  }
}

Web Component JavaScript

We will implement the following Web Component JavaScript functions:

  • constructor() It is fired when the Web Component is initialized. 
    In this function, we create the shadow DOM root element, copy of the template element is added as a child element to the shadow DOM root element, create the unique ID (guid) for the HTML div and set the variable _firstConnection.

    constructor() {
        super();
    
        _shadowRoot = this.attachShadow({
            mode: "open"
        });
        _shadowRoot.appendChild(tmpl.content.cloneNode(true));
    
        this._id = createGuid();
    
        _shadowRoot.querySelector("#export_div").id = this._id + "_export_div";
    
        this._firstConnection = 0;
    }​

     

  • connectedCallback() It is fired when the widget is added to the html DOM of the page. 
    In this function we check whether the app is in the edit mode or not and to list the available components (e.g,. Button, Text, Input Field, Panel, etc) in variable (this.metadata)["components"] for future use.

    connectedCallback() {
        try {
            if (window.commonApp) {
                let outlineContainer = commonApp.getShell().findElements(true, ele => ele.hasStyleClass && ele.hasStyleClass("sapAppBuildingOutline"))[0]; // sId: "__container0"
    
                if (outlineContainer && outlineContainer.getReactProps) {
                    let parseReactState = state => {
                        let components = {};
    
                        let globalState = state.globalState;
                        let instances = globalState.instances;
                        let app = instances.app["[{\"app\":\"MAIN_APPLICATION\"}]"];
                        let names = app.names;
    
                        for (let key in names) {
                            let name = names[key];
    
                            let obj = JSON.parse(key).pop();
                            let type = Object.keys(obj)[0];
                            let id = obj[type];
    
                            components[id] = {
                                type: type,
                                name: name
                            };
                        }
    
                        for (let componentId in components) {
                            let component = components[componentId];
                        }
    
                        let metadata = JSON.stringify({
                            components: components,
                            vars: app.globalVars
                        });
    
                        if (metadata != this.metadata) {
                            this.metadata = metadata;
    
                            this.dispatchEvent(new CustomEvent("propertiesChanged", {
                                detail: {
                                    properties: {
                                        metadata: metadata
                                    }
                                }
                            }));
                        }
                    };
    
                    let subscribeReactStore = store => {
                        this._subscription = store.subscribe({
                            effect: state => {
                                parseReactState(state);
                                return {
                                    result: 1
                                };
                            }
                        });
                    };
    
                    let props = outlineContainer.getReactProps();
                    if (props) {
                        subscribeReactStore(props.store);
                    } else {
                        let oldRenderReactComponent = outlineContainer.renderReactComponent;
                        outlineContainer.renderReactComponent = e => {
                            let props = outlineContainer.getReactProps();
                            subscribeReactStore(props.store);
    
                            oldRenderReactComponent.call(outlineContainer, e);
                        }
                    }
                }
            }
        } catch (e) {}
    }

     

  • disconnectedCallback() It is fired when the widget is removed from the html DOM of the page (e.g. by hide).
    In this function we reset the state of the “react store subscription” subscribeReactStore that we defined in connectedCallback().

    disconnectedCallback() {
        if (this._subscription) { // react store subscription
            this._subscription();
            this._subscription = null;
        }
    }​
  • onCustomWidgetBeforeUpdate() When the custom widget is updated, the Custom Widget SDK framework executes this function first.
    In this function we set the status of _designMode() as in changedProperties.

    onCustomWidgetBeforeUpdate(changedProperties) {
        if ("designMode" in changedProperties) {
            this._designMode = changedProperties["designMode"];
        }
    }
  • onCustomWidgetAfterUpdate() When the custom widget is updated, the Custom Widget SDK framework executes this function after the update.
    In this function we will check if the widget instance loads for the first time. If yes then it will load the quaggaJS library and call the function loadthis().

    onCustomWidgetAfterUpdate(changedProperties) {
        var that = this;
        if (this._firstConnection === 0) {
            this._firstConnection = 1;
            let quaggaminjs = "http://localhost/SAC/sacbarcodescanner/quagga.min.js";
            async function LoadLibs() {
                try {
                    await loadScript(quaggaminjs, _shadowRoot);
                } catch (e) {
                    alert(e);
                } finally {
                    loadthis(that);
                }
            }
            LoadLibs();
        }
    }

 

JavaScript Functions

  • loadthis()
    This function creates the div HTML element to place the UI elements: sap.m.Label, sap.m.Button and sap.m.Input. If is in design mode, the UI elements will be disabled.

    function loadthis(that) {
        var that_ = that;
    
        let buttonSlot = document.createElement('div');
        buttonSlot.slot = "export_button";
        that_.appendChild(buttonSlot);
    
        that_._Label = new sap.m.Label({
            required: false,
            text: "Barcode value",
            design: "Bold"
        });
    
        that_._exportButton = new sap.m.Button({
            id: "scan",
            text: "Scan",
            icon: "sap-icon://bar-code",
            visible: true,
            tooltip: "Scan Barcode",
            press: function() {
                startScan();
            }
        });
    
        that_._Input = new sap.m.Input({
            id: "scannedValue",
            type: sap.m.InputType.Text,
            placeholder: ''
        });
    
        that_._simpleForm = new sap.ui.layout.form.SimpleForm({
            labelSpanL: 3,
            labelSpanM: 3,
            emptySpanL: 3,
            emptySpanM: 3,
            columnsL: 1,
            columnsM: 1,
            editable: true,
            content: [
                that_._Label,
                that_._Input,
                that_._exportButton
            ]
        })
    
        that_._simpleForm.placeAt(buttonSlot);
        that_._renderExportButton();
    
        if (that_._designMode) {
            sap.ui.getCore().byId("scan").setEnabled(false);
            sap.ui.getCore().byId("scannedValue").setEditable(false);
        }
    }​

    In design mode, the UI elements are disabled.

  • _initQuagga()
    Initialize the Quagga library with a given configuration and registers the method  onProcessed() and onDetected().

    function _initQuagga(oTarget, that) {
        var oDeferred = jQuery.Deferred();
    
        // Initialise Quagga plugin - see https://serratus.github.io/quaggaJS/#configobject for details
        Quagga.init({
            inputStream: {
                type: "LiveStream",
                target: oTarget,
                constraints: {
                    width: {
                        min: 640
                    },
                    height: {
                        min: 480
                    },
                    facingMode: "environment"
                }
            },
            locator: {
                patchSize: "medium",
                halfSample: true
            },
            numOfWorkers: 2,
            frequency: 10,
            decoder: {
                readers: [{
                    format: "code_128_reader",
                    config: {}
                }]
            },
            locate: true
        }, function(error) {
            if (error) {
                oDeferred.reject(error);
            } else {
                oDeferred.resolve();
            }
        });
    
        if (!this._bQuaggaEventHandlersAttached) {
            // Attach event handlers...
    
            Quagga.onProcessed(function(result) {
                var drawingCtx = Quagga.canvas.ctx.overlay,
                    drawingCanvas = Quagga.canvas.dom.overlay;
    
                if (result) {
                    // The following will attempt to draw boxes around detected barcodes
                    if (result.boxes) {
                        drawingCtx.clearRect(0, 0, parseInt(drawingCanvas.getAttribute("width")), parseInt(drawingCanvas.getAttribute("height")));
                        result.boxes.filter(function(box) {
                            return box !== result.box;
                        }).forEach(function(box) {
                            Quagga.ImageDebug.drawPath(box, {
                                x: 0,
                                y: 1
                            }, drawingCtx, {
                                color: "green",
                                lineWidth: 2
                            });
                        });
                    }
    
                    if (result.box) {
                        Quagga.ImageDebug.drawPath(result.box, {
                            x: 0,
                            y: 1
                        }, drawingCtx, {
                            color: "#00F",
                            lineWidth: 2
                        });
                    }
    
                    if (result.codeResult && result.codeResult.code) {
                        Quagga.ImageDebug.drawPath(result.line, {
                            x: 'x',
                            y: 'y'
                        }, drawingCtx, {
                            color: 'red',
                            lineWidth: 3
                        });
                    }
                }
            }.bind(this));
    
            Quagga.onDetected(function(result) {
                // Barcode has been detected, value will be in result.codeResult.code. If requierd, validations can be done 
                // on result.codeResult.code to ensure the correct format/type of barcode value has been picked up
    
                // Set barcode value in input field
                sap.ui.getCore().byId("scannedValue").setValue(result.codeResult.code);
                // Close dialog
                that._oScanDialog.close();
            }.bind(this));
    
            // Set flag so that event handlers are only attached once...
            this._bQuaggaEventHandlersAttached = true;
        }
    
        return oDeferred.promise();
    }
  • startScan()
    This function will be called once user presses the scan button. It creates the dialog box to scan the barcode and initiates the Quagga start() method. The start() method starts the video stream and begins locating and decoding the images.

    function startScan() {
        if (!this._oScanDialog) {
            this._oScanDialog = new sap.m.Dialog({
                title: "Scan Barcode",
                contentWidth: "670px",
                contentHeight: "480px",
                horizontalScrolling: false,
                verticalScrolling: false,
                stretchOnPhone: true,
                content: [new sap.ui.core.HTML({
                    id: "scanContainer",
                    content: "<div />"
                })],
                endButton: new sap.m.Button({
                    text: "Cancel",
                    press: function(oEvent) {
                        this._oScanDialog.close();
                    }.bind(this)
                }),
                afterOpen: function() {
                    // TODO: Investigate why Quagga.init needs to be called every time...possibly because DOM 
                    // element is destroyed each time dialog is closed
                    _initQuagga(sap.ui.getCore().byId("scanContainer").getDomRef(), this).done(function() {
                        // Initialisation done, start Quagga
                        Quagga.start();
                    }).fail(function(oError) {
                        // Failed to initialise, show message and close dialog...this should not happen as we have
                        // already checked for camera device ni /model/models.js and hidden the scan button if none detected
                        MessageBox.error(oError.message.length ? oError.message : ("Failed to initialise Quagga with reason code " + oError.name), {
                            onClose: function() {
                                this._oScanDialog.close();
                            }.bind(this)
                        });
                    }.bind(this));
    
                }.bind(this),
                afterClose: function() {
                    // Dialog closed, stop Quagga
                    Quagga.stop();
                }
            });
        }
    
        this._oScanDialog.open();
    }​
  • loadScript()
    This function loads the external JavaScript library, in this case is quaggaJS.

    function loadScript(src, shadowRoot) {
        return new Promise(function(resolve, reject) {
            let script = document.createElement('script');
            script.src = src;
    
            script.onload = () => {
                console.log("Load: " + src);
                resolve(script);
            }
            script.onerror = () => reject(new Error(`Script load error for ${src}`));
    
            shadowRoot.appendChild(script)
        });
    }
  • createGuid()
    This function is to create the unique ID for HTML div element.

    function createGuid() {
        return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, c => {
            let r = Math.random() * 16 | 0,
                v = c === "x" ? r : (r & 0x3 | 0x8);
            return v.toString(16);
        });
    }

Usage

Upload the Custom Widget

  1. Select Browse and Custom Widgets.
  2. Click + button to upload.
  3. Upload the barcodescanner.json.
  4. Widget is registered and ready to use.

Create and Insert the Widget in Analytics Application

  1. Select Create and Analytics Application.
  2. Insert the widget barcodeScan.
  3. You are ready to go.

 

References:

 

Assigned tags

      2 Comments
      You must be Logged on to comment or reply to a post.
      Author's profile photo Durairaj Athavan Raja
      Durairaj Athavan Raja

      Thanks for the excellent blog. Have couple of clarifications, hope you can help me.

      1. When you use UI5 controls within the custom widget (shadow dom) how’s the global styles work with your controls inside shadow dom? . I tried with microchart and it didnt work. Finally I had to load the library inside the shadow dom using sap.ui.getCore().loadlibrary so that required library and styles are loaded within the shadow dom to make it work.
      2. Also i see that you are using
        that_._simpleForm = new sap.ui.layout.form.SimpleForm({​

      whereas the sap.ui.layout library is not loaded in app.html . Are you loading dynamically? What is the best way to handle dynamic library load?

      Thanks for you time.

      Regards

      Raja

       

       

       

      Author's profile photo Ferry Djaja
      Ferry Djaja
      Blog Post Author

      Hi,

      As far as I understand, the UI5 components are loaded as part of the SAC bootstraps process. So you do not need to load them manually. It is dynamically loaded.

      For my case is: https://sapui5.hana.ondemand.com/1.60.13/resources/

      Regards,

      Ferry