我是微软Dynamics 365 & Power Platform方面的工程师罗勇,也是2015年7月到2018年6月连续三年Dynamics CRM/Business Solutions方面的微软最有价值专家(Microsoft MVP),欢迎关注我的微信公众号 MSFTDynamics365erLuoYong ,回复393或者20200212可方便获取本文,同时可以在第一间得到我发布的最新博文信息,follow me!

Dynamics 365 Customer Engagement V9版本新增了客户端API:Xrm.WebApi.online.execute ,官方文档请参考:​​Xrm.WebApi.online.execute (Client API reference)​​ 。

示例中举了非绑定操作(Action)的调用,但是没有绑定操作的调用示例,Dynamics MVP Debajit Dutta 的文章 ​​Calling bound actions (entity actions) using Xrm.WebApi.execute in Dynamics V9​​ 有介绍,我这里稍微改动一点转换过来。

1. 调用绑定action必须传递参数 entity 的值,这个entity的值就是一个类似 { entityType: "account", id: accountid } 的对象,它是操作绑定的实体要执行该操作的记录。

2. 操作的为EntityReference类型的输入参数和调用时候传递的entity参数,其对应的structuralProperty的值为5,也就是 EntityType 。

3.参数的typeName的值请参考 Settings –> Customizations –> Developer Sesources -> Instance Web API 中 Download OData Metadata 链接下载下来的操作的输入参数的Type值,类似如下图。如果是EntityReference类型的参数,typeName的值是该实体的逻辑名称前面加上mscrm. 。

D365 V9版本新的客户端API Xrm.WebApi.online.execute_Action

  4. 操作返回参数值的获取请参考我后面的代码,是 result.json().then 中能获取到返回值。

话不多说,上个调用示例,我这个操作是绑定在客户(account)实体上。

var ly_accountid = formContext.getAttribute("lvo_accountid").getValue();
if (ly_accountid !== null) {
var accountid = ly_accountid[0].id.replace("{", "").replace("}", "");
var target = { entityType: "account", id: accountid };
var reqObject = {};
reqObject.getMetadata = function () {
return {
boundParameter: "entity",
operationType: 0,
operationName: "ly_LuoYongBoundAction",
parameterTypes: {
"entity": {
typeName: "mscrm.account",
structuralProperty: 5
}
}
}
};
reqObject.entity = target;
Xrm.WebApi.online.execute(reqObject).then(
function (result) {
if (result.ok) {
console.log("Status: %s %s", result.status, result.statusText);
result.json().then(
function (response) {
//此处获取到了操作返回参数的值
console.log(response);
alert(response.OutCountry.ly_testid);
});
}
},
function (error) {
console.log(error);
Xrm.Utility.alertDialog(error.message);
}
);
}


再举个例子,调用 ​​RemoveFromQueue​​ 这个绑定Action的例子。

    var queueitemid = "f372ba20-dcbb-ec11-983e-0017fa04af7b";
var target = { entityType: "queueitem", id: queueitemid };
var reqObject = {};
reqObject.getMetadata = function () {
return {
boundParameter: "entity",
operationType: 0,
operationName: "RemoveFromQueue",
parameterTypes: {
"entity": {
typeName: "mscrm.queueitem",
structuralProperty: 5
}
}
}
};
reqObject.entity = target;
Xrm.WebApi.online.execute(reqObject).then(
function (result) {
if (result.ok) {
console.log("Status: %s %s", result.status, result.statusText);
}
},
function (error) {
console.log(error);
Xrm.Utility.alertDialog(error.message);
}
);


我这里还记录下自己写HTTP请求调用非绑定操作的代码示例:

var clientURL = Xrm.Utility.getGlobalContext().getClientUrl();
var req = new XMLHttpRequest()
req.open("POST", encodeURI(clientURL + "/api/data/v9.1/WinOpportunity"), true);
req.setRequestHeader("Accept", "application/json");
req.setRequestHeader("Content-Type", "application/json; charset=utf-8");
req.setRequestHeader("OData-MaxVersion", "4.0");
req.setRequestHeader("OData-Version", "4.0");
req.onreadystatechange = function () {
if (this.readyState == 4 /* complete */) {
req.onreadystatechange = null;
if (this.status == 204) {
Xrm.Utility.alertDialog("将商机作为赢单关闭成功!");
}
else {
var error = JSON.parse(this.response).error;
Xrm.Utility.alertDialog("将商机作为赢单关闭出错." + error.message);
}
}
};
var requestmsg = {};
requestmsg.Status = 3;
requestmsg.OpportunityClose = {};
requestmsg.OpportunityClose.subject = "罗勇结束了这个商机";
requestmsg.OpportunityClose["opportunityid@odata.bind"] = "/opportunities(6025165A-3AA3-E511-80C7-000D3A807EC7)";
req.send(JSON.stringify(requestmsg));


调用绑定操作的示例:

var clientURL = Xrm.Utility.getGlobalContext().getClientUrl();
var req = new XMLHttpRequest()
req.open("POST", encodeURI(clientURL + "/api/data/v9.1/teams(E4CC382D-02B9-E511-80DC-000D3A804C3F)/Microsoft.Dynamics.CRM.RemoveMembersTeam"), true);
req.setRequestHeader("Accept", "application/json");
req.setRequestHeader("Content-Type", "application/json; charset=utf-8");
req.setRequestHeader("OData-MaxVersion", "4.0");
req.setRequestHeader("OData-Version", "4.0");
req.onreadystatechange = function () {
if (this.readyState == 4 /* complete */) {
req.onreadystatechange = null;
if (this.status == 204) {
Xrm.Utility.alertDialog("将用户移出团队成功!");
}
else {
var error = JSON.parse(this.response).error;
Xrm.Utility.alertDialog("将用户移出团队出错." + error.message);
}
}
};
var requestmsg = {};
requestmsg.Members = [];
requestmsg.Members[0] = {};
requestmsg.Members[0].systemuserid = "A576C4B5-44A9-E511-80CF-000D3A806074";
requestmsg.Members[1] = {};
requestmsg.Members[1].systemuserid = "CEFE67E5-44A9-E511-80CF-000D3A806074";
req.send(JSON.stringify(requestmsg));


当然通过组织服务也是可以调用Action的,示例如下:

OrganizationRequest req = new OrganizationRequest("ly_AssignAction");
req["InRequestType"] = new EntityReference("ly_casetype",Guid.Parse("4deeef82-4830-ea11-a810-000d3a3786fc"));
req["InRequestSubType"] = new EntityReference("ly_casetype", Guid.Parse("7a66ea89-4930-ea11-a810-000d3a3786fc"));
req["Target"] = new EntityReference("ly_servicerequest", Guid.Parse("d5315ee2-ca53-ea11-a812-000d3a378f47"));//global action则无需此参数
crmSvc.Execute(req);

 再举个调用非绑定action的例子,这两个输入参数都是EntityReference类型,注意传递值的方法:

var AssignActionRequest = function (entityLogicalName, entityId, businessGroup, group, requestType) {
this.InEntityLogicalName = entityLogicalName;
this.InEntityId = entityId;
this.InGroup = group;
this.InRequestType = requestType;
};

AssignActionRequest.prototype.getMetadata = function () {
return {
boundParameter: null,
parameterTypes: {
"InEntityLogicalName": {
typeName: "Edm.String",
structuralProperty: 1
},
"InEntityId": {
typeName: "Edm.String",
structuralProperty: 1
},
"InGroup": {
typeName: "mscrm.ly_demoentitylogicalname",
structuralProperty: 5
},
"InRequestType": {
typeName: "mscrm.lvo_casetype",
structuralProperty: 5
}
},
operationType: 0,
operationName: "ly_AssignAction",
};
};

var requestType = {
"ly_casetypeid": "4d1eef82-4830-ea11-a810-000d3a3786fc"
}

var assignActionRequest = new AssignActionRequest("lvo_orderchangerequest", "{d5315ee2-ca53-ea11-a812-000d3a378f47}", null, equestType);
//如下也可以
//var assignActionRequest = new AssignActionRequest("lvo_orderchangerequest", "d5315ee2-ca53-ea11-a812-000d3a378f47",null,equestType);
Xrm.WebApi.online.execute(assignActionRequest).then(
function (result) {
if (result.ok) {
var alertStrings = { confirmButtonLabel: "OK", text: "Operation compeleted successfully.", title: "Info" };
var alertOptions = { height: 300, width: 400 };
Xrm.Navigation.openAlertDialog(alertStrings, alertOptions);
}
},
function (error) {
console.log(error);
var alertStrings = { confirmButtonLabel: "OK", text: "Operation Failed. " + error.message, title: "Error" };
var alertOptions = { height: 300, width: 400 };
Xrm.Navigation.openAlertDialog(alertStrings, alertOptions);
}
);