OLD | NEW |
| (Empty) |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | |
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. | |
4 | |
5 part of polymer; | |
6 | |
7 /// *Warning* this class is experimental and subject to change. | |
8 /// | |
9 /// The data associated with a polymer-element declaration, if it is backed | |
10 /// by a Dart class instead of a JavaScript prototype. | |
11 class PolymerDeclaration { | |
12 /// The one syntax to rule them all. | |
13 static final BindingDelegate _polymerSyntax = new PolymerExpressions(); | |
14 | |
15 /// The polymer-element for this declaration. | |
16 final HtmlElement element; | |
17 | |
18 /// The Dart type corresponding to this custom element declaration. | |
19 final Type type; | |
20 | |
21 /// If we extend another custom element, this points to the super declaration. | |
22 final PolymerDeclaration superDeclaration; | |
23 | |
24 /// The name of the custom element. | |
25 final String name; | |
26 | |
27 /// Map of publish properties. Can be a field or a property getter, but if | |
28 /// this map contains a getter, is because it also has a corresponding setter. | |
29 /// | |
30 /// Note: technically these are always single properties, so we could use a | |
31 /// Symbol instead of a PropertyPath. However there are lookups between this | |
32 /// map and [_observe] so it is easier to just track paths. | |
33 Map<PropertyPath, smoke.Declaration> _publish; | |
34 | |
35 /// The names of published properties for this polymer-element. | |
36 Iterable<String> get publishedProperties => | |
37 _publish != null ? _publish.keys.map((p) => '$p') : const []; | |
38 | |
39 /// Same as [_publish] but with lower case names. | |
40 Map<String, smoke.Declaration> _publishLC; | |
41 | |
42 Map<PropertyPath, List<Symbol>> _observe; | |
43 | |
44 /// Name and expression for each computed property. | |
45 Map<Symbol, String> _computed = {}; | |
46 | |
47 Map<String, Object> _instanceAttributes; | |
48 | |
49 /// A set of properties that should be automatically reflected to attributes. | |
50 /// Typically this is used for CSS styling. If none, this variable will be | |
51 /// left as null. | |
52 Set<String> _reflect; | |
53 | |
54 List<Element> _sheets; | |
55 List<Element> get sheets => _sheets; | |
56 | |
57 List<Element> _styles; | |
58 List<Element> get styles => _styles; | |
59 | |
60 // The default syntax for polymer-elements. | |
61 PolymerExpressions syntax = _polymerSyntax; | |
62 | |
63 DocumentFragment get templateContent { | |
64 final template = fetchTemplate(); | |
65 return template != null ? templateBind(template).content : null; | |
66 } | |
67 | |
68 /// Maps event names and their associated method in the element class. | |
69 final Map<String, String> _eventDelegates = {}; | |
70 | |
71 /// Expected events per element node. | |
72 // TODO(sigmund): investigate whether we need more than 1 set of local events | |
73 // per element (why does the js implementation stores 1 per template node?) | |
74 Expando<Set<String>> _templateDelegates; | |
75 | |
76 String get extendee => | |
77 superDeclaration != null ? superDeclaration.name : null; | |
78 | |
79 /// The root URI for assets. | |
80 Uri _rootUri; | |
81 | |
82 /// List of properties to ignore for observation. | |
83 static Set<Symbol> _OBSERVATION_BLACKLIST = | |
84 new HashSet.from(const [#attribute]); | |
85 | |
86 static bool _canObserveProperty(Symbol property) => | |
87 !_OBSERVATION_BLACKLIST.contains(property); | |
88 | |
89 /// This list contains some property names that people commonly want to use, | |
90 /// but won't work because of Chrome/Safari bugs. It isn't an exhaustive | |
91 /// list. In particular it doesn't contain any property names found on | |
92 /// subtypes of HTMLElement (e.g. name, value). Rather it attempts to catch | |
93 /// some common cases. | |
94 /// | |
95 /// Dart Note: `class` is left out since its an invalid symbol in dart. This | |
96 /// means that nobody could make a property by this name anyways though. | |
97 /// Dart Note: We have added `classes` to this list, which is the dart:html | |
98 /// equivalent of `classList` but more likely to have conflicts. | |
99 static Set<Symbol> _PROPERTY_NAME_BLACKLIST = new HashSet.from([ | |
100 const Symbol('children'), | |
101 const Symbol('id'), | |
102 const Symbol('hidden'), | |
103 const Symbol('style'), | |
104 const Symbol('title'), | |
105 const Symbol('classes') | |
106 ]); | |
107 | |
108 bool _checkPropertyBlacklist(Symbol name) { | |
109 if (_PROPERTY_NAME_BLACKLIST.contains(name)) { | |
110 print('Cannot define property "$name" for element "${this.name}" ' | |
111 'because it has the same name as an HTMLElement property, and not ' | |
112 'all browsers support overriding that. Consider giving it a ' | |
113 'different name. '); | |
114 return true; | |
115 } | |
116 return false; | |
117 } | |
118 | |
119 // Dart note: since polymer-element is handled in JS now, we have a simplified | |
120 // flow for registering. We don't need to wait for the supertype or the code | |
121 // to be noticed. | |
122 PolymerDeclaration(this.element, this.name, this.type, this.superDeclaration); | |
123 | |
124 void register() { | |
125 // more declarative features | |
126 desugar(); | |
127 // register our custom element | |
128 registerType(name); | |
129 | |
130 // NOTE: skip in Dart because we don't have mutable global scope. | |
131 // reference constructor in a global named by 'constructor' attribute | |
132 // publishConstructor(); | |
133 } | |
134 | |
135 /// Implement various declarative features. | |
136 // Dart note: this merges "buildPrototype" "desugarBeforeChaining" and | |
137 // "desugarAfterChaining", because we don't have prototypes. | |
138 void desugar() { | |
139 | |
140 // back reference declaration element | |
141 _declarations[name] = this; | |
142 | |
143 // transcribe `attributes` declarations onto own prototype's `publish` | |
144 publishAttributes(superDeclaration); | |
145 | |
146 publishProperties(); | |
147 | |
148 inferObservers(); | |
149 | |
150 // desugar compound observer syntax, e.g. @ObserveProperty('a b c') | |
151 explodeObservers(); | |
152 | |
153 createPropertyAccessors(); | |
154 // install mdv delegate on template | |
155 installBindingDelegate(fetchTemplate()); | |
156 // install external stylesheets as if they are inline | |
157 installSheets(); | |
158 // adjust any paths in dom from imports | |
159 resolveElementPaths(element); | |
160 // compile list of attributes to copy to instances | |
161 accumulateInstanceAttributes(); | |
162 // parse on-* delegates declared on `this` element | |
163 parseHostEvents(); | |
164 // install a helper method this.resolvePath to aid in | |
165 // setting resource urls. e.g. | |
166 // this.$.image.src = this.resolvePath('images/foo.png') | |
167 initResolvePath(); | |
168 // under ShadowDOMPolyfill, transforms to approximate missing CSS features | |
169 _shimShadowDomStyling(templateContent, name, extendee); | |
170 | |
171 // TODO(jmesserly): this feels unnatrual in Dart. Since we have convenient | |
172 // lazy static initialization, can we get by without it? | |
173 if (smoke.hasStaticMethod(type, #registerCallback)) { | |
174 smoke.invoke(type, #registerCallback, [this]); | |
175 } | |
176 } | |
177 | |
178 void registerType(String name) { | |
179 var baseTag; | |
180 var decl = this; | |
181 while (decl != null) { | |
182 baseTag = decl.element.attributes['extends']; | |
183 decl = decl.superDeclaration; | |
184 } | |
185 document.registerElement(name, type, extendsTag: baseTag); | |
186 } | |
187 | |
188 // from declaration/mdv.js | |
189 Element fetchTemplate() => element.querySelector('template'); | |
190 | |
191 void installBindingDelegate(Element template) { | |
192 if (template != null) { | |
193 templateBind(template).bindingDelegate = this.syntax; | |
194 } | |
195 } | |
196 | |
197 // from declaration/path.js | |
198 void resolveElementPaths(Node node) => PolymerJs.resolveElementPaths(node); | |
199 | |
200 // Dart note: renamed from "addResolvePathApi". | |
201 void initResolvePath() { | |
202 // let assetpath attribute modify the resolve path | |
203 var assetPath = element.attributes['assetpath']; | |
204 if (assetPath == null) assetPath = ''; | |
205 var base = Uri.parse(element.ownerDocument.baseUri); | |
206 _rootUri = base.resolve(assetPath); | |
207 } | |
208 | |
209 String resolvePath(String urlPath, [baseUrlOrString]) { | |
210 Uri base; | |
211 if (baseUrlOrString == null) { | |
212 // Dart note: this enforces the same invariant as JS, where you need to | |
213 // call addResolvePathApi first. | |
214 if (_rootUri == null) { | |
215 throw new StateError('call initResolvePath before calling resolvePath'); | |
216 } | |
217 base = _rootUri; | |
218 } else if (baseUrlOrString is Uri) { | |
219 base = baseUrlOrString; | |
220 } else { | |
221 base = Uri.parse(baseUrlOrString); | |
222 } | |
223 return base.resolve(urlPath).toString(); | |
224 } | |
225 | |
226 void publishAttributes(PolymerDeclaration superDecl) { | |
227 // get properties to publish | |
228 if (superDecl != null) { | |
229 // Dart note: even though we walk the type hierarchy in | |
230 // _getPublishedProperties, this will additionally include any names | |
231 // published via the `attributes` attribute. | |
232 if (superDecl._publish != null) { | |
233 _publish = new Map.from(superDecl._publish); | |
234 } | |
235 if (superDecl._reflect != null) { | |
236 _reflect = new Set.from(superDecl._reflect); | |
237 } | |
238 } | |
239 | |
240 _getPublishedProperties(type); | |
241 | |
242 // merge names from 'attributes' attribute into the '_publish' object | |
243 var attrs = element.attributes['attributes']; | |
244 if (attrs != null) { | |
245 // names='a b c' or names='a,b,c' | |
246 // record each name for publishing | |
247 for (var attr in attrs.split(_ATTRIBUTES_REGEX)) { | |
248 // remove excess ws | |
249 attr = attr.trim(); | |
250 | |
251 // if the user hasn't specified a value, we want to use the | |
252 // default, unless a superclass has already chosen one | |
253 if (attr == '') continue; | |
254 | |
255 var decl, path; | |
256 var property = smoke.nameToSymbol(attr); | |
257 if (property != null) { | |
258 path = new PropertyPath([property]); | |
259 if (_publish != null && _publish.containsKey(path)) { | |
260 continue; | |
261 } | |
262 decl = smoke.getDeclaration(type, property); | |
263 } | |
264 | |
265 if (property == null || decl == null || decl.isMethod || decl.isFinal) { | |
266 window.console.warn('property for attribute $attr of polymer-element ' | |
267 'name=$name not found.'); | |
268 continue; | |
269 } | |
270 if (_publish == null) _publish = {}; | |
271 _publish[path] = decl; | |
272 } | |
273 } | |
274 | |
275 // NOTE: the following is not possible in Dart; fields must be declared. | |
276 // install 'attributes' as properties on the prototype, | |
277 // but don't override | |
278 } | |
279 | |
280 void _getPublishedProperties(Type type) { | |
281 var options = const smoke.QueryOptions( | |
282 includeInherited: true, | |
283 includeUpTo: HtmlElement, | |
284 withAnnotations: const [PublishedProperty]); | |
285 for (var decl in smoke.query(type, options)) { | |
286 if (decl.isFinal) continue; | |
287 if (_checkPropertyBlacklist(decl.name)) continue; | |
288 if (_publish == null) _publish = {}; | |
289 _publish[new PropertyPath([decl.name])] = decl; | |
290 | |
291 // Should we reflect the property value to the attribute automatically? | |
292 if (decl.annotations | |
293 .where((a) => a is PublishedProperty) | |
294 .any((a) => a.reflect)) { | |
295 if (_reflect == null) _reflect = new Set(); | |
296 _reflect.add(smoke.symbolToName(decl.name)); | |
297 } | |
298 } | |
299 } | |
300 | |
301 void accumulateInstanceAttributes() { | |
302 // inherit instance attributes | |
303 _instanceAttributes = new Map<String, Object>(); | |
304 if (superDeclaration != null) { | |
305 _instanceAttributes.addAll(superDeclaration._instanceAttributes); | |
306 } | |
307 | |
308 // merge attributes from element | |
309 element.attributes.forEach((name, value) { | |
310 if (isInstanceAttribute(name)) { | |
311 _instanceAttributes[name] = value; | |
312 } | |
313 }); | |
314 } | |
315 | |
316 static bool isInstanceAttribute(name) { | |
317 // do not clone these attributes onto instances | |
318 final blackList = const { | |
319 'name': 1, | |
320 'extends': 1, | |
321 'constructor': 1, | |
322 'noscript': 1, | |
323 'assetpath': 1, | |
324 'cache-csstext': 1, | |
325 // add ATTRIBUTES_ATTRIBUTE to the blacklist | |
326 'attributes': 1, | |
327 }; | |
328 | |
329 return !blackList.containsKey(name) && !name.startsWith('on-'); | |
330 } | |
331 | |
332 /// Extracts events from the element tag attributes. | |
333 void parseHostEvents() { | |
334 addAttributeDelegates(_eventDelegates); | |
335 } | |
336 | |
337 void addAttributeDelegates(Map<String, String> delegates) { | |
338 element.attributes.forEach((name, value) { | |
339 if (_hasEventPrefix(name)) { | |
340 var start = value.indexOf('{{'); | |
341 var end = value.lastIndexOf('}}'); | |
342 if (start >= 0 && end >= 0) { | |
343 delegates[_removeEventPrefix(name)] = | |
344 value.substring(start + 2, end).trim(); | |
345 } | |
346 } | |
347 }); | |
348 } | |
349 | |
350 String urlToPath(String url) { | |
351 if (url == null) return ''; | |
352 return (url.split('/') | |
353 ..removeLast() | |
354 ..add('')).join('/'); | |
355 } | |
356 | |
357 // Dart note: loadStyles, convertSheetsToStyles, copySheetAttribute and | |
358 // findLoadableStyles are not ported because they're handled by Polymer JS | |
359 // before we get into [register]. | |
360 | |
361 /// Install external stylesheets loaded in <element> elements into the | |
362 /// element's template. | |
363 void installSheets() { | |
364 cacheSheets(); | |
365 cacheStyles(); | |
366 installLocalSheets(); | |
367 installGlobalStyles(); | |
368 } | |
369 | |
370 void cacheSheets() { | |
371 _sheets = findNodes(_SHEET_SELECTOR); | |
372 for (var s in sheets) s.remove(); | |
373 } | |
374 | |
375 void cacheStyles() { | |
376 _styles = findNodes('$_STYLE_SELECTOR[$_SCOPE_ATTR]'); | |
377 for (var s in styles) s.remove(); | |
378 } | |
379 | |
380 /// Takes external stylesheets loaded in an `<element>` element and moves | |
381 /// their content into a style element inside the `<element>`'s template. | |
382 /// The sheet is then removed from the `<element>`. This is done only so | |
383 /// that if the element is loaded in the main document, the sheet does | |
384 /// not become active. | |
385 /// Note, ignores sheets with the attribute 'polymer-scope'. | |
386 void installLocalSheets() { | |
387 var sheets = | |
388 this.sheets.where((s) => !s.attributes.containsKey(_SCOPE_ATTR)); | |
389 var content = templateContent; | |
390 if (content != null) { | |
391 var cssText = new StringBuffer(); | |
392 for (var sheet in sheets) { | |
393 cssText | |
394 ..write(_cssTextFromSheet(sheet)) | |
395 ..write('\n'); | |
396 } | |
397 if (cssText.length > 0) { | |
398 var style = element.ownerDocument.createElement('style') | |
399 ..text = '$cssText'; | |
400 | |
401 content.insertBefore(style, content.firstChild); | |
402 } | |
403 } | |
404 } | |
405 | |
406 List<Element> findNodes(String selector, [bool matcher(Element e)]) { | |
407 var nodes = element.querySelectorAll(selector).toList(); | |
408 var content = templateContent; | |
409 if (content != null) { | |
410 nodes = nodes..addAll(content.querySelectorAll(selector)); | |
411 } | |
412 if (matcher != null) return nodes.where(matcher).toList(); | |
413 return nodes; | |
414 } | |
415 | |
416 /// Promotes external stylesheets and style elements with the attribute | |
417 /// polymer-scope='global' into global scope. | |
418 /// This is particularly useful for defining @keyframe rules which | |
419 /// currently do not function in scoped or shadow style elements. | |
420 /// (See wkb.ug/72462) | |
421 // TODO(sorvell): remove when wkb.ug/72462 is addressed. | |
422 void installGlobalStyles() { | |
423 var style = styleForScope(_STYLE_GLOBAL_SCOPE); | |
424 Polymer.applyStyleToScope(style, document.head); | |
425 } | |
426 | |
427 String cssTextForScope(String scopeDescriptor) { | |
428 var cssText = new StringBuffer(); | |
429 // handle stylesheets | |
430 var selector = '[$_SCOPE_ATTR=$scopeDescriptor]'; | |
431 matcher(s) => s.matches(selector); | |
432 | |
433 for (var sheet in sheets.where(matcher)) { | |
434 cssText | |
435 ..write(_cssTextFromSheet(sheet)) | |
436 ..write('\n\n'); | |
437 } | |
438 // handle cached style elements | |
439 for (var style in styles.where(matcher)) { | |
440 cssText | |
441 ..write(style.text) | |
442 ..write('\n\n'); | |
443 } | |
444 return cssText.toString(); | |
445 } | |
446 | |
447 StyleElement styleForScope(String scopeDescriptor) { | |
448 var cssText = cssTextForScope(scopeDescriptor); | |
449 return cssTextToScopeStyle(cssText, scopeDescriptor); | |
450 } | |
451 | |
452 StyleElement cssTextToScopeStyle(String cssText, String scopeDescriptor) { | |
453 if (cssText == '') return null; | |
454 | |
455 return new StyleElement() | |
456 ..text = cssText | |
457 ..attributes[_STYLE_SCOPE_ATTRIBUTE] = '$name-$scopeDescriptor'; | |
458 } | |
459 | |
460 /// Fetch a list of all *Changed methods so we can observe the associated | |
461 /// properties. | |
462 void inferObservers() { | |
463 for (var decl in smoke.query(type, _changedMethodQueryOptions)) { | |
464 // TODO(jmesserly): now that we have a better system, should we | |
465 // deprecate *Changed methods? | |
466 if (_observe == null) _observe = new HashMap(); | |
467 var name = smoke.symbolToName(decl.name); | |
468 name = name.substring(0, name.length - 7); | |
469 if (!_canObserveProperty(decl.name)) continue; | |
470 _observe[new PropertyPath(name)] = [decl.name]; | |
471 } | |
472 } | |
473 | |
474 /// Fetch a list of all methods annotated with [ObserveProperty] so we can | |
475 /// observe the associated properties. | |
476 void explodeObservers() { | |
477 var options = const smoke.QueryOptions( | |
478 includeFields: false, | |
479 includeProperties: false, | |
480 includeMethods: true, | |
481 includeInherited: true, | |
482 includeUpTo: HtmlElement, | |
483 withAnnotations: const [ObserveProperty]); | |
484 for (var decl in smoke.query(type, options)) { | |
485 for (var meta in decl.annotations) { | |
486 if (meta is! ObserveProperty) continue; | |
487 if (_observe == null) _observe = new HashMap(); | |
488 for (String name in meta.names) { | |
489 _observe.putIfAbsent(new PropertyPath(name), () => []).add(decl.name); | |
490 } | |
491 } | |
492 } | |
493 } | |
494 | |
495 void publishProperties() { | |
496 // Dart note: _publish was already populated by publishAttributes | |
497 if (_publish != null) _publishLC = _lowerCaseMap(_publish); | |
498 } | |
499 | |
500 Map<String, dynamic> _lowerCaseMap(Map<PropertyPath, dynamic> properties) { | |
501 final map = new Map<String, dynamic>(); | |
502 properties.forEach((PropertyPath path, value) { | |
503 map['$path'.toLowerCase()] = value; | |
504 }); | |
505 return map; | |
506 } | |
507 | |
508 void createPropertyAccessors() { | |
509 // Dart note: since we don't have a prototype in Dart, most of the work of | |
510 // createPolymerAccessors is done lazily on the first access of properties. | |
511 // Here we just extract the information from annotations and store it as | |
512 // properties on the declaration. | |
513 | |
514 // Dart Note: The js side makes computed properties read only, and does | |
515 // special logic right here for them. For us they are automatically read | |
516 // only unless you define a setter for them, so we left that out. | |
517 var options = const smoke.QueryOptions( | |
518 includeInherited: true, | |
519 includeUpTo: HtmlElement, | |
520 withAnnotations: const [ComputedProperty]); | |
521 var existing = {}; | |
522 for (var decl in smoke.query(type, options)) { | |
523 var name = decl.name; | |
524 if (_checkPropertyBlacklist(name)) continue; | |
525 var meta = decl.annotations.firstWhere((e) => e is ComputedProperty); | |
526 var prev = existing[name]; | |
527 // The definition of a child class takes priority. | |
528 if (prev == null || smoke.isSubclassOf(decl.type, prev.type)) { | |
529 _computed[name] = meta.expression; | |
530 existing[name] = decl; | |
531 } | |
532 } | |
533 } | |
534 } | |
535 | |
536 /// maps tag names to prototypes | |
537 final Map _typesByName = new Map<String, Type>(); | |
538 | |
539 Type _getRegisteredType(String name) => _typesByName[name]; | |
540 | |
541 /// Dart Note: instanceOfType not implemented for dart, its not needed. | |
542 | |
543 /// track document.register'ed tag names and their declarations | |
544 final Map _declarations = new Map<String, PolymerDeclaration>(); | |
545 | |
546 bool _isRegistered(String name) => _declarations.containsKey(name); | |
547 PolymerDeclaration _getDeclaration(String name) => _declarations[name]; | |
548 | |
549 /// Using Polymer's web_components/src/ShadowCSS.js passing the style tag's | |
550 /// content. | |
551 void _shimShadowDomStyling( | |
552 DocumentFragment template, String name, String extendee) { | |
553 if (_ShadowCss == null || !_hasShadowDomPolyfill) return; | |
554 | |
555 _ShadowCss.callMethod('shimStyling', [template, name, extendee]); | |
556 } | |
557 | |
558 final bool _hasShadowDomPolyfill = js.context.hasProperty('ShadowDOMPolyfill'); | |
559 final JsObject _ShadowCss = | |
560 _WebComponents != null ? _WebComponents['ShadowCSS'] : null; | |
561 | |
562 const _STYLE_SELECTOR = 'style'; | |
563 const _SHEET_SELECTOR = 'link[rel=stylesheet]'; | |
564 const _STYLE_GLOBAL_SCOPE = 'global'; | |
565 const _SCOPE_ATTR = 'polymer-scope'; | |
566 const _STYLE_SCOPE_ATTRIBUTE = 'element'; | |
567 const _STYLE_CONTROLLER_SCOPE = 'controller'; | |
568 | |
569 String _cssTextFromSheet(LinkElement sheet) { | |
570 if (sheet == null) return ''; | |
571 | |
572 // In deploy mode we should never do a sync XHR; link rel=stylesheet will | |
573 // be inlined into a <style> tag by ImportInliner. | |
574 if (_deployMode) return ''; | |
575 | |
576 // TODO(jmesserly): sometimes the href property is wrong after deployment. | |
577 var href = sheet.href; | |
578 if (href == '') href = sheet.attributes["href"]; | |
579 | |
580 // TODO(jmesserly): it seems like polymer-js is always polyfilling | |
581 // HTMLImports, because their code depends on "__resource" to work, so I | |
582 // don't see how it can work with native HTML Imports. We use a sync-XHR | |
583 // under the assumption that the file is likely to have been already | |
584 // downloaded and cached by HTML Imports. | |
585 try { | |
586 return (new HttpRequest() | |
587 ..open('GET', href, async: false) | |
588 ..send()).responseText; | |
589 } on DomException catch (e, t) { | |
590 _sheetLog.fine('failed to XHR stylesheet text href="$href" error: ' | |
591 '$e, trace: $t'); | |
592 return ''; | |
593 } | |
594 } | |
595 | |
596 final Logger _sheetLog = new Logger('polymer.stylesheet'); | |
597 | |
598 final smoke.QueryOptions _changedMethodQueryOptions = new smoke.QueryOptions( | |
599 includeFields: false, | |
600 includeProperties: false, | |
601 includeMethods: true, | |
602 includeInherited: true, | |
603 includeUpTo: HtmlElement, | |
604 matches: _isObserverMethod); | |
605 | |
606 bool _isObserverMethod(Symbol symbol) { | |
607 String name = smoke.symbolToName(symbol); | |
608 if (name == null) return false; | |
609 return name.endsWith('Changed') && name != 'attributeChanged'; | |
610 } | |
611 | |
612 final _ATTRIBUTES_REGEX = new RegExp(r'\s|,'); | |
613 | |
614 final JsObject _WebComponents = js.context['WebComponents']; | |
OLD | NEW |