Thursday, May 1, 2014

Salesforce1 Developer Week Challenge walk-through....

Hopefully Salesforce doesn't mind me posting this but I was working on it with someone and it was getting late and I said I'd post my solution.  Anyway, I went to the Salesforce1 developer week Meetup, in Denver, CO, last night and it was awesome!  As usual it was a fun event with a great atmosphere and a lot of cool people to mingle with.  Anyway, onto the code!

Here was the challenge and the requirements:

Challenge 
Take the shipment application page included and integrate it into Salesforce1.  Add the ability to quickly access the page from the mobile app, and use the mobiles app's native functionality to create a shipment request.

Requirements
  1. Install schema package : http://bit.ly/sfdc-shipment
    • If you already have the Warehouse package and the Mobile design Templates installed, this has a few components that will conflict, you may want to get a new org, at least that's what I did.     
  1.  Add a Javascript publisher method to the globalShipmentPage that will enable the 'Submit' button when someone selects a merchandise record from the start page.
  2. Add a Javascript subscription method to the page that will invoke the remote action 'insertShipment' in the included globalShipmentController with the appropriate fields from the form and close the publisher on success.
  3. Create a Visualforce global action for the globalShipmentPage and add it to the global layout.
  
 Example of the completed action.
The Work

Installing the package in your org will create a new App called 'Shipments' with the following tabs : Merchandise and Shipments.  You'll want to go in and add a few Merchandise items first.  Next, I went and skipped ahead to step 4, so that I could do testing while I worked on my code, using /one/one.app appended on to the end of my org url, in Chrome.  (ie : https://na10.salesforce.com/one/one.app)

So, for step 4, I went to Develop > Pages > globalShipmentPage : Edit, and checked the Available for Salesforce mobile apps checkbox, to make it available in Salesforce1.  The next step was to go create a Global Action.  To do this navigate to Create > Global Actions : New Action.  As long as the mobile checkbox has been checked on the VF page, you should have it listed in the Visualforce Page dropdown on the setup screen. Here is an example :
 

Once you are done with this, click Save.  Now navigate to Chatter > Publisher Layouts and Edit the Global Publisher Layout.  Drag your newly created Publisher Action to the Publisher Actions area and click SaveAny Page layout that is using the Global Layout will now have this Action Available when clicking the "+" in the lower corner of the screen.

Now back to step 2, adding the code to enable the "Select" button!  Open the developer console and from there : File > Open > Pages : globalShipmentPage.  To enable the submit button or any of the other publisher events, the first thing you have to do is include the following javascript library UPDATED

Apparently, this script link no longer works, or I was getting errors in my dev org while using it for something else.
<script type='text/javascript' src='/canvas/sdk/js/publisher.js'></script> 

The new link is :

<script type='text/javascript' src='/canvas/sdk/js/29.0/publisher.js'></script>

This has actually already been included on line 18 of the provided code!

You can find all of the publisher Events in the SF1 Cheat sheet, which can be found here.
To enable the 'Submit' button you need to include the following code somewhere in your code :

Sfdc.canvas.publisher.publish({ name : "publisher.setValidForSubmit", payload:"true"});

Since we don't want the publisher button to be available before a piece of Merchandise has been selected, I placed the code in the function that gets appended to the link created for each merchandise item in the search results, at line 112 : newLink.click(function(e) { ...  

 newLink.click(function(e) {
                                e.preventDefault();
                                           
                                $("#merchName_result").text(mName);
                                $("#merchId_result").text(mId);
                                $("#searchPage").hide();
                                $("#resultPage").show();
                                Sfdc.canvas.publisher.publish({ name : "publisher.setValidForSubmit", payload:"true"});
                            });


Now when a user selects a merchandise item, it will enable the submit button.

The next step is to add the code which  will run when the user clicks the "Submit" button.  I added this at line 38, inside of the jquery.ready function 

Sfdc.canvas.publisher.subscribe({name: "publisher.post", onData:function(e) {submitShipment(); }});  

The submitShipment function, is the callback that is called when the button is clicked.  This is where we will capture our values and then call our remote action from, in the following step!

Step 3!  Inside the submitShipment function, the first thing we need to do is grab our values from the user input.  I created some variables to store the data in and then went ahead and just used jQuery selectors to assign them.  There are a few ways that it could be done though...

 var merchId, quantity, status, receivedAt;           
           
 //get doc id values and assign to variables...
        merchId = $("#merchId_result").text();
        quantity = $('input[id$=shipment_quantity]').val();
        status = $('select[id$=shipment_status] option:selected').val();
        receivedAt = $('input[id$=shipment_receivedAt]').val();
Next, I checked to make sure the quantity, status, and received date, all had values.  If they do not, I alert the user.  If they do, I call my remote action from the globalShipmentController.

if(quantity == '' || status == '' || receivedAt == '')
            {
                alert("All values must be entered to submit a shipment!");
             } else {
               
        //visulaforce remote action call with parameters passed to it
        Visualforce.remoting.Manager.invokeAction(
        '{!$RemoteAction.globalShipmentController.insertShipment}',
        merchId, quantity, status, receivedAt,
        function(result, event){
            if (event.status) {
               
                //if the result is good, this will close the publisher action
                Sfdc.canvas.publisher.publish({ name : "publisher.close", payload : { refresh:"true" }});
            } else if (event.type === 'exception') {
                alert(event.message);
               
            } else {
                alert(event.message);
              
            }
        },
        {escape: true}
    );
       
      } 

If you are unfamiliar with  VF remote Actions, you can read about them here and there is a nice example here.

You are pretty much done now.  If you haven't been testing this, you can do so either by logging into Salesforce1 on your mobile device or as I said earlier, by going to https://yourinstance.salesforce.com/one/one.app, then by navigating to a record for an object that has the global publisher layout and clicking on the publisher action icon.  An easy place is from the Feed!  Chrome is a nice way to test this since you can use the web dev tools, to see what is going on. Once you create a new shipment, you can click on your Feed from the top of the Salesforce1 navigation Menu, at the left of the screen; and will see a feed item for your new shipment record!

Hopefully this walkthrough has made sense and was fun to use while completing the Salesforce1 Code Challenge!

I want to thank Salesforce for a fun night and event, the free drinks, the shirt, and the SF1 Hoodie!

If you would like to  learn more about how Salesforce1 works and the framework it is developed on, check out this video on Aura.   



          

 



 



1 comment: