Technology Blogs by Members
Explore a vibrant mix of technical expertise, industry insights, and tech buzz in member blogs covering SAP products, technology, and events. Get in the mix!
cancel
Showing results for 
Search instead for 
Did you mean: 
ferrygun18
Contributor
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:

 
3 Comments
Labels in this area