| OLD | NEW |
| (Empty) |
| 1 #!/usr/bin/python | |
| 2 # Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | |
| 3 # for details. All rights reserved. Use of this source code is governed by a | |
| 4 # BSD-style license that can be found in the LICENSE file. | |
| 5 | |
| 6 """This module provides shared functionality for the systems to generate | |
| 7 frog binding from the IDL database.""" | |
| 8 | |
| 9 import os | |
| 10 from generator import * | |
| 11 from systembase import * | |
| 12 | |
| 13 # Members (getters, setters, and methods) to suppress. These are | |
| 14 # either removed or custom implemented. | |
| 15 _dom_frog_omitted_members = set([ | |
| 16 # Replace with custom. | |
| 17 'DOMWindow.get:top', | |
| 18 'HTMLIFrameElement.get:contentWindow', | |
| 19 | |
| 20 # Remove. | |
| 21 'DOMWindow.get:frameElement', | |
| 22 'HTMLIFrameElement.get:contentDocument', | |
| 23 ]) | |
| 24 | |
| 25 class FrogSystem(System): | |
| 26 | |
| 27 def __init__(self, templates, database, emitters, output_dir): | |
| 28 super(FrogSystem, self).__init__( | |
| 29 templates, database, emitters, output_dir) | |
| 30 self._impl_file_paths = [] | |
| 31 | |
| 32 def InterfaceGenerator(self, | |
| 33 interface, | |
| 34 common_prefix, | |
| 35 super_interface_name, | |
| 36 source_filter): | |
| 37 """.""" | |
| 38 template_file = 'impl_%s.darttemplate' % interface.id | |
| 39 template = self._templates.TryLoad(template_file) | |
| 40 if not template: | |
| 41 template = self._templates.Load('frog_impl.darttemplate') | |
| 42 | |
| 43 dart_code = self._ImplFileEmitter(interface.id) | |
| 44 return FrogInterfaceGenerator(self, interface, template, | |
| 45 super_interface_name, dart_code) | |
| 46 | |
| 47 def GenerateLibraries(self, lib_dir): | |
| 48 self._GenerateLibFile( | |
| 49 'frog_dom.darttemplate', | |
| 50 os.path.join(lib_dir, 'dom_frog.dart'), | |
| 51 (self._interface_system._dart_interface_file_paths + | |
| 52 self._interface_system._dart_callback_file_paths + | |
| 53 self._impl_file_paths)) | |
| 54 | |
| 55 def Finish(self): | |
| 56 pass | |
| 57 | |
| 58 def _ImplFileEmitter(self, name): | |
| 59 """Returns the file emitter of the Frog implementation file.""" | |
| 60 path = os.path.join(self._output_dir, 'src', 'frog', '%s.dart' % name) | |
| 61 self._impl_file_paths.append(path) | |
| 62 return self._emitters.FileEmitter(path) | |
| 63 | |
| 64 # ------------------------------------------------------------------------------ | |
| 65 | |
| 66 class FrogInterfaceGenerator(object): | |
| 67 """Generates a Frog class for a DOM IDL interface.""" | |
| 68 | |
| 69 def __init__(self, system, interface, template, super_interface, dart_code): | |
| 70 """Generates Dart code for the given interface. | |
| 71 | |
| 72 Args: | |
| 73 | |
| 74 interface: an IDLInterface instance. It is assumed that all types have | |
| 75 been converted to Dart types (e.g. int, String), unless they are in | |
| 76 the same package as the interface. | |
| 77 template: A string template. | |
| 78 super_interface: A string or None, the name of the common interface that | |
| 79 this interface implements, if any. | |
| 80 dart_code: an Emitter for the file containing the Dart implementation | |
| 81 class. | |
| 82 """ | |
| 83 self._system = system | |
| 84 self._interface = interface | |
| 85 self._template = template | |
| 86 self._super_interface = super_interface | |
| 87 self._dart_code = dart_code | |
| 88 self._current_secondary_parent = None | |
| 89 | |
| 90 | |
| 91 def StartInterface(self): | |
| 92 interface = self._interface | |
| 93 interface_name = interface.id | |
| 94 self._class_name = self._ImplClassName(interface_name) | |
| 95 | |
| 96 base = None | |
| 97 if interface.parents: | |
| 98 supertype = interface.parents[0].type.id | |
| 99 if IsDartCollectionType(supertype): | |
| 100 # List methods are injected in AddIndexer. | |
| 101 pass | |
| 102 else: | |
| 103 base = self._ImplClassName(supertype) | |
| 104 | |
| 105 native_spec = MakeNativeSpec(interface.javascript_binding_name) | |
| 106 | |
| 107 if base: | |
| 108 extends = ' extends ' + base | |
| 109 elif native_spec[0] == '=': | |
| 110 # The implementation is a singleton with no prototype. | |
| 111 extends = '' | |
| 112 else: | |
| 113 extends = ' extends _DOMTypeJs' | |
| 114 | |
| 115 # TODO: Include all implemented interfaces, including other Lists. | |
| 116 implements = [interface_name] | |
| 117 element_type = MaybeTypedArrayElementType(self._interface) | |
| 118 if element_type: | |
| 119 implements.append('List<%s>' % DartType(element_type)) | |
| 120 | |
| 121 self._members_emitter = self._dart_code.Emit( | |
| 122 self._template, | |
| 123 #class $CLASSNAME$EXTENDS$IMPLEMENTS$NATIVESPEC { | |
| 124 #$!MEMBERS | |
| 125 #} | |
| 126 CLASSNAME=self._class_name, | |
| 127 EXTENDS=extends, | |
| 128 IMPLEMENTS=' implements ' + ', '.join(implements), | |
| 129 NATIVESPEC=' native "' + native_spec + '"') | |
| 130 | |
| 131 element_type = MaybeTypedArrayElementType(interface) | |
| 132 if element_type: | |
| 133 self.AddTypedArrayConstructors(element_type) | |
| 134 | |
| 135 # Emit a factory provider class for the constructor. | |
| 136 constructor_info = AnalyzeConstructor(interface) | |
| 137 if constructor_info: | |
| 138 self._EmitFactoryProvider(interface_name, constructor_info) | |
| 139 | |
| 140 | |
| 141 def FinishInterface(self): | |
| 142 """.""" | |
| 143 pass | |
| 144 | |
| 145 def _ImplClassName(self, type_name): | |
| 146 return '_' + type_name + 'Js' | |
| 147 | |
| 148 def _EmitFactoryProvider(self, interface_name, constructor_info): | |
| 149 template_file = 'factoryprovider_%s.darttemplate' % interface_name | |
| 150 template = self._system._templates.TryLoad(template_file) | |
| 151 if not template: | |
| 152 template = self._system._templates.Load('factoryprovider.darttemplate') | |
| 153 | |
| 154 factory_provider = '_' + interface_name + 'FactoryProvider' | |
| 155 emitter = self._system._ImplFileEmitter(factory_provider) | |
| 156 emitter.Emit( | |
| 157 template, | |
| 158 FACTORYPROVIDER=factory_provider, | |
| 159 CONSTRUCTOR=interface_name, | |
| 160 PARAMETERS=constructor_info.ParametersImplementationDeclaration(), | |
| 161 NAMEDCONSTRUCTOR=constructor_info.name or interface_name, | |
| 162 ARGUMENTS=constructor_info.ParametersAsArgumentList()) | |
| 163 | |
| 164 def _NarrowToImplementationType(self, type_name): | |
| 165 # TODO(sra): Move into the 'system' and cache the result. | |
| 166 if type_name == 'EventListener': | |
| 167 # Callbacks are typedef functions so don't have a class. | |
| 168 return type_name | |
| 169 if self._system._database.HasInterface(type_name): | |
| 170 interface = self._system._database.GetInterface(type_name) | |
| 171 if RecognizeCallback(interface): | |
| 172 # Callbacks are typedef functions so don't have a class. | |
| 173 return type_name | |
| 174 else: | |
| 175 return self._ImplClassName(type_name) | |
| 176 return type_name | |
| 177 | |
| 178 def _NarrowInputType(self, type_name): | |
| 179 return self._NarrowToImplementationType(DartType(type_name)) | |
| 180 | |
| 181 def _NarrowOutputType(self, type_name): | |
| 182 return self._NarrowToImplementationType(DartType(type_name)) | |
| 183 | |
| 184 def AddConstant(self, constant): | |
| 185 # Since we are currently generating native classes without interfaces, | |
| 186 # generate the constants as part of the class. This will need to go away | |
| 187 # if we revert back to generating interfaces. | |
| 188 self._members_emitter.Emit('\n static final $TYPE $NAME = $VALUE;\n', | |
| 189 NAME=constant.id, | |
| 190 TYPE=DartType(constant.type.id), | |
| 191 VALUE=constant.value) | |
| 192 | |
| 193 pass | |
| 194 | |
| 195 def OverrideMember(self, member): | |
| 196 return self._interface.id + '.' + member in _dom_frog_omitted_members | |
| 197 | |
| 198 def AddAttribute(self, getter, setter): | |
| 199 if getter and self.OverrideMember('get:' + getter.id): | |
| 200 getter = None | |
| 201 if setter and self.OverrideMember('set:' + setter.id): | |
| 202 setter = None | |
| 203 if not getter and not setter: | |
| 204 return | |
| 205 | |
| 206 output_type = getter and self._NarrowOutputType(getter.type.id) | |
| 207 input_type = setter and self._NarrowInputType(setter.type.id) | |
| 208 | |
| 209 # If the (getter, setter) pair is shadowing, we can't generate a shadowing | |
| 210 # field (Issue 1633). | |
| 211 (super_getter, super_getter_interface) = self._FindShadowedAttribute(getter) | |
| 212 (super_setter, super_setter_interface) = self._FindShadowedAttribute(setter) | |
| 213 if super_getter or super_setter: | |
| 214 if getter and not setter and super_getter and not super_setter: | |
| 215 if DartType(getter.type.id) == DartType(super_getter.type.id): | |
| 216 # Compatible getter, use the superclass property. This works because | |
| 217 # JavaScript will do its own dynamic dispatch. | |
| 218 self._members_emitter.Emit( | |
| 219 '\n' | |
| 220 ' // Use implementation from $SUPER.\n' | |
| 221 ' // final $TYPE $NAME;\n', | |
| 222 SUPER=super_getter_interface.id, | |
| 223 NAME=DartDomNameOfAttribute(getter), TYPE=output_type) | |
| 224 return | |
| 225 | |
| 226 self._members_emitter.Emit('\n // Shadowing definition.') | |
| 227 self._AddAttributeUsingProperties(getter, setter) | |
| 228 return | |
| 229 | |
| 230 # Can't generate field if attribute has different name in JS and Dart. | |
| 231 if self._AttributeChangesName(getter or setter): | |
| 232 self._AddAttributeUsingProperties(getter, setter) | |
| 233 return | |
| 234 | |
| 235 if getter and setter and input_type == output_type: | |
| 236 self._members_emitter.Emit( | |
| 237 '\n $TYPE $NAME;\n', | |
| 238 NAME=DartDomNameOfAttribute(getter), TYPE=output_type) | |
| 239 return | |
| 240 if getter and not setter: | |
| 241 self._members_emitter.Emit( | |
| 242 '\n final $TYPE $NAME;\n', | |
| 243 NAME=DartDomNameOfAttribute(getter), TYPE=output_type) | |
| 244 return | |
| 245 self._AddAttributeUsingProperties(getter, setter) | |
| 246 | |
| 247 def _AttributeChangesName(self, attr): | |
| 248 return attr.id != DartDomNameOfAttribute(attr) | |
| 249 | |
| 250 def _AddAttributeUsingProperties(self, getter, setter): | |
| 251 if getter: | |
| 252 self._AddGetter(getter) | |
| 253 if setter: | |
| 254 self._AddSetter(setter) | |
| 255 | |
| 256 def _AddGetter(self, attr): | |
| 257 # TODO(sra): Remove native body when Issue 829 fixed. | |
| 258 self._members_emitter.Emit( | |
| 259 '\n $TYPE get $NAME() native "return this.$NATIVE_NAME;";\n', | |
| 260 NAME=DartDomNameOfAttribute(attr), | |
| 261 NATIVE_NAME=attr.id, | |
| 262 TYPE=self._NarrowOutputType(attr.type.id)) | |
| 263 | |
| 264 def _AddSetter(self, attr): | |
| 265 # TODO(sra): Remove native body when Issue 829 fixed. | |
| 266 self._members_emitter.Emit( | |
| 267 ' void set $NAME($TYPE value) native "this.$NATIVE_NAME = value;";\n', | |
| 268 NAME=DartDomNameOfAttribute(attr), | |
| 269 NATIVE_NAME=attr.id, | |
| 270 TYPE=self._NarrowInputType(attr.type.id)) | |
| 271 | |
| 272 def _FindShadowedAttribute(self, attr): | |
| 273 """Returns (attribute, superinterface) or (None, None).""" | |
| 274 def FindInParent(interface): | |
| 275 """Returns matching attribute in parent, or None.""" | |
| 276 if interface.parents: | |
| 277 parent = interface.parents[0] | |
| 278 if IsDartCollectionType(parent.type.id): | |
| 279 return (None, None) | |
| 280 if self._system._database.HasInterface(parent.type.id): | |
| 281 parent_interface = self._system._database.GetInterface(parent.type.id) | |
| 282 attr2 = FindMatchingAttribute(parent_interface, attr) | |
| 283 if attr2: | |
| 284 return (attr2, parent_interface) | |
| 285 return FindInParent(parent_interface) | |
| 286 return (None, None) | |
| 287 | |
| 288 return FindInParent(self._interface) if attr else (None, None) | |
| 289 | |
| 290 | |
| 291 def AddSecondaryAttribute(self, interface, getter, setter): | |
| 292 self._SecondaryContext(interface) | |
| 293 self.AddAttribute(getter, setter) | |
| 294 | |
| 295 def AddSecondaryOperation(self, interface, info): | |
| 296 self._SecondaryContext(interface) | |
| 297 self.AddOperation(info) | |
| 298 | |
| 299 def _SecondaryContext(self, interface): | |
| 300 if interface is not self._current_secondary_parent: | |
| 301 self._current_secondary_parent = interface | |
| 302 self._members_emitter.Emit('\n // From $WHERE\n', WHERE=interface.id) | |
| 303 | |
| 304 def AddIndexer(self, element_type): | |
| 305 """Adds all the methods required to complete implementation of List.""" | |
| 306 # We would like to simply inherit the implementation of everything except | |
| 307 # get length(), [], and maybe []=. It is possible to extend from a base | |
| 308 # array implementation class only when there is no other implementation | |
| 309 # inheritance. There might be no implementation inheritance other than | |
| 310 # DOMBaseWrapper for many classes, but there might be some where the | |
| 311 # array-ness is introduced by a non-root interface: | |
| 312 # | |
| 313 # interface Y extends X, List<T> ... | |
| 314 # | |
| 315 # In the non-root case we have to choose between: | |
| 316 # | |
| 317 # class YImpl extends XImpl { add List<T> methods; } | |
| 318 # | |
| 319 # and | |
| 320 # | |
| 321 # class YImpl extends ListBase<T> { copies of transitive XImpl methods; } | |
| 322 # | |
| 323 self._members_emitter.Emit( | |
| 324 '\n' | |
| 325 ' $TYPE operator[](int index) native "return this[index];";\n', | |
| 326 TYPE=self._NarrowOutputType(element_type)) | |
| 327 | |
| 328 if 'CustomIndexedSetter' in self._interface.ext_attrs: | |
| 329 self._members_emitter.Emit( | |
| 330 '\n' | |
| 331 ' void operator[]=(int index, $TYPE value) native "this[index] = valu
e";\n', | |
| 332 TYPE=self._NarrowInputType(element_type)) | |
| 333 else: | |
| 334 self._members_emitter.Emit( | |
| 335 '\n' | |
| 336 ' void operator[]=(int index, $TYPE value) {\n' | |
| 337 ' throw new UnsupportedOperationException("Cannot assign element of
immutable List.");\n' | |
| 338 ' }\n', | |
| 339 TYPE=self._NarrowInputType(element_type)) | |
| 340 | |
| 341 # TODO(sra): Use separate mixins for mutable implementations of List<T>. | |
| 342 # TODO(sra): Use separate mixins for typed array implementations of List<T>. | |
| 343 template_file = 'immutable_list_mixin.darttemplate' | |
| 344 template = self._system._templates.Load(template_file) | |
| 345 self._members_emitter.Emit(template, E=DartType(element_type)) | |
| 346 | |
| 347 | |
| 348 def AddTypedArrayConstructors(self, element_type): | |
| 349 self._members_emitter.Emit( | |
| 350 '\n' | |
| 351 ' factory $CTOR(int length) => _construct_$CTOR(length);\n' | |
| 352 '\n' | |
| 353 ' factory $CTOR.fromList(List<$TYPE> list) => _construct_$CTOR(list);\n
' | |
| 354 '\n' | |
| 355 ' factory $CTOR.fromBuffer(ArrayBuffer buffer) => _construct_$CTOR(buff
er);\n' | |
| 356 '\n' | |
| 357 ' static _construct_$CTOR(arg) native \'return new $CTOR(arg);\';\n', | |
| 358 CTOR=self._interface.id, | |
| 359 TYPE=DartType(element_type)) | |
| 360 | |
| 361 | |
| 362 def AddOperation(self, info): | |
| 363 """ | |
| 364 Arguments: | |
| 365 info: An OperationInfo object. | |
| 366 """ | |
| 367 # TODO(vsm): Handle overloads. | |
| 368 params = info.ParametersImplementationDeclaration( | |
| 369 lambda type_name: self._NarrowInputType(type_name)) | |
| 370 | |
| 371 native_body = dom_frog_native_bodies.get( | |
| 372 self._interface.id + '.' + info.name, '') | |
| 373 if native_body: | |
| 374 native_body = " '''" + native_body + "'''" | |
| 375 | |
| 376 self._members_emitter.Emit( | |
| 377 '\n' | |
| 378 ' $TYPE $NAME($PARAMS) native$NATIVESTRING;\n', | |
| 379 TYPE=self._NarrowOutputType(info.type_name), | |
| 380 NAME=info.name, | |
| 381 PARAMS=params, | |
| 382 NATIVESTRING=native_body) | |
| OLD | NEW |