2008-10-08 Marek Habersack <mhabersack@novell.com>
[mono.git] / mcs / class / System.Web / System.Web.UI / DataBinder.cs
1 //
2 // System.Web.UI.DataBinder.cs
3 //
4 // Authors:
5 //      Duncan Mak  (duncan@ximian.com)
6 //      Gonzalo Paniagua Javier (gonzalo@ximian.com)
7 //
8 // (C) 2002 Ximian, Inc. (http://www.ximian.com)
9 // Copyright (C) 2005 Novell, Inc (http://www.novell.com)
10 //
11 // Permission is hereby granted, free of charge, to any person obtaining
12 // a copy of this software and associated documentation files (the
13 // "Software"), to deal in the Software without restriction, including
14 // without limitation the rights to use, copy, modify, merge, publish,
15 // distribute, sublicense, and/or sell copies of the Software, and to
16 // permit persons to whom the Software is furnished to do so, subject to
17 // the following conditions:
18 // 
19 // The above copyright notice and this permission notice shall be
20 // included in all copies or substantial portions of the Software.
21 // 
22 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
23 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
24 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
25 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
26 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
27 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
28 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 //
30
31 using System.Collections;
32 using System.ComponentModel;
33 using System.Reflection;
34 using System.Security.Permissions;
35 #if NET_2_0
36 using System.Collections.Generic;
37 #endif
38
39 namespace System.Web.UI {
40
41         // CAS - no InheritanceDemand here as the class is sealed
42         [AspNetHostingPermission (SecurityAction.LinkDemand, Level = AspNetHostingPermissionLevel.Minimal)]
43         public sealed class DataBinder {
44                 public DataBinder ()
45                 {
46                 }
47
48                 internal static string FormatResult (object result, string format)
49                 {
50                         if (result == null)
51                                 return String.Empty;
52
53                         if (format == null || format.Length == 0)
54                                 return result.ToString ();
55
56                         return String.Format (format, result);
57                 }
58                 
59                 public static object Eval (object container, string expression)
60                 {
61                         expression = expression != null ? expression.Trim () : null;
62                         if (expression == null || expression.Length == 0)
63                                 throw new ArgumentNullException ("expression");
64
65                         object current = container;
66
67                         while (current != null) {
68                                 int dot = expression.IndexOf ('.');
69                                 int size = (dot == -1) ? expression.Length : dot;
70                                 string prop = expression.Substring (0, size);
71                                 if (prop.IndexOf ('[') != -1)
72                                         current = GetIndexedPropertyValue (current, prop);
73                                 else
74                                         current = GetPropertyValue (current, prop);
75
76                                 if (dot == -1)
77                                         break;
78                                 
79                                 expression = expression.Substring (prop.Length + 1);
80                         }
81
82                         return current;
83                 }
84
85                 public static string Eval (object container, string expression, string format)
86                 {
87                         object result = Eval (container, expression);
88                         return FormatResult (result, format);
89                 }
90
91                 public static object GetIndexedPropertyValue (object container, string expr)
92                 {
93                         if (container == null)
94                                 throw new ArgumentNullException ("container");
95                         if ((expr == null) || (expr.Length == 0))
96                                 throw new ArgumentNullException ("expr");
97
98                         int openIdx = expr.IndexOf ('[');
99                         int closeIdx = expr.IndexOf (']'); // see the test case. MS ignores all after the first ]
100                         if (openIdx < 0 || closeIdx < 0 || closeIdx - openIdx <= 1)
101                                 throw new ArgumentException (expr + " is not a valid indexed expression.");
102
103                         string val = expr.Substring (openIdx + 1, closeIdx - openIdx - 1);
104                         val = val.Trim ();
105                         if (val.Length == 0)
106                                 throw new ArgumentException (expr + " is not a valid indexed expression.");
107
108                         bool is_string = false;
109                         // a quoted val means we have a string
110                         if ((val[0] == '\'' && val[val.Length - 1] == '\'') ||
111                                 (val[0] == '\"' && val[val.Length - 1] == '\"')) {
112                                 is_string = true;
113                                 val = val.Substring(1, val.Length - 2);
114                         } else {
115                                 // if all chars are digits, then we have a int
116                                 for(int i = 0; i < val.Length; i++)
117                                         if (!Char.IsDigit(val[i])) {
118                                                 is_string = true;
119                                                 break;
120                                         }
121                         }
122
123                         int intVal = 0;
124                         if (!is_string) {
125                                 try {
126                                         intVal = Int32.Parse (val);
127                                 } catch {
128                                         throw new ArgumentException (expr + " is not a valid indexed expression.");
129                                 }
130                         }
131
132                         string property = null;
133                         if (openIdx > 0) {
134                                 property = expr.Substring (0, openIdx);
135                                 if (property != null && property != String.Empty)
136                                         container = GetPropertyValue (container, property);
137                         }
138
139                         if (container == null)
140                                 return null;
141
142                         if (container is System.Collections.IList) {
143                                 IList l = (IList) container;
144                                 return l [intVal];
145                         }
146
147                         Type t = container.GetType ();
148
149                         // MS does not seem to look for any other than "Item"!!!
150                         object [] atts = t.GetCustomAttributes (typeof (DefaultMemberAttribute), false);
151                         if (atts.Length != 1)
152                                 property = "Item";
153                         else
154                                 property = ((DefaultMemberAttribute) atts [0]).MemberName;
155
156                         Type [] argTypes = new Type [] { (is_string) ? typeof (string) : typeof (int) };
157                         PropertyInfo prop = t.GetProperty (property, argTypes);
158                         if (prop == null)
159                                 throw new ArgumentException (expr + " indexer not found.");
160
161                         object [] args = new object [1];
162                         if (is_string)
163                                 args [0] = val;
164                         else
165                                 args [0] = intVal;
166
167                         return prop.GetValue (container, args);
168                 }
169
170                 public static string GetIndexedPropertyValue (object container, string expr, string format)
171                 {
172                         object result = GetIndexedPropertyValue (container, expr);
173                         return FormatResult (result, format);
174                 }
175
176                 public static object GetPropertyValue (object container, string propName)
177                 {
178                         if (container == null)
179                                 throw new ArgumentNullException ("container");
180                         if (propName == null)
181                                 throw new ArgumentNullException ("propName");
182
183                         PropertyDescriptor prop = TypeDescriptor.GetProperties (container).Find (propName, true);
184                         if (prop == null) {
185                                 throw new HttpException ("Property " + propName + " not found in " +
186                                                          container.GetType ());
187                         }
188
189                         return prop.GetValue (container);
190                 }
191
192                 public static string GetPropertyValue (object container, string propName, string format)
193                 {
194                         object result = GetPropertyValue (container, propName);
195                         return FormatResult (result, format);
196                 }
197
198                 #if NET_2_0
199                 [ThreadStatic]
200                 static Dictionary<Type, PropertyInfo> dataItemCache;
201         
202                 public static object GetDataItem (object container, out bool foundDataItem)
203                 {       
204                         foundDataItem = false;
205                         if (container == null)                  
206                                 return null;
207                         
208                         if (container is IDataItemContainer) {
209                                 foundDataItem = true;
210                                 return ((IDataItemContainer)container).DataItem;
211                         }
212                         
213                         PropertyInfo pi = null;
214                         if (dataItemCache == null)
215                                 dataItemCache = new Dictionary<Type, PropertyInfo> ();
216                         
217                         Type type = container.GetType ();
218                         if (!dataItemCache.TryGetValue (type, out pi)) {
219                                 pi = type.GetProperty ("DataItem", BindingFlags.Public | BindingFlags.Instance);
220                                 dataItemCache [type] = pi;
221                         }
222
223                         if (pi == null)
224                                 return null;
225                         
226                         foundDataItem = true;
227
228                         return pi.GetValue (container, null); 
229                 } 
230                 
231                 
232                 public static object GetDataItem (object container)
233                 {
234                         bool flag;
235                         return GetDataItem (container, out flag); 
236                 }
237                 #endif
238         }
239 }
240