OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 /** | 5 /** |
6 * Provides some additional convenience methods on top of the basic mirrors | 6 * Provides some additional convenience methods on top of the basic mirrors |
7 */ | 7 */ |
8 library mirrors_helpers; | 8 library mirrors_helpers; |
9 | 9 |
10 // Import and re-export mirrors here to minimize both dependence on mirrors | 10 // Import and re-export mirrors here to minimize both dependence on mirrors |
11 // and the number of times we have to be told that mirrors aren't finished yet. | 11 // and the number of times we have to be told that mirrors aren't finished yet. |
12 import 'dart:mirrors'; | 12 import 'dart:mirrors'; |
13 export 'dart:mirrors'; | 13 export 'dart:mirrors'; |
14 import 'serialization_helpers.dart'; | 14 import 'serialization_helpers.dart'; |
15 | 15 |
16 // TODO(alanknight): Remove this method. It is working around a bug | 16 // TODO(alanknight): Remove this method. It is working around a bug |
17 // in the Dart VM which incorrectly returns Object as the superclass | 17 // in the Dart VM which incorrectly returns Object as the superclass |
18 // of Object. | 18 // of Object. |
19 _getSuperclass(ClassMirror mirror) { | 19 _getSuperclass(ClassMirror mirror) { |
20 var superclass = mirror.superclass; | 20 var superclass = mirror.superclass; |
21 return (superclass == mirror) ? null : superclass; | 21 return (superclass == mirror) ? null : superclass; |
22 } | 22 } |
23 | 23 |
24 /** | 24 /** |
25 * Return a list of all the public fields of a class, including inherited | 25 * Return a list of all the public fields of a class, including inherited |
26 * fields. | 26 * fields. |
27 */ | 27 */ |
28 Iterable<VariableMirror> publicFields(ClassMirror mirror) { | 28 Iterable<VariableMirror> publicFields(ClassMirror mirror) { |
29 var mine = mirror.variables.values.where( | 29 var mine = mirror.declarations.values.where( |
30 (x) => !(x.isPrivate || x.isStatic)); | 30 (x) => x is VariableMirror && !(x.isPrivate || x.isStatic)); |
31 var mySuperclass = _getSuperclass(mirror); | 31 var mySuperclass = _getSuperclass(mirror); |
32 if (mySuperclass != null) { | 32 if (mySuperclass != null) { |
33 return append(publicFields(mySuperclass), mine); | 33 return append(publicFields(mySuperclass), mine); |
34 } else { | 34 } else { |
35 return mine; | 35 return new List<VariableMirror>.from(mine); |
36 } | 36 } |
37 } | 37 } |
38 | 38 |
39 /** Return true if the class has a field named [name]. Note that this | 39 /** Return true if the class has a field named [name]. Note that this |
40 * includes private fields, but excludes statics. */ | 40 * includes private fields, but excludes statics. */ |
41 bool hasField(Symbol name, ClassMirror mirror) { | 41 bool hasField(Symbol name, ClassMirror mirror) { |
42 if (name == null) return false; | 42 if (name == null) return false; |
43 var field = mirror.variables[name]; | 43 var field = mirror.declarations[name]; |
44 if (field != null && !field.isStatic) return true; | 44 if (field is VariableMirror && !field.isStatic) return true; |
45 var superclass = _getSuperclass(mirror); | 45 var superclass = _getSuperclass(mirror); |
46 if (superclass == null) return false; | 46 if (superclass == null) return false; |
47 return hasField(name, superclass); | 47 return hasField(name, superclass); |
48 } | 48 } |
49 | 49 |
50 /** | 50 /** |
51 * Return a list of all the getters of a class, including inherited | 51 * Return a list of all the getters of a class, including inherited |
52 * getters. Note that this allows private getters, but excludes statics. | 52 * getters. Note that this allows private getters, but excludes statics. |
53 */ | 53 */ |
54 Iterable<MethodMirror> publicGetters(ClassMirror mirror) { | 54 Iterable<MethodMirror> publicGetters(ClassMirror mirror) { |
55 var mine = mirror.getters.values.where((x) => !(x.isPrivate || x.isStatic)); | 55 var mine = mirror.declarations.values.where( |
| 56 (x) => x is MethodMirror && x.isGetter && !(x.isPrivate || x.isStatic)); |
56 var mySuperclass = _getSuperclass(mirror); | 57 var mySuperclass = _getSuperclass(mirror); |
57 if (mySuperclass != null) { | 58 if (mySuperclass != null) { |
58 return append(publicGetters(mySuperclass), mine); | 59 return append(publicGetters(mySuperclass), mine); |
59 } else { | 60 } else { |
60 return mine.toList(); | 61 return new List<MethodMirror>.from(mine); |
61 } | 62 } |
62 } | 63 } |
63 | 64 |
64 /** Return true if the class has a getter named [name] */ | 65 /** Return true if the class has a getter named [name] */ |
65 bool hasGetter(Symbol name, ClassMirror mirror) { | 66 bool hasGetter(Symbol name, ClassMirror mirror) { |
66 if (name == null) return false; | 67 if (name == null) return false; |
67 var getter = mirror.getters[name]; | 68 var getter = mirror.declarations[name]; |
68 if (getter != null && !getter.isStatic) return true; | 69 if (getter is MethodMirror && getter.isGetter && !getter.isStatic) { |
| 70 return true; |
| 71 } |
69 var superclass = _getSuperclass(mirror); | 72 var superclass = _getSuperclass(mirror); |
70 if (superclass == null) return false; | 73 if (superclass == null) return false; |
71 return hasField(name, superclass); | 74 return hasField(name, superclass); |
72 } | 75 } |
73 | 76 |
74 /** | 77 /** |
75 * Return a list of all the public getters of a class which have corresponding | 78 * Return a list of all the public getters of a class which have corresponding |
76 * setters. | 79 * setters. |
77 */ | 80 */ |
78 Iterable<MethodMirror> publicGettersWithMatchingSetters(ClassMirror mirror) { | 81 Iterable<MethodMirror> publicGettersWithMatchingSetters(ClassMirror mirror) { |
79 var setters = mirror.setters; | 82 var declarations = mirror.declarations; |
80 return publicGetters(mirror).where((each) => | 83 return publicGetters(mirror).where((each) => |
81 setters["${each.simpleName}="] != null); | 84 // TODO(alanknight): Use new Symbol here? |
| 85 declarations["${each.simpleName}="] != null); |
82 } | 86 } |
83 | 87 |
84 /** | 88 /** |
85 * Given either an instance or a type, returns the type. Instances of Type | 89 * Given either an instance or a type, returns the type. Instances of Type |
86 * will be treated as types. Passing in an instance is really just backward | 90 * will be treated as types. Passing in an instance is really just backward |
87 * compatibility. | 91 * compatibility. |
88 */ | 92 */ |
89 ClassMirror turnInstanceIntoSomethingWeCanUse(x) { | 93 ClassMirror turnInstanceIntoSomethingWeCanUse(x) { |
90 if (x is Type) return reflectClass(x); | 94 if (x is Type) return reflectClass(x); |
91 return reflect(x).type; | 95 return reflect(x).type; |
92 } | 96 } |
OLD | NEW |