逐步解說:使用現代應用程式 SOAP 端點與 JavaScript
發行︰ 2016年11月
適用於: Dynamics CRM 2015
此逐步解說示範如何建立 JavaScript 程式庫,用於 SOAP 端點與 Web 資源。 此時 Microsoft Dynamics CRM 2015 和 Microsoft Dynamics CRM Online 2015 更新 不提供用於使用 SOAP 端點呼叫訊息的 JavaScript 程式庫。 此逐步解說示範如何建立 JScript 程式庫,用於使用組織服務 Execute 方法的 Assign 訊息。 使用 Execute 方法其他訊息會依照類似的模式。 若要建立任何訊息的 JavaScript 程式庫將需要類似的程序。 如需使用這個程序中開發的 JavaScript 程式庫範例,請參閱 範例:使用 JavaScript,擷取實體中繼資料。
注意
如需使用 SOAP 端點的完整範例 JavaScript 程式庫,請參閱 Sdk.Soap.js 程式庫。 Sdk.Soap.js 程式庫是使用此處描述的程序建立。
程序包含擷取 HTTP 要求和從 Managed 程式碼主控台應用程式傳送的回應,然後使用函數建立 JavaScript 程式庫以傳送相同的要求和處理類似的回應。 使用應用程式如 Fiddler 取得 HTTP 需要和回應很常見,但 Fiddler 和類似應用程式無法存取用於內部部署Microsoft Dynamics CRM 2015 的加密 HTTP 流量。 提供的 SOAPLogger 範例解決方案擷取加密前的 HTTP 要求,以及解密後的 HTTP 回應。SOAPLogger 範例解決方案同時適用於 Microsoft Dynamics CRM 2015 和 Microsoft Dynamics CRM Online 2015 更新。SOAPLogger 也會篩選與建立 JScript 程式庫工作無關的所有 HTTP 流量。
此逐步解說引導您進行下列工作:
在 擷取範例 HTTP 要求和回應,您會編輯 SOAPLogger 範例的 Run 方法,以使用 Managed 程式碼執行 Assign 要求。
然後執行 SOAPLogger 解決方案,產生關於所傳送 HTTP 要求以及所接收的回應的資訊。
在 建立 JScript 程式庫 中,您將根據擷取的 HTTP 流量建立 JScript 程式庫,以使用組織服務 Execute 方法傳送和接收 Assign 訊息。
在 測試 JScript 程式庫 中,您將測試 JScript 程式庫,確認其可以用來指派記錄。
在 建立互動 Web 資源,您將建立 HTML Web 資源,其使用程式庫指派 Account 記錄給選取的使用者,如下列螢幕擷取畫面所示。
代表此完整逐步解說的受管理的解決方案可在 SDK 套件的下列位置找到:SDK\SampleCode\JS\SOAPForJScript\SOAPEndpointforJScript_1_0_0_1_managed.zip。 下載 Microsoft Dynamics CRM SDK 套件。 您可以安裝 (匯入) 此解決方案和檢視完整結果。 因為它是受管理的解決方案,您可以輕鬆解除安裝 (刪除) 它,從系統完全移除它。
SDK\SampleCode\JS\SOAPForJScript\SOAPForJScript.sln 是 Visual Studio 解決方案,具有此逐步解說程式碼的 HTML 與 JS 檔案。
必要條件
您必須可使用 JScript 撰寫 Managed 程式碼,對應於要執行之函數。
您需要存取 SDK 下載檔案 (位於 SDK\SampleCode\CS\Client\SOAPLogger) 中的可用 SOAPLogger 範例。
注意
Microsoft Dynamics CRM SDK 中的 SOAPLogger 範例只適用於 Microsoft Visual C#。 如果您偏好 Microsoft Visual Basic .NET 版本,請參閱 CodePlex 上的 Microsoft Dynamics CRM VB.Net SoapLogger。
您需要 Microsoft Visual Studio 才能執行 SOAPLogger 解決方案。
您需要了解 JScript 以及使用 XMLHttpRequest進行非同步程式設計。
擷取範例 HTTP 要求和回應
使用 SoapLoggerOrganizationService,編輯 SOAPLogger 解決方案新增程式碼執行 AssignRequest。SoapLoggerOrganizationService 從 IOrganizationService 繼承並新增產生的記錄檔的功能,會顯示傳送和收到的相關 HTTP 流量。
使用 Microsoft Visual Studio,開啟位於 SDK\SampleCode\CS\Client\SOAPLogger\SOAPLogger.sln 的 SOAPLogger 解決方案。
在 SoapLogger.cs 檔案,找出 Run 方法,如下列範例所示:
public void Run(ServerConnection.Configuration serverConfig) { try { // Connect to the Organization service. // The using statement assures that the service proxy will be properly disposed. using (_serviceProxy = new OrganizationServiceProxy(serverConfig.OrganizationUri, serverConfig.HomeRealmUri, serverConfig.Credentials, serverConfig.DeviceCredentials)) { // This statement is required to enable early-bound type support. _serviceProxy.EnableProxyTypes(); IOrganizationService service = (IOrganizationService)_serviceProxy; using (StreamWriter output = new StreamWriter("output.txt")) { SoapLoggerOrganizationService slos = new SoapLoggerOrganizationService(serverConfig.OrganizationUri, service, output); //Add the code you want to test here: // You must use the SoapLoggerOrganizationService 'slos' proxy rather than the IOrganizationService proxy you would normally use. } } } // Catch any service fault exceptions that Microsoft Dynamics CRM throws. catch (FaultException<Microsoft.Xrm.Sdk.OrganizationServiceFault>) { // You can handle an exception here or pass it back to the calling method. throw; } }
識別 SystemUser 記錄的有效 id 和未指派給該使用者的 Account 記錄的識別碼。
提示
若要從應用程式取得任何記錄的 Id,請開啟該記錄,並使用 [複製連結] 命令。 將連結貼至 記事本 並尋找包含 id 參數值的部分,排除編碼括號 (“%7b” = “{”) 和 (“%7d” = “}”)。 例如,下列表示 Account 記錄的 URL。<your organization root url>/main.aspx?etc=1&id=%7bF2CA52DE-552D-E011-A8FB-00155DB059BE%7d&pagetype=entityrecord>。 id 值是 F2CA52DE-552D-E011-A8FB-00155DB059BE。
使用這些值取代以下程式碼的預留位置值,並將程式碼新增至執行方法,依指示。 如需 Assign 要求的更多資訊,請參閱 AssignRequest。
Guid userId = new Guid("XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX"); Guid accountId = new Guid("XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX"); AssignRequest assign = new AssignRequest{Assignee = new EntityReference(SystemUser.EntityLogicalName, userId),Target = new EntityReference(Account.EntityLogicalName, accountId)}; AssignResponse assignResp = (AssignResponse)slos.Execute(assign);
按 F5 偵錯應用程式。 輸入關於伺服器的資訊和您的驗證認證。 如果程式碼成功執行,客戶記錄現在會指派給您指定的使用者。 如需執行範例程式碼的詳細資訊,請參閱 使用 Microsoft Dynamics CRM 2015 Web 服務執行簡單的程式
在 SOAPLogger 專案資料夾,請瀏覽至 bin 和 Debug 資料夾並尋找 output.txt 檔案。
開啟 output.txt 檔案,然後檢閱內容。 它們應該類似下列,不過實際 GUID 值會取代預留位置“XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX”。
HTTP REQUEST -------------------------------------------------- POST <your server root>/XRMServices/2011/Organization.svc/web Content-Type: text/xml; charset=utf-8 SOAPAction: https://schemas.microsoft.com/xrm/2011/Contracts/Services/IOrganizationService/Execute <s:Envelope xmlns:s="https://schemas.xmlsoap.org/soap/envelope/"> <s:Body> <Execute xmlns="https://schemas.microsoft.com/xrm/2011/Contracts/Services" xmlns:i="http://www.w3.org/2001/XMLSchema-instance"> <request i:type="b:AssignRequest" xmlns:a="https://schemas.microsoft.com/xrm/2011/Contracts" xmlns:b="https://schemas.microsoft.com/crm/2011/Contracts"> <a:Parameters xmlns:c="http://schemas.datacontract.org/2004/07/System.Collections.Generic"> <a:KeyValuePairOfstringanyType> <c:key>Target</c:key> <c:value i:type="a:EntityReference"> <a:Id>XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX</a:Id> <a:LogicalName>account</a:LogicalName> <a:Name i:nil="true" /> </c:value> </a:KeyValuePairOfstringanyType> <a:KeyValuePairOfstringanyType> <c:key>Assignee</c:key> <c:value i:type="a:EntityReference"> <a:Id>XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX</a:Id> <a:LogicalName>systemuser</a:LogicalName> <a:Name i:nil="true" /> </c:value> </a:KeyValuePairOfstringanyType> </a:Parameters> <a:RequestId i:nil="true" /> <a:RequestName>Assign</a:RequestName> </request> </Execute> </s:Body> </s:Envelope> -------------------------------------------------- HTTP RESPONSE -------------------------------------------------- <s:Envelope xmlns:s="https://schemas.xmlsoap.org/soap/envelope/"> <s:Body> <ExecuteResponse xmlns="https://schemas.microsoft.com/xrm/2011/Contracts/Services" xmlns:i="http://www.w3.org/2001/XMLSchema-instance"> <ExecuteResult i:type="b:AssignResponse" xmlns:a="https://schemas.microsoft.com/xrm/2011/Contracts" xmlns:b="https://schemas.microsoft.com/crm/2011/Contracts"> <a:ResponseName>Assign</a:ResponseName> <a:Results xmlns:c="http://schemas.datacontract.org/2004/07/System.Collections.Generic" /> </ExecuteResult> </ExecuteResponse> </s:Body> </s:Envelope> --------------------------------------------------
注意您在使用程式碼傳入變數的位置,包括 Account 實體邏輯名稱。
您已經順利擷取執行指派作業的 HTTP 流量的相關功能。
建立 JScript 程式庫
由您決定如何建構程式庫的方式。 每個 JScript 應具有下列項目:
協助確保所有函數名稱是唯一的策略。
擷取 URL 至 Web 資源的 SOAP 端點的方法。
分隔方法,以非同步方式傳送要求和接收回應。
處理所有傳回的 WCF 錯誤的函數。
下列 JScript 程式庫含有這些項目,並包含 assignRequest 和 assignResponse 函數,根據使用 HTTP 擷取的 SOAPLogger 要求和回應來使用 Assign 訊息。
if (typeof (SDK) == "undefined")
{ SDK = { __namespace: true }; }
//This will establish a more unique namespace for functions in this library. This will reduce the
// potential for functions to be overwritten due to a duplicate name when the library is loaded.
SDK.SOAPSamples = {
_getClientUrl: function () {
///<summary>
/// Returns the URL for the SOAP endpoint using the context information available in the form
/// or HTML Web resource.
///</summary
var OrgServicePath = "/XRMServices/2011/Organization.svc/web";
var clientUrl = "";
if (typeof GetGlobalContext == "function") {
var context = GetGlobalContext();
clientUrl = context.getClientUrl();
}
else {
if (typeof Xrm.Page.context == "object") {
clientUrl = Xrm.Page.context.getClientUrl();
}
else
{ throw new Error("Unable to access the server URL"); }
}
return clientUrl + OrgServicePath;
},
assignRequest: function (Assignee, Target, Type, successCallback, errorCallback) {
///<summary>
/// Sends the Assign Request
///</summary>
this._parameterCheck(Assignee, "GUID", "The SDK.SOAPSamples.assignRequest method Assignee parameter must be a string Representing a GUID value.");
///<param name="Assignee" Type="String">
/// The GUID representing the System user that the record will be assigned to.
///</param>
this._parameterCheck(Target, "GUID", "The SDK.SOAPSamples.assignRequest method Target parameter must be a string Representing a GUID value.");
///<param name="Target" Type="String">
/// The GUID representing the user-owned entity record that will be assigne to the Assignee.
///</param>
this._parameterCheck(Type, "String", "The SDK.SOAPSamples.assignRequest method Type parameter must be a string value.");
Type = Type.toLowerCase();
///<param name="Type" Type="String">
/// The Logical name of the user-owned entity. For example, 'account'.
///</param>
if (successCallback != null)
this._parameterCheck(successCallback, "Function", "The SDK.SOAPSamples.assignRequest method successCallback parameter must be a function.");
///<param name="successCallback" Type="Function">
/// The function to perform when an successfult response is returned.
///</param>
this._parameterCheck(errorCallback, "Function", "The SDK.SOAPSamples.assignRequest method errorCallback parameter must be a function.");
///<param name="errorCallback" Type="Function">
/// The function to perform when an error is returned.
///</param>
//The request is simply the soap envelope captured by the SOAPLogger with variables added for the
// values passed. All quotations must be escaped to create valid JScript strings.
var request = [];
request.push("<s:Envelope xmlns:s=\"https://schemas.xmlsoap.org/soap/envelope/\">");
request.push("<s:Body>");
request.push("<Execute xmlns=\"https://schemas.microsoft.com/xrm/2011/Contracts/Services\"");
request.push(" xmlns:i=\"http://www.w3.org/2001/XMLSchema-instance\">");
request.push("<request i:type=\"b:AssignRequest\"");
request.push(" xmlns:a=\"https://schemas.microsoft.com/xrm/2011/Contracts\"");
request.push(" xmlns:b=\"https://schemas.microsoft.com/crm/2011/Contracts\">");
request.push("<a:Parameters xmlns:c=\"http://schemas.datacontract.org/2004/07/System.Collections.Generic\">");
request.push("<a:KeyValuePairOfstringanyType>");
request.push("<c:key>Target</c:key>");
request.push("<c:value i:type=\"a:EntityReference\">");
request.push("<a:Id>" + this._xmlEncode(Target) + "</a:Id>");
request.push("<a:LogicalName>" + this._xmlEncode(Type) + "</a:LogicalName>");
request.push("<a:Name i:nil=\"true\" />");
request.push("</c:value>");
request.push("</a:KeyValuePairOfstringanyType>");
request.push("<a:KeyValuePairOfstringanyType>");
request.push("<c:key>Assignee</c:key>");
request.push("<c:value i:type=\"a:EntityReference\">");
request.push("<a:Id>" + this._xmlEncode(Assignee) + "</a:Id>");
request.push("<a:LogicalName>systemuser</a:LogicalName>");
request.push("<a:Name i:nil=\"true\" />");
request.push("</c:value>");
request.push("</a:KeyValuePairOfstringanyType>");
request.push("</a:Parameters>");
request.push("<a:RequestId i:nil=\"true\" />");
request.push("<a:RequestName>Assign</a:RequestName>");
request.push("</request>");
request.push("</Execute>");
request.push("</s:Body>");
request.push("</s:Envelope>");
var req = new XMLHttpRequest();
req.open("POST", SDK.SOAPSamples._getClientUrl(), true)
// Responses will return XML. It isn't possible to return JSON.
req.setRequestHeader("Accept", "application/xml, text/xml, */*");
req.setRequestHeader("Content-Type", "text/xml; charset=utf-8");
req.setRequestHeader("SOAPAction", "https://schemas.microsoft.com/xrm/2011/Contracts/Services/IOrganizationService/Execute");
req.onreadystatechange = function () { SDK.SOAPSamples.assignResponse(req, successCallback, errorCallback); };
req.send(request.join(""));
},
assignResponse: function (req, successCallback, errorCallback) {
///<summary>
/// Recieves the assign response
///</summary>
///<param name="req" Type="XMLHttpRequest">
/// The XMLHttpRequest response
///</param>
///<param name="successCallback" Type="Function">
/// The function to perform when an successfult response is returned.
/// For this message no data is returned so a success callback is not really necessary.
///</param>
///<param name="errorCallback" Type="Function">
/// The function to perform when an error is returned.
/// This function accepts a JScript error returned by the _getError function
///</param>
if (req.readyState == 4) {
req.onreadystatechange = null; //avoids memory leaks
if (req.status == 200) {
if (successCallback != null)
{ successCallback(); }
}
else {
errorCallback(SDK.SOAPSamples._getError(req.responseXML));
}
}
},
_getError: function (faultXml) {
///<summary>
/// Parses the WCF fault returned in the event of an error.
///</summary>
///<param name="faultXml" Type="XML">
/// The responseXML property of the XMLHttpRequest response.
///</param>
var errorMessage = "Unknown Error (Unable to parse the fault)";
if (typeof faultXml == "object") {
try {
var bodyNode = faultXml.firstChild.firstChild;
//Retrieve the fault node
for (var i = 0; i < bodyNode.childNodes.length; i++) {
var node = bodyNode.childNodes[i];
//NOTE: This comparison does not handle the case where the XML namespace changes
if ("s:Fault" == node.nodeName) {
for (var j = 0; j < node.childNodes.length; j++) {
var faultStringNode = node.childNodes[j];
if ("faultstring" == faultStringNode.nodeName) {
errorMessage = faultStringNode.textContent;
break;
}
}
break;
}
}
}
catch (e) { };
}
return new Error(errorMessage);
},
_xmlEncode: function (strInput) {
var c;
var XmlEncode = '';
if (strInput == null) {
return null;
}
if (strInput == '') {
return '';
}
for (var cnt = 0; cnt < strInput.length; cnt++) {
c = strInput.charCodeAt(cnt);
if (((c > 96) && (c < 123)) ||
((c > 64) && (c < 91)) ||
(c == 32) ||
((c > 47) && (c < 58)) ||
(c == 46) ||
(c == 44) ||
(c == 45) ||
(c == 95)) {
XmlEncode = XmlEncode + String.fromCharCode(c);
}
else {
XmlEncode = XmlEncode + '&#' + c + ';';
}
}
return XmlEncode;
},
_parameterCheck: function (parameter, type, errorMessage) {
switch (type) {
case "String":
if (typeof parameter != "string") {
throw new Error(errorMessage);
}
break;
case "Function":
if (typeof parameter != "function") {
throw new Error(errorMessage);
}
break;
case "EntityFilters":
var found = false;
for (x in this.EntityFilters) {
if (this.EntityFilters[x] == parameter) {
found = true;
break;
}
}
if (!found) {
throw new Error(errorMessage);
}
break;
case "Boolean":
if (typeof parameter != "boolean") {
throw new Error(errorMessage);
}
break;
case "GUID":
var re = new RegExp("[0-9a-fA-F]{8}\-[0-9a-fA-F]{4}\-[0-9a-fA-F]{4}\-[0-9a-fA-F]{4}\-[0-9a-fA-F]{12}");
if (!(typeof parameter == "string" && re.test(parameter))) {
throw new Error(errorMessage);
}
break;
default:
throw new Error("An invalid type parameter value was passed to the SDK.MetaData._parameterCheck function.");
break;
}
},
__namespace: true
};
測試 JScript 程式庫
在準備您的 JScript 程式庫後,應執行某些初始測試,確認如預期運作。 因為只在應用程式中進行驗證,您必須使用程式庫內容,在 Microsoft Dynamics CRM 2015 中建立指令碼 Web 資源。 若要在程式庫中使用函數,您需要特定 UI 元素 (表單或 HTML Web 資源) 來呼叫程式庫中的 JScript 函數。 下列程序說明使用簡單 HTML Web 資源來測試程式庫中的函數。
使用下列程式碼,建立 HTML 文件:
<html xmlns="http://www.w3.org/1999/xhtml"> <head> <title>Assign Test</title> <meta http-equiv="X-UA-Compatible" content="IE=edge" /> <script src="../ClientGlobalContext.js.aspx" type="text/javascript"></script> <script src="Scripts/SDK.REST.js" type="text/javascript"></script> <script src="Scripts/SDK.SOAPSample.Assign.js" type="text/javascript"></script> <script type="text/javascript"> var resultsArea = null; var userid; var accountid; var userLookup; var accountLookup; var accountsToShow = 10; var usersToShow = 10; var users = []; var accounts = []; document.onreadystatechange = function () { ///<summary> /// Initializes the sample when the document is ready ///</summary> if (document.readyState == "complete") { userid = document.getElementById("userid"); accountid = document.getElementById("accountid"); resultsArea = document.getElementById("results"); userLookup = document.getElementById("userLookup"); accountLookup = document.getElementById("accountLookup"); populateUserLookup(); populateAccountLookup(); } } function testAssign() { //The field to enter the user id of the person the account record should be assigned to. userid.value; // The field to enter the account id of the account record to assign to the user accountid.value; // Since the response does not include any data to parse, simply display a success message. var successCallback = function () { setText(resultsArea, "Success!"); }; // Display the error from the message passed back from the response. var errorCallback = function (error) { setText(resultsArea,"errorCallback error: "+error.message); }; // Call the function try { SDK.SOAPSamples.assignRequest(userid.value, accountid.value, "account", successCallback, errorCallback); } catch (e) { setText(resultsArea, e.message); } } function populateUserLookup() { SDK.REST.retrieveMultipleRecords("SystemUser", "$top=" + usersToShow + "&$select=FullName,SystemUserId&$filter=FullName ne 'INTEGRATION' and FullName ne 'SYSTEM'", function (results) { for (var i = 0; i < results.length; i++) { users.push(results[i]); } }, function (error) { alert(error.message); }, function () { for (var i = 0; i < users.length; i++) { var user = users[i]; userLookup.options[i] = new Option(user.FullName, user.SystemUserId); } userid.value = userLookup.options[0].value; userLookup.onchange = function () { userid.value = userLookup.options[userLookup.selectedIndex].value; clearMessage(); }; }); } function populateAccountLookup() { SDK.REST.retrieveMultipleRecords("Account", "$top=" + accountsToShow + "&$select=AccountId,Name,OwnerId", function (results) { for (var i = 0; i < results.length; i++) { accounts.push(results[i]); } }, function (error) { alert(error.message); }, function () { for (var i = 0; i < accounts.length; i++) { var account = accounts[i]; accountLookup.options[i] = new Option(account.Name + " : " + account.OwnerId.Name, account.AccountId); } accountid.value = accountLookup.options[0].value; accountLookup.onchange = function () { accountid.value = accountLookup.options[accountLookup.selectedIndex].value; clearMessage(); }; }); } function clearMessage() { setText(resultsArea, ""); } function setText(element, text) { if (typeof (element.innerText) != "undefined") { element.innerText = text; } else { element.textContent = text; } } </script> </head> <body> <table> <tr> <td> <label for="userid"> SystemUserId:</label> </td> <td> <input id="userid" type="text" style="width:270px;" /> </td> <td> <select id="userLookup"></select> </td> </tr> <tr> <td> <label for="accountid"> AccountId:</label> </td> <td> <input id="accountid" type="text" style="width:270px;" /> </td> <td> <select id="accountLookup"></select> </td> </tr> </table> <button id="testAssign" title="Click this button to test the testAssign function." onclick="testAssign();"> Test Assign</button> <div id="results"> &nbsp;</div> </body> </html>
使用此程式碼,建立 HTML Web 資源。HTML Web 資源名稱的完整名稱取決於解決方案自訂首碼。 假設您的解決方案自訂首碼為「new」,將此 HTML Web 資源命名維「new_/AssignTest.htm」。 只要所有 Web 資源使用相同的自訂首碼,使用的特定自訂首碼對此範例沒有影響。
使用 JScript 程式庫的內容,建立指令碼 Web 資源。 將此 Web 資源命名為「new_/Scripts/SDK.SOAPSample.Assign.js」。
發行所有自訂並開啟 new_/AssignTest.htm Web 資源。 按一下 [預覽] 按鈕。
使用您在 No text is specified for bookmark or legacy link 'c9a435ab-a4cb-4e0c-9ef7-b7ba66278407#BKMK_EditSoapLogger'. 程序的步驟 3 相同的程序,找出有效系統使用者和客戶記錄識別碼值並貼至頁面。 然後按一下 [Test Assign] 確認您取得成功回應。
輸入無效值,確認錯誤正確剖析。
例如,如果輸入「A Store (sample)」而不是客戶的識別碼,應該會顯示下列錯誤訊息:
The formatter threw an exception while trying to deserialize the message: There was an error while trying to deserialize parameter https://schemas.microsoft.com/xrm/2011/Contracts/Services:request. The InnerException message was 'There was an error deserializing the object of type Microsoft.Xrm.Sdk.OrganizationRequest. The value 'A Store (sample)' cannot be parsed as the type 'Guid'.'. Please see InnerException for more details.
如果沒有 AccountId 的客戶記錄,訊息應該是:
Account With Id = 883c3084-1f2f-e011-ad66-00155dba3814 Does Not Exist
建立互動 Web 資源
下一個步驟是在表單指令碼、功能區命令或 Web 資源中使用 JScript 程式庫。 下列程式碼範例顯示 HTML Web 資源使用 REST 端點擷取 SystemUser 和 Account 記錄清單,提供互動式使用者介面將客戶記錄指派給使用者。 此頁面使用 Scripts/SDK.SOAPSample.Assign.js 指令碼 Web 資源和使用 json2.js 程式庫的指令碼 Web 資源,支援使用 JSON 端點與 REST 物件。SDK.REST.js 程式庫用以擷取使用 REST 端點的資料。
<html lang="en-us">
<head>
<title>Assign Accounts Page</title>
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<style type="text/css">
body
{
font-family: Segoe UI;
background-color: #EFF2F6;
}
.dataTable
{
border-collapse: collapse;
border: 1px solid black;
}
.tableHead
{
background-color: #C0C0C0;
width: 100%;
}
.grid
{
border-bottom: 1px solid black;
}
td
{
padding-left: 5px;
padding-right: 5px;
}
#accountList
{
background-color: White;
}
#checkboxCol
{
border-right: 1px solid black;
width: 20px;
}
#nameCol
{
border-right: 1px solid black;
width: 300px;
}
#ownerCol
{
width: 300px;
}
</style>
<script src="../ClientGlobalContext.js.aspx" type="text/javascript"></script>
<script src="Scripts/SDK.REST.js" type="text/javascript"></script>
<script src="Scripts/SDK.SOAPSample.Assign.js" type="text/javascript"></script>
<script type="text/javascript">
//Set variables for page elements
var userSelect; //The select control used to select the user to assign records to.
var accountList; //The tbody element that rows will be added to for each retrieved account
var selectAll; //The checkbox to select all the retrieved accounts
var btnAssign; //The button to assign assign the accounts
var tableCaption; //A label hidden on load
var dataTable; //the table element hidden on load
var alertFlag; // Alert flag to indicate the changes
var users = []; //SystemUser records retrieved
var accounts = []; //Accounts not assigned to the currently selected user.
var accountsToShow = 5;
var suppressRetrievedAccountsAlert = false;
var accountsToAssign = [];
var userId = null;
function startSample() {
///<summary>
/// Starts the sample
///</summary>
alertFlag = document.getElementById("dispalert");
userSelect = document.getElementById("userList");
//A new set of a 5 accounts will be retrieved when the user changes
userSelect.onchange = getAccounts;
accountList = document.getElementById("accountList");
selectAll = document.getElementById("selectAll");
//When the select all checkbox is clicked, toggle the selection for each row of the table.
selectAll.onclick = toggleSelectAllRecords;
btnAssign = document.getElementById("btnAssign");
//Add a helper function to enable or disable the assign button.
btnAssign.setEnabled = setEnabled;
//Set the event handler to the Assign button
btnAssign.onclick = assignAccounts;
tableCaption = document.getElementById("tableCaption");
dataTable = document.getElementById("dataTable");
var divSample = document.getElementById("divSample");
//Load the list of users
getUsers();
divSample.style.display = "block";
document.getElementById("btnStart").setAttribute("disabled", "disabled");
}
function getUsers() {
SDK.REST.retrieveMultipleRecords("SystemUser",
"$select=FullName,SystemUserId&$filter=FullName ne 'INTEGRATION' and FullName ne 'SYSTEM'",
function (results) {
for (var i = 0; i < results.length; i++) {
users.push(results[i]);
}
},
function (error) {
showMessage(error.message);
},
function () {
if (users.length > 1) {
for (var i = 0; i < users.length; i++) {
var user = users[i];
userSelect.options[i + 1] = new Option(user.FullName, user.SystemUserId);
}
userSelect.removeAttribute("disabled");
if (alertFlag.checked == true) {
alert(users.length + " system users retrieved");
}
}
else {
var notification = "This sample requires that more than one user is available in the organization.";
showMessage(notification);
if (alertFlag.checked == true) {
alert("This sample requires that more than one user is available in the organization.");
}
}
});
}
function getAccounts() {
//Clear out any records displayed in the table
clearAccountList();
var selectedUserId = userSelect.options[userSelect.selectedIndex].value;
if (selectedUserId != "none") {
SDK.REST.retrieveMultipleRecords("Account",
"$top=" + accountsToShow + "&$select=AccountId,Name,OwnerId&$filter=OwnerId/Id ne (guid'" + encodeURIComponent(selectedUserId) + "')",
function (results) {
accounts.length = 0;
for (var i = 0; i < results.length; i++) {
accounts.push(results[i]);
}
},
function (error) {
showMessage(error.message);
},
function () {
//onComplete
if (accounts.length > 0) {
for (var i = 0; i < accounts.length; i++) {
var account = accounts[i];
var row = document.createElement("tr");
row.setAttribute("class", "grid");
row.setAttribute("id", account.AccountId);
var checkboxCell = document.createElement("td");
var checkbox = document.createElement("input");
checkbox.setAttribute("type", "checkbox");
checkbox.onclick = validateAssignButton;
checkboxCell.appendChild(checkbox);
row.appendChild(checkboxCell);
var nameCell = document.createElement("td");
setText(nameCell, account.Name);
row.appendChild(nameCell);
var userNameCell = document.createElement("td");
setText(userNameCell, account.OwnerId.Name);
row.appendChild(userNameCell);
accountList.appendChild(row);
}
if (alertFlag.checked == true && !suppressRetrievedAccountsAlert) {
alert(accounts.length + " account records retrieved.");
suppressRetrievedAccountsAlert = false;
}
}
else {
//If no records are returned display a message.
clearAccountList();
var row = document.createElement("tr");
var cell = document.createElement("td");
cell.setAttribute("colSpan", "3");
setText(cell, "No Accounts were retrieved");
row.appendChild(cell);
accountList.appendChild(row);
}
//Show any of the UI elements that are initially hidden
setVisibleUIElements(true);
});
}
else { setVisibleUIElements(false); }
}
function assignAccounts() {
///<summary>
/// queues the selected accounts to be assigned sequentially
///</summary>
userId = userSelect.options[userSelect.selectedIndex].value;
var checkboxes = accountList.getElementsByTagName("input");
for (var i = checkboxes.length - 1; i >= 0; i--) {
if (checkboxes[i].checked) {
var accountId = checkboxes[i].parentElement.parentElement.id;
accountsToAssign.push(accountId);
}
}
assignNextAccount();
selectAll.checked = false;
}
function assignNextAccount() {
/// <summary>Assigns the queued accounts</summary>
//Prevents a generic SQL error that can occur when too many assign requests occur in rapid succession
if (accountsToAssign.length > 0) {
SDK.SOAPSamples.assignRequest(userId, accountsToAssign.shift(), "account", function () {
assignNextAccount();
}, function (error) {
showMessage("There was an error assigning the account with Id :" + accountId + ". The error message is " + error.message + ".");
assignNextAccount();
});
}
else {
suppressRetrievedAccountsAlert = true;
getAccounts();
btnAssign.setEnabled(false)
if (alertFlag.checked == true) {
alert("Record assignment completed and next set of records retrieved.");
}
}
}
function showMessage(message) {
///<summary>
/// Helper function to display message on the page if necessary.
///</summary
var dvMessage = document.createElement("div");
dvMessage.innerHTML = SDK.SOAPSamples._xmlEncode(message);
document.getElementById("status").appendChild(dvMessage);
}
function clearAccountList() {
///<summary>
/// Helper function remove rows from the Account List table.
///</summary
for (var i = (accountList.rows.length - 1) ; i >= 0; i--) {
accountList.deleteRow(i);
}
accounts.length = 0;
}
function toggleSelectAllRecords() {
///<summary>
/// Helper function to toggle all selected rows in the account list table.
///</summary
var checkboxes = accountList.getElementsByTagName("input");
for (var i = 0; i < checkboxes.length; i++) {
checkboxes[i].checked = this.checked;
}
btnAssign.setEnabled(this.checked);
}
function validateAssignButton() {
///<summary>
/// Helper function to enable the Assign Records button when rows are selected.
///</summary
if (this.checked == true)
{ btnAssign.setEnabled(true); }
else {
selectAll.checked = false;
var checkboxes = accountList.getElementsByTagName("input");
var checked = false;
for (var i = 0; i < checkboxes.length; i++) {
if (checkboxes[i].checked == true) {
checked = true;
break;
}
}
btnAssign.setEnabled(checked);
}
}
function setEnabled(bool) {
///<summary>
/// Helper method attached to the Assign button to make it easier to enable/disable the button.
///</summary
if (bool)
{ this.removeAttribute("disabled"); }
else
{ this.setAttribute("disabled", "disabled"); }
}
function setVisibleUIElements(display) {
///<summary>
/// Helper function to show those UI elements initially hidden.
///</summary
if (display) {
show(tableCaption);
show(dataTable);
show(btnAssign);
}
else {
hide(tableCaption);
hide(dataTable);
hide(btnAssign);
}
}
function show(element) {
if (element.tagName.toLowerCase() == "table") {
element.style.display = "table";
}
else {
element.style.display = "block";
}
}
function hide(element) {
element.style.display = "none";
}
// setText mitigate differences in how browsers set or get text content.
function setText(node, text) {
if (typeof (node.innerText) != "undefined") {
node.innerText = text;
}
else {
node.textContent = text;
}
}
</script>
</head>
<body>
<h1>
Assign Accounts Sample
</h1>
<p>
This page requires JavaScript and will update dynamically.
</p>
<p>
<input id="dispalert" name="dispalert" type="checkbox" value="alert" /><label for="dispalert">Display alert window when data changes.</label>
</p>
<p>
Click the <b>Start</b> button to begin the sample.
</p>
<input type="button" id="btnStart" name="btnStart" value="Start" onclick="startSample()" />
<div id="divSample" style="display: none">
<label for="userList">
User:
</label>
<select id="userList" name="userList" title="Select a system user from this list." disabled>
<option value="none">Select a User...</option>
</select>
<p id="tableCaption" style="display: none;">
Top 5 Accounts not assigned to the selected user:
</p>
<table class="dataTable" id="dataTable" style="display: none; width: 100%;">
<thead>
<tr class="tableHead">
<th scope="col" id="checkboxCol">
<input id="selectAll" name="selectAll" title="Select this to select all records" type="checkbox" /><label for="selectAll">Select&nbsp;All</label>
</th>
<th scope="col" id="nameCol">
Name
</th>
<th scope="col" id="ownerCol">
Owner
</th>
</tr>
</thead>
<tbody id="accountList"></tbody>
</table>
<label style="display: none;" for="btnAssign">
Click to assign selected records
</label>
<button id="btnAssign" name="btnAssign" disabled style="display: none; float: right;">
Assign Records
</button>
<label style="display: none;" for="btnAssign">
Click to assign selected records
</label>
<button id="Button1" name="btnAssign" disabled style="display: none; float: right;">
Assign Records
</button>
</div>
<div id="status">
&nbsp;
</div>
</body>
</html>
另請參閱
AssignRequest
範例:使用 JavaScript,擷取實體中繼資料
針對具有 Web 資源的現代應用程式使用現代應用程式 SOAP 端點
在 Web 資源中使用 Web 服務資料 (OData 和現代應用程式 SOAP 端點)
撰寫應用程式和伺服器擴充功能
Microsoft Dynamics CRM 2015 適用的 JavaScript 程式庫
技術文章:使用選項組選項與 REST 端點 - JScript
© 2017 Microsoft. 著作權所有,並保留一切權利。 著作權