| // Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| /** |
| * Tools for generating code in analyzer and analysis server. |
| */ |
| library analyzer.src.codegen.tools; |
| |
| import 'package:analyzer/src/codegen/html.dart'; |
| import 'package:analyzer/src/codegen/text_formatter.dart'; |
| import 'package:html/dom.dart' as dom; |
| |
| final RegExp trailingSpacesInLineRegExp = new RegExp(r' +$', multiLine: true); |
| final RegExp trailingWhitespaceRegExp = new RegExp(r'[\n ]+$'); |
| |
| /** |
| * Join the given strings using camelCase. If [doCapitalize] is true, the first |
| * part will be capitalized as well. |
| */ |
| String camelJoin(List<String> parts, {bool doCapitalize: false}) { |
| List<String> upcasedParts = <String>[]; |
| for (int i = 0; i < parts.length; i++) { |
| if (i == 0 && !doCapitalize) { |
| upcasedParts.add(parts[i]); |
| } else { |
| upcasedParts.add(capitalize(parts[i])); |
| } |
| } |
| return upcasedParts.join(); |
| } |
| |
| /** |
| * Capitalize and return the passed String. |
| */ |
| String capitalize(String string) { |
| return string[0].toUpperCase() + string.substring(1); |
| } |
| |
| /** |
| * Mixin class for generating code. |
| */ |
| class CodeGenerator { |
| _CodeGeneratorState _state; |
| |
| /** |
| * Settings that specialize code generation behavior for a given |
| * programming language. |
| */ |
| CodeGeneratorSettings codeGeneratorSettings = new CodeGeneratorSettings(); |
| |
| /** |
| * Measure the width of the current indentation level. |
| */ |
| int get indentWidth => _state.nextIndent.length; |
| |
| /** |
| * Execute [callback], collecting any code that is output using [write] |
| * or [writeln], and return the result as a string. |
| */ |
| String collectCode(void callback(), {bool removeTrailingNewLine: false}) { |
| _CodeGeneratorState oldState = _state; |
| try { |
| _state = new _CodeGeneratorState(); |
| callback(); |
| var text = |
| _state.buffer.toString().replaceAll(trailingSpacesInLineRegExp, ''); |
| if (!removeTrailingNewLine) { |
| return text; |
| } else { |
| return text.replaceAll(trailingWhitespaceRegExp, ''); |
| } |
| } finally { |
| _state = oldState; |
| } |
| } |
| |
| /** |
| * Generate a doc comment based on the HTML in [docs]. |
| * |
| * When generating java code, the output is compatible with Javadoc, which |
| * understands certain HTML constructs. |
| */ |
| void docComment(List<dom.Node> docs, {bool removeTrailingNewLine: false}) { |
| if (containsOnlyWhitespace(docs)) return; |
| if (codeGeneratorSettings.docCommentStartMarker != null) |
| writeln(codeGeneratorSettings.docCommentStartMarker); |
| int width = codeGeneratorSettings.commentLineLength; |
| bool javadocStyle = codeGeneratorSettings.languageName == 'java'; |
| indentBy(codeGeneratorSettings.docCommentLineLeader, () { |
| write(nodesToText(docs, width - _state.indent.length, javadocStyle, |
| removeTrailingNewLine: removeTrailingNewLine)); |
| }); |
| if (codeGeneratorSettings.docCommentEndMarker != null) |
| writeln(codeGeneratorSettings.docCommentEndMarker); |
| } |
| |
| /** |
| * Execute [callback], indenting any code it outputs. |
| */ |
| void indent(void callback()) { |
| indentSpecial( |
| codeGeneratorSettings.indent, codeGeneratorSettings.indent, callback); |
| } |
| |
| /** |
| * Execute [callback], using [additionalIndent] to indent any code it outputs. |
| */ |
| void indentBy(String additionalIndent, void callback()) => |
| indentSpecial(additionalIndent, additionalIndent, callback); |
| |
| /** |
| * Execute [callback], using [additionalIndent] to indent any code it outputs. |
| * The first line of output is indented by [firstAdditionalIndent] instead of |
| * [additionalIndent]. |
| */ |
| void indentSpecial( |
| String firstAdditionalIndent, String additionalIndent, void callback()) { |
| String oldNextIndent = _state.nextIndent; |
| String oldIndent = _state.indent; |
| try { |
| _state.nextIndent += firstAdditionalIndent; |
| _state.indent += additionalIndent; |
| callback(); |
| } finally { |
| _state.nextIndent = oldNextIndent; |
| _state.indent = oldIndent; |
| } |
| } |
| |
| void lineComment(List<dom.Node> docs) { |
| if (containsOnlyWhitespace(docs)) { |
| return; |
| } |
| write(codeGeneratorSettings.lineCommentLineLeader); |
| int width = codeGeneratorSettings.commentLineLength; |
| indentBy(codeGeneratorSettings.lineCommentLineLeader, () { |
| write(nodesToText(docs, width - _state.indent.length, false)); |
| }); |
| } |
| |
| void outputHeader({bool javaStyle: false, String year = null}) { |
| String header; |
| if (codeGeneratorSettings.languageName == 'java') { |
| header = ''' |
| /* |
| * Copyright (c) ${year ?? '2015'}, the Dart project authors. |
| * |
| * Licensed under the Eclipse Public License v1.0 (the "License"); you may not use this file except |
| * in compliance with the License. You may obtain a copy of the License at |
| * |
| * http://www.eclipse.org/legal/epl-v10.html |
| * |
| * Unless required by applicable law or agreed to in writing, software distributed under the License |
| * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express |
| * or implied. See the License for the specific language governing permissions and limitations under |
| * the License. |
| * |
| * This file has been automatically generated. Please do not edit it manually. |
| * To regenerate the file, use the script "pkg/analysis_server/tool/spec/generate_files". |
| */'''; |
| } else if (codeGeneratorSettings.languageName == 'python') { |
| header = ''' |
| # Copyright (c) ${year ?? '2014'}, the Dart project authors. Please see the AUTHORS file |
| # for details. All rights reserved. Use of this source code is governed by a |
| # BSD-style license that can be found in the LICENSE file. |
| # |
| # This file has been automatically generated. Please do not edit it manually. |
| # To regenerate the file, use the script |
| # "pkg/analysis_server/tool/spec/generate_files". |
| '''; |
| } else { |
| header = ''' |
| // Copyright (c) ${year ?? '2014'}, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| // |
| // This file has been automatically generated. Please do not edit it manually. |
| // To regenerate the file, use the script |
| // "pkg/analysis_server/tool/spec/generate_files". |
| '''; |
| } |
| writeln(header.trim()); |
| } |
| |
| /** |
| * Output text without ending the current line. |
| */ |
| void write(Object obj) { |
| _state.write(obj.toString()); |
| } |
| |
| /** |
| * Output text, ending the current line. |
| */ |
| void writeln([Object obj = '']) { |
| _state.write('$obj\n'); |
| } |
| } |
| |
| /** |
| * Controls several settings of [CodeGenerator]. |
| * |
| * The default settings are valid for generating Java and Dart code. |
| */ |
| class CodeGeneratorSettings { |
| /** |
| * Name of the language being generated. Lowercase. |
| */ |
| String languageName; |
| |
| /** |
| * Marker used in line comments. |
| */ |
| String lineCommentLineLeader; |
| |
| /** |
| * Start marker for doc comments. |
| */ |
| String docCommentStartMarker; |
| |
| /** |
| * Line leader for body lines in doc comments. |
| */ |
| String docCommentLineLeader; |
| |
| /** |
| * End marker for doc comments. |
| */ |
| String docCommentEndMarker; |
| |
| /** |
| * Line length for doc comment lines. |
| */ |
| int commentLineLength; |
| |
| /** |
| * String used for indenting code. |
| */ |
| String indent; |
| |
| CodeGeneratorSettings( |
| {this.languageName: 'java', |
| this.lineCommentLineLeader: '// ', |
| this.docCommentStartMarker: '/**', |
| this.docCommentLineLeader: ' * ', |
| this.docCommentEndMarker: ' */', |
| this.commentLineLength: 99, |
| this.indent: ' '}); |
| } |
| |
| /** |
| * Mixin class for generating HTML representations of code that are suitable |
| * for enclosing inside a <pre> element. |
| */ |
| abstract class HtmlCodeGenerator { |
| _HtmlCodeGeneratorState _state; |
| |
| /** |
| * Add the given [node] to the HTML output. |
| */ |
| void add(dom.Node node) { |
| _state.add(node); |
| } |
| |
| /** |
| * Add the given [nodes] to the HTML output. |
| */ |
| void addAll(Iterable<dom.Node> nodes) { |
| for (dom.Node node in nodes) { |
| _state.add(node); |
| } |
| } |
| |
| /** |
| * Execute [callback], collecting any code that is output using [write], |
| * [writeln], [add], or [addAll], and return the result as a list of DOM |
| * nodes. |
| */ |
| List<dom.Node> collectHtml(void callback()) { |
| _HtmlCodeGeneratorState oldState = _state; |
| try { |
| _state = new _HtmlCodeGeneratorState(); |
| if (callback != null) { |
| callback(); |
| } |
| return _state.buffer; |
| } finally { |
| _state = oldState; |
| } |
| } |
| |
| /** |
| * Execute [callback], wrapping its output in an element with the given |
| * [name] and [attributes]. |
| */ |
| void element(String name, Map<dynamic, String> attributes, |
| [void callback()]) { |
| add(makeElement(name, attributes, collectHtml(callback))); |
| } |
| |
| /** |
| * Execute [callback], indenting any code it outputs by two spaces. |
| */ |
| void indent(void callback()) { |
| String oldIndent = _state.indent; |
| try { |
| _state.indent += ' '; |
| callback(); |
| } finally { |
| _state.indent = oldIndent; |
| } |
| } |
| |
| /** |
| * Output text without ending the current line. |
| */ |
| void write(Object obj) { |
| _state.write(obj.toString()); |
| } |
| |
| /** |
| * Output text, ending the current line. |
| */ |
| void writeln([Object obj = '']) { |
| _state.write('$obj\n'); |
| } |
| } |
| |
| /** |
| * State used by [CodeGenerator]. |
| */ |
| class _CodeGeneratorState { |
| StringBuffer buffer = new StringBuffer(); |
| String nextIndent = ''; |
| String indent = ''; |
| bool indentNeeded = true; |
| |
| void write(String text) { |
| List<String> lines = text.split('\n'); |
| for (int i = 0; i < lines.length; i++) { |
| if (i == lines.length - 1 && lines[i].isEmpty) { |
| break; |
| } |
| if (indentNeeded) { |
| buffer.write(nextIndent); |
| nextIndent = indent; |
| } |
| indentNeeded = false; |
| buffer.write(lines[i]); |
| if (i != lines.length - 1) { |
| buffer.writeln(); |
| indentNeeded = true; |
| } |
| } |
| } |
| } |
| |
| /** |
| * State used by [HtmlCodeGenerator]. |
| */ |
| class _HtmlCodeGeneratorState { |
| List<dom.Node> buffer = <dom.Node>[]; |
| String indent = ''; |
| bool indentNeeded = true; |
| |
| void add(dom.Node node) { |
| if (node is dom.Text) { |
| write(node.text); |
| } else { |
| buffer.add(node); |
| } |
| } |
| |
| void write(String text) { |
| if (text.isEmpty) { |
| return; |
| } |
| if (indentNeeded) { |
| buffer.add(new dom.Text(indent)); |
| } |
| List<String> lines = text.split('\n'); |
| if (lines.last.isEmpty) { |
| lines.removeLast(); |
| buffer.add(new dom.Text(lines.join('\n$indent') + '\n')); |
| indentNeeded = true; |
| } else { |
| buffer.add(new dom.Text(lines.join('\n$indent'))); |
| indentNeeded = false; |
| } |
| } |
| } |