This tutorial will outline the steps required to create a sample inventory management application using RhoElements and RhoMobile Suite. The application will allow a user to scan barcodes and keep a running tally of how many times each item was scanned. To keep things simple, we will not be doing any synchronization in this part of the tutorial and will cover that topic later on.
Before we begin, make sure you have downloaded and installed RhoMobile Suite. Recall that RhoStudio is an Eclipse based IDE that allows you to write a mobile application using HTML, JavaScript and Ruby that will run on multiple mobile platforms. Each mobile platform’s development environment is used in the build process to compile a final executable for that platform. It is imperative that you have this complete development environment setup for each platform you will be building on. The prerequisites and setup for building on each mobile platform (for example, setting up an Android SDK) can be found here. In this tutorial we will begin by developing using RhoSimulator, but the final application will be running on a device.
After launching Zebra RhoStudio, let’s create a new project inside by selecting ‘File/New Project’. You should see a dialog appear like the one below. Select ‘RhoMobile application’ and then click the ‘Next’ button.
In the next screen, name the project “InventoryManagement” – be sure to keep ‘Use RhoElements’ checked and click Finish.
This will create a blank RhoMobile project with a directory structure similar to the one shown in the following image.
Let’s take a look at the file structure for a moment and explain what is going on. The ‘app’ folder, as it may sound, is where we will put the core application files including all of our data models and their corresponding views and controllers (more on that in a moment). The ‘icon’ folder contains the image use for the application icon. The ‘public’ folder will contain some subfolders to hold our CSS and JavaScript files. By default, RhoStudio includes jQuery Mobile/Jquery and corresponding CSS files. For the sake of this example, we will not be replacing any of these and we will be focusing on the ‘app’ folder contents. Recall that a Rhodes application follows a MVC (Model/View/Controller) approach and essentially the application is running a self contained web server. The user interface is primarily determined by the ‘Views’ or ‘.erb’ files. The ‘layout.erb’ file, as the name implies, defines the main layout for the application. If you open it up in RhoStudio, you will see something that looks very familiar to a typical HTML file.
You will notice typical html tags like:
<title>InventoryManagement</title>
You will also see Ruby syntax included in tags that contain percent symbols.
<% if is_bb6 %>
This is what is called inline Ruby syntax. If this file was called ‘layout.html’ then the ruby syntax would not be recognized, but since it has a ‘.erb’ extension, we can include ruby syntax. Typically this is done for conditional reasons, where you would like to change the HTML tags based on some conditions. You can see from the layout.erb file that there are several conditional statements that determine what style sheet to use based on the running operating system the application is executing on. We will come back to this file later on to modify the look of the application, but for now we will leave it alone.
Towards the bottom of the ‘layout.erb’ file you will see <% content %>
This is where the ‘main’ ui will be placed. Initially when the app is launched, the contents of ‘/app/index/erb’ will be used to populate the ‘content’ section. If you open that file up, you will see a small snippet of HTML and inline Ruby:
Before we add the data models we will use for the application, let’s compile the application and see what happens. Highlight your project folder in the Project Explorer (in this example, it is named InventoryMangement). Then from the RhoStudio menu, select ‘Run/Run Configurations’:
Select ‘RhoMobile Application’ and the ‘New_Configuration’ icon underneath. (You may need to right-click ‘RhoMobile Application’ and select New to get the ‘New_Configuration’ icon.) The Run Configurations window appears. Rename the run configuration from New_configuration to InventoryMangement.
Click ‘Run’. Be sure to keep the ‘Simulator type’ as ‘RhoSimulator’. You will see indications in the console tab that the project is compiling:
After a few moments you will see two new windows open: the simulator showing the application, and the ‘Web Inspector’. The simulator window shows how the application will look on the device. The ‘Web Inspector’ is a tool that can be used to help troubleshoot your application.
As you can see, the application does not really do anything except to show the contents of ‘/app/index.erb’:
<div data-role="content"> <ul data-role="listview"> <li><a href="#">Add link here...</a></li> </ul> </div>
The code snippet above is what is shown in the simulator. The ‘data-role’ part of the HTML tags is jQuery syntax that controls the look and behavior of the list item. Let’s close both of these windows so we can start to add the data models we will use for the application.
Right-click on your project folder in the Project Explorer. Select ‘New’ followed by ‘Rhodes model’ and name it ProductCatalog. It will have three attributes – Name, SKU and Image – and it will store information about the products in our inventory. Do not enter spaces between the attributes.
Pay attention to the case you are using for model attributes. Ruby is case sensitive, so it is recommended that you follow the same case sensitive approach through your code. |
Creating a model will add a new folder to the ‘app’ directory with the name of that model along with a set of prebuilt files for different views as well as the model controller to handle the logic for each of these views: index.erb: lists all ProductCatalog items new.erb: form for creating a new ProductCatalog item edit.erb: form for updating an existing ProductCatalog item show.erb: page that shows an existing ProductCatalog item
If you open these files, you will see they look very similar to the /app/index.erb we mentioned before. These files define how the page will look using a combination of HTML and inline Ruby syntax. All of the logic for this model is stored in the product_catalog_controller.rb:
You can think of these as functions or methods that can be executed for this model. As you can see, there is a corresponding method for each view that was created before. When a view is loaded, it will look for the corresponding method in the controller to execute prior to rendering the view. By default, ‘index.erb’ is loaded when no specific view is referenced. As you can see in the example, this will execute the ‘index’ method inside the product_catalog_controller.rb. This file is purely Ruby syntax. It will have Rhodes Mobile APIs available to the application as well as standard Ruby syntax.
Now let’s add this model to our ‘app/index.erb’ so that we can see how these methods are already plugged into the application. We will use these CRUD methods for our real application in the upcoming steps, but in a different manor. Open up the ‘/app/index.erb’ file and replace the line:
Select ‘Run/RunConfigurations’ again and click ‘Run’ after you selected the run project we set up before (it should be selected already). This time we see the new ‘Catalog’ entry:
If you click this, it will change the page to the ‘/app/ProductCatalog/index.erb’ page. We did not need to specify the full URL in the HREF attribute because we are already at the ‘/app’ level. So just saying ‘ProductCatalog’ was enough. Like with traditional web sites, index.html is the default file that is loaded when none is specified and likewise in our case, index.erb is loaded by default. Since we do not have any ProductCatalog records yet, nothing is displayed:
If you click the ‘New’ button it will load the ‘ProductCatalog/new.erb’ file and likewise execute the ‘new’ method in the ‘product_catalog_controller.erb’ file:
This essentially creates a new ProductCatalog item and assigns it to the variable @productcatalog. It also defines what actions to take if the ‘back’ button is pressed on that page.
Go ahead and create an item. It does not matter what you type in, we will be deleting it after you see that the model’s CRUD views and methods are working. Notice that in the ‘new.erb’ file, the POST action for the form is calling the ‘create’ method.
Select that item. Press the ‘Edit’ button. This will change the view to the ‘ProductCatalog/edit.erb’ file along with executing the ‘edit’ method inside the controller. Likewise the POST action for the ‘edit.erb’ form is to call the ‘Update’ method in the controller:
def update @productcatalog = ProductCatalog.find(@params['id']) @productcatalog.update_attributes(@params['productcatalog']) if @productcatalog redirect :action => :index end
Go ahead and delete that record as we will write some code to preload the ProductCatalog with some sample data.
To seed our initial database of products under the ProductCatalog, we will have a JSON file with sample data. In a real world scenario, you may choose to use AJAX and Web Services to retrieve your product catalog or you may also use RhoConnect to keep your product catalog synchronized (we will show you how to do this later on). To create the file, right-click on the ‘public’ folder under the application folder and click on New -> File to bring up the new file dialog. Name the file Products.txt and click Finish. This will create an empty text file which we will populate with sample data. Copy and paste the text below.
[ {"SKU": "12345", "Name": "Converse Hi-Top", "Image" : "%3D%3D"}, {"SKU": "67890", "Name": "Skechers Womens", "Image" : ""}, {"SKU": "12121", "Name": "Pointer Lace-up", "Image" : "%3D"}, {"SKU": "22222", "Name": "Asics Pacewalker", "Image" : "%3D%3D"}, ]
‘SKU’ is the barcode for an item, ‘Name’ is the name of the item and ‘Image’ contains the base64 string representation of the image related to that product. There are tools available online to convert an image to its base64 representation which you can use. This is also referred to as ‘DataURI’ and can be a nice way to handle images without having to have a whole bunch of physical files, you can keep it all in the data scheme. As you may recall, all of the UI is rendered though a browser and the <IMG>
tag displays base64 strings very nicely.
Save the file after copying the JSON string and open the ‘application.rb’ file under the ‘app’ directory. This code gets executed when the application starts up.
Add the following code before the first ‘end’ statement:
#Seed the database with initial data catalog = ProductCatalog.find(:all) if catalog.empty? fileName = File.join(Rho::RhoApplication::get_base_app_path(), '/public/Products.txt') lines = File.read(fileName) jsonContent = Rho::JSON.parse(lines) jsonContent.each { |json| ProductCatalog.create("SKU" => json['SKU'], "Name" => json['Name'], "Image" => json['Image']) } #finished creating database for real application you may # want to display a loading page to indicate to the user # something is going on else #The catalog has been loaded already there is nothing to do end
Save the file and close it. The first line of this code snippet tries to get all the objects inside of the ProductCatalog model. It then checks if the resulting collection is empty, in which case we have no records and we need to seed it with the JSON file. It then opens the ‘Products.txt’ file using the ‘File’ module. The line
Rho::RhoApplication::get_base_app_path()
refers to the Rhodes API for getting the path to the root of the application. After reading and parsing each line, a ProductCatalog entry is created using the data returned from the JSON file.
Let’s run the app in RhoSimulator again (Run/RunConfigurations), and see if the Product Catalog gets loaded. If you do not see the set of products that we had in the seed data, then most likely you did not delete all records from the previous steps. Recall that our code that we added to the initialize will load the seed data if there are no records in the ProductCatalog model. You should see something that looks like:
If you click on a item, you should see that all three properties have data from the seed file. If you do not, then you probably misspelled the properties when creating the ProductCatalog model (ex: ‘Sku’ instead of ‘SKU’). No need to worry, simply delete the ProductCatalog folder from the project and recreate the ProductCatalog model (recall we did not modify any of the prebuilt pages for the ProductCatalog yet).
Notice that the Image field in the detail view will show the raw base64 representation of the image instead of the actual image. To fix this, open up ‘show.erb’ under the ProductCatalog model and change the line
Now that we have our ProductCatalog created and pre-populated, let’s add a second model that will hold our Inventory information. For this model we will keep track of what SKU is scanned as well as the total amount of times it was scanned. Right click on the project’s ‘app’ folder and select ‘New/RhoMobile Model’ and call it ‘Inventory’ with attributes SKU and Quantity.
Open the ‘index.erb’ file under the ‘app’ folder again to add a link to the Inventory viewer. Do this by adding the following line of code under the “View Catalog” list item:
<li><a href="Inventory">View Inventory</a></li>
Save the file, run the project again, and click on the “View Inventory” link inside the app. The resultant page will be blank because we do not have any entries in the Inventory model yet. Click on the ‘New’ button on the top right of this screen to bring up the New Inventory screen with an option to add an SKU and a quantity for that item. Add an SKU number that you have used in the ProductCatalog model, set its quantity to 1 and click on ‘Create’.
This redirects back to the Inventory list with an entry for the inventory item that we just created. Repeat the creation process for two more SKUs.
Let’s now change the Inventory List view slightly to show the quantity and the item image in the same list. By default, you should see a collection of all the entries inside the Inventory model returned in the ‘@inventories’ variable. We will use this collection to change the display of the inventory list. Open the ‘index.erb’ file inside the Inventory folder.
Add the following code after the line <% @inventories.each do |inventory| %>
<% product = ProductCatalog.find( :first, :conditions => {:SKU => inventory.SKU} ) if product name = product.Name if product.Image isrc = product.Image end else name = inventory.SKU end %>
The <% @inventories.each do |inventory| %>
line picks up each Inventory entry using the ‘@inventories’ variable we declared in the code behind file and performs the operations listed above. We look up the ProductCatalog model to find the item’s catalog by SKU using the ‘ProductCatalog.find’ function. If we find an item for it, we get the name of the product and if it has an image, get the base64 of that image from the catalog. If we do not find the product in our catalogs, we simply set the name to be the SKU. To show the name, image and quantity in the list view, change the list item on this page as described below:
The code above sets up list item linking to the ‘edit.erb’ page for an Inventory object and sets up the HTML/CSS elements required to display a thumbnail of the image, the item name and the quantity in the same line. RhoMobile uses jQuery Mobile so we can use CSS class elements from it to construct our UI. Save the file and re-deploy the project. On the home screen, click on ‘View Inventory’ and you should see a list of the three Inventory items we created above, with their image, name and quantity scanned as a list. A sample is shown below:
Clicking on each item will show a details page with the SKU and Quantity as editable fields for that Inventory item. This page, in fact, is the ‘edit.erb’ page under ‘Inventory’. Now that we have setup a catalog, an inventory and a mechanism to display the inventory, we will setup a page that will allow us to scan a barcode to make entries in the Inventory model rather than manually adding them in.
Add a new file to the Inventory model and call it ‘scanPage.erb’. Open the file and add to it the following code:
<div data-role="page"> <div data-role="header" data-position="inline"> <h1>Scan a barcode</h1> <script type="text/javascript"> </script> </div> <div data-role="content"> <div id="divCountTotal"> <a href="/app/Inventory/index" data-role="button" id="numberScanned">0 Items Scanned</a></li> </div> <div id="divImageInventory" style="text-align:center" margin-top="10px"> <img id="iImage" src="/public/images/barcode.jpg" style="max-width:320px"/> </div> <div id="barcode" style="font-size:1.2em"> </div> <div id="productName" style="font-size:1.7em;"> READY TO SCAN </div> <div id="count" style="text-align:center;font-size:4em"> </div> </div> </div>
This code sets up our scan page with a section to add JavaScript (more on that later), a div for displaying the scanned barcode, the name of the item scanned, the number of times a product has been scanned, and the image of the scanned product along with the number of items scanned in total. Notice how we start off with an image ‘barcode.jpg’ as the placeholder for the product’s image. This is just a sample barcode image retrieved from the internet and placed under the ‘public/images’ folder under the root of the project. To test this, add a link to this scan page in the ‘index.erb’ file under the root of the app. Add the following list item below the ‘View Inventory’ link:
<li><a href="Inventory/scanPage">Scan a barcode</a></li>
Deploy the application and click on the ‘Scan a barcode’ link to view the page we just created.
Notice that we setup a button.
<a href="/app/Inventory/index" data-role="button" id="numberScanned">0 Items Scanned</a></li>
This will let the user click on the button to see the complete list of items inventoried (basically the default Inventory/index.erb file we modified before). Notice how we specify the path to the page.
Next, we will setup the controller to trigger the Scanner to start as soon as the scanPage is loaded. Open up ‘inventory_controller.rb’ under the Inventory model. Notice how there are functions for each ‘.erb’ file in the controller. This indicates the behavior of the application when a page is called so that the developer can add additional functions before the page loads. For example, we need to add code to turn the Scanner on before the app directs to ‘scanPage.erb’. To do this, we add a function inside the controller called ‘scanPage’ and turn the Scanner on before rendering ‘scanPage.erb’. The ‘scanPage’ function looks like:
def scanPage Scanner.decodeEvent = url_for(:action => :decodeEventCallback) Scanner.enable render :back => '/app' end
This function calls the Rhodes ‘Scanner’ class and firsts assigns a decode event callback. This function is what gets executed on a successful barcode decode. The ‘Scanner.enable’ line makes the scanner active for this page and the user can press the scanner button on the device to bring up the viewfinder and perform a scan. After enabling the scanner, the function renders the ‘scanPage.erb’ page and attaches the Back action to display the home screen, i.e. the ‘index.erb’ at the root of the ‘app’.
At the time of writing this tutorial, the Scanner object is only available on Zebra Hardware. You will need a device at this point to complete the remainder of the project. Alternatively, I could have chosen the Barcode object with slightly different syntax. Please consult the complete list of Mobile APIs as well as the API Compatibility Matrix to see which APIs are supported on which platforms. |
The ‘decodeEventCallback’ function is given below:
def decodeEventCallback barcode = @params['data'] product = ProductCatalog.find( :first, :conditions => {:SKU => barcode} ) if product.nil? product = ProductCatalog.create({"SKU" => barcode, "Name" => "Unknown Product", "Image" => "%3D"}) end name = product.Name if product.Image WebView.execute_js("setImgSrc('"+product.Image+"'); ") end inventoryItem = Inventory.find( :first, :conditions => {:SKU => barcode} ) if inventoryItem.nil? inventoryItem = Inventory.create( {"SKU" => barcode, "Quantity" => 1} ) else itemCount = inventoryItem.Quantity.to_i itemCount += 1 inventoryItem.update_attributes( {"Quantity" => itemCount.to_s} ) end inventoryItem.save if itemCount.nil? itemCount = 1 end getInventoryCount WebView.execute_js("setProductData('"+name+"','"+barcode+"','"+itemCount.to_s+"'); ") end
Let’s look at this function line by line. When the decode event returns after a successful decode, it contains a few parameters (a full list is provided in the RhoElements API for Scanner) out of which we extract the ‘data’ element which contains the barcode that was just scanned. With that barcode, we find the ProductCatalog entry we may have for that product by matching the ‘barcode’ variable to each product’s ‘SKU’. If we do not have an entry for that barcode, we create a new one with a default ‘Name’ and ‘Image’.
If we have an entry in our ProductCatalog model, we fetch the item’s name and image to be displayed on the page. If we find an image, we want to display it on the scanPage.erb. In order to communicate from the Ruby controller to the rendered page, you must use the Rhodes WebView object. This object has a function called ‘execute_js’ which allows you to call JavaScript functions as well as pass variables to that function:
WebView.execute_js("setImgSrc('"+product.Image+"'); ")
This will execute the JavaScript function ‘setImgSrc’ (which we will create in a second) from code behind and pass as arguments the base64 representation of the image we fetched from the ProductCatalog model.
The decode event callback then tries to find an entry in the Inventory model for the product that we just scanned so that we can fetch the Quantity parameter to determine how many of this product have we already scanned. If the result of the find is nil, it means we have no entry for this product in our Inventory model in which case we create an entry by using ‘Inventory.create’ and set its Quantity to 1. This ‘create’ function, as well as others, can be executed on any data model that you have created. We pass in a Ruby hash that essentially has a list of property names and their values.
If we do find an entry in the Inventory model, we fetch the Quantity saved in the database, convert it to an integer by using ‘Quantity.to_i’, increment it by one and update that record by using ‘inventoryItem.update_attribute’. Finally we save this change by executing an ‘Inventory.save’. The block of code after this step that checks to see if the itemCount is nil is used when we scan an item for the first time. In that case, the itemCount will be nil and we initialize it to 1 just for ease of displaying on the screen. Then we call a ‘getInventoryCount’ method in the controller to count the total number of items we have scanned. Create this method in the inventory_controller.rb:
def getInventoryCount items= Inventory.find( :all ) scanCount = 0 items.each do |item| scanCount = scanCount + item.Quantity.to_i() end WebView.execute_js("setProductCount('"+scanCount.to_s+"'); ") end
This method first gets a collection of all the items inside of the Inventory model, iterating through each of them and fetching the Quantity field and adding it to our ‘scanCount’ parameter. We then call another JavaScript function to display the total.
Now we have added all of the code to the controller, let’s go back to the scanPage.erb and add the JavaScript functions we are calling.
$('[data-role=page]').live('pageshow', function (event) { $.get('/app/Inventory/getInventoryCount', { }); }); function setProductData(name, SKU, count) { if (!document.getElementsByTagName) return; document.getElementById("barcode").innerHTML = SKU; document.getElementById("productName").innerHTML = name; document.getElementById("count").innerHTML = count; } function setProductCount(scanCount) { var itemtext = scanCount + " Items Scanned"; $('#numberScanned .ui-btn-text').text(itemtext) } function setImgSrc(data) { document.getElementById("iImage").src = data; }
In the above code you will see mention of one JavaScript function we did not speak about before.
$('[data-role=page]').live('pageshow', function (event) { $.get('/app/Inventory/getInventoryCount', { }); });
This statement is jQuery syntax that effectively executes a function when the page is completely ready to handle JavaScript. The $.get method is an Ajax passing the URL as a parameter. In order to communicate from the webpage via JavaScript to the Ruby controller methods, is through Ajax. The URL we pass is calling the getInventoryCount method we added inside the controller above. Like we mentioned before, this method in turn uses the WebView.execute_js function to talk back to JavaScript and the rendered page.
One other thing we will do here is remove the default Rhodes toolbar, since we want our UI to take up most of the real estate. This is easily accomplish by going back into the ‘app/application.rb’ file. Look for the line:
#@@toolbar = nil
And remove the ‘#’ to uncomment the line.
That’s it! We just finished creating our inventory management application. Since we are using the ‘Scanner’ class from the Rhodes framework, we will need to deploy this application on a device in order to see the Scanner library. Attach a device to your computer and re-deploy the application, only this time selecting ‘device’ as the simulator type and the platform corresponding to your device. While creating this application, a Windows Mobile device was used and the screenshot below reflects that. The application will take longer to deploy on a device than a simulator and hence it is advisable to keep an eye on the Console window inside of Eclipse to determine when deployment has completed before using the application.
If you close the application and reload it, you will see that all items in the data models have been retained. We could now spend some time modifying the style of the pages and changing the page flow, but for now we will stop here. In another tutorial we will show you have to add RhoConnect data synchronization services so that the ProductCatalog can some from a centralized source as well as show how to get the Inventoried records up to a backend for storage.