web-dev-qa-db-ja.com

CRM 2011:Javascriptでエンティティを取得する

CRM 2011オンラインのカスタマイズに取り組んでおり、JavaScriptを使用してエンティティを取得する必要があります。

必要なエンティティは、別のフィールドのID値(連絡先エンティティ)に基づいています。この連絡先IDは問題なく取得できます。

必要なエンティティはカスタムエンティティです。連絡先IDに基づいて複数の一致がある可能性があるため、リストの最初の一致を取得したいだけです(順序は重要ではありません)

これまで、これを行うためのいくつかの方法を検討してきました...

  • OData-作成できるクエリ式について十分な例を見つけることができませんでした。また、カスタムエンティティでこれを機能させるかどうか/方法もわかりません。

  • FetchXML-組み込みの「高度な検索」を使用してNiceFetchXMLクエリを作成することもできます。誰かが助けてくれれば、JavaScriptからこれを呼び出して喜んでいますか?有望な答えが1つ見つかりました ここ しかし、「結果」の戻りデータがどのように設定されているかがわかりませんでした(Service.Fetch関数)

  • SOAPリクエスト-私が最初に試したのは、CRM 4で実行できたのと同様の方法ですが、これは機能していないようです。リクエストは実行されますが、結果データは空のようです。これが私が持っているすべてのコードなので、誰かが以下のコードで問題を見つけることができれば、それは素晴らしいことです。

[〜#〜] edit [〜#〜]:いくつかの冗長なクエリデータを見つけました(リンク開始タグを削除しましたが、終了タグを残しました)-これを削除してから、XML結果データを取得します...ただし、where句は適用されないようです(すべてのエンティティのリストを取得するだけです)

    var xml = "<?xml version='1.0' encoding='utf-8'?>" + 
    "<soap:Envelope xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\">" + 
    GenerateAuthenticationHeader() +
    "<soap:Body>" + 
    "<RetrieveMultiple xmlns=\"http://schemas.Microsoft.com/crm/2007/WebServices\">" + 
    "<query xmlns:q1=\"http://schemas.Microsoft.com/crm/2006/Query\" xsi:type=\"q1:QueryExpression\">" + 
    "<q1:EntityName>new_vehicle</q1:EntityName>" + 
    "<q1:ColumnSet xsi:type='q1:ColumnSet'>" + 
    "<q1:Attributes>" + 
    "<q1:Attribute>new_vehicleid</q1:Attribute>" +
    "<q1:Attribute>new_primarydriver</q1:Attribute>" +
    "<q1:Attribute>statuscode</q1:Attribute>" +
    "<q1:Attribute>new_registration</q1:Attribute>" +
    "</q1:Attributes>" + 
    "</q1:ColumnSet>" + 
    "<q1:Distinct>false</q1:Distinct>" + 

    "<q1:Conditions>" + 

                     "<q1:Condition>" +
                     "<q1:AttributeName>new_primarydriver</q1:AttributeName>" +
    "<q1:Operator>Equal</q1:Operator>" +
    "<q1:Values>" +
    "<q1:Value xmlns:q2='http://Microsoft.com/wsdl/types/' xsi:type='q2:guid'>" +
    customerID +
    "</q1:Value></q1:Values></q1:Condition>" + 

    "</q1:Conditions>" +

    "</query></RetrieveMultiple>" +
    "</soap:Body></soap:Envelope>";


    var xmlHttpRequest = new ActiveXObject("Msxml2.XMLHTTP");

    xmlHttpRequest.Open("POST", "/mscrmservices/2007/CrmService.asmx", false);
    xmlHttpRequest.setRequestHeader("SOAPAction", "http://schemas.Microsoft.com/crm/2007/WebServices/RetrieveMultiple");
    xmlHttpRequest.setRequestHeader("Content-Type", "text/xml; charset=utf-8");
    xmlHttpRequest.setRequestHeader("Content-Length", xml.length);
    xmlHttpRequest.send(xml);

    var result = xmlHttpRequest.responseXML.xml;
    var doc = new ActiveXObject("MSXML2.DOMDocument");
    doc.async = false;
    doc.loadXML(result);

    var id = doc.selectSingleNode("//new_vehicleid");
    var registration = doc.selectSingleNode("//new_registration");

    if(id == null)
       return null;

    var vehicle = new Array();
                     value[0] = new Object();
                     value[0].id = id;
                     value[0].name = registration;
                     value[0].entityType = "new_vehicle";

    return vehicle;

大きなコード投稿については申し訳ありませんが、うまくいけば、よりよく理解している誰かが助けることができます

14
musefan

まず、GlennFerrieLiveの回答投稿に感謝します。 Dynamics CRM 2011 SDK (特に1つだけ)で見つけたサンプルは本当に役に立ち、含まれているJSONパーサーは仕事に最適でした!

この回答を投稿して、SDKの例からはそれほど明白ではないかもしれない、注意を払うためのいくつかの重要なコメントを付けて、それをどのように行ったかの完全な例を示します。


ルックアップフィールドから選択したID値を取得します

私のタスクの目的は、javascriptを使用して、別のルックアップエンティティの選択されたデータに基づいてルックアップフィールドを設定することでした。設定するエンティティは「new_vehicle」であり、クエリを実行するエンティティは「customer」です。

最初の仕事は、連絡先検索フィールドのID値を取得することです。

var customerItem = Xrm.Page.getAttribute("customerid").getValue();
var customerID = customerItem[0].id;

IDを使用したエンティティのクエリ

次は、customerID値を使用して、現在割り当てられている車両(ルックアップフィールドを設定するために使用するエンティティ)を検索した部分です。

私が見つけた最初の問題は、ODataでクエリを実行すると、ID値が中括弧{}で機能しないように見えることでした。したがって、これらを削除する必要があります...

customerID = customerID.replace('{', '').replace('}', '');

次に、oDataPathを取得します...

var oDataPath = Xrm.Page.context.getServerUrl() + "/xrmservices/2011/organizationdata.svc";

次に、ODataクエリを作成できます...

var filter = "/new_vehicleSet?" +
  "$select=new_vehicleId,new_Registration" + 
  "&$filter=new_PrimaryDriver/Id eq (guid'" + customerID + "')" + 
  "&$orderby=new_LastAllocationDate desc" + 
  "&$top=1";

注:ここで注意すべき重要なことがいくつかあります...

  1. Guid値を使用する場合は、(guid'xxx')を使用してそれがguidであることを明示的に指定する必要があります。
  2. ルックアップエンティティ(例:new_PrimaryDriver)でフィルタリングする場合は、クエリに値(例:Id)を追加する必要があります-これにより、new_PrimaryDriver/Idになります

クエリを設定したら、次のようにデータをリクエストできます...

var retrieveRecordsReq = new XMLHttpRequest();
retrieveRecordsReq.open("GET", oDataPath + filter, true);
retrieveRecordsReq.setRequestHeader("Accept", "application/json");
retrieveRecordsReq.setRequestHeader("Content-Type", "application/json; charset=utf-8");
retrieveRecordsReq.onreadystatechange = function () {
    if (this.readyState == 4) {
       if (this.status == 200) {
           var retrievedRecords = JSON.parse(retrieveRecordsReq.responseText).d;
           if(retrievedRecords.results.length > 0)
           {
               var vehicle = retrievedRecords.results[0];
               SetLookup("new_replacedvehicle", vehicle.new_vehicleId, vehicle.new_Registration, "new_vehicle");
           }
       }
    }
};
retrieveRecordsReq.send();

これは非同期呼び出しであり、onreadystatechange関数は完了時に処理されることに注意してください。この関数では、成功したかどうかを確認するためにいくつかのチェックを行い、結果のJSONデータを解析します-JSON.Parse関数が含まれていますこの投稿の下部にあります(ただし、 [〜#〜] sdk [〜#〜] から入手できます)


上記でクエリしたエンティティを使用してルックアップフィールドを設定する

ここで注意すべきもう1つの関数は、ルックアップフィールドを設定するために追加した単純なヘルパー関数であるSetLookupです。これは次のとおりです...

function SetLookup(fieldName, idValue, textValue, typeValue)
{
    var value = new Array();
    value[0] = new Object();
    value[0].id = idValue;
    value[0].name = textValue;
    value[0].typename = typeValue;

    Xrm.Page.getAttribute(fieldName).setValue(value);
}

JSON解析関数

これは、上記のコード(JSON.parse)で使用されたJSONヘルパー関数であり、SDKで見つかったとおりに貼り付けられています...

if (!this.JSON) { this.JSON = {}; } (function () { function f(n) { return n < 10 ? '0' + n : n; } if (typeof Date.prototype.toJSON !== 'function') { Date.prototype.toJSON = function (key) { return isFinite(this.valueOf()) ? this.getUTCFullYear() + '-' + f(this.getUTCMonth() + 1) + '-' + f(this.getUTCDate()) + 'T' + f(this.getUTCHours()) + ':' + f(this.getUTCMinutes()) + ':' + f(this.getUTCSeconds()) + 'Z' : null; }; String.prototype.toJSON = Number.prototype.toJSON = Boolean.prototype.toJSON = function (key) { return this.valueOf(); }; } var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g, escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g, gap, indent, meta = { '\b': '\\b', '\t': '\\t', '\n': '\\n', '\f': '\\f', '\r': '\\r', '"': '\\"', '\\': '\\\\' }, rep; function quote(string) { escapable.lastIndex = 0; return escapable.test(string) ? '"' + string.replace(escapable, function (a) { var c = meta[a]; return typeof c === 'string' ? c : '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4); }) + '"' : '"' + string + '"'; } function str(key, holder) { var i, k, v, length, mind = gap, partial, value = holder[key]; if (value && typeof value === 'object' && typeof value.toJSON === 'function') { value = value.toJSON(key); } if (typeof rep === 'function') { value = rep.call(holder, key, value); } switch (typeof value) { case 'string': return quote(value); case 'number': return isFinite(value) ? String(value) : 'null'; case 'boolean': case 'null': return String(value); case 'object': if (!value) { return 'null'; } gap += indent; partial = []; if (Object.prototype.toString.apply(value) === '[object Array]') { length = value.length; for (i = 0; i < length; i += 1) { partial[i] = str(i, value) || 'null'; } v = partial.length === 0 ? '[]' : gap ? '[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']' : '[' + partial.join(',') + ']'; gap = mind; return v; } if (rep && typeof rep === 'object') { length = rep.length; for (i = 0; i < length; i += 1) { k = rep[i]; if (typeof k === 'string') { v = str(k, value); if (v) { partial.Push(quote(k) + (gap ? ': ' : ':') + v); } } } } else { for (k in value) { if (Object.hasOwnProperty.call(value, k)) { v = str(k, value); if (v) { partial.Push(quote(k) + (gap ? ': ' : ':') + v); } } } } v = partial.length === 0 ? '{}' : gap ? '{\n' + gap + partial.join(',\n' + gap) + '\n' + mind + '}' : '{' + partial.join(',') + '}'; gap = mind; return v; } } if (typeof JSON.stringify !== 'function') { JSON.stringify = function (value, replacer, space) { var i; gap = ''; indent = ''; if (typeof space === 'number') { for (i = 0; i < space; i += 1) { indent += ' '; } } else if (typeof space === 'string') { indent = space; } rep = replacer; if (replacer && typeof replacer !== 'function' && (typeof replacer !== 'object' || typeof replacer.length !== 'number')) { throw new Error('JSON.stringify'); } return str('', { '': value }); }; } if (typeof JSON.parse !== 'function') { JSON.parse = function (text, reviver) { var j; function walk(holder, key) { var k, v, value = holder[key]; if (value && typeof value === 'object') { for (k in value) { if (Object.hasOwnProperty.call(value, k)) { v = walk(value, k); if (v !== undefined) { value[k] = v; } else { delete value[k]; } } } } return reviver.call(holder, key, value); } text = String(text); cx.lastIndex = 0; if (cx.test(text)) { text = text.replace(cx, function (a) { return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4); }); } if (/^[\],:{}\s]*$/.test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@').replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { j = eval('(' + text + ')'); return typeof reviver === 'function' ? walk({ '': j }, '') : j; } throw new SyntaxError('JSON.parse'); }; } } ());
28
musefan