Technical Articles
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 inconnectedCallback()
.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 inchangedProperties
.
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 functionloadthis()
.
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 methodonProcessed()
andonDetected()
.
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 Quaggastart()
method. Thestart()
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
- Select Browse and Custom Widgets.
- Click + button to upload.
- Upload the barcodescanner.json.
- Widget is registered and ready to use.
Create and Insert the Widget in Analytics Application
- Select Create and Analytics Application.
- Insert the widget barcodeScan.
- You are ready to go.
References:
- Build a Custom Widget in SAP Analytics Cloud, Analytics Application
- Barcode scanning with device camera in SAPUI5 applications (without a native container)
- github.com/ferrygun/SACBarcodeScannerCustomWidget
- SAP Analytics Cloud Custom Widget Facebook Page
Thanks for the excellent blog. Have couple of clarifications, hope you can help me.
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
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
Hi Ferry
Sorry, one question. If I want to use UPC/ EAN-13, what a need to modify to read this type of barcode?
Regards!!!