Tuesday, February 14, 2017

Using Salesforce standard modal in a Detail page button - Part 2

As illustrated in Part 1, we looked at opening the modal which come with Salesforce and changing its content.

But most of the time, you want the content in the JavaScript detail button to be dynamic. So instead of writing a lot of JavaScript code and encoding the HTML content into a string to then pass it to the setContentInnerHTML function, we could display a Visualforce page.

You just need to modify it as below and change the source (src attribute).
box.setContentInnerHTML(
  '<iframe src="/apex/modaldemo" style="border:none;" width="100%" height="'+ (height - 30)+'px" />'
);

We have now added a Visualforce page, which is kind of awesome. But what if we want to some processing and at the end close the modal from within the Visualforce page?

Well, this is not possible since you don't have access to the modal instance in the Visualforce page.
Fortunately we do have an alternative way which is the use of postMessage.

How does it work? 
First we need to the Visualforce page to send the message once the action is done. Which can be a click on button, a progress bar reaching 100%, an action function which has completed or many other scenarios. 

You just need to execute the following JavaScript code:
window.parent.postMessage('close', '*');

Now we need to capture this message from the JavaScript on the button and then close the modal. Here is how we do it. 
window.addEventListener('message', function(event){
  if (event.data === 'close'){
    box.hide();
  }
});

Below is the full implementation: 
Your visualforce page
<apex:page showHeader="false" sidebar="false" >
    <script type="text/javascript">
      var closePopup = function(){
        window.parent.postMessage('close', '*');
      }
    </script>
    <apex:form >
        <apex:pageBlock title="This is a visualforce page">
            <apex:pageblockButtons location="bottom" >
                <apex:commandButton onClick="closePopup();return false;" value="Close" rerender="none" />
            </apex:pageblockButtons>
        </apex:pageBlock>
    </apex:form>
</apex:page>
JavaScript to be included in the detail page button
(function() {
    var width = 700;
    var height = 200;
    var title = "Dialog title here";

    var box = new SimpleDialog("salesforce" + Math.random(), true);
    box.setTitle(title);
    box.displayX = true;
    box.isMovable = false;
    box.createDialog();
    box.setWidth(width);
    // set your html content here
    box.setContentInnerHTML(
      '<iframe src="/apex/modaldemo" style="border:none;" width="100%" height="'+ (height - 30)+'px" />'
    );
    //set the height of the modal
    box.dialog.children[1].style.height = height + 'px';
    box.show();

    //if displayX is set to true, then override standard close event by this code
    box.dialog.getElementsByClassName('dialogClose')[0].onclick = function() {
      box.hide();
      // you can add code to reload the page or redirect to another page 
    };
    //optional : add this to close the modal onclick of the overlay background
    box.background.addEventListener('click', function() {
      box.hide();
    });
    
    // if you want to be able to close the modal from the Visualforce page
    window.addEventListener('message', function(event){
      if (event.data === 'close'){
        box.hide();
      }
    });
})();


Hope you like it. :)

3 comments:

  1. Hi Kevan,

    The showbox function is not working in lightning. is there any alternative for this or any comments from you.
    I am getting DOM exception.
    ERROR: Uncaught DOMException: Blocked a frame with origin "https://hp--ngitdev--c.cs8.visual.force.com" from accessing a cross-origin frame.
    at showBox (https://hp--ngitdev--c.cs8.visual.force.com/apex/OMGetOpportunityOrderListPage?id=0061V00000h7FvXQAU&tour=&isdtp=p1&sfdcIFrameOrigin=https://hp--ngitdev.lightning.force.com&sfdcIFrameHost=web&nonce=0e2b097567482f81256e066fc4dfa917350111858dbe54a803b1d80747009f42&ltn_app_id=06m270000002x6ZAAQ&clc=0:30:30)

    Thanks and Regards,
    Sadik

    ReplyDelete
  2. HI Kevan
    Showbox functionalityis not working on onclick function in vf page

    ReplyDelete
  3. Hi Rocky,

    It has been a long time since I haven't used this. There may have been a change by Salesforce.
    I shall try to reproduce this again and keep you updated.

    ReplyDelete