Thursday 31 January 2013

Microsoft Dynamics CRM Compatibility List

Ever wanted to know if a specific product version or service pack are considered to be compatible and supported with CRM 4.0 or 2011? Microsoft have now released a new "living" KB article that describes the current support stance for most of the products you would consider using with CRM.

This is especially useful for situations where you want for example to know what latest Exchange Server major and minor versions are supported etc.

The article is here: KB2669061

Update Rollup 6 is now part of the installation files

Happy days!!!

As of the latest update (20/01/2013) to the downloadable server installation files for CRM 2011, the installation files now include CRM Update Rollup 6 (Build 05.00.9690.1992)... yay!

This will definitely help with the long winded multi-step patching process we have had to use up to now, especially when installing multiple front end servers.

You can find the installation files on the Microsoft Download Center here.

Friday 3 December 2010

Infinite Loop Detection in CRM Workflows

Annoyingly, if you have a workflow that calls itself or a child workflow more than 7 times in an hour, you've probably runi in to CRM infinit loop detection. Basically, if CRM detects this happening it will step in and shut down the whole workflow.

However, you can chage the setting (provided of course that you know what you are doing and that your workflows are well written) to a greater number. You do this using the CRM Deployment Config Tool.

For example, if oyu need to up the number of loos from 7 to 20, you could use the following command:

microsoft.crm.deploymentconfigtool.exe workflowsettings update -maximumdepth:20

Happy testing!

Tuesday 21 September 2010

Internet Explorer 8 and broken menus in CRM 4.0

** Please note, this issue is fixed un Update Rollup 9**

I noticed this one ages ago, but I've never really taken the time to sit down and work out what was going on.

Ever since the release of IE 8, I've noticed that the menus seem to overlap in very odd ways. In certain circumstances you even get pop-out menus completely covering the menus underneath them like the image on the side of the page.

Now this is obviously less than idea. Functionality it doesnt make a big difference, but aesthetically it can be very annoying to users... and annoyed users are never fun to deal with! ;)

So I noticed that this was happening predominantly when the CRM server was in the local client's Trusted Sites internet security zone. So after a bit of trial and error I managed to isolate the specific setting causing my headaches... There is a setting called 'Allow script initiated windows without size or position constraints'. This is set to Disable by default in the Trusted Sites security zone. Setting it to Enable solves the issue:

Saturday 21 August 2010

Can I use a picklist/lookup as my primary attribute?

In my experience it's safe to say that CRM's approach of always needing the primary attribute of a custom entity to be an nvarchar attribute  unfortunately doesn't fit all business scenarios.

Take the example where an organisation has a complex product management scenario where the standard product functionality has to be suppressed and replaced with something more suitable. Within the Account record you may want to see all of the products owned by the organisation. If you have a standard catalogue of products, having to name each one whenever a customer acquires said product is far from ideal.

There are a couple of ways of dealing with this, but I generally like to use JavaScript to plug the 'gap'.

So say we want to have a picklist of 'Product' for our primary attribute on our new product entity, we could do the following:
  1. Create the new entity, schema name of new_product.
  2. In the new entity, give the primary attribute a name of new_productname (Name this 'Product Name').
  3. Create a new pick-list attribute called new_productnamepicklist (Also name this 'Product Name').
  4. Place the picklist attribute front & centre on your entity
  5. Place the primary attribute elsewhere and disable it... I generally use a 'System' section on an 'Admin' tab.
Now we just need some JavaScript to automatically populate the new_productname attribute with the value from new_productnamepicklist attribute. Place the following code in the onChange event of the new_productnamepicklist attribute:

var CRM_FORM_TYPE_CREATE = 1;
var CRM_FORM_TYPE_UPDATE = 2;

if ((crmForm.FormType == CRM_FORM_TYPE_CREATE) || (crmForm.FormType == CRM_FORM_TYPE_UPDATE)){
    //get our value from the picklist - make sure you use SelectedValue, not DataValue as we want the text, not the item code.
    var recName = crmForm.all.new_productnamepicklist.SelectedValue;

    //Set the attribute name
    crmForm.all.new_productname.DataValue = recName;
    //Don't forget to force submit as the attribute is disabled on the form.
    crmForm.all.new_productname.ForceSubmit = true;    
}

Please be aware that while this particular approach is very simple to implement, it has its limitations. As the work is being done on the form, the onChange event will not fire if the record is updated as part of a bulk edit, or if it is created via an import or through the web services etc. If you need a more sophisticated solution to handle those scenarios, you will want to use a plugin - I'll try and write a follow-up with more detail on that approach when I get a chance.

Lastly, if you face this challenge, please be sure to give some thought to your views. If you look at a list of products, there are few things less helpful than seeing row after row of a single product. You will need to add extra attributes to your view columns to make it easy for your user to identify the data they need.

Thursday 19 August 2010

CRM build and Update Rollup numbers

Ever needed to figure out what Update Rollup was running on a CRM server. Hopefully this table will help you.

VersionBuild NumberReleased on
RTM4.0.7333.319/12/2007
Rollup 14.0.7333.111324/11/2008
Rollup 24.0.7333.1312 or 4.0.7333.131615/01/2009, 08/02/2009
Rollup 34.0.7333.140812/03/2009
Rollup 44.0.7333.15510705//2009
Rollup 54.0.7333.1644 or 4.0.7333.164502/07/2009
Rollup 64.0.7333.175027/09/2009
Rollup 74.0.7333.213822/10/2009
Rollup 84.0.7333.254217/12/2009
Rollup 94.0.7333.264411/02/2010
Rollup 104.0.7333.274108/04/2010
Rollup 114.0.7333.2861 or 4.0.7333.286203/06/2010
Rollup 124.0.7333.293502/08/2010

I'll try to keep it as up to date as possible.

Changing the label of a field and a section

I've previously written about changing the name of a field dynamically using client script. Well now a requirement has come up to not only change the label of an attribute, but also a section heading... easy right? No chance!

As it turns out, with an attribute we have a nice and easy element Id that we can always use to address that field in confidence that we are looking at the right thing. We don't have this with section labels. So my problem was two-fold:
  1. Find a way to iterate through all the sections until I found the one that I wanted, and then change its name.
  2. Then I had to come up with a way of storing that change, so that next time I needed to rename it I knew what to look for; once I've  renamed the section, the original name I used to find it obviously wont work anymore (Duh).
Blatant plagiarism note: Big thanks to Irfan Saeed who did all the heavy lifting with regards the JavaScript to actually rename the sections.

So now that I have a way to find and rename the section, all I need is a way to keep a handle on what the current name is... enter the global variable:

While generally speaking best practice says that you should always prefix your JavaScript variable declarations with the 'var' keyword, it is legal to omit said keyword. If you don't use the 'var' keyword, then the script interpreter will grant that variable global scope... meaning that we can access it and use it anywhere on our form. Sounds like a perfect way to keep a handle on the current name of our se3ction doesn't it?

So in order to do this, create a variable in the forms onLoad event, omitting the 'var' keyword as discussed:

currentSectionName = "";

Now in the onChange and onLoad events of our attribute and form respectively, paste the following code (obviously updating the field names to your own requirements:

//only do something if there is data in the attribute
if (crmForm.all.new_projectcategory.DataValue != null) {
    //get the attribute value
    var pCat = crmForm.all.new_projectcategory.DataValue;

    //define the default section name
    var defaultSectionName = "Delivery";

    //check if the current section name is different to the default name
    if (currentSectionName == "")
        currentSectionName = defaultSectionName;

    //work through the values
    switch (pCat) {
        case "1":
            crmForm.all.ownerid_c.innerText = "Project Manager";
            if (currentSectionName != "Delivery") { UpdateSectionName("tab0", currentSectionName, "Delivery"); }
            currentSectionName = "Delivery";
            break;
        case "2":
            crmForm.all.ownerid_c.innerText = "Project Administrator";
            if (currentSectionName != "Implementation") { UpdateSectionName("tab0", currentSectionName, "Implementation"); }
            currentSectionName = "Implementation";
            break;
    }
}

function UpdateSectionName(TabNumber, CurrentSectionName, NewSectionName) {
    var anchorNode = document.getElementById(TabNumber);
    var secBarCss = "ms-crm-Form-SectionBar";
    var addrSecElm = null;
    var results = getElementsByClassName(secBarCss, anchorNode);
    for (var i = 0; i < results.length; i++) {
        if (results[i].innerText == CurrentSectionName) {
            addrSecElm = results[i];
            break;
        }
    }
    if (addrSecElm != null) {
        addrSecElm.innerText = NewSectionName;
    }
}

function getElementsByClassName(className, anchorNode) {
    if (!anchorNode) anchorNode = document.body;
    var result = [];
    var regEx = new RegExp("\\b" + className + "\\b");
    var children = anchorNode.getElementsByTagName("*");
    for (var i = 0; i < children.length; i++) {
        if (regEx.test(children[i].className)) result.push(children[i]);
    }
    return result;
}

And there you have it; now when we change our field, we are dynamically updating both the attribute and section labels.

Before:


After:



Note: Please be aware, while this approach works, it is currently unsupported by Microsoft so it may stop working in a future release.