Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(176)

Side by Side Diff: pkg/polymer/lib/src/info.dart

Issue 23224003: move polymer.dart into dart svn (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: add --deploy to todomvc sample Created 7 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « pkg/polymer/lib/src/html5_utils.dart ('k') | pkg/polymer/lib/src/messages.dart » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(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 /**
6 * Datatypes holding information extracted by the analyzer and used by later
7 * phases of the compiler.
8 */
9 library polymer.src.info;
10
11 import 'dart:collection' show SplayTreeMap, LinkedHashMap;
12
13 import 'package:analyzer_experimental/src/generated/ast.dart';
14 import 'package:csslib/visitor.dart';
15 import 'package:html5lib/dom.dart';
16 import 'package:source_maps/span.dart' show Span;
17
18 import 'dart_parser.dart' show DartCodeInfo;
19 import 'messages.dart';
20 import 'summary.dart';
21 import 'utils.dart';
22
23 /**
24 * Information that is global. Roughly corresponds to `window` and `document`.
25 */
26 class GlobalInfo {
27 /**
28 * Pseudo-element names exposed in a component via a pseudo attribute.
29 * The name is only available from CSS (not Dart code) so they're mangled.
30 * The same pseudo-element in different components maps to the same
31 * mangled name (as the pseudo-element is scoped inside of the component).
32 */
33 final Map<String, String> pseudoElements = <String, String>{};
34
35 /** All components declared in the application. */
36 final Map<String, ComponentInfo> components = new SplayTreeMap();
37 }
38
39 /**
40 * Information for any library-like input. We consider each HTML file a library,
41 * and each component declaration a library as well. Hence we use this as a base
42 * class for both [FileInfo] and [ComponentInfo]. Both HTML files and components
43 * can have .dart code provided by the user for top-level user scripts and
44 * component-level behavior code. This code can either be inlined in the HTML
45 * file or included in a script tag with the "src" attribute.
46 */
47 abstract class LibraryInfo implements LibrarySummary {
48
49 /** Whether there is any code associated with the page/component. */
50 bool get codeAttached => inlinedCode != null || externalFile != null;
51
52 /**
53 * The actual inlined code. Use [userCode] if you want the code from this file
54 * or from an external file.
55 */
56 DartCodeInfo inlinedCode;
57
58 /**
59 * If this library's code was loaded using a script tag (e.g. in a component),
60 * [externalFile] has the path to such Dart file relative from the compiler's
61 * base directory.
62 */
63 UrlInfo externalFile;
64
65 /** Info asscociated with [externalFile], if any. */
66 FileInfo externalCode;
67
68 /**
69 * The inverse of [externalCode]. If this .dart file was imported via a script
70 * tag, this refers to the HTML file that imported it.
71 */
72 LibraryInfo htmlFile;
73
74 /** File where the top-level code was defined. */
75 UrlInfo get dartCodeUrl;
76
77 /**
78 * Name of the file that will hold any generated Dart code for this library
79 * unit. Note this is initialized after parsing.
80 */
81 String outputFilename;
82
83 /** Parsed cssSource. */
84 List<StyleSheet> styleSheets = [];
85
86 /** This is used in transforming Dart code to track modified files. */
87 bool modified = false;
88
89 /**
90 * This is used in transforming Dart code to compute files that reference
91 * [modified] files.
92 */
93 List<FileInfo> referencedBy = [];
94
95 /**
96 * Components used within this library unit. For [FileInfo] these are
97 * components used directly in the page. For [ComponentInfo] these are
98 * components used within their shadowed template.
99 */
100 final Map<ComponentSummary, bool> usedComponents =
101 new LinkedHashMap<ComponentSummary, bool>();
102
103 /**
104 * The actual code, either inlined or from an external file, or `null` if none
105 * was defined.
106 */
107 DartCodeInfo get userCode =>
108 externalCode != null ? externalCode.inlinedCode : inlinedCode;
109 }
110
111 /** Information extracted at the file-level. */
112 class FileInfo extends LibraryInfo implements HtmlFileSummary {
113 /** Relative path to this file from the compiler's base directory. */
114 final UrlInfo inputUrl;
115
116 /**
117 * Whether this file should be treated as the entry point of the web app, i.e.
118 * the file users navigate to in their browser. This will be true if this file
119 * was passed in the command line to the dwc compiler, and the
120 * `--components_only` flag was omitted.
121 */
122 final bool isEntryPoint;
123
124 // TODO(terry): Ensure that that the libraryName is a valid identifier:
125 // a..z || A..Z || _ [a..z || A..Z || 0..9 || _]*
126 String get libraryName =>
127 path.basename(inputUrl.resolvedPath).replaceAll('.', '_');
128
129 /** File where the top-level code was defined. */
130 UrlInfo get dartCodeUrl => externalFile != null ? externalFile : inputUrl;
131
132 /**
133 * All custom element definitions in this file. This may contain duplicates.
134 * Normally you should use [components] for lookup.
135 */
136 final List<ComponentInfo> declaredComponents = new List<ComponentInfo>();
137
138 /**
139 * All custom element definitions defined in this file or imported via
140 *`<link rel='components'>` tag. Maps from the tag name to the component
141 * information. This map is sorted by the tag name.
142 */
143 final Map<String, ComponentSummary> components =
144 new SplayTreeMap<String, ComponentSummary>();
145
146 /** Files imported with `<link rel="import">` */
147 final List<UrlInfo> componentLinks = <UrlInfo>[];
148
149 /** Files imported with `<link rel="stylesheet">` */
150 final List<UrlInfo> styleSheetHrefs = <UrlInfo>[];
151
152 /** Root is associated with the body node. */
153 Element body;
154
155 FileInfo(this.inputUrl, [this.isEntryPoint = false]);
156
157 /**
158 * Query for an [Element] matching the provided [tag], starting from the
159 * [body].
160 */
161 Element query(String tag) => body.query(tag);
162 }
163
164
165 /** Information about a web component definition declared locally. */
166 // TODO(sigmund): use a mixin to pull in ComponentSummary.
167 class ComponentInfo extends LibraryInfo implements ComponentSummary {
168 /** The file that declares this component. */
169 final FileInfo declaringFile;
170
171 /** The component tag name, defined with the `name` attribute on `element`. */
172 final String tagName;
173
174 /**
175 * The tag name that this component extends, defined with the `extends`
176 * attribute on `element`.
177 */
178 final String extendsTag;
179
180 /**
181 * The component info associated with the [extendsTag] name, if any.
182 * This will be `null` if the component extends a built-in HTML tag, or
183 * if the analyzer has not run yet.
184 */
185 ComponentSummary extendsComponent;
186
187 /** The Dart class containing the component's behavior. */
188 String className;
189
190 /** The Dart class declaration. */
191 ClassDeclaration get classDeclaration => _classDeclaration;
192 ClassDeclaration _classDeclaration;
193
194 /** The declaring `<element>` tag. */
195 final Node element;
196
197 /** File where this component was defined. */
198 UrlInfo get dartCodeUrl => externalFile != null
199 ? externalFile : declaringFile.inputUrl;
200
201 /**
202 * True if [tagName] was defined by more than one component. If this happened
203 * we will skip over the component.
204 */
205 bool hasConflict = false;
206
207 ComponentInfo(this.element, this.declaringFile, this.tagName,
208 this.extendsTag);
209
210 /**
211 * Gets the HTML tag extended by the base of the component hierarchy.
212 * Equivalent to [extendsTag] if this inherits directly from an HTML element,
213 * in other words, if [extendsComponent] is null.
214 */
215 String get baseExtendsTag =>
216 extendsComponent == null ? extendsTag : extendsComponent.baseExtendsTag;
217
218 Span get sourceSpan => element.sourceSpan;
219
220 /** Is apply-author-styles enabled. */
221 bool get hasAuthorStyles =>
222 element.attributes.containsKey('apply-author-styles');
223
224 /**
225 * Finds the declaring class, and initializes [className] and
226 * [classDeclaration]. Also [userCode] is generated if there was no script.
227 */
228 void findClassDeclaration(Messages messages) {
229 var constructor = element.attributes['constructor'];
230 className = constructor != null ? constructor :
231 toCamelCase(tagName, startUppercase: true);
232
233 // If we don't have any code, generate a small class definition, and
234 // pretend the user wrote it as inlined code.
235 if (userCode == null) {
236 var superclass = extendsComponent != null ? extendsComponent.className
237 : 'autogenerated.PolymerElement';
238 inlinedCode = new DartCodeInfo(null, null, [],
239 'class $className extends $superclass {\n}', null);
240 }
241
242 var code = userCode.code;
243 _classDeclaration = userCode.findClass(className);
244 if (_classDeclaration == null) {
245 // Check for deprecated x-tags implied constructor.
246 if (tagName.startsWith('x-') && constructor == null) {
247 var oldCtor = toCamelCase(tagName.substring(2), startUppercase: true);
248 _classDeclaration = userCode.findClass(oldCtor);
249 if (_classDeclaration != null) {
250 messages.warning('Implied constructor name for x-tags has changed to '
251 '"$className". You should rename your class or add a '
252 'constructor="$oldCtor" attribute to the element declaration. '
253 'Also custom tags are not required to start with "x-" if their '
254 'name has at least one dash.',
255 element.sourceSpan);
256 className = oldCtor;
257 }
258 }
259
260 if (_classDeclaration == null) {
261 messages.error('please provide a class definition '
262 'for $className:\n $code', element.sourceSpan);
263 return;
264 }
265 }
266 }
267
268 String toString() => '#<ComponentInfo $tagName '
269 '${inlinedCode != null ? "inline" : "from ${dartCodeUrl.resolvedPath}"}>';
270 }
271
272
273 /**
274 * Information extracted about a URL that refers to another file. This is
275 * mainly introduced to be able to trace back where URLs come from when
276 * reporting errors.
277 */
278 class UrlInfo {
279 /** Original url. */
280 final String url;
281
282 /** Path that the URL points to. */
283 final String resolvedPath;
284
285 /** Original source location where the URL was extracted from. */
286 final Span sourceSpan;
287
288 UrlInfo(this.url, this.resolvedPath, this.sourceSpan);
289
290 /**
291 * Resolve a path from an [url] found in a file located at [inputUrl].
292 * Returns null for absolute [url]. Unless [ignoreAbsolute] is true, reports
293 * an error message if the url is an absolute url.
294 */
295 static UrlInfo resolve(String url, UrlInfo inputUrl, Span span,
296 String packageRoot, Messages messages, {bool ignoreAbsolute: false}) {
297
298 var uri = Uri.parse(url);
299 if (uri.host != '' || (uri.scheme != '' && uri.scheme != 'package')) {
300 if (!ignoreAbsolute) {
301 messages.error('absolute paths not allowed here: "$url"', span);
302 }
303 return null;
304 }
305
306 var target;
307 if (url.startsWith('package:')) {
308 target = path.join(packageRoot, url.substring(8));
309 } else if (path.isAbsolute(url)) {
310 if (!ignoreAbsolute) {
311 messages.error('absolute paths not allowed here: "$url"', span);
312 }
313 return null;
314 } else {
315 target = path.join(path.dirname(inputUrl.resolvedPath), url);
316 url = pathToUrl(path.normalize(path.join(
317 path.dirname(inputUrl.url), url)));
318 }
319 target = path.normalize(target);
320
321 return new UrlInfo(url, target, span);
322 }
323
324 bool operator ==(UrlInfo other) =>
325 url == other.url && resolvedPath == other.resolvedPath;
326
327 int get hashCode => resolvedPath.hashCode;
328
329 String toString() => "#<UrlInfo url: $url, resolvedPath: $resolvedPath>";
330 }
OLDNEW
« no previous file with comments | « pkg/polymer/lib/src/html5_utils.dart ('k') | pkg/polymer/lib/src/messages.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698