Tuesday, May 7, 2013

Close Opportunity using Jscript


function UpdateStatus(state, status, entityname) {

    var request = "<s:Envelope xmlns:s=\"http://schemas.xmlsoap.org/soap/envelope/\">";
    request += "<s:Body>";
    request += "<Execute xmlns=\"http://schemas.microsoft.com/xrm/2011/Contracts/Services\" xmlns:i=\"http://www.w3.org/2001/XMLSchema-instance\">";
    request += "<request i:type=\"b:SetStateRequest\" xmlns:a=\"http://schemas.microsoft.com/xrm/2011/Contracts\" xmlns:b=\"http://schemas.microsoft.com/crm/2011/Contracts\">";
    request += "<a:Parameters xmlns:c=\"http://schemas.datacontract.org/2004/07/System.Collections.Generic\">";
    request += "<a:KeyValuePairOfstringanyType>";
    request += "<c:key>EntityMoniker</c:key>";
    request += "<c:value i:type=\"a:EntityReference\">";
    request += "<a:Id>" + Xrm.Page.data.entity.getId() + "</a:Id>";
    request += "<a:LogicalName>" + entityname + "</a:LogicalName>";
    request += "<a:Name i:nil=\"true\" />";
    request += "</c:value>";
    request += "</a:KeyValuePairOfstringanyType>";
    request += "<a:KeyValuePairOfstringanyType>";
    request += "<c:key>State</c:key>";
    request += "<c:value i:type=\"a:OptionSetValue\">";
    request += "<a:Value>" + state + "</a:Value>";
    request += "</c:value>";
    request += "</a:KeyValuePairOfstringanyType>";
    request += "<a:KeyValuePairOfstringanyType>";
    request += "<c:key>Status</c:key>";
    request += "<c:value i:type=\"a:OptionSetValue\">";
    request += "<a:Value>" + status + "</a:Value>";
    request += "</c:value>";
    request += "</a:KeyValuePairOfstringanyType>";
    request += "</a:Parameters>";
    request += "<a:RequestId i:nil=\"true\" />";
    request += "<a:RequestName>SetState</a:RequestName>";
    request += "</request>";
    request += "</Execute>";
    request += "</s:Body>";
    request += "</s:Envelope>";

    //send set state request
    $.ajax({
        type: "POST",
        contentType: "text/xml; charset=utf-8",
        datatype: "xml",
        url: Xrm.Page.context.getServerUrl() + "/XRMServices/2011/Organization.svc/web",
        data: request,
        beforeSend: function (XMLHttpRequest) {
            XMLHttpRequest.setRequestHeader("Accept", "application/xml, text/xml, */*");
            XMLHttpRequest.setRequestHeader("SOAPAction", "http://schemas.microsoft.com/xrm/2011/Contracts/Services/IOrganizationService/Execute");
        },
        success: function (data, textStatus, XmlHttpRequest) {
       window.location.reload(true);
        },
        error: function (XMLHttpRequest, textStatus, errorThrown) {
            alert(errorThrown);
        }
    });

}

XRM 2011 - Microsoft Dynamics CRM 2011 Style Buttons


Adding a button to a Form is a great way to add additional functionality right where the user is already focused. Today we are looking at a few different approaches to adding a button. First, we’re looking at handling the upgrade for those of you who have already used the 4.0 button inside of 2011 or if you had a 4.0 environment and are upgrading. If you have a clean 2011 environment, then feel free to skip down to the “From Scratch” section.

There is already some code floating around to create a 4.0 style button inside of CRM 2011. The button we’re creating will instead create a CRM 2011 style button (i.e. “Example Button”).
image

The CRM 2011 button starts gray and then when the user hovers over it, the button will look like:
image

Creating our Web Resource

First,name your Web Resource whatever you’d like. In my case, I’m planning to re-use it in various ways, so I’m throwing it in our JS root directory.
image

   1: // CRM 2011 Style Button 
   2: // Creates a button from a form field 
   3: // Paul Way - 1/3/2012
   4: function ConvertToButton(fldName, btnLabel, btnWidth, evt){ 
   5:    var btn = '<button id="btn_' + fldName + '" ' + 
   6:                     ' style="width:' + btnWidth + '" ' + 
   7:                     ' class="ms-crm-Button" ' + 
   8:                     ' onmouseover="Mscrm.ButtonUtils.hoverOn(this);" ' + 
   9:                     ' onmouseout="Mscrm.ButtonUtils.hoverOff(this);" ' + 
  10:                  '>' + btnLabel + '</button>'; 
  11:  
  12:    var ctrl = Xrm.Page.ui.controls.get(fldName)._control;
  13:  
  14:    // Add the new button 
  15:    ctrl.get_element().innerHTML += btn;
  16:  
  17:    // Hide the textbox 
  18:    ctrl.get_element().firstChild.style.display = 'none';
  19:  
  20:    // Hide the label (optional) 
  21:    Xrm.Page.ui.controls.get('pager').setLabel('');
  22:  
  23:    // Add Event to the newly created button 
  24:    ctrl.get_element().childNodes[1].attachEvent('onclick', evt);
  25:  
  26: }

Modifying our Form

To use this new function, you’ll first need to add the newly created web resource to the form. You’ll also want to have a separate JS web resource to call the ConvertToButton function. In my case, I already have a web resource for contact specific JavaScript. If I didn’t, I would need to create a new web resource and then place my contact specific code there. Here’s an example screenshot:
image

By having two web resources for this, you’ll have only one version of your ConvertToButton function throughout your CRM 2011 environment. Let’s say down the road you’ll need to update the function to change the label to “Button”, then you only need to modify one file. In CRM 4.0, you probably had a lot of duplicate code. With CRM 2011, your browser can cache the JavaScript for better performance and it is easier to maintain when organized appropriately.
In our other web resource, the one we setup with the OnLoad, we need the following code.
   1: function contactsOnLoad(){ 
   2:    convertToButton('pager', 'Example Button', '150px', function(){alert("test")}); 
   3: }

Why is this so different then the 4.0 code?

If you’ve used the CRM 4.0 code, you’ll notice the code here is a lot shorter. It’s actually pretty different as well.
  • For one, instead of modifying the input element we are actually creating a button HTML element. This just means that we can now use about any kind of field instead of just pure textboxes. Not a huge deal, but opens up the email address 3 and other attributes.
  • Both are really unsupported but the other uses the deprecated crmForm.all.
  • Finally, the button style was meant for 4.0 whereas now we have the 2011 look-n-feel.

From Scratch

So far we’ve mainly focused on if you already were using the 4.0 code and were upgrading. But what if you don’t have the existing code structure? I’d argue against creating a new attribute just to have a button. When adding a new attribute, you are also adding the attribute to the underlying SQL tables and views. Using an existing field isn’t really a great option either because there is always the chance you will need the field or it will overlap with an application from the MS marketplace.
Instead, I’d recommend creating a button place holder web resource and then embedding the web resource on the page. The web resource is just a JPG image like this:
image

We then place the image wherever we want on the form. Make sure to set the formatting to one column and one row.
image
image

Finally, we need a little bit of code added to the /js/formButton.js web resource:
   1: // CRM 2011 Style Button 
   2: // Creates a button from a form field 
   3: // Paul Way - 1/3/2012
   4:  
   5: function convertWebResourceToButton(fldName, btnLabel, btnWidth, leftMargin, evt){ 
   6:    var btn = '<button id="btn_' + fldName + '" ' + 
   7:                        ' style="margin-left:' + leftMargin + ';width:' + btnWidth + ';" ' + 
   8:                        ' class="ms-crm-Button" ' + 
   9:                        ' onmouseover="Mscrm.ButtonUtils.hoverOn(this);" ' + 
  10:                        ' onmouseout="Mscrm.ButtonUtils.hoverOff(this);" ' + 
  11:                      '>' + btnLabel + '</button>'; 
  12:    
  13:    var ctrl = Xrm.Page.ui.controls.get(fldName)._control.get_element().childNodes[1];
  14:  
  15:    // Replace image with buttom 
  16:    ctrl.innerHTML = btn; 
  17:    
  18:    // Add Event to the newly created button 
  19:    ctrl.firstChild.attachEvent('onclick', evt);
  20:  
  21: }
And as for our OnLoad function, we’ll need to use this instead:
   1: function contactsOnLoad(){ 
   2:    convertWebResourceToButton('WebResource_btnProfInfo', 'Another Approach', '150px', '119px', function(){alert("test2")}); 
   3: }
Thanks to:  Paul Way