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

Side by Side Diff: client/html/scripts/html_diff.dart

Issue 8771054: Add a script to generate HTML and DOM docs with cross-links to one another. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 9 years 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
OLDNEW
(Empty)
1 // Copyright (c) 2011, 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 * A script to assist in documenting the difference between the dart:html API
7 * and the old DOM API.
8 */
9 #library('html_diff');
10
11 #import('../../../frog/lang.dart');
12 #import('../../../frog/file_system_node.dart');
13 #import('../../../frog/file_system.dart');
14 #import('../../../utils/dartdoc/dartdoc.dart');
15
16 void main() {
17 HtmlDiff.initWorld('../../frog', new NodeFileSystem());
18 initializeDartDoc();
19 var diff = new HtmlDiff();
20 diff.run();
21
22 diff.domToHtml.forEach((domMember, htmlMembers) {
23 for (var htmlMember in htmlMembers) {
24 if (diff.sameName(domMember, htmlMember)) continue;
25 var htmlTypeName = htmlMember.declaringType.name;
26 var htmlName = '$htmlTypeName.${htmlMember.name}';
27 if (htmlMember.isConstructor || htmlMember.isFactory) {
28 final separator = htmlMember.constructorName == '' ? '' : '.';
29 htmlName = 'new $htmlTypeName$separator${htmlMember.constructorName}';
30 }
31 print('${domMember.declaringType.name}.${domMember.name} -> ${htmlName}');
32 }
33 });
34
35 for (var type in world.dom.types.getValues()) {
36 if (type.name == null) continue;
37 if (type.definition is FunctionTypeDefinition) continue;
38 for (var member in type.members.getValues()) {
39 if (!member.isPrivate && member.name != 'typeName' &&
40 !diff.domToHtml.containsKey(member) &&
41 (member is MethodMember || member is PropertyMember)) {
42 print('No dart:html wrapper for ${type.name}.${member.name}');
43 }
44 }
45 }
46 }
47
48 class HtmlDiff {
49 final Map<Member, Set<Member>> domToHtml;
50 final Map<Member, Set<Member>> htmlToDom;
51 final Map<Type, Set<Type>> domTypesToHtml;
52 final Map<Type, Set<Type>> htmlTypesToDom;
53
54 static void initWorld(String frogDir, FileSystem files) {
55 parseOptions(frogDir, [] /* args */, files);
56 initializeWorld(files);
57 world.processDartScript('dart:htmlimpl');
58 world.resolveAll();
59 }
60
61 HtmlDiff() :
62 domToHtml = <Member, Set<Member>>{},
63 htmlToDom = <Member, Set<Member>>{},
64 domTypesToHtml = <Type, Set<Type>>{},
65 htmlTypesToDom = <Type, Set<Type>>{};
66
67 void run() {
68 final htmlLib = world.libraries['dart:htmlimpl'];
69 for (var implType in htmlLib.types.getValues()) {
70 final domTypes = htmlToDomTypes(implType);
71 final htmlType = htmlImplToHtmlType(implType);
72 if (htmlType == null) continue;
73
74 htmlTypesToDom.putIfAbsent(htmlType, () => new Set()).addAll(domTypes);
75 domTypes.forEach((t) =>
76 domTypesToHtml.putIfAbsent(t, () => new Set()).add(htmlType));
77
78 final members = new List.from(implType.members.getValues());
79 members.addAll(implType.constructors.getValues());
80 implType.factories.forEach((f) => members.add(f));
81 members.forEach((m) => _addMemberDiff(m, domTypes));
82 }
83 }
84
85 bool sameName(Member domMember, Member htmlMember) {
86 var domTypeName = domMember.declaringType.name;
87 if (domTypeName == 'DOMWindow') domTypeName = 'Window';
88 domTypeName = domTypeName.replaceFirst(new RegExp('^(HTML|WebKit)'), '');
89 var htmlTypeName = htmlMember.declaringType.name;
90
91 var domName = domMember.name;
92 var htmlName = htmlMember.name;
93 if (htmlName.startsWith('get:') || htmlName.startsWith('set:')) {
94 htmlName = htmlName.substring(4);
95 }
96
97 return domTypeName == htmlTypeName && domName == htmlName;
98 }
99
100 void _addMemberDiff(Member implMember, List<Type> domTypes) {
101 if (implMember.isProperty) {
102 if (implMember.canGet) _addMemberDiff(implMember.getter, domTypes);
103 if (implMember.canSet) _addMemberDiff(implMember.setter, domTypes);
104 }
105
106 var domMembers = htmlToDomMembers(implMember, domTypes);
107 var htmlMember = htmlImplToHtmlMember(implMember);
108 if (htmlMember == null && !domMembers.isEmpty()) {
109 print('Warning: dart:htmlimpl member ${implMember.declaringType.name}.' +
110 '${implMember.name} has no corresponding dart:html member.');
111 }
112
113 if (htmlMember == null) return;
114 if (!domMembers.isEmpty()) htmlToDom[htmlMember] = domMembers;
115 domMembers.forEach((m) =>
116 domToHtml.putIfAbsent(m, () => new Set()).add(htmlMember));
117 }
118
119 Type htmlImplToHtmlType(Type implType) {
120 if (implType == null || implType.isTop || implType.interfaces.isEmpty() ||
121 implType.interfaces[0].library.name != 'html') {
122 return null;
123 }
124
125 return implType.interfaces[0];
126 }
127
128 Member htmlImplToHtmlMember(Member implMember) {
129 var htmlType = htmlImplToHtmlType(implMember.declaringType);
130 if (htmlType == null) return null;
131
132 bool getter, setter;
133 if (implMember.isConstructor || implMember.isFactory) {
134 var constructor = htmlType.constructors[implMember.name];
135 if (constructor != null) return constructor;
136 return htmlType.factories[implMember.name];
137 } else if ((getter = implMember.name.startsWith('get:')) ||
138 (setter = implMember.name.startsWith('set:'))) {
139 // Use getMember to follow interface inheritance chains. If it's a
140 // ConcreteMember, though, it's an implementation of some data structure
141 // and we don't care about it.
142 var htmlProperty = htmlType.getMember(implMember.name.substring(4));
143 if (htmlProperty != null && htmlProperty is! ConcreteMember) {
144 return getter ? htmlProperty.getter : htmlProperty.setter;
145 } else {
146 return null;
147 }
148 } else {
149 return htmlType.getMember(implMember.name);
150 }
151 }
152
153 List<Type> htmlToDomTypes(Type htmlType) {
154 if (htmlType.name == null) return [];
155 final tags = _getTags(findComment(htmlType.span));
156
157 if (tags.containsKey('domName')) {
158 var domNames = map(tags['domName'].split(','), (s) => s.trim());
159 if (domNames.length == 1 && domNames[0] == 'none') return [];
160 return map(domNames, (domName) {
161 // DOMWindow is Chrome-specific, so we don't use it in our annotations.
162 if (domName == 'Window') domName = 'DOMWindow';
163 final domType = world.dom.types[domName];
164 if (domType == null) print('Warning: no dart:dom type named $domName');
165 return domType;
166 });
167 } else {
168 if (!htmlType.name.endsWith('WrappingImplementation')) return [];
169 final domName = htmlType.name.replaceFirst('WrappingImplementation', '');
170 var domType = world.dom.types[domName];
171 if (domType == null && domName.endsWith('Element')) {
172 domType = world.dom.types['HTML$domName'];
173 }
174 if (domType == null) domType = world.dom.types['WebKit$domName'];
175 if (domType == null) {
176 print('Warning: no dart:dom type matches dart:htmlimpl ' +
177 htmlType.name);
178 return [];
179 }
180 return [domType];
181 }
182 }
183
184 Set<Member> htmlToDomMembers(Member htmlMember, List<Type> domTypes) {
185 if (htmlMember.isPrivate || htmlMember is! MethodMember) return new Set();
186 final tags = _getTags(findComment(htmlMember.span));
187 if (tags.containsKey('domName')) {
188 final domNames = map(tags['domName'].split(','), (s) => s.trim());
189 if (domNames.length == 1 && domNames[0] == 'none') return new Set();
190 final members = new Set();
191 domNames.forEach((name) {
192 var nameMembers = _membersFromName(name, domTypes);
193 if (nameMembers.isEmpty()) {
194 if (name.contains('.')) {
195 print('Warning: no member $name');
196 } else {
197 final options = Strings.join(
198 map(domTypes, (t) => "${t.name}.$name"), ' or ');
199 print('Warning: no member $options');
200 }
201 }
202 members.addAll(nameMembers);
203 });
204 return members;
205 }
206
207 if (domTypes.isEmpty() || htmlMember.definition == null) return new Set();
208 if (htmlMember.name == 'get:on') {
209 final members = _membersFromName('addEventListener', domTypes);
210 members.addAll(_membersFromName('dispatchEvent', domTypes));
211 members.addAll(_membersFromName('removeEventListener', domTypes));
212 return members;
213 }
214
215 if (htmlMember.isFactory && htmlMember.name == '' &&
216 domTypes.length == 1 && domTypes[0].name.endsWith('Event')) {
217 return _membersFromName('init${domTypes[0].name}', domTypes);
218 }
219
220 return _getDomMembers(htmlMember.definition.body, domTypes);
221 }
222
223 Set<Member> _membersFromName(String name, List<Type> defaultTypes) {
224 if (!name.contains('.', 0)) {
225 if (defaultTypes.isEmpty()) {
226 print('Warning: no default type for ${name}');
227 return new Set();
228 }
229 final members = new Set<Member>();
230 defaultTypes.forEach((t) {
231 if (t.members.containsKey(name)) members.add(t.members[name]);
232 });
233 return members;
234 }
235
236 final splitName = name.split('.');
237 if (splitName.length != 2) {
238 print('Warning: invalid member name ${name}');
239 return new Set();
240 }
241 var typeName = splitName[0];
242 if (typeName == 'Window') typeName = 'DOMWindow';
243 final type = world.dom.types[typeName];
244 if (type == null) return new Set();
245 final member = type.members[splitName[1]];
246 if (member == null) return new Set();
247 return new Set.from([member]);
248 }
249
250 Set<Member> _getDomMembers(Statement stmt, List<Type> domTypes) {
251 if (stmt is BlockStatement) {
252 final body = stmt.body.filter((s) => !_ignorableStatement(s));
253 if (body.length != 1) return new Set();
254 return _getDomMembers(stmt.body[0], domTypes);
255 } else if (stmt is ReturnStatement) {
256 return _domMembersFromExpression(stmt.value, domTypes);
257 } else if (stmt is ExpressionStatement) {
258 return _domMembersFromExpression(stmt.body, domTypes);
259 } else if (stmt is TryStatement) {
260 return _getDomMembers(stmt.body, domTypes);
261 } else if (stmt is IfStatement) {
262 final members = _getDomMembers(stmt.trueBranch, domTypes);
263 members.addAll(_getDomMembers(stmt.falseBranch, domTypes));
264 return members;
265 } else {
266 return new Set();
267 }
268 }
269
270 /**
271 * Whether a statement can be ignored for the purpose of determining the DOM
272 * name of the enclosing method. The Webkit-to-Dart conversion process leaves
273 * behind various throws and returns that we want to ignore.
274 */
275 bool _ignorableStatement(Statement stmt) {
276 if (stmt is BlockStatement) {
277 return Collections.every(stmt.body, (s) => _ignorableStatement(s));
278 } else if (stmt is TryStatement) {
279 return _ignorableStatement(stmt.body);
280 } else if (stmt is IfStatement) {
281 return _ignorableStatement(stmt.trueBranch) &&
282 _ignorableStatement(stmt.falseBranch);
283 } else if (stmt is ReturnStatement) {
284 return stmt.value == null || stmt.value is ThisExpression;
285 } else {
286 return stmt is ThrowStatement;
287 }
288 }
289
290 Set<Member> _domMembersFromExpression(Expression expr, List<Type> domTypes) {
291 if (expr is BinaryExpression && expr.op.kind == TokenKind.ASSIGN) {
292 return _domMembersFromExpression(expr.x, domTypes);
293 } else if (expr is CallExpression) {
294 if (expr.target is DotExpression && expr.target.self is VarExpression &&
295 expr.target.self.name.name == 'LevelDom' &&
296 (expr.target.name.name.startsWith('wrap') ||
297 expr.target.name.name == 'unwrap')) {
298 return _domMembersFromExpression(expr.arguments[0].value, domTypes);
299 }
300 return _domMembersFromExpression(expr.target, domTypes);
301 } else if (expr is DotExpression) {
302 if (expr.self is NewExpression && expr.name.name == '_wrap' &&
303 expr.self.arguments.length == 1) {
304 return _domMembersFromExpression(expr.self.arguments[0].value, domTypes) ;
305 } else if (expr.self is VarExpression && expr.self.name.name == '_ptr') {
306 return _membersFromName(expr.name.name, domTypes);
307 }
308 final bases = _domMembersFromExpression(expr.self, domTypes);
309 return new Set.from(map(bases, (base) {
310 if (base == null || base.returnType == null) return null;
311 return base.returnType.members[expr.name.name];
312 }).filter((m) => m != null));
313 } else if (expr is NewExpression && expr.arguments.length == 1) {
314 return _domMembersFromExpression(expr.arguments[0].value, domTypes);
315 } else {
316 return new Set();
317 }
318 }
319
320 Map<String, String> _getTags(String comment) {
321 if (comment == null) return const <String, String>{};
322 final re = new RegExp("@([a-zA-Z]+) ([^;]+)(?:;|\$)");
323 final tags = <String, String>{};
324 for (var m in re.allMatches(comment.trim())) {
325 tags[m[1]] = m[2];
326 }
327 return tags;
328 }
329 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698