$Action を使用した action メソッドへの動的参照
$Action グローバル変数を使用すると、オブジェクトの種別または特定のレコードに対する有効なアクションを動的に参照できます。これは、そのアクションを実行するために URL を作成する場合によく使用されます。
たとえば、<apex:outputLink> で式 {!URLFOR($Action[objectName].New)} を、sObject の名前を提供するコントローラメソッド getObjectName() と一緒に使用できます。
その例を次に示します。コントローラ拡張は、システムをクエリして、ユーザがアクセスできるすべてのカスタムオブジェクトの名前を取得し、そのリストを、新規レコードを作成するためのリンクと一緒に表示します。最初に、DynamicActionsHandler という名前でコントローラ拡張を作成します。
この拡張では、次の点に留意してください。
1swfobject.registerObject("clippy.codeblock-0", "9");
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17public with sharing class DynamicActionsHandler {
18
19 public List<CustomObjectDetails> customObjectDetails { get; private set; }
20
21 public DynamicActionsHandler(ApexPages.StandardController cont) {
22 this.loadCustomObjects();
23 }
24
25 public void loadCustomObjects() {
26 List<CustomObjectDetails> cObjects = new List<CustomObjectDetails>();
27 // Schema.getGlobalDescribe() returns lightweight tokens with minimal metadata
28 Map<String, Schema.SObjectType> gd = Schema.getGlobalDescribe();
29 for(String obj : gd.keySet()) {
30 if(obj.endsWith('__c')) {
31 // Get the full metadata details only for custom items
32 Schema.DescribeSObjectResult objD = gd.get(obj).getDescribe();
33 if( ! objD.isCustomSetting()) {
34 // Save details for custom objects, not custom settings
35 CustomObjectDetails objDetails = new CustomObjectDetails(
36 obj, objD.getLabel(), objD.isCreateable());
37 cObjects.add(objDetails);
38 }
39 }
40 }
41 cObjects.sort();
42 this.customObjectDetails = cObjects;
43 }
44
45 public class CustomObjectDetails implements Comparable {
46 public String nameStr { get; set; }
47 public String labelStr { get; set; }
48 public Boolean creatable { get; set; }
49
50 public CustomObjectDetails(String aName, String aLabel, Boolean isCreatable) {
51 this.nameStr = aName;
52 this.labelStr = aLabel;
53 this.creatable = isCreatable;
54 }
55
56 public Integer compareTo(Object objToCompare) {
57 CustomObjectDetails cod = (CustomObjectDetails)objToCompare;
58 return(this.nameStr.compareTo(cod.nameStr));
59 }
60 }
61}- loadCustomObjects メソッドは Apex スキーマメソッドを使用して、使用可能なカスタムオブジェクトに関するメタデータ情報を取得します。Schema.getGlobalDescribe メソッドは、使用可能なオブジェクトとカスタム設定に関する小さいメタデータセットを取得するための軽量の操作です。メソッドは、コレクションをスキャンして名前の末尾が「__c」(カスタムオブジェクトまたは設定であることを示す) の項目を探します。これらの項目は、getDescribe を使用してより詳細に調査され、選択したメタデータがカスタムオブジェクト用に保存されます。
- if(obj.endsWith('__c')) を使用して、項目がカスタムオブジェクトであるかどうかをテストするのは、「ハッキング」のように感じられるかもしれませんが、もう 1 つの選択肢である obj.getDescribe().isCustom() コールにはコストがかかり、getDescribe へのコー��数にガバナ制限があります。大量のオブジェクトが含まれている可能性があるリストに対しては、「__c」文字列のスキャンを最初に行うのが効率的です。
- このメタデータは、内部クラス CustomObjectDetails に保存されます。このクラスは、保存すべき項目の単純な構造化されたコンテナとして機能します。
- CustomObjectDetails には Comparable インターフェースが実装されており、各オブジェクトの属性 (この場合はカスタムオブジェクト名) でカスタムオブジェクト詳細のリストを並び替えることができます。
ここで、次のマークアップで Visualforce ページを作成します。
特定のレコードが割り当てられていないページで使用できる便利なアクションは、New と List の 2 つのみです。レコードをクエリするページで、$Action グローバル変数は、View、Clone、Edit、Delete などのメソッドを提供します。特定の標準オブジェクトには、データ型に適した追加のアクションがあります。
1swfobject.registerObject("clippy.codeblock-1", "9");
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17<apex:page standardController="Account"
18 extensions="DynamicActionsHandler">
19 <br/>
20
21 <apex:dataTable value="{!customObjectDetails}" var="coDetails">
22 <apex:column >
23 <apex:facet name="header">Custom Object</apex:facet>
24 <apex:outputText value="{!coDetails.labelStr}"/>
25 </apex:column>
26 <apex:column >
27 <apex:facet name="header">Actions</apex:facet>
28 <apex:outputLink value="{!URLFOR($Action[coDetails.nameStr].New)}"
29 rendered="{!coDetails.creatable}">[Create]</apex:outputLink><br/>
30 <apex:outputLink value="{!URLFOR($Action[coDetails.nameStr].List,
31 $ObjectType[coDetails.nameStr].keyPrefix)}">[List]</apex:outputLink>
32 </apex:column>
33 </apex:dataTable>
34
35</apex:page>