diff --git a/spec/compilers/component_with_provider b/spec/compilers/component_with_provider index 004a2ba0b..8832c899d 100644 --- a/spec/compilers/component_with_provider +++ b/spec/compilers/component_with_provider @@ -43,13 +43,13 @@ class C extends _C { componentDidUpdate() { if (false) { B._subscribe(this, new A({ - moves: (a) => { - return null - }, - ups: (b) => { - return null - } - })) + moves: (a) => { + return null + }, + ups: (b) => { + return null + } + })) } else { B._unsubscribe(this) }; @@ -58,13 +58,13 @@ class C extends _C { componentDidMount() { if (false) { B._subscribe(this, new A({ - moves: (a) => { - return null - }, - ups: (b) => { - return null - } - })) + moves: (a) => { + return null + }, + ups: (b) => { + return null + } + })) } else { B._unsubscribe(this) }; diff --git a/spec/compilers/component_with_provider_and_lifecycle_functions b/spec/compilers/component_with_provider_and_lifecycle_functions index f567bd1b2..28c15f678 100644 --- a/spec/compilers/component_with_provider_and_lifecycle_functions +++ b/spec/compilers/component_with_provider_and_lifecycle_functions @@ -56,13 +56,13 @@ class C extends _C { componentDidUpdate() { if (false) { B._subscribe(this, new A({ - moves: (a) => { - return null - }, - ups: (b) => { - return null - } - })) + moves: (a) => { + return null + }, + ups: (b) => { + return null + } + })) } else { B._unsubscribe(this) }; @@ -73,13 +73,13 @@ class C extends _C { componentDidMount() { if (false) { B._subscribe(this, new A({ - moves: (a) => { - return null - }, - ups: (b) => { - return null - } - })) + moves: (a) => { + return null + }, + ups: (b) => { + return null + } + })) } else { B._unsubscribe(this) }; diff --git a/spec/compilers/component_with_provider_and_store b/spec/compilers/component_with_provider_and_store index f6c853e9e..44bf0e95d 100644 --- a/spec/compilers/component_with_provider_and_store +++ b/spec/compilers/component_with_provider_and_store @@ -60,13 +60,13 @@ class C extends _C { componentDidUpdate() { if (false) { B._subscribe(this, new A({ - moves: (a) => { - return null - }, - ups: (b) => { - return null - } - })) + moves: (a) => { + return null + }, + ups: (b) => { + return null + } + })) } else { B._unsubscribe(this) }; @@ -77,13 +77,13 @@ class C extends _C { if (false) { B._subscribe(this, new A({ - moves: (a) => { - return null - }, - ups: (b) => { - return null - } - })) + moves: (a) => { + return null + }, + ups: (b) => { + return null + } + })) } else { B._unsubscribe(this) }; diff --git a/spec/compilers/directives/format b/spec/compilers/directives/format index 59c313d95..2651297ad 100644 --- a/spec/compilers/directives/format +++ b/spec/compilers/directives/format @@ -17,6 +17,7 @@ class A extends _C { return (() => { const [a,b] = [`HelloHelloHelloHelloHelloHelloHelloHelloHelloHelloHelloHBello`, `"HelloHelloHelloHelloHelloHelloHelloHelloHelloHelloHelloH" \\ "Bello"`]; + return a + b; })(); } diff --git a/spec/compilers/module_access_subscriptions b/spec/compilers/module_access_subscriptions index 2f34db631..92527ee6a 100644 --- a/spec/compilers/module_access_subscriptions +++ b/spec/compilers/module_access_subscriptions @@ -47,8 +47,8 @@ class C extends _C { componentDidUpdate() { if (true) { B._subscribe(this, new A({ - test: `` - })) + test: `` + })) } else { B._unsubscribe(this) }; @@ -57,8 +57,8 @@ class C extends _C { componentDidMount() { if (true) { B._subscribe(this, new A({ - test: `` - })) + test: `` + })) } else { B._unsubscribe(this) }; diff --git a/spec/indented_string_builder_spec.cr b/spec/indented_string_builder_spec.cr new file mode 100644 index 000000000..2b1aace8a --- /dev/null +++ b/spec/indented_string_builder_spec.cr @@ -0,0 +1,67 @@ +require "./spec_helper" + +INDENT = 2 + +describe Mint::IndentedStringBuilder do + it "indent class with constructor and display name" do + b = Mint::IndentedStringBuilder.new + + b << "class " << "A" << " extends " << "_C" << " " << "{\n" + b.indent_size += INDENT + b << "constructor" << "(" << "props" << ") " + b << "{\n" + b.indent_size += INDENT + b << "super" << "(" << "props" << ")" << ";" + # js.call + b << "\n\n" + b << "this._d" << "(" + b << "{\n" + b.indent_size += INDENT + b << "a" << ": " << "[\n" + b.indent_size += INDENT + b << "null" << "," << "\n" << "`Hello`" + b.indent_size -= INDENT + b << "\n]" + b.indent_size -= INDENT + b << "\n}" + b << ")" << ";" + b.indent_size -= INDENT + b << "\n}" + b.indent_size -= INDENT + b << "\n}" + b << ";" + b << "\n\n" + b << "A.displayName = \"Test\"" + b << ";" + + expected = + <<-STR + class A extends _C { + constructor(props) { + super(props); + + this._d({ + a: [ + null, + `Hello` + ] + }); + } + }; + + A.displayName = "Test"; + STR + + result = b.build + + pos = b.get_position_for_next_input + pos[:line].should eq(13) + pos[:column].should eq(23) + + begin + result.should eq(expected) + rescue error + fail diff(expected, result) + end + end +end diff --git a/spec/js_spec.cr b/spec/js_spec.cr index 82a50917a..076c76667 100644 --- a/spec/js_spec.cr +++ b/spec/js_spec.cr @@ -6,7 +6,13 @@ describe "JS" do context "object" do it "renders object Optimized" do - optimized.object({"a" => "b", "c" => "d"}).should eq("{a:b,c:d}") + subject = + optimized.object({"a" => "b", "c" => "d"}) + + result = + Mint::Codegen.build(subject)[:code] + + result.should eq("{a:b,c:d}") end end end diff --git a/src/builder.cr b/src/builder.cr index fed9f9451..9aba5c9b9 100644 --- a/src/builder.cr +++ b/src/builder.cr @@ -1,6 +1,6 @@ module Mint class Builder - def initialize(relative, skip_service_worker, skip_icons) + def initialize(relative, skip_service_worker, skip_icons, source_map) json = MintJson.parse_current terminal.measure "#{COG} Ensuring dependencies... " do @@ -20,7 +20,15 @@ module Mint end terminal.puts "#{COG} Compiling your application:" - File.write Path[DIST_DIR, "index.js"], index(json.application.css_prefix) + build = index(json.application.css_prefix, source_map) + File.write Path[DIST_DIR, "index.js"], build[:code] + + if source_map + terminal.puts "#{COG} Generating source map:" + build[:source_map].try do |map| + File.write Path[DIST_DIR, "index.js.map"], map.build_json + end + end if SourceFiles.external_javascripts? terminal.measure "#{COG} Writing external javascripts..." do @@ -91,7 +99,7 @@ module Mint end end - def index(css_prefix) + def index(css_prefix, source_map) runtime = Assets.read("runtime.js") @@ -125,7 +133,7 @@ module Mint } end - runtime + compiled + Codegen.build(Codegen.join([runtime, compiled]), source_map) end def terminal diff --git a/src/commands/build.cr b/src/commands/build.cr index 973424608..1fe3da942 100644 --- a/src/commands/build.cr +++ b/src/commands/build.cr @@ -18,9 +18,14 @@ module Mint description: "If specified the application icons will not be generated", default: false + define_flag source_map : Bool, + description: "If specified generate source mappings for debugging", + default: false, + short: "m" + def run execute "Building for production" do - Builder.new(flags.relative, flags.skip_service_worker, flags.skip_icons) + Builder.new(flags.relative, flags.skip_service_worker, flags.skip_icons, flags.source_map) end end end diff --git a/src/commands/compile.cr b/src/commands/compile.cr index bdf1487ac..7c4c1a6f9 100644 --- a/src/commands/compile.cr +++ b/src/commands/compile.cr @@ -8,21 +8,32 @@ module Mint define_flag output : String, description: "The output file", default: "program.js", - required: false, short: "o" + define_flag source_map : Bool, + description: "If specified generate source mappings for debugging", + default: false, + short: "m" + def run execute "Compiling" do - File.write(flags.output, compile) + result = + compile(flags.source_map) + + File.write(flags.output, result[:code]) + + result[:source_map].try do |map| + File.write("#{flags.output}.map", map) + end end end - def compile + def compile(generate_source_map : Bool) json = MintJson.parse_current runtime = - Assets.read("runtime.js") + Assets.read("runtime.js").as(Codegen::Node) sources = Dir.glob(SourceFiles.all) @@ -31,7 +42,7 @@ module Mint Ast.new .merge(Core.ast) - compiled = "" + compiled = "".as(Codegen::Node) terminal.measure " #{ARROW} Parsing #{sources.size} source files... " do sources.reduce(ast) do |memo, file| @@ -54,7 +65,18 @@ module Mint } end - runtime + compiled + result = + Codegen.build(Codegen.join([runtime, compiled]), generate_source_map) + + source_map : String? = nil + + result[:source_map].try do |map| + terminal.measure " #{ARROW} Generating source map: " do + source_map = map.build_json + end + end + + {code: result[:code], source_map: source_map} end end end diff --git a/src/commands/start.cr b/src/commands/start.cr index 748819abe..923f54e33 100644 --- a/src/commands/start.cr +++ b/src/commands/start.cr @@ -22,9 +22,15 @@ module Mint required: false, short: "p" + define_flag source_map : Bool, + description: "If specified generate source mappings for debugging", + default: true, + required: false, + short: "m" + def run execute "Running the development server" do - Reactor.start flags.host, flags.port, flags.auto_format + Reactor.start flags.host, flags.port, flags.auto_format, flags.source_map end end end diff --git a/src/compiler.cr b/src/compiler.cr index d66b427d5..372136a7f 100644 --- a/src/compiler.cr +++ b/src/compiler.cr @@ -1,14 +1,12 @@ module Mint class Compiler - include Skippable - delegate lookups, checked, cache, component_records, to: @artifacts delegate ast, types, variables, resolve_order, to: @artifacts delegate record_field_lookup, to: @artifacts getter js, style_builder, static_components, static_components_pool - @static_components = {} of String => String + @static_components = {} of String => Codegen::Node @static_components_pool = NamePool(String, Nil).new def initialize(@artifacts : TypeChecker::Artifacts, @optimize = false, css_prefix = nil) @@ -25,15 +23,17 @@ module Mint # Helpers for compiling things # ---------------------------------------------------------------------------- - def compile(nodes : Array(Ast::Node), separator : String) - compile(nodes).join(separator) + def compile(nodes : Array(Ast::Node), separator : String) : Codegen::Node + Codegen.join(nodes.map { |node| compile(node).as(Codegen::Node) }, separator) end - def compile(nodes : Array(Ast::Node)) - nodes.map { |node| compile(node).as(String) }.reject!(&.empty?) + def compile(nodes : Array(Ast::Node)) : Array(Codegen::Node) + nodes + .map { |node| compile(node).as(Codegen::Node) } + .reject! { |node| Codegen.empty? node } end - def compile(node : Ast::Node) : String + def compile(node : Ast::Node) : Codegen::Node if checked.includes?(node) _compile(node) else @@ -41,7 +41,7 @@ module Mint end end - def _compile(node : Ast::Node) : String + def _compile(node : Ast::Node) : Codegen::Node raise "Compiler not implemented for node #{node}!" end end diff --git a/src/compilers/access.cr b/src/compilers/access.cr index f7a5ccdbf..42df95601 100644 --- a/src/compilers/access.cr +++ b/src/compilers/access.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Access) : String + def _compile(node : Ast::Access) : Codegen::Node first = compile node.lhs @@ -13,13 +13,16 @@ module Mint if node.safe? js.iif do + access = + Codegen.symbol_mapped(node.lhs, node, Codegen.join ["_.", field]) + js.statements([ js.const("_", first), - js.return(js.call("_s", ["_", "(_) => _.#{field}"])), + js.return(js.call("_s", ["_", Codegen.join ["(_) => ", access]])), ]) end else - "#{first}.#{field}" + Codegen.symbol_mapped(node.lhs, node, Codegen.join [first, ".", field]) end end end diff --git a/src/compilers/argument.cr b/src/compilers/argument.cr index 021a1aa27..5d08f5be8 100644 --- a/src/compilers/argument.cr +++ b/src/compilers/argument.cr @@ -1,7 +1,7 @@ module Mint class Compiler - def _compile(node : Ast::Argument) : String - js.variable_of(node) + def _compile(node : Ast::Argument) : Codegen::Node + Codegen.symbol_mapped(node, js.variable_of(node)) end end end diff --git a/src/compilers/array_access.cr b/src/compilers/array_access.cr index 4d33a2e48..d3b49e85e 100644 --- a/src/compilers/array_access.cr +++ b/src/compilers/array_access.cr @@ -1,24 +1,25 @@ module Mint class Compiler - def _compile(node : Ast::ArrayAccess) : String + def _compile(node : Ast::ArrayAccess) : Codegen::Node type = cache[node.lhs] lhs = - compile node.lhs + Codegen.source_mapped(node.lhs, compile node.lhs) index = case node.index - when Int64 - node.index - when Ast::Node - compile node.index.as(Ast::Node) + in Int64 + node.index.to_s + in Ast::Node + node_index = node.index.as(Ast::Node) + Codegen.source_mapped(node_index, compile node_index) end if type.name == "Tuple" && node.index.is_a?(Int64) - "#{lhs}[#{index}]" + Codegen.join [lhs, "[", index, "]"] else - "_at(#{lhs}, #{index})" + Codegen.join ["_at(", lhs, ", ", index, ")"] end end end diff --git a/src/compilers/array_destructuring.cr b/src/compilers/array_destructuring.cr index fbefa1562..a65208a65 100644 --- a/src/compilers/array_destructuring.cr +++ b/src/compilers/array_destructuring.cr @@ -1,16 +1,16 @@ module Mint class Compiler - def _compile(node : Ast::ArrayDestructuring, value) + def _compile(node : Ast::ArrayDestructuring, value : Codegen::Node) : Array(Codegen::Node) if node.spread? statements = [ - "const __ = Array.from(#{value})", - ] + Codegen.join ["const __ = Array.from(", value, ")"], + ] of Codegen::Node node .items .take_while(&.is_a?(Ast::Variable)) .each do |var| - statements << "const #{js.variable_of(var)} = __.shift()" + statements << Codegen.join ["const ", js.variable_of(var), " = __.shift()"] end node @@ -18,18 +18,16 @@ module Mint .reverse .take_while(&.is_a?(Ast::Variable)) .each do |var| - statements << "const #{js.variable_of(var)} = __.pop()" + statements << Codegen.join ["const ", js.variable_of(var), " = __.pop()"] end - statements << "const #{js.variable_of(node.items.select(Ast::Spread).first.variable)} = __" + statements << Codegen.join ["const ", js.variable_of(node.items.select(Ast::Spread).first.variable), " = __"] statements else variables = - node - .items - .join(',') { |param| js.variable_of(param) } + Codegen.join(node.items.map { |param| js.variable_of(param) }, ",") - ["const [#{variables}] = #{value}"] + [Codegen.join ["const [", variables, "] = ", value]] of Codegen::Node end end end diff --git a/src/compilers/array_literal.cr b/src/compilers/array_literal.cr index d4ebd59f9..8b922beb1 100644 --- a/src/compilers/array_literal.cr +++ b/src/compilers/array_literal.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::ArrayLiteral) : String + def _compile(node : Ast::ArrayLiteral) : Codegen::Node items = compile node.items, ", " - "[#{items}]" + Codegen.join ["[", items, "]"] end end end diff --git a/src/compilers/bool_literal.cr b/src/compilers/bool_literal.cr index 4c6227352..0b0da72aa 100644 --- a/src/compilers/bool_literal.cr +++ b/src/compilers/bool_literal.cr @@ -1,7 +1,7 @@ module Mint class Compiler - def _compile(node : Ast::BoolLiteral) : String - node.value.to_s + def _compile(node : Ast::BoolLiteral) : Codegen::Node + Codegen.symbol_mapped(node, node.value.to_s) end end end diff --git a/src/compilers/call.cr b/src/compilers/call.cr index 49fe2525f..659e66d60 100644 --- a/src/compilers/call.cr +++ b/src/compilers/call.cr @@ -1,11 +1,11 @@ module Mint class Compiler - def _compile(node : Ast::Call) : String + def _compile(node : Ast::Call) : Codegen::Node expression = compile node.expression arguments = - compile node.arguments, ", " + Codegen.join(node.arguments, ", ") { |arg| Codegen.source_mapped(arg, compile arg) } if node.safe? js.iif do @@ -14,10 +14,10 @@ module Mint if node.arguments.empty? "(_) => _" else - "((..._) => _(#{arguments}, ..._))" + Codegen.join ["((..._) => _(", arguments, ", ..._))"] end else - "(_) => _(#{arguments})" + Codegen.join ["(_) => _(", arguments, ")"] end js.statements([ @@ -31,12 +31,12 @@ module Mint if node.arguments.empty? expression else - "((..._) => #{expression}(#{arguments}, ..._))" + Codegen.join ["((..._) => ", expression, "(", arguments, ", ..._))"] end when node.expression.is_a?(Ast::InlineFunction) - "(#{expression})(#{arguments})" + Codegen.join ["(", expression, ")(", arguments, ")"] else - "#{expression}(#{arguments})" + Codegen.join [expression, "(", arguments, ")"] end end end diff --git a/src/compilers/case.cr b/src/compilers/case.cr index d3ef2196c..df558e713 100644 --- a/src/compilers/case.cr +++ b/src/compilers/case.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def compile(node : Ast::Case, block : Proc(String, String)? = nil) : String + def compile(node : Ast::Case, block : Proc(Codegen::Node, Codegen::Node)? = nil) : Codegen::Node if checked.includes?(node) _compile node, block else @@ -8,13 +8,16 @@ module Mint end end - def _compile(node : Ast::Case, block : Proc(String, String)? = nil) : String + def _compile(node : Ast::Case, block : Proc(Codegen::Node, Codegen::Node)? = nil) : Codegen::Node condition = compile node.condition variable, condition_let = js.let condition + condition_let = + Codegen.source_mapped(node, condition_let) + body = node .branches diff --git a/src/compilers/case_branch.cr b/src/compilers/case_branch.cr index 15c6c36d3..77aad7547 100644 --- a/src/compilers/case_branch.cr +++ b/src/compilers/case_branch.cr @@ -2,8 +2,8 @@ module Mint class Compiler def _compile(node : Ast::CaseBranch, index : Int32, - variable : String, - block : Proc(String, String)? = nil) : Tuple(String?, String) + variable : Codegen::Node, + block : Proc(Codegen::Node, Codegen::Node)? = nil) : Tuple(Codegen::Node?, Codegen::Node) expression = case item = node.expression when Array(Ast::CssDefinition) @@ -14,7 +14,7 @@ module Mint "{}" end when Ast::Node - js.return(compile(item)) + Codegen.source_mapped(item, js.return(compile(item))) else "" end @@ -40,14 +40,14 @@ module Mint end when Ast::TupleDestructuring variables = - match - .parameters - .join(',') { |param| js.variable_of(param) } + Codegen.join(match.parameters, ",") { |param| + Codegen.source_mapped(param, js.variable_of(param)) + } { "Array.isArray(#{variable})", js.statements([ - "const [#{variables}] = #{variable}", + Codegen.join(["const [", variables, "] = ", variable]), expression, ]), } @@ -56,11 +56,11 @@ module Mint case lookups[match].as(Ast::EnumOption).parameters[0]? when Ast::EnumRecordDefinition match.parameters.map do |param| - "const #{js.variable_of(param)} = #{variable}._0.#{param.value}" + Codegen.join ["const ", js.variable_of(param), " = ", variable, "._0.#{param.value}"] end else match.parameters.map_with_index do |param, index1| - "const #{js.variable_of(param)} = #{variable}._#{index1}" + Codegen.join ["const ", js.variable_of(param), " = ", variable, "._#{index1}"] end end @@ -68,7 +68,7 @@ module Mint js.class_of(lookups[match]) { - "#{variable} instanceof #{name}", + Codegen.join([variable, " instanceof ", name]), js.statements(variables + [expression]), } else @@ -76,7 +76,7 @@ module Mint compile match { - "_compare(#{variable}, #{compiled})", + Codegen.join(["_compare(", variable, ", ", compiled, ")"]), expression, } end diff --git a/src/compilers/catch.cr b/src/compilers/catch.cr index d8edd17fe..4f94c8c25 100644 --- a/src/compilers/catch.cr +++ b/src/compilers/catch.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Catch) : String + def _compile(node : Ast::Catch) : Codegen::Node body = compile node.expression diff --git a/src/compilers/component.cr b/src/compilers/component.cr index ce4b65ff9..dc4da9387 100644 --- a/src/compilers/component.cr +++ b/src/compilers/component.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Component) : String + def _compile(node : Ast::Component) : Codegen::Node name = js.class_of(node) @@ -12,14 +12,14 @@ module Mint end global_let = - "let #{name}" if node.global? + Codegen.join ["let ", name] if node.global? compile node.styles, node styles = node.styles.map do |style_node| style_builder.compile_style(style_node, self) - end.reject!(&.empty?) + end.reject! { |item| Codegen.empty? item } functions = compile_component_functions node @@ -44,10 +44,10 @@ module Mint store_stuff = compile_component_store_data node - constructor_body = %w[] + constructor_body = [] of Codegen::Node default_props = - node.properties.each_with_object({} of String => String) do |prop, memo| + node.properties.each_with_object({} of Codegen::Node => Codegen::Node) do |prop, memo| prop_name = if prop.name.value == "children" %("children") @@ -79,11 +79,11 @@ module Mint values = node .states - .each_with_object({} of String => String) do |item, memo| + .each_with_object({} of Codegen::Node => Codegen::Node) do |item, memo| memo[js.variable_of(item)] = compile item.default end - constructor_body << "this.state = new Record(#{js.object(values)})" + constructor_body << Codegen.join ["this.state = new Record(", js.object(values), ")"] end constructor = @@ -95,7 +95,7 @@ module Mint end end - functions << js.function("_persist", %w[], js.assign(name, "this")) if node.global? + functions << js.function("_persist", [] of Codegen::Node, js.assign(name, "this")) if node.global? body = ([constructor] + styles + gets + refs + states + store_stuff + functions) @@ -108,8 +108,8 @@ module Mint ].compact) end - def compile_component_store_data(node : Ast::Component) : Array(String) - node.connects.reduce(%w[]) do |memo, item| + def compile_component_store_data(node : Ast::Component) : Array(Codegen::Node) + node.connects.reduce([] of Codegen::Node) do |memo, item| store = ast.stores.find { |entity| entity.name == item.store } if store @@ -125,9 +125,9 @@ module Mint when store.constants.any? { |constant| constant.name == original }, store.gets.any? { |get| get.name.value == original }, store.states.find(&.name.value.==(original)) - memo << js.get(name, "return #{store_name}.#{id};") + memo << js.get(name, Codegen.join ["return ", store_name, ".", id, ";"]) when store.functions.any? { |func| func.name.value == original } - memo << "#{name} (...params) { return #{store_name}.#{id}(...params); }" + memo << Codegen.join [name, " (...params) { return ", store_name, ".", id, "(...params); }"] end end end @@ -136,11 +136,11 @@ module Mint end end - def compile_component_functions(node : Ast::Component) : Array(String) + def compile_component_functions(node : Ast::Component) : Array(Codegen::Node) heads = { - "componentWillUnmount" => %w[], - "componentDidUpdate" => %w[], - "componentDidMount" => %w[], + "componentWillUnmount" => [] of Codegen::Node, + "componentDidUpdate" => [] of Codegen::Node, + "componentDidMount" => [] of Codegen::Node, } node.connects.each do |item| @@ -151,8 +151,8 @@ module Mint name = js.class_of(store) - heads["componentWillUnmount"] << "#{name}._unsubscribe(this)" - heads["componentDidMount"] << "#{name}._subscribe(this)" + heads["componentWillUnmount"] << Codegen.join [name, "._unsubscribe(this)"] + heads["componentDidMount"] << Codegen.join [name, "._subscribe(this)"] end end @@ -166,14 +166,15 @@ module Mint data = compile use.data - body = - "if (#{condition}) {\n" \ - " #{name}._subscribe(this, #{data})\n" \ - "} else {\n" \ - " #{name}._unsubscribe(this)\n" \ - "}" + body = Codegen.join [ + "if (", condition, ") {\n", + Codegen.indent([name, "._subscribe(this, ", data, ")\n"]), + "} else {\n", + Codegen.indent([name, "._unsubscribe(this)\n"]), + "}", + ] - heads["componentWillUnmount"] << "#{name}._unsubscribe(this)" + heads["componentWillUnmount"] << Codegen.join([name, "._unsubscribe(this)"]) heads["componentDidUpdate"] << body heads["componentDidMount"] << body end @@ -182,7 +183,7 @@ module Mint node .functions .reject { |function| heads[function.name.value]? } - .map { |function| compile(function, "").as(String) } + .map { |function| compile(function, "") } specials = heads.map do |key, value| @@ -201,7 +202,7 @@ module Mint end end - (specials + others).compact.reject!(&.empty?) + (specials + others).compact.reject! { |item| Codegen.empty? item } end end end diff --git a/src/compilers/constant.cr b/src/compilers/constant.cr index 0c54e9202..5fa35943d 100644 --- a/src/compilers/constant.cr +++ b/src/compilers/constant.cr @@ -1,11 +1,11 @@ module Mint class Compiler - def compile_constants(nodes : Array(Ast::Constant)) : Hash(String, String) + def compile_constants(nodes : Array(Ast::Constant)) : Hash(Codegen::Node, Codegen::Node) nodes .select(&.in?(checked)) - .each_with_object({} of String => String) do |node, memo| + .each_with_object({} of Codegen::Node => Codegen::Node) do |node, memo| memo[js.variable_of(node)] = - js.arrow_function(%w[], js.return(compile(node.value))) + js.arrow_function([] of Codegen::Node, js.return(compile(node.value))) end end end diff --git a/src/compilers/css_definition.cr b/src/compilers/css_definition.cr new file mode 100644 index 000000000..d7ae4b7c4 --- /dev/null +++ b/src/compilers/css_definition.cr @@ -0,0 +1,22 @@ +module Mint + class Compiler + def _compile(items : Array(Ast::CssDefinition), block : Proc(Codegen::Node, Codegen::Node)?) : Codegen::Node + compiled = + items.each_with_object({} of Codegen::Node => Codegen::Node) do |definition, memo| + variable = + if block + block.call(definition.name) + else + "" + end + + value = + compile definition.value + + memo["[`#{variable}`]"] = value + end + + Codegen.join ["Object.assign(_, ", js.object(compiled), ")"] + end + end +end diff --git a/src/compilers/decode.cr b/src/compilers/decode.cr index 481b6abc4..0d068cf21 100644 --- a/src/compilers/decode.cr +++ b/src/compilers/decode.cr @@ -1,13 +1,13 @@ module Mint class Compiler - def _compile(node : Ast::Decode) : String + def _compile(node : Ast::Decode) : Codegen::Node expression = compile node.expression code = @serializer.decoder types[node] - "#{code}(#{expression})" + Codegen.join [code, "(", expression, ")"] end end end diff --git a/src/compilers/directives/documentation.cr b/src/compilers/directives/documentation.cr index 945e4e358..6c8fa1092 100644 --- a/src/compilers/directives/documentation.cr +++ b/src/compilers/directives/documentation.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Directives::Documentation) : String + def _compile(node : Ast::Directives::Documentation) : Codegen::Node entity = lookups[node] diff --git a/src/compilers/directives/format.cr b/src/compilers/directives/format.cr index 17b58cfa7..b688df5f7 100644 --- a/src/compilers/directives/format.cr +++ b/src/compilers/directives/format.cr @@ -1,19 +1,18 @@ module Mint class Compiler - def _compile(node : Ast::Directives::Format) : String + def _compile(node : Ast::Directives::Format) : Codegen::Node content = compile node.content formatted = - skip do + Codegen.no_indent( Formatter.new(ast) .format(node.content) .gsub('\\', "\\\\") .gsub('`', "\\`") - .gsub("${", "\\${") - end + .gsub("${", "\\${")) - "[#{content}, `#{formatted}`]" + Codegen.join ["[", content, ", `", formatted, "`]"] end end end diff --git a/src/compilers/directives/svg.cr b/src/compilers/directives/svg.cr index a56899b38..452e3bfac 100644 --- a/src/compilers/directives/svg.cr +++ b/src/compilers/directives/svg.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Directives::Svg) : String + def _compile(node : Ast::Directives::Svg) : Codegen::Node directory = Path[node.input.file].dirname diff --git a/src/compilers/encode.cr b/src/compilers/encode.cr index 0a5ea50de..9e8d36a5a 100644 --- a/src/compilers/encode.cr +++ b/src/compilers/encode.cr @@ -1,13 +1,13 @@ module Mint class Compiler - def _compile(node : Ast::Encode) : String + def _compile(node : Ast::Encode) : Codegen::Node expression = compile node.expression code = @serializer.encoder cache[node.expression] - "#{code}(#{expression})" + Codegen.join [code || "", "(", expression, ")"] end end end diff --git a/src/compilers/enum.cr b/src/compilers/enum.cr index a7c647524..fa1871be0 100644 --- a/src/compilers/enum.cr +++ b/src/compilers/enum.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Enum) : String + def _compile(node : Ast::Enum) : Codegen::Node enum_ids = node.options.map do |option| name = @@ -11,14 +11,14 @@ module Mint .map { |index| "_#{index - 1}" } assignments = - ids.map { |item| "this.#{item} = #{item}" } + ids.map { |item| Codegen.join ["this.", item, " = ", item] } js.class( name, extends: "_E", body: [js.function("constructor", ids) do js.statements([ - js.call("super", %w[]), + js.call("super", [] of Codegen::Node), assignments, "this.length = #{option.parameters.size}", ].flatten) diff --git a/src/compilers/enum_id.cr b/src/compilers/enum_id.cr index 051cd6a18..51f7fd693 100644 --- a/src/compilers/enum_id.cr +++ b/src/compilers/enum_id.cr @@ -1,13 +1,13 @@ module Mint class Compiler - def _compile(node : Ast::EnumId) : String + def _compile(node : Ast::EnumId) : Codegen::Node name = js.class_of(lookups[node]) expressions = compile node.expressions, "," - "new #{name}(#{expressions})" + Codegen.join ["new ", name, "(", expressions, ")"] end end end diff --git a/src/compilers/env.cr b/src/compilers/env.cr index 8276b1064..cb405ddb3 100644 --- a/src/compilers/env.cr +++ b/src/compilers/env.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def compile(node : Ast::Env) : String + def compile(node : Ast::Env) : Codegen::Node value = MINT_ENV[node.name].to_s.gsub('`', "\\`") diff --git a/src/compilers/finally.cr b/src/compilers/finally.cr index ba82c684d..d9c35e685 100644 --- a/src/compilers/finally.cr +++ b/src/compilers/finally.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::Finally) : String + def _compile(node : Ast::Finally) : Codegen::Node body = compile node.expression - "finally {\n#{body.indent}\n}" + Codegen.join ["finally {\n", Codegen.indent(body), "\n}"] end end end diff --git a/src/compilers/for_expression.cr b/src/compilers/for_expression.cr index d8ca327ea..269230ec9 100644 --- a/src/compilers/for_expression.cr +++ b/src/compilers/for_expression.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::For) : String + def _compile(node : Ast::For) : Codegen::Node body = compile node.body @@ -16,24 +16,26 @@ module Mint condition = node.condition.try do |item| - <<-JS - const _2 = #{compile(item.condition)} - if (!_2) { continue } - JS + Codegen.join [ + "const _2 = ", compile(item.condition), + "\nif (!_2) { continue }", + ] end + contents_push_body = Codegen.join ["_0.push(", body, ")"] + contents = if condition - js.statements([condition, "_0.push(#{body})"]) + js.statements([condition, contents_push_body]) else - "_0.push(#{body})" + contents_push_body end js.iif do js.statements([ "const _0 = []", - "const _1 = #{subject}", - js.for("let #{arguments} of _1", contents), + Codegen.join(["const _1 = ", subject]), + js.for(Codegen.join(["let ", arguments, " of _1"]), contents), js.return("_0"), ]) end diff --git a/src/compilers/function.cr b/src/compilers/function.cr index f2952044a..2db3625b6 100644 --- a/src/compilers/function.cr +++ b/src/compilers/function.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def compile(node : Ast::Function, contents = "") : String + def compile(node : Ast::Function, contents : Codegen::Node = "") : Codegen::Node if checked.includes?(node) _compile node, contents else @@ -8,9 +8,9 @@ module Mint end end - def _compile(node : Ast::Function, contents = "") : String + def _compile(node : Ast::Function, contents : Codegen::Node = "") : Codegen::Node name = - js.variable_of(node) + Codegen.source_mapped(node.name, js.variable_of(node)) expression = compile node.body @@ -21,15 +21,15 @@ module Mint .sort_by { |item| resolve_order.index(item) || -1 }) arguments = - compile node.arguments + node.arguments.map { |arg| Codegen.source_mapped(arg, compile arg) } last = - [js.return(expression)] + [Codegen.source_mapped(node.body, js.return(expression)).as(Codegen::Node)] - last.unshift(contents) unless contents.empty? + last.unshift(contents) unless Codegen.empty? contents body = - js.statements(wheres + last) + Codegen.source_mapped(node.where || node.body, js.statements(wheres + last)) js.function(name, arguments, body) end diff --git a/src/compilers/get.cr b/src/compilers/get.cr index 8bd562ffe..dd6281e9f 100644 --- a/src/compilers/get.cr +++ b/src/compilers/get.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Get) : String + def _compile(node : Ast::Get) : Codegen::Node body = compile node.body diff --git a/src/compilers/html_attribute.cr b/src/compilers/html_attribute.cr index 11373d75e..d0afcccac 100644 --- a/src/compilers/html_attribute.cr +++ b/src/compilers/html_attribute.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def resolve(node : Ast::HtmlAttribute, is_element = true) : Hash(String, String) + def resolve(node : Ast::HtmlAttribute, is_element = true) : Hash(Codegen::Node, Codegen::Node) value = compile node.value @@ -10,20 +10,20 @@ module Mint case downcase_name when .starts_with?("on") if is_element - value = "(event => (#{value})(_normalizeEvent(event)))" + value = Codegen.join ["(event => (", value, ")(_normalizeEvent(event)))"] end when "ref" - value = "(ref => { ref ? #{value}.call(this, ref) : null })" + value = Codegen.join ["(ref => { ref ? ", value, ".call(this, ref) : null })"] else # ignore end if downcase_name == "readonly" && is_element - {"readOnly" => value} + {"readOnly".as(Codegen::Node) => value} elsif lookups[node]? {js.variable_of(lookups[node]) => value} else - { %("#{node.name.value}") => value } + {(%("#{node.name.value}").as(Codegen::Node)) => value} end end end diff --git a/src/compilers/html_component.cr b/src/compilers/html_component.cr index 3cff0ee06..4c67fbc19 100644 --- a/src/compilers/html_component.cr +++ b/src/compilers/html_component.cr @@ -1,19 +1,19 @@ module Mint class Compiler - def _compile(node : Ast::HtmlComponent) : String + def _compile(node : Ast::HtmlComponent) : Codegen::Node if node.static? name = static_components_pool.of(node.static_hash, nil) static_components[name] ||= compile_html_component(node) - "$" + name + "()" + Codegen.join ["$", name, "()"] else compile_html_component(node) end end - def compile_html_component(node : Ast::HtmlComponent) : String + def compile_html_component(node : Ast::HtmlComponent) : Codegen::Node name = js.class_of(lookups[node]) @@ -24,27 +24,25 @@ module Mint items = compile node.children, ", " - "_array(#{items})" + Codegen.join ["_array(", items, ")"] end attributes = node .attributes .map { |item| resolve(item, false) } - .reduce({} of String => String) { |memo, item| memo.merge(item) } + .reduce({} of Codegen::Node => Codegen::Node) { |memo, item| memo.merge(item) } node.ref.try do |ref| attributes["ref"] = "(instance) => { this._#{ref.value} = instance }" end contents = - ["#{name}", - js.object(attributes), - children] - .reject!(&.empty?) - .join(", ") + Codegen.join( + [name, js.object(attributes), children].reject! { |item| Codegen.empty? item }, + ", ") - "_h(#{contents})" + Codegen.join ["_h(", contents, ")"] end end end diff --git a/src/compilers/html_element.cr b/src/compilers/html_element.cr index b6bbd41ea..7507d090e 100644 --- a/src/compilers/html_element.cr +++ b/src/compilers/html_element.cr @@ -1,8 +1,8 @@ module Mint class Compiler - def compile(value : Array(Ast::Node | String), quote_string : Bool = false) + def compile(value : Array(Ast::Node | String), quote_string : Bool = false) : Codegen::Node if value.any?(Ast::Node) - value.map do |part| + Codegen.join(value, " + ") do |part| case part when Ast::StringLiteral compile part, quote: quote_string @@ -11,19 +11,16 @@ module Mint else compile part end - end.reject!(&.empty?) - .join(" + ") + end else result = - value - .select(String) - .join(' ') + Codegen.join(value.select(String), " ") - "`#{result}`" + Codegen.join ["`", result, "`"] end end - def _compile(node : Ast::HtmlElement) : String + def _compile(node : Ast::HtmlElement) : Codegen::Node tag = node.tag.value @@ -42,7 +39,7 @@ module Mint .attributes .reject(&.name.value.in?("class", "style")) .map { |attribute| resolve(attribute) } - .reduce({} of String => String) { |memo, item| memo.merge(item) } + .reduce({} of String => Codegen::Node) { |memo, item| memo.merge(item) } style_nodes = node.styles.map { |item| lookups[item] } @@ -64,11 +61,11 @@ module Mint classes = if class_name && class_name_attribute_value - "#{class_name_attribute_value} + ` #{class_name}`" + Codegen.join [class_name_attribute_value, " + ` ", class_name, "`"] elsif class_name_attribute_value - "#{class_name_attribute_value}" + class_name_attribute_value elsif class_name - "`#{class_name}`" + Codegen.join ["`", class_name, "`"] end attributes["className"] = classes if classes @@ -78,7 +75,7 @@ module Mint .find(&.name.value.==("style")) .try { |attribute| compile(attribute.value) } - styles = %w[] + styles = [] of Codegen::Node node.styles.each do |item| if style_builder.any?(lookups[item]) @@ -95,7 +92,7 @@ module Mint styles << custom_styles if custom_styles unless styles.empty? - attributes["style"] = "_style([#{styles.join(", ")}])" + attributes["style"] = Codegen.join ["_style([", Codegen.join(styles, ", "), "])"] end node.ref.try do |ref| @@ -110,13 +107,11 @@ module Mint end contents = - [%("#{tag}"), - attributes, - children] - .reject!(&.empty?) - .join(", ") + Codegen.join( + [%("#{tag}"), attributes, children].reject! { |item| Codegen.empty? item }, + ", ") - "_h(#{contents})" + Codegen.join ["_h(", contents, ")"] end end end diff --git a/src/compilers/html_expression.cr b/src/compilers/html_expression.cr index 52d08949c..0c4ab010a 100644 --- a/src/compilers/html_expression.cr +++ b/src/compilers/html_expression.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::HtmlExpression) : String + def _compile(node : Ast::HtmlExpression) : Codegen::Node if node.expressions.empty? "null" elsif node.expressions.size == 1 @@ -9,7 +9,7 @@ module Mint children = compile node.expressions - "_h(React.Fragment, {}, #{js.array(children)})" + Codegen.join ["_h(React.Fragment, {}, ", js.array(children), ")"] end end end diff --git a/src/compilers/html_fragment.cr b/src/compilers/html_fragment.cr index 26180be0e..b0dedb861 100644 --- a/src/compilers/html_fragment.cr +++ b/src/compilers/html_fragment.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::HtmlFragment) : String + def _compile(node : Ast::HtmlFragment) : Codegen::Node attributes = if key = node.key js.object({"key" => compile key.value}) @@ -14,7 +14,7 @@ module Mint items = compile node.children - "_h(React.Fragment, #{attributes}, #{js.array(items)})" + Codegen.join ["_h(React.Fragment, ", attributes, ", ", js.array(items), ")"] end end end diff --git a/src/compilers/if.cr b/src/compilers/if.cr index 0befc1122..cbd6ce3c0 100644 --- a/src/compilers/if.cr +++ b/src/compilers/if.cr @@ -1,25 +1,6 @@ module Mint class Compiler - def _compile(items : Array(Ast::CssDefinition), block : Proc(String, String)?) - compiled = - items.each_with_object({} of String => String) do |definition, memo| - variable = - if block - block.call(definition.name) - else - "" - end - - value = - compile definition.value - - memo["[`#{variable}`]"] = value - end - - "Object.assign(_, #{js.object(compiled)})" - end - - def compile(node : Ast::If, block : Proc(String, String)? = nil) : String + def compile(node : Ast::If, block : Proc(Codegen::Node, Codegen::Node)? = nil) : Codegen::Node if checked.includes?(node) _compile node, block else @@ -27,7 +8,7 @@ module Mint end end - def _compile(node : Ast::If, block : Proc(String, String)? = nil) : String + def _compile(node : Ast::If, block : Proc(Codegen::Node, Codegen::Node)? = nil) : Codegen::Node condition = compile node.condition @@ -39,7 +20,7 @@ module Mint when Array(Ast::CssDefinition) _compile item, block: block else - compile item + Codegen.source_mapped(item, compile item) end falsy = @@ -47,14 +28,14 @@ module Mint when Array(Ast::CssDefinition) _compile item, block: block when Ast::If - compile item, block: block + Codegen.source_mapped(item, compile item, block: block) when Ast::Node - compile item + Codegen.source_mapped(item, compile item) else "null" end - "(#{condition} ? #{truthy} : #{falsy})" + Codegen.source_mapped(node, Codegen.join ["(", condition, " ? ", truthy, " : ", falsy, ")"]) end end end diff --git a/src/compilers/inline_function.cr b/src/compilers/inline_function.cr index 6dada9498..5f360b7c0 100644 --- a/src/compilers/inline_function.cr +++ b/src/compilers/inline_function.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::InlineFunction) : String + def _compile(node : Ast::InlineFunction) : Codegen::Node body = compile node.body diff --git a/src/compilers/interpolation.cr b/src/compilers/interpolation.cr index 6447f0e0d..c85994cc9 100644 --- a/src/compilers/interpolation.cr +++ b/src/compilers/interpolation.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Interpolation) : String + def _compile(node : Ast::Interpolation) : Codegen::Node compile node.expression end end diff --git a/src/compilers/js.cr b/src/compilers/js.cr index 6457488a6..d7e41ecfb 100644 --- a/src/compilers/js.cr +++ b/src/compilers/js.cr @@ -1,20 +1,20 @@ module Mint class Compiler - def _compile(node : Ast::Js) : String + def _compile(node : Ast::Js) : Codegen::Node value = - node.value.join do |item| + Codegen.strip(Codegen.join(node.value) do |item| case item when Ast::Node compile item else item end - end.strip + end) - if value.empty? + if Codegen.empty? value "" else - "(#{value})" + Codegen.join ["(", Codegen.source_mapped(node, value), ")"] end end end diff --git a/src/compilers/member_access.cr b/src/compilers/member_access.cr index bc2e7d257..ed400e2cc 100644 --- a/src/compilers/member_access.cr +++ b/src/compilers/member_access.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::MemberAccess) : String + def _compile(node : Ast::MemberAccess) : Codegen::Node "((_) => _.#{node.name.value})" end end diff --git a/src/compilers/module.cr b/src/compilers/module.cr index c495bf940..5c12ac68f 100644 --- a/src/compilers/module.cr +++ b/src/compilers/module.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Module) : String + def _compile(node : Ast::Module) : Codegen::Node name = js.class_of(node) @@ -12,14 +12,14 @@ module Mint constructor = if !constants.empty? - [js.function("constructor", %w[]) do + [js.function("constructor", [] of Codegen::Node) do js.statements([ - js.call("super", %w[]), + js.call("super", [] of Codegen::Node), js.call("this._d", [js.object(constants)]), ]) end] else - [] of String + [] of Codegen::Node end js.module(name, functions + constructor) diff --git a/src/compilers/module_access.cr b/src/compilers/module_access.cr index d0e61fe21..713b75a0d 100644 --- a/src/compilers/module_access.cr +++ b/src/compilers/module_access.cr @@ -1,13 +1,13 @@ module Mint class Compiler - def _compile(node : Ast::ModuleAccess) : String + def _compile(node : Ast::ModuleAccess) : Codegen::Node name = js.class_of(lookups[node]) case lookups[node] when Ast::Provider if node.variable.value == "subscriptions" - return "#{name}._subscriptions" + return Codegen.join [name, "._subscriptions"] end else # ignore @@ -16,7 +16,7 @@ module Mint variable = js.variable_of(lookups[node.variable]) - "#{name}.#{variable}" + Codegen.join [name, ".", variable] end end end diff --git a/src/compilers/negated_expression.cr b/src/compilers/negated_expression.cr index fbff1ef84..8f0c33166 100644 --- a/src/compilers/negated_expression.cr +++ b/src/compilers/negated_expression.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::NegatedExpression) : String + def _compile(node : Ast::NegatedExpression) : Codegen::Node expression = compile node.expression - "#{node.negations}#{expression}" + Codegen.join [node.negations, expression] end end end diff --git a/src/compilers/next_call.cr b/src/compilers/next_call.cr index fc562f4eb..4ee4aa684 100644 --- a/src/compilers/next_call.cr +++ b/src/compilers/next_call.cr @@ -1,11 +1,11 @@ module Mint class Compiler - def _compile(node : Ast::NextCall) : String + def _compile(node : Ast::NextCall) : Codegen::Node entity = lookups[node]? state = - node.data.fields.each_with_object({} of String => String) do |item, memo| + node.data.fields.each_with_object({} of Codegen::Node => Codegen::Node) do |item, memo| field = case entity when Ast::Provider @@ -21,7 +21,8 @@ module Mint end if field - memo[js.variable_of(field)] = compile item.value + memo[js.variable_of(field)] = + Codegen.source_mapped(item.value, compile item.value) else memo end @@ -29,7 +30,9 @@ module Mint js.promise do js.arrow_function(["_resolve"]) do - "this.setState(_u(this.state, new Record(#{js.object(state)})), _resolve)\n" + Codegen.join [ + "this.setState(_u(this.state, new Record(", js.object(state), ")), _resolve)", + ] end end end diff --git a/src/compilers/number_literal.cr b/src/compilers/number_literal.cr index c405529ce..d052691b5 100644 --- a/src/compilers/number_literal.cr +++ b/src/compilers/number_literal.cr @@ -1,7 +1,7 @@ module Mint class Compiler - def _compile(node : Ast::NumberLiteral) : String - node.static_value + def _compile(node : Ast::NumberLiteral) : Codegen::Node + Codegen.symbol_mapped(node, node.static_value) end end end diff --git a/src/compilers/operation.cr b/src/compilers/operation.cr index 030e8648c..7abebe09a 100644 --- a/src/compilers/operation.cr +++ b/src/compilers/operation.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Operation) : String + def _compile(node : Ast::Operation) : Codegen::Node left = compile node.left @@ -9,13 +9,13 @@ module Mint case node.operator when "or" - "(#{left}._0 || #{right})" + Codegen.join ["(", left, "._0 || ", right, ")"] when "==" - "_compare(#{left}, #{right})" + Codegen.join ["_compare(", left, ", ", right, ")"] when "!=" - "!_compare(#{left}, #{right})" + Codegen.join ["!_compare(", left, ", ", right, ")"] else - "#{left} #{node.operator} #{right}" + Codegen.join [left, node.operator, right], " " end end end diff --git a/src/compilers/parallel.cr b/src/compilers/parallel.cr index 0f12a07e7..fa007becc 100644 --- a/src/compilers/parallel.cr +++ b/src/compilers/parallel.cr @@ -1,18 +1,19 @@ module Mint class Compiler - def _compile(node : Ast::Parallel) : String + def _compile(node : Ast::Parallel) : Codegen::Node body = node.statements.map do |statement| - prefix = ->(value : String) { + prefix = ->(value : Codegen::Node) { case target = statement.target when Ast::Variable - js.assign(js.variable_of(target), value) + js.assign(Codegen.symbol_mapped(target, js.variable_of(target)), value) when Ast::TupleDestructuring + params = + target.parameters.map { |param| Codegen.symbol_mapped(param, js.variable_of(param)) } + variables = - target - .parameters - .join(',') { |param| js.variable_of(param) } + Codegen.join(params, ",") - "[#{variables}] = #{value}" + Codegen.join ["[", variables, "] = ", value] else value end @@ -31,11 +32,11 @@ module Mint node .catches .select { |item| item.type == type.parameters[0].name } - .map { |item| compile(item).as(String) } + .map { |item| compile(item) } end else # ignore - end || %w[] + end || [] of Codegen::Node case type when TypeChecker::Type @@ -67,7 +68,7 @@ module Mint when "Promise" if catches && !catches.empty? js.asynciif do - js.try(prefix.call("await #{expression}"), + js.try(prefix.call(Codegen.join(["await ", expression])), [js.catch("_error", js.statements(catches))], "") end @@ -78,13 +79,13 @@ module Mint else # ignore end || js.asynciif do - prefix.call("await #{expression}") + prefix.call(Codegen.join ["await ", expression]) end end catch_all = node.catch_all.try do |catch| - "return #{compile catch.expression}" + Codegen.join ["return ", compile catch.expression] end || js.statements([ "console.warn(`Unhandled error in parallel expression:`)", @@ -105,10 +106,10 @@ module Mint node.statements.map do |statement| case target = statement.target when Ast::Variable - js.let(js.variable_of(target), "null") + js.let(Codegen.symbol_mapped(target, js.variable_of(target)), "null") when Ast::TupleDestructuring target.parameters.map do |variable| - js.let(js.variable_of(variable), "null") + js.let(Codegen.symbol_mapped(variable, js.variable_of(variable)), "null") end else # ignore diff --git a/src/compilers/parenthesized_expression.cr b/src/compilers/parenthesized_expression.cr index cfac88078..edf9366f9 100644 --- a/src/compilers/parenthesized_expression.cr +++ b/src/compilers/parenthesized_expression.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::ParenthesizedExpression) : String + def _compile(node : Ast::ParenthesizedExpression) : Codegen::Node expression = compile node.expression - "(#{expression})" + Codegen.join ["(", expression, ")"] end end end diff --git a/src/compilers/pipe.cr b/src/compilers/pipe.cr index c01ca501b..1a2b1b0f2 100644 --- a/src/compilers/pipe.cr +++ b/src/compilers/pipe.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Pipe) : String + def _compile(node : Ast::Pipe) : Codegen::Node compile node.call end end diff --git a/src/compilers/property.cr b/src/compilers/property.cr index eee82642e..e83bd2fdf 100644 --- a/src/compilers/property.cr +++ b/src/compilers/property.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Property) : String + def _compile(node : Ast::Property) : Codegen::Node prop_name = if node.name.value == "children" "children" @@ -12,7 +12,7 @@ module Mint js.variable_of(node) body = - "return this._p('#{prop_name}');" + Codegen.join ["return this._p('", prop_name, "');"] js.get(name, body) end diff --git a/src/compilers/provider.cr b/src/compilers/provider.cr index b1a5bea8e..a97e0c20c 100644 --- a/src/compilers/provider.cr +++ b/src/compilers/provider.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Provider) : String + def _compile(node : Ast::Provider) : Codegen::Node functions = compile node.functions diff --git a/src/compilers/record.cr b/src/compilers/record.cr index bb2cda5e9..8fe38f77e 100644 --- a/src/compilers/record.cr +++ b/src/compilers/record.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::Record) : String + def _compile(node : Ast::Record) : Codegen::Node fields = node.fields .map { |item| resolve(item) } - .reduce({} of String => String) { |memo, item| memo.merge(item) } + .reduce({} of String => Codegen::Node) { |memo, item| memo.merge(item) } type = types[node]? @@ -13,9 +13,9 @@ module Mint name = js.class_of(type.name) - "new #{name}(#{js.object(fields)})" + Codegen.join ["new ", name, "(", js.object(fields), ")"] else - "new Record(#{js.object(fields)})" + Codegen.join ["new Record(", js.object(fields), ")"] end end end diff --git a/src/compilers/record_constructor.cr b/src/compilers/record_constructor.cr index 8d1855ae4..630aa1022 100644 --- a/src/compilers/record_constructor.cr +++ b/src/compilers/record_constructor.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::RecordConstructor) : String + def _compile(node : Ast::RecordConstructor) : Codegen::Node type = types[node]? @@ -9,13 +9,13 @@ module Mint name = js.class_of(type.name) - args = %w[] + args = [] of Codegen::Node fields = type .fields .each_with_index - .reduce({} of String => String) do |memo, value| + .reduce({} of String => Codegen::Node) do |memo, value| field, index = value key, _ = field @@ -32,7 +32,7 @@ module Mint end body = - "new #{name}(#{js.object(fields)})" + Codegen.join ["new ", name, "(", js.object(fields), ")"] if args.empty? body diff --git a/src/compilers/record_definition.cr b/src/compilers/record_definition.cr index 79b0474d5..9994f9f38 100644 --- a/src/compilers/record_definition.cr +++ b/src/compilers/record_definition.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::RecordDefinition) : String + def _compile(node : Ast::RecordDefinition) : Codegen::Node type = types[node] name = @@ -15,7 +15,7 @@ module Mint "{}" end - "const #{name} = _R(#{mappings})" + Codegen.join ["const ", name, " = _R(", mappings, ")"] else "" end diff --git a/src/compilers/record_field.cr b/src/compilers/record_field.cr index 1aeb22028..7f240db63 100644 --- a/src/compilers/record_field.cr +++ b/src/compilers/record_field.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def resolve(node : Ast::RecordField) : Hash(String, String) + def resolve(node : Ast::RecordField) : Hash(String, Codegen::Node) value = compile node.value diff --git a/src/compilers/record_update.cr b/src/compilers/record_update.cr index 8c3bb2f10..89ee7f9d8 100644 --- a/src/compilers/record_update.cr +++ b/src/compilers/record_update.cr @@ -1,15 +1,15 @@ module Mint class Compiler - def _compile(node : Ast::RecordUpdate) : String + def _compile(node : Ast::RecordUpdate) : Codegen::Node expression = compile node.expression fields = node.fields .map { |item| resolve(item) } - .reduce({} of String => String) { |memo, item| memo.merge(item) } + .reduce({} of Codegen::Node => Codegen::Node) { |memo, item| memo.merge(item) } - "_u(#{expression}, #{js.object(fields)})" + Codegen.join ["_u(", expression, ", ", js.object(fields), ")"] end end end diff --git a/src/compilers/regexp_literal.cr b/src/compilers/regexp_literal.cr index 83d137850..959fe4625 100644 --- a/src/compilers/regexp_literal.cr +++ b/src/compilers/regexp_literal.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::RegexpLiteral) : String + def _compile(node : Ast::RegexpLiteral) : Codegen::Node node.static_value end end diff --git a/src/compilers/route.cr b/src/compilers/route.cr index 91aa5c2c0..22d172cfc 100644 --- a/src/compilers/route.cr +++ b/src/compilers/route.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Route) : String + def _compile(node : Ast::Route) : Codegen::Node expression = compile node.expression diff --git a/src/compilers/routes.cr b/src/compilers/routes.cr index 53d0adc89..3654fe0fa 100644 --- a/src/compilers/routes.cr +++ b/src/compilers/routes.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::Routes) : String + def _compile(node : Ast::Routes) : Codegen::Node routes = compile node.routes - "_program.addRoutes(#{js.array(routes)})" + Codegen.join ["_program.addRoutes(", js.array(routes), ")"] end end end diff --git a/src/compilers/sequence.cr b/src/compilers/sequence.cr index 153233e71..c6e5a276b 100644 --- a/src/compilers/sequence.cr +++ b/src/compilers/sequence.cr @@ -1,25 +1,26 @@ module Mint class Compiler - def _compile(node : Ast::Sequence) : String + def _compile(node : Ast::Sequence) : Codegen::Node body = node .statements .sort_by { |item| resolve_order.index(item) || -1 } .map_with_index do |statement, index| - prefix = ->(value : String) { + prefix = ->(value : Codegen::Node) { if (index + 1) == node.statements.size - "_ = #{value}" + Codegen.join ["_ = ", value] else case target = statement.target when Ast::Variable - js.let(js.variable_of(target), value) + js.let(Codegen.symbol_mapped(target, js.variable_of(target)), value) when Ast::TupleDestructuring + params = + target.parameters.map { |param| Codegen.symbol_mapped(param, js.variable_of(param)) } + variables = - target - .parameters - .join(',') { |param| js.variable_of(param) } + Codegen.join(params, ",") - "const [#{variables}] = #{value}" + Codegen.join ["const [", variables, "] = ", value] else value end @@ -39,12 +40,12 @@ module Mint node .catches .select { |item| item.type == type.parameters[0].name } - .map { |item| compile(item).as(String) } + .map { |item| compile(item) } else - %w[] + [] of Codegen::Node end else - %w[] + [] of Codegen::Node end case type @@ -74,26 +75,26 @@ module Mint unless catches.empty? try = js.asynciif do js.try( - body: "return await #{expression}", + body: Codegen.join(["return await ", expression]), catches: [ js.catch("_error") { js.statements(catches) }, ], finally: "") - end + end.as(Codegen::Node) - prefix.call("await #{try}") + prefix.call(Codegen.join ["await ", try]) end else # ignore end else # ignore - end || prefix.call("await #{expression}") + end || prefix.call(Codegen.join ["await ", expression]) end catch_all = node.catch_all.try do |catch| - "_ = #{compile catch.expression}" + Codegen.join ["_ = ", compile catch.expression] end || js.statements([ "console.warn(`Unhandled error in sequence expression:`)", "console.warn(_error)", @@ -114,7 +115,7 @@ module Mint js.if("!(_error instanceof DoError)", catch_all) end, ], - finally: finally.to_s), + finally: finally), "return _", ]) end diff --git a/src/compilers/state.cr b/src/compilers/state.cr index e9b22b2f4..404d66a4f 100644 --- a/src/compilers/state.cr +++ b/src/compilers/state.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::State) : String + def _compile(node : Ast::State) : Codegen::Node name = js.variable_of(node) - js.get(name, "return this.state.#{name};") + js.get(name, Codegen.join ["return this.state.", name, ";"]) end end end diff --git a/src/compilers/statement.cr b/src/compilers/statement.cr index 9c0c74f03..5e8afd845 100644 --- a/src/compilers/statement.cr +++ b/src/compilers/statement.cr @@ -1,7 +1,7 @@ module Mint class Compiler - def _compile(node : Ast::Statement) : String - compile node.expression + def _compile(node : Ast::Statement) : Codegen::Node + Codegen.source_mapped(node, compile node.expression) end end end diff --git a/src/compilers/store.cr b/src/compilers/store.cr index 54e0e026e..193e353f5 100644 --- a/src/compilers/store.cr +++ b/src/compilers/store.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Store) : String + def _compile(node : Ast::Store) : Codegen::Node functions = compile node.functions @@ -22,12 +22,12 @@ module Mint js.store(name, body) end - def compile_constructor(node : Ast::Store | Ast::Provider) : String + def compile_constructor(node : Ast::Store | Ast::Provider) : Codegen::Node states = node .states .select(&.in?(checked)) - .each_with_object({} of String => String) do |state, memo| + .each_with_object({} of Codegen::Node => Codegen::Node) do |state, memo| name = js.variable_of(state) @@ -42,9 +42,9 @@ module Mint js.call("this._d", [js.object(compile_constants(node.constants))]) end - js.function("constructor", %w[]) do + js.function("constructor", [] of Codegen::Node) do js.statements([ - js.call("super", %w[]), + js.call("super", [] of Codegen::Node), js.assign("this.state", js.object(states)), constants, ].compact) diff --git a/src/compilers/string_literal.cr b/src/compilers/string_literal.cr index 08898c20c..1850dba16 100644 --- a/src/compilers/string_literal.cr +++ b/src/compilers/string_literal.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def compile(node : Ast::StringLiteral, quote : Bool = false) : String + def compile(node : Ast::StringLiteral, quote : Bool = false) : Codegen::Node if checked.includes?(node) _compile(node, quote) else @@ -8,27 +8,25 @@ module Mint end end - def _compile(node : Ast::StringLiteral, quote : Bool = false) : String + def _compile(node : Ast::StringLiteral, quote : Bool = false) : Codegen::Node value = - node - .value - .join do |item| - case item - when Ast::Node - "${#{compile(item)}}" - when String - item - .gsub('`', "\\`") - .gsub("${", "\\${") - else - "" - end + Codegen.join(node.value) do |item| + case item + when Ast::Node + Codegen.join ["${", compile(item), "}"] + when String + item + .gsub('`', "\\`") + .gsub("${", "\\${") + else + "" end + end if quote - %(`"#{value}"`) + Codegen.join ["`\"", value, "\"`"] else - "`#{value}`" + Codegen.join ["`", value, "`"] end end end diff --git a/src/compilers/suite.cr b/src/compilers/suite.cr index c3dd19587..0073e6cc1 100644 --- a/src/compilers/suite.cr +++ b/src/compilers/suite.cr @@ -1,13 +1,13 @@ module Mint class Compiler - def _compile(node : Ast::Suite) : String + def _compile(node : Ast::Suite) : Codegen::Node name = - compile node.name + (compile node.name).as(Codegen::Node) tests = - compile node.tests, "," + (compile node.tests, ",").as(Codegen::Node) - "{ name: #{name}, tests: [#{tests}] }" + Codegen.join ["{ name: ", name, ", tests: [", tests, "] }"] end end end diff --git a/src/compilers/test.cr b/src/compilers/test.cr index df99eb96f..f4b70519a 100644 --- a/src/compilers/test.cr +++ b/src/compilers/test.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Test) : String + def _compile(node : Ast::Test) : Codegen::Node raw_expression = node.expression @@ -17,19 +17,21 @@ module Mint left = compile raw_expression.left - "(() => { - const context = new TestContext(#{left}) - const right = #{right} + Codegen.join [ + "(() => { + const context = new TestContext(", left, ") + const right = ", right, " - context.step((subject) => { - if (_compare(subject, right)) { - return true - } else { - throw `Assertion failed ${right.toString()} != ${subject.toString()}` - } - }) - return context - })()" + context.step((subject) => { + if (_compare(subject, right)) { + return true + } else { + throw `Assertion failed ${right.toString()} != ${subject.toString()}` + } + }) + return context + })()", + ] end else # ignore @@ -37,7 +39,7 @@ module Mint expression = compile raw_expression unless expression - "{ name: #{name}, proc: () => { return #{expression} } }" + Codegen.join ["{ name: ", name, ", proc: () => { return ", expression, " } }"] end end end diff --git a/src/compilers/top_level.cr b/src/compilers/top_level.cr index 0ce2214b4..0f5afadba 100644 --- a/src/compilers/top_level.cr +++ b/src/compilers/top_level.cr @@ -8,7 +8,7 @@ module Mint # Compiles the application with the runtime and the rendering of the $Main # component. - def self.compile(artifacts : TypeChecker::Artifacts, options = DEFAULT_OPTIONS) : String + def self.compile(artifacts : TypeChecker::Artifacts, options = DEFAULT_OPTIONS) : Codegen::Node compiler = new(artifacts, options[:optimize], options[:css_prefix]) @@ -19,11 +19,11 @@ module Mint .ast .components .select(&.global?) - .each_with_object({} of String => String) do |item, memo| + .each_with_object({} of String => Codegen::Node) do |item, memo| name = compiler.js.class_of(item) - memo[name] = "$#{name}" + memo[name] = Codegen.join ["$", name] end main_class = @@ -32,13 +32,13 @@ module Mint globals_object = compiler.js.object(globals) - "\n_program.render(#{main_class}, #{globals_object})" + Codegen.join ["\n_program.render(", main_class, ", ", globals_object, ")"] end || "" compiler.wrap_runtime(compiler.compile, main) end - def self.compile_embed(artifacts : TypeChecker::Artifacts, options = DEFAULT_OPTIONS) : String + def self.compile_embed(artifacts : TypeChecker::Artifacts, options = DEFAULT_OPTIONS) : Codegen::Node compiler = new(artifacts, options[:optimize]) @@ -49,11 +49,11 @@ module Mint .ast .components .select(&.global?) - .each_with_object({} of String => String) do |item, memo| + .each_with_object({} of String => Codegen::Node) do |item, memo| name = compiler.js.class_of(item) - memo[name] = "$#{name}" + memo[name] = Codegen.join ["$", name] end main_class = @@ -62,7 +62,7 @@ module Mint globals_object = compiler.js.object(globals) - "\n Mint.embed = (base) => (new mint.EmbeddedProgram(base)).render(#{main_class}, #{globals_object})" + Codegen.join ["\n Mint.embed = (base) => (new mint.EmbeddedProgram(base)).render(", main_class, ", ", globals_object, ")"] end || "" compiler.wrap_runtime(compiler.compile, main) @@ -71,7 +71,7 @@ module Mint # Compiles the application without the runtime. def self.compile_bare(artifacts : TypeChecker::Artifacts, options = DEFAULT_OPTIONS) : String compiler = new(artifacts, options[:optimize], options[:css_prefix]) - compiler.compile + Codegen.build(compiler.compile)[:code] end # Compiles the application with the runtime and the tests @@ -79,11 +79,14 @@ module Mint compiler = new(artifacts) - compiler.wrap_runtime(compiler.compile(include_tests: true)) + compiled = + compiler.wrap_runtime(compiler.compile(include_tests: true)) + + (Codegen.build compiled)[:code] end # Compiles the application - def compile(include_tests : Bool = false) : String + def compile(include_tests : Bool = false) : Codegen::Node records = compile ast.records @@ -112,26 +115,28 @@ module Mint if all_css.empty? "" else - "_insertStyles(`\n#{all_css}\n`)" + Codegen.join ["_insertStyles(`\n", all_css, "\n`)"] end suites = if include_tests - ["SUITES = [#{compile(ast.suites, ",")}]"] + [Codegen.join ["SUITES = [", compile(ast.suites, ","), "]"]] else - %w[] + [] of Codegen::Node end static = static_components.map do |name, compiled| - js.const("$#{name}", "_m(() => #{compiled})") + js.const( + "$#{name}", + Codegen.join(["_m(() => ", compiled, ")"])) end elements = (enums + records + modules + providers + routes + components + static + stores + [footer] + suites) - .reject!(&.empty?) + .reject! { |n| Codegen.empty? n } - replace_skipped(js.statements(elements)) + js.statements(elements) end # -------------------------------------------------------------------------- @@ -177,7 +182,7 @@ module Mint # -------------------------------------------------------------------------- # Wraps the application with the runtime - def wrap_runtime(body, main = "") + def wrap_runtime(body : Codegen::Node, main : Codegen::Node = "") : Codegen::Node html_event_module = ast.modules.find(&.name.==("Html.Event")).not_nil! @@ -185,81 +190,100 @@ module Mint html_event_module.functions.find(&.name.value.==("fromEvent")).not_nil! from_event_call = - js.class_of(html_event_module) + "." + js.variable_of(from_event) - - <<-RESULT - (() => { - const _enums = {} - const mint = Mint(_enums) - - const _normalizeEvent = function (event) { - return #{from_event_call}(mint.normalizeEvent(event)) - }; - - const _R = mint.createRecord; - const _h = mint.createElement; - const _createPortal = mint.createPortal; - const _insertStyles = mint.insertStyles; - const _navigate = mint.navigate; - const _compare = mint.compare; - const _program = mint.program; - const _encode = mint.encode; - const _style = mint.style; - const _array = mint.array; - const _u = mint.update; - const _at = mint.at; - - window.TestContext = mint.TestContext; - const TestContext = mint.TestContext; - const ReactDOM = mint.ReactDOM; - const Decoder = mint.Decoder; - const Encoder = mint.Encoder; - const DateFNS = mint.DateFNS; - const Record = mint.Record; - const React = mint.React; - - const _C = mint.Component; - const _P = mint.Provider; - const _M = mint.Module; - const _S = mint.Store; - const _E = mint.Enum; - - const _m = (method) => { - let value; - return () => { - if (value) { return value } - value = method() - return value - } - } - - const _s = (item, callback) => { - if (item instanceof #{nothing}) { - return item - } else if (item instanceof #{just}) { - return new #{just}(callback(item._0)) - } else { - return callback(item) - } - } - - class DoError extends Error {} - - #{body} - - const Nothing = #{nothing} - const Just = #{just} - const Err = #{err} - const Ok = #{ok} - - _enums.nothing = #{nothing} - _enums.just = #{just} - _enums.err = #{err} - _enums.ok = #{ok} - - #{main} - })() - RESULT + Codegen.join [js.class_of(html_event_module), ".", js.variable_of(from_event)] + + Codegen.join [ + ( + <<-JS1 + (() => { + const _enums = {} + const mint = Mint(_enums) + + const _normalizeEvent = function (event) { + return + JS1 + ), + from_event_call, + ( + <<-JS2 + (mint.normalizeEvent(event)) + }; + + const _R = mint.createRecord; + const _h = mint.createElement; + const _createPortal = mint.createPortal; + const _insertStyles = mint.insertStyles; + const _navigate = mint.navigate; + const _compare = mint.compare; + const _program = mint.program; + const _encode = mint.encode; + const _style = mint.style; + const _array = mint.array; + const _u = mint.update; + const _at = mint.at; + + window.TestContext = mint.TestContext; + const TestContext = mint.TestContext; + const ReactDOM = mint.ReactDOM; + const Decoder = mint.Decoder; + const Encoder = mint.Encoder; + const DateFNS = mint.DateFNS; + const Record = mint.Record; + const React = mint.React; + + const _C = mint.Component; + const _P = mint.Provider; + const _M = mint.Module; + const _S = mint.Store; + const _E = mint.Enum; + + const _m = (method) => { + let value; + return () => { + if (value) { return value } + value = method() + return value + } + } + + const _s = (item, callback) => { + if (item instanceof #{nothing}) { + return item + } else if (item instanceof #{just}) { + return new #{just}(callback(item._0)) + } else { + return callback(item) + } + } + + class DoError extends Error {} + + JS2 + ), + body, + ( + <<-JS3 + + const Nothing = #{nothing} + const Just = #{just} + const Err = #{err} + const Ok = #{ok} + + _enums.nothing = #{nothing} + _enums.just = #{just} + _enums.err = #{err} + _enums.ok = #{ok} + + JS3 + ), + main, + ( + <<-JS4 + + })() + JS4 + ), + ] end end end diff --git a/src/compilers/try.cr b/src/compilers/try.cr index 37e4a9e9d..a3932384d 100644 --- a/src/compilers/try.cr +++ b/src/compilers/try.cr @@ -1,9 +1,14 @@ module Mint class Compiler - def _compile(node : Ast::Try) : String + def _compile(node : Ast::Try) : Codegen::Node catch_all = node.catch_all.try do |catch| - js.let("_catch_all", js.arrow_function(%w[], "return #{compile(catch.expression)}")) + "\n\n" + Codegen.join [ + js.let("_catch_all", + js.arrow_function([] of Codegen::Node, + Codegen.join(["return ", compile(catch.expression)]))), + "\n\n", + ] end body = @@ -14,20 +19,21 @@ module Mint is_last = (index + 1) == node.statements.size - prefix = ->(value : String) { + prefix = ->(value : Codegen::Node) { if is_last - "return #{value}" + Codegen.source_mapped(statement, Codegen.join ["return ", value]) else case target = statement.target when Ast::Variable - js.let(js.variable_of(target), value) + js.let(Codegen.symbol_mapped(target, js.variable_of(target)), value) when Ast::TupleDestructuring + params = + target.parameters.map { |param| Codegen.symbol_mapped(param, js.variable_of(param)) } + variables = - target - .parameters - .join(',') { |param| js.variable_of(param) } + Codegen.join(params, ",") - "const [#{variables}] = #{value}" + Codegen.join ["const [", variables, "] = ", value] else value end @@ -54,7 +60,7 @@ module Mint js.statements([ js.let(variable, "_error"), - "return #{catch_body}", + Codegen.join ["return ", catch_body], ]) end end @@ -69,7 +75,7 @@ module Mint # ignore end - if catches && !catches.empty? + if catches && !Codegen.empty?(catches) js.statements([ js.let("_#{index}", expression), js.if("_#{index} instanceof Err") do @@ -85,7 +91,7 @@ module Mint end end - js.iif("#{catch_all}#{js.statements(body)}") + js.iif(Codegen.join [catch_all, js.statements(body)].compact) end end end diff --git a/src/compilers/tuple_literal.cr b/src/compilers/tuple_literal.cr index c48153ea5..a43af258f 100644 --- a/src/compilers/tuple_literal.cr +++ b/src/compilers/tuple_literal.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::TupleLiteral) : String + def _compile(node : Ast::TupleLiteral) : Codegen::Node items = compile node.items, ", " - "[#{items}]" + Codegen.join ["[", items, "]"] end end end diff --git a/src/compilers/unary_minus.cr b/src/compilers/unary_minus.cr index 91e7fbfd1..fd22e4b2e 100644 --- a/src/compilers/unary_minus.cr +++ b/src/compilers/unary_minus.cr @@ -1,10 +1,10 @@ module Mint class Compiler - def _compile(node : Ast::UnaryMinus) : String + def _compile(node : Ast::UnaryMinus) : Codegen::Node expression = compile node.expression - "-(#{expression})" + Codegen.join ["-(", expression, ")"] end end end diff --git a/src/compilers/variable.cr b/src/compilers/variable.cr index b410094d4..b263d9f8e 100644 --- a/src/compilers/variable.cr +++ b/src/compilers/variable.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Variable) : String + def _compile(node : Ast::Variable) : Codegen::Node entity, parent = variables[node] # Subscriptions for providers are handled here @@ -41,75 +41,79 @@ module Mint # ignore end - case parent - when Tuple(String, TypeChecker::Checkable, Ast::Node) - js.variable_of(parent[2]) - else - case entity - when Ast::Component, Ast::HtmlElement - case parent - when Ast::Component - ref = - parent - .refs - .find { |(ref, _)| ref.value == node.value } - .try { |(ref, _)| js.variable_of(ref) } + result = + case parent + when Tuple(String, TypeChecker::Checkable, Ast::Node) + js.variable_of(parent[2]) + else + case entity + when Ast::Component, Ast::HtmlElement + case parent + when Ast::Component + ref = + parent + .refs + .find { |(ref, _)| ref.value == node.value } + .try { |(ref, _)| js.variable_of(ref) } + .as(Codegen::Node) - "this.#{ref}" - else - raise "SHOULD NOT HAPPEN" - end - when Ast::Function - function = - if connected - js.variable_of(connected) + Codegen.join ["this.", ref] else - js.variable_of(entity.as(Ast::Node)) + raise "SHOULD NOT HAPPEN" end + when Ast::Function + function = + if connected + js.variable_of(connected) + else + js.variable_of(entity.as(Ast::Node)) + end - case parent - when Ast::Module, Ast::Store - name = - js.class_of(parent.as(Ast::Node)) + case parent + when Ast::Module, Ast::Store + name = + js.class_of(parent.as(Ast::Node)) - "#{name}.#{function}" - else - "this.#{function}" - end - when Ast::Property, Ast::Get, Ast::State, Ast::Constant - name = - if connected - js.variable_of(connected) + Codegen.join [name, ".", function] else - js.variable_of(entity.as(Ast::Node)) + Codegen.join ["this.", function] end + when Ast::Property, Ast::Get, Ast::State, Ast::Constant + name = + if connected + js.variable_of(connected) + else + js.variable_of(entity.as(Ast::Node)) + end - "this.#{name}" - when Ast::Argument - compile entity - when Ast::Statement, Ast::WhereStatement - case target = entity.target - when Ast::Variable - js.variable_of(target) - else - "SHOULD NEVER HAPPEN" - end - when Tuple(Ast::Node, Int32) - case item = entity[0] - when Ast::WhereStatement, Ast::Statement - case target = item.target - when Ast::TupleDestructuring - js.variable_of(target.parameters[entity[1]]) + Codegen.join ["this.", name] + when Ast::Argument + compile entity + when Ast::Statement, Ast::WhereStatement + case target = entity.target + when Ast::Variable + js.variable_of(target) + else + "SHOULD NEVER HAPPEN" + end + when Tuple(Ast::Node, Int32) + case item = entity[0] + when Ast::WhereStatement, Ast::Statement + case target = item.target + when Ast::TupleDestructuring + js.variable_of(target.parameters[entity[1]]) + else + js.variable_of(node) + end else js.variable_of(node) end else - js.variable_of(node) + Codegen.join ["this.", node.value] end - else - "this.#{node.value}" end - end + + Codegen.source_mapped(node, result) end end end diff --git a/src/compilers/void.cr b/src/compilers/void.cr index 517cdf09d..bf14637ac 100644 --- a/src/compilers/void.cr +++ b/src/compilers/void.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::Void) : String + def _compile(node : Ast::Void) : Codegen::Node "null" end end diff --git a/src/compilers/where_statement.cr b/src/compilers/where_statement.cr index 8d16664de..a1d240630 100644 --- a/src/compilers/where_statement.cr +++ b/src/compilers/where_statement.cr @@ -1,22 +1,20 @@ module Mint class Compiler - def _compile(node : Ast::WhereStatement) : String + def _compile(node : Ast::WhereStatement) : Codegen::Node expression = compile node.expression case target = node.target when Ast::Variable name = - js.variable_of(target) + Codegen.symbol_mapped(target, js.variable_of(target)) - "let #{name} = #{expression}" + Codegen.join ["let ", name, " = ", expression] when Ast::TupleDestructuring variables = - target - .parameters - .join(',') { |param| js.variable_of(param) } + Codegen.join(target.parameters, ",") { |param| Codegen.symbol_mapped(param, js.variable_of(param)) } - "const [#{variables}] = #{expression}" + Codegen.join ["const [", variables, "] = ", expression] else "" end diff --git a/src/compilers/with.cr b/src/compilers/with.cr index bb04176a9..8181e6a20 100644 --- a/src/compilers/with.cr +++ b/src/compilers/with.cr @@ -1,6 +1,6 @@ module Mint class Compiler - def _compile(node : Ast::With) : String + def _compile(node : Ast::With) : Codegen::Node compile node.body end end diff --git a/src/js.cr b/src/js.cr index 7df117d23..68d63b5f9 100644 --- a/src/js.cr +++ b/src/js.cr @@ -1,53 +1,55 @@ module Mint abstract class Renderer - abstract def object(hash : Hash(String, String)) : String - abstract def function(name : String, arguments : Array(String), body : String) : String - abstract def arrow_function(arguments : Array(String), body : String) : String - abstract def const(name : String, value : String) : String - abstract def class(name : String, extends : String, body : Array(String)) : String - abstract def assign(name : String, value : String) : String - abstract def statements(items : Array(String)) : String - abstract def ifchain(items : Array(Tuple(String?, String))) : String - abstract def store(name : String, body : Array(String)) : String - abstract def module(name : String, body : Array(String)) : String - abstract def provider(name : String, body : Array(String)) : String - abstract def iic(body : Array(String)) : String - abstract def iif(body : String) : String - abstract def asynciif(body : String) : String - abstract def get(name : String, body : String) : String - abstract def if(condition : String, body : String) : String - abstract def elseif(condition, &block : Proc(String)) : String - abstract def else(&block : Proc(String)) : String - abstract def catch(condition : String, body : String) : String - abstract def try(body : String, catches : Array(String), finally : String) : String - abstract def promise(body : String) : String - abstract def array(items : Array(String)) : String + abstract def object(hash : Hash(Codegen::Node, Codegen::Node)) : Codegen::Node + abstract def function(name : Codegen::Node, arguments : Array(Codegen::Node), body : Codegen::Node) : Codegen::Node + abstract def arrow_function(arguments : Array(Codegen::Node), body : Codegen::Node) : Codegen::Node + abstract def const(name : String, value : Codegen::Node) : Codegen::Node + abstract def class(name : String | Codegen::SourceMappable, extends : Codegen::Node, body : Array(Codegen::Node)) : Codegen::Node + abstract def assign(name : Codegen::Node, value : Codegen::Node) : Codegen::Node + abstract def statements(items : Array(Codegen::Node)) : Codegen::Node + abstract def ifchain(items : Array(Tuple(Codegen::Node?, Codegen::Node))) : Codegen::Node + abstract def store(name : Codegen::Node, body : Array(Codegen::Node)) : Codegen::Node + abstract def module(name : String, body : Array(Codegen::Node)) : Codegen::Node + abstract def provider(name : String, body : Array(Codegen::Node)) : Codegen::Node + abstract def iif(body : Codegen::Node) : Codegen::Node + abstract def asynciif(body : Codegen::Node) : Codegen::Node + abstract def get(name : Codegen::Node, body : Codegen::Node) : Codegen::Node + abstract def if(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + abstract def elseif(condition : Codegen::Node, &block : Proc(Codegen::Node)) : Codegen::Node + abstract def else(&block : Proc(Codegen::Node)) : Codegen::Node + abstract def catch(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + abstract def try(body : Codegen::Node, catches : Array(Codegen::Node), finally : Codegen::Node?) : Codegen::Node + abstract def promise(body : Codegen::Node) : Codegen::Node + abstract def array(items : Array(Codegen::Node)) : Codegen::Node abstract def display_name(name : String, real_name : String) : String abstract def css_rule(name : String, definitions : Array(String)) : String abstract def css_rules(rules : Array(String)) : String - abstract def for(condition : String, body : String) : String - - def ifchain(items : Array(Tuple(String?, String))) : String - items - .sort_by { |(condition, _)| condition.nil? ? 1 : -1 } - .map_with_index do |(condition, body), index| - case {index, condition} - when {0, nil} - body # This branch handles only one item which does not have condition - when {_, nil} - self.else { body } - when {0, _} - self.if(condition.to_s, body) - else - self.elseif(condition) { body } + abstract def for(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + + def ifchain(items : Array(Tuple(Codegen::Node?, Codegen::Node))) : Codegen::Node + converted = + items + .sort_by { |(condition, _)| condition.nil? ? 1 : -1 } + .map_with_index do |(condition, body), index| + case {index, condition} + when {0, nil} + body # This branch handles only one item which does not have condition + when {_, nil} + self.else { body } + when {0, _} + self.if(condition, body) + else + self.elseif(condition) { body } + end end - end.join(' ') + + Codegen.join(converted, " ") end end class Optimized < Renderer - def for(condition, body) : String - "for(#{condition}){#{body}}" + def for(condition, body) : Codegen::Node + Codegen.join ["for(", condition, "){", body, "}"] end def css_rule(name, definitions) : String @@ -62,97 +64,95 @@ module Mint "" end - def object(hash : Hash(String, String)) : String - body = - hash.join(',') { |key, value| "#{key}:#{value}" } - - "{#{body}}" - end + def object(hash : Hash(Codegen::Node, Codegen::Node)) : Codegen::Node + body_parts = [] of Codegen::Node + hash.each do |key, value| + body_parts << Codegen.join [key, ":", value] + end - def function(name : String, arguments : Array(String), body : String) : String - "#{name}(#{arguments.join(',')}){#{body}}" + Codegen.join ["{", Codegen.join(body_parts, ","), "}"] end - def arrow_function(arguments : Array(String), body : String) : String - "((#{arguments.join(", ")})=>{#{body}})" + def function(name : Codegen::Node, arguments : Array(Codegen::Node), body : Codegen::Node) : Codegen::Node + Codegen.join [name, "(", Codegen.join(arguments, ","), "){", body, "}"] end - def const(name : String, value : String) : String - "const #{name}=#{value}" + def arrow_function(arguments : Array(Codegen::Node), body : Codegen::Node) : Codegen::Node + Codegen.join ["((", Codegen.join(arguments, ", "), ")=>{", body, "})"] end - def assign(name : String, value : String) : String - "#{name}=#{value}" + def const(name : String, value : Codegen::Node) : Codegen::Node + Codegen.join ["const ", name, "=", value] end - def class(name, extends : String, body : Array(String)) : String - "class #{name} extends #{extends}{#{body.join}}" + def assign(name : Codegen::Node, value : Codegen::Node) : Codegen::Node + Codegen.join [name, "=", value] end - def statements(items : Array(String)) : String - items.join(';') + def class(name : String | Codegen::SourceMappable, extends : Codegen::Node, body : Array(Codegen::Node)) : Codegen::Node + Codegen.join ["class ", name, " extends ", extends, "{", Codegen.join(body), "}"] end - def store(name : String, body : Array(String)) : String - const(name, "new(class extends _S{#{body.join}})") + def statements(items : Array(Codegen::Node)) : Codegen::Node + Codegen.join(items, ";") end - def module(name : String, body : Array(String)) : String - const(name, "new(class extends _M{#{body.join}})") + def store(name : Codegen::Node, body : Array(Codegen::Node)) : Codegen::Node + const(name, Codegen.join ["new(class extends _S{", Codegen.join(body), "})"]) end - def provider(name : String, body : Array(String)) : String - const(name, "new(class extends _P{#{body.join}})") + def module(name : String, body : Array(Codegen::Node)) : Codegen::Node + const(name, Codegen.join ["new(class extends _M{", Codegen.join(body), "})"]) end - def iic(body : Array(String)) : String - "new(class{#{body.join}})" + def provider(name : String, body : Array(Codegen::Node)) : Codegen::Node + const(name, Codegen.join ["new(class extends _P{", Codegen.join(body), "})"]) end - def iif(body : String) : String - "(()=>{#{body}})()" + def iif(body : Codegen::Node) : Codegen::Node + Codegen.join ["(()=>{", body, "})()"] end - def asynciif(body : String) : String - "(async()=>{#{body}})()" + def asynciif(body : Codegen::Node) : Codegen::Node + Codegen.join ["(async()=>{", body, "})()"] end - def get(name : String, body : String) : String - "get #{name}(){#{body}}" + def get(name : Codegen::Node, body : Codegen::Node) : Codegen::Node + Codegen.join ["get ", name, "(){", body, "}"] end - def if(condition : String, body : String) : String - "if(#{condition}){#{body}}" + def if(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + Codegen.join ["if(", condition, "){", body, "}"] end - def elseif(condition, &block : Proc(String)) : String - "else if(#{condition}){#{yield}}" + def elseif(condition : Codegen::Node, &block : Proc(Codegen::Node)) : Codegen::Node + Codegen.join ["else if(", condition, "){", yield, "}"] end - def else(&block : Proc(String)) : String - "else{#{yield}}" + def else(&block : Proc(Codegen::Node)) : Codegen::Node + Codegen.join ["else{", yield, "}"] end - def catch(condition : String, body : String) : String - "catch(#{condition}){#{body}}" + def catch(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + Codegen.join ["catch(", condition, "){", body, "}"] end - def try(body : String, catches : Array(String), finally : String) : String - "try{#{body}}#{catches.join}#{finally}" + def try(body : Codegen::Node, catches : Array(Codegen::Node), finally : Codegen::Node?) : Codegen::Node + Codegen.join ["try{", body, "}", Codegen.join(catches), finally].compact end - def promise(body : String) : String - "new Promise(#{body})" + def promise(body : Codegen::Node) : Codegen::Node + Codegen.join ["new Promise(", body, ")"] end - def array(items : Array(String)) : String - "[#{items.join(',')}]" + def array(items : Array(Codegen::Node)) : Codegen::Node + Codegen.join ["[", Codegen.join(items, ","), "]"] end end class Normal < Renderer - def for(condition, body) : String - "for (#{condition}) #{class_body(body)}" + def for(condition, body) : Codegen::Node + Codegen.join ["for (", condition, ") ", class_body(body)] end def css_rule(name, definitions) : String @@ -167,124 +167,125 @@ module Mint "#{name}.displayName = \"#{real_name}\"" end - def object(hash : Hash(String, String)) : String + def object(hash : Hash(Codegen::Node, Codegen::Node)) : Codegen::Node if hash.empty? "{}" else - body = - hash.join(",\n") { |key, value| "#{key}: #{value}" } + body_parts = [] of Codegen::Node + hash.each do |key, value| + body_parts << Codegen.join [key, ": ", value] + end - "{\n#{body.indent}\n}" + Codegen.join ["{\n", Codegen.indent(Codegen.join(body_parts, ",\n")), "\n}"] end end - def function(name : String, arguments : Array(String), body : String) : String - "#{name}(#{arguments.join(", ")}) #{class_body(body)}" + def function(name : Codegen::Node, arguments : Array(Codegen::Node), body : Codegen::Node) : Codegen::Node + Codegen.join [name, "(", Codegen.join(arguments, ", "), ") ", class_body(body)] end - def arrow_function(arguments : Array(String), body : String) : String - "(#{arguments.join(", ")}) => #{class_body(body)}" + def arrow_function(arguments : Array(Codegen::Node), body : Codegen::Node) : Codegen::Node + Codegen.join ["(", Codegen.join(arguments, ", "), ") => ", class_body(body)] end - def const(name : String, value : String) : String - "const #{name} = #{value}" + def const(name : String, value : Codegen::Node) : Codegen::Node + Codegen.join ["const ", name, " = ", value] end - def class(name : String, extends : String, body : Array(String)) : String - "class #{name} extends #{extends} #{class_body(body)}" + def class(name : String | Codegen::SourceMappable, extends : Codegen::Node, body : Array(Codegen::Node)) : Codegen::Node + Codegen.join ["class ", name, " extends ", extends, " ", class_body(body)] end - def assign(name : String, value : String) : String - "#{name} = #{value}" + def assign(name : Codegen::Node, value : Codegen::Node) : Codegen::Node + Codegen.join [name, " = ", value] end - def statements(items : Array(String)) : String - items.each_with_index.reduce("") do |memo, (item, index)| - last = items[index - 1]? if index > 0 + def statements(items : Array(Codegen::Node)) : Codegen::Node + nodes = + items.each_with_index.reduce([] of Codegen::Node) do |memo, (item, index)| + last = items[index - 1]? if index > 0 - if last - if last.includes?('\n') - memo += "\n\n" - elsif item.includes?('\n') - memo += "\n\n" - else - memo += "\n" + if last + if Codegen.includes_endl? last + memo << "\n\n" + elsif Codegen.includes_endl? item + memo << "\n\n" + else + memo << "\n" + end end - end - memo += item - memo += ";" unless memo.ends_with?(';') - memo - end - end + memo << item + memo << ";" unless Codegen.ends_with?(';', Codegen.join memo) + memo + end - def store(name : String, body : Array(String)) : String - const(name, "new(class extends _S #{class_body(body)})") + Codegen.join nodes end - def module(name : String, body : Array(String)) : String - const(name, "new(class extends _M #{class_body(body)})") + def store(name : Codegen::Node, body : Array(Codegen::Node)) : Codegen::Node + const(name, Codegen.join ["new(class extends _S ", class_body(body), ")"]) end - def provider(name : String, body : Array(String)) : String - const(name, "new(class extends _P #{class_body(body)})") + def module(name : String, body : Array(Codegen::Node)) : Codegen::Node + const(name, Codegen.join ["new(class extends _M ", class_body(body), ")"]) end - def iic(body : Array(String)) : String - "new(class #{class_body(body)})" + def provider(name : String, body : Array(Codegen::Node)) : Codegen::Node + const(name, Codegen.join ["new(class extends _P ", class_body(body), ")"]) end - def iif(body : String) : String - "(() => #{class_body(body)})()" + def iif(body : Codegen::Node) : Codegen::Node + Codegen.join ["(() => ", class_body(body), ")()"] end - def asynciif(body : String) : String - "(async () => #{class_body(body)})()" + def asynciif(body : Codegen::Node) : Codegen::Node + Codegen.join ["(async () => ", class_body(body), ")()"] end - def get(name : String, body : String) : String - "get #{name}() #{class_body(body)}" + def get(name : Codegen::Node, body : Codegen::Node) : Codegen::Node + Codegen.join ["get ", name, "() ", class_body(body)] end - def if(condition : String, body : String) : String - "if (#{condition}) #{class_body(body)}" + def if(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + Codegen.join ["if (", condition, ") ", class_body(body)] end - def elseif(condition, &block : Proc(String)) : String - "else if (#{condition}) #{class_body(yield)}" + def elseif(condition, &block : Proc(Codegen::Node)) : Codegen::Node + Codegen.join ["else if (", condition, ") ", class_body(yield)] end - def else(&block : Proc(String)) : String - "else #{class_body(yield)}" + def else(&block : Proc(Codegen::Node)) : Codegen::Node + Codegen.join ["else ", class_body(yield)] end - def catch(condition : String, body : String) : String - "catch (#{condition}) #{class_body(body)}" + def catch(condition : Codegen::Node, body : Codegen::Node) : Codegen::Node + Codegen.join ["catch (", condition, ") ", class_body(body)] end - def try(body : String, catches : Array(String), finally : String) : String - finally = " #{finally}" unless finally.empty? - "try #{class_body(body)} #{catches.join("\n ")}#{finally}" + def try(body : Codegen::Node, catches : Array(Codegen::Node), finally : Codegen::Node?) : Codegen::Node + finally = Codegen.join [" ", finally] unless finally.nil? || Codegen.empty? finally + Codegen.join ["try ", class_body(body), " ", Codegen.join(catches, "\n "), finally].compact end - def promise(body : String) : String - "new Promise(#{body})" + def promise(body : Codegen::Node) : Codegen::Node + Codegen.join ["new Promise(", body, ")"] end - def array(items : Array(String)) : String + def array(items : Array(Codegen::Node)) : Codegen::Node if items.empty? "[]" else - "[\n#{items.join(",\n").indent}\n]" + Codegen.join ["[\n", Codegen.indent(Codegen.join(items, ",\n")), "\n]"] end end - private def class_body(body : String) - "{\n#{body.indent}\n}" + private def class_body(body : Codegen::Node) + Codegen.join ["{\n", Codegen.indent(body), "\n}"] end - private def class_body(body : Array(String)) - "{\n" + body.join("\n\n").indent + "\n}" + private def class_body(body : Array(Codegen::Node)) + Codegen.join ["{\n", Codegen.indent(Codegen.join(body, "\n\n")), "\n}"] end end @@ -294,7 +295,7 @@ module Mint @style_prop_cache : Hash(String, String) = {} of String => String @style_cache : Hash(Ast::Node, String) = {} of Ast::Node => String - @cache : Hash(Ast::Node, String) = {} of Ast::Node => String + @cache : Hash(Ast::Node, Codegen::Node) = {} of Ast::Node => Codegen::Node @type_cache : Hash(String, String) = {} of String => String @@ -311,7 +312,7 @@ module Mint @renderer = optimize ? Optimized.new : Normal.new end - def variable_of(node) + def variable_of(node : Ast::Node) : Codegen::Node case node when Ast::Function return node.name.value if node.keep_name? @@ -322,12 +323,17 @@ module Mint @cache[node] ||= next_variable end - def class_of(name : String) + def class_of(name : String) : String @type_cache[name] ||= next_class end - def class_of(node : Ast::Node) - @cache[node] ||= next_class + def class_of(node : Ast::Node) : String + val = (@cache[node] ||= next_class) + case val + when String + val + else raise "Can't determine class of #{typeof(node)}" + end end def style_of(node : Ast::Node) @@ -342,29 +348,30 @@ module Mint next_variable end - def let(name, value) - "let #{name} = #{value}" + def let(name : Codegen::Node, value : Codegen::Node) + Codegen.join ["let ", name, " = ", value] end - def let(value) + def let(value : Codegen::Node) variable = next_variable - {variable, "let #{variable} = #{value}"} + {variable, Codegen.join ["let ", variable, " = ", value]} end - def call(name, props) - "#{name}(#{props.join(',')})" + def call(name : Codegen::Node, props : Array(Codegen::Node)) + props_joined = Codegen.join(props, ",") + Codegen.join [name, "(", props_joined, ")"] end - def function(name, arguments = %w[]) : String + def function(name, arguments = [] of Codegen::Node) : Codegen::Node function(name, arguments, yield) end - def return(value) - if value.empty? + def return(value : Codegen::Node) : Codegen::Node + if Codegen.empty? value "return" else - "return #{value}" + Codegen.join ["return ", value] end end @@ -384,7 +391,7 @@ module Mint promise(yield) end - def arrow_function(arguments : Array(String)) + def arrow_function(arguments : Array(Codegen::Node)) arrow_function(arguments, yield) end diff --git a/src/reactor.cr b/src/reactor.cr index d139f4d26..4a1b60061 100644 --- a/src/reactor.cr +++ b/src/reactor.cr @@ -8,6 +8,7 @@ module Mint # * Keeps a cache of ASTs of the parsed files for faster recompilation # * When --auto-format flag is passed all source files are watched and if # any changes it formats the file + # * By default generates source map for debugging (--source_map flag) class Reactor @sockets = [] of HTTP::WebSocket @error : String? @@ -15,15 +16,18 @@ module Mint @host : String @port : Int32 @auto_format : Bool + @source_map : Bool getter ast : Ast = Ast.new - getter script = "" - def self.start(host : String, port : Int32, auto_format : Bool) - new host, port, auto_format + def self.start(host : String, port : Int32, auto_format : Bool, mappings : Bool) + new host, port, auto_format, mappings end - def initialize(@host, @port, @auto_format) + def initialize(@host, @port, @auto_format, @source_map) + @script = "" + @script_map = "" + terminal.measure "#{COG} Ensuring dependencies... " do MintJson.parse_current.check_dependencies! end @@ -69,14 +73,18 @@ module Mint type_checker.check # Compile. - @script = Compiler.compile type_checker.artifacts, { + compiled = Compiler.compile type_checker.artifacts, { optimize: false, css_prefix: json.application.css_prefix, } + build = Codegen.build(compiled, @source_map) + @script = build[:code] + @script_map = build[:source_map].try { |m| m.build_json } || "" @error = nil rescue exception : Error @error = exception.to_html @script = "" + @script_map = "" end def index @@ -102,7 +110,19 @@ module Mint get "/index.js" do |env| env.response.content_type = "application/javascript" - script + if @source_map + env.response.headers["SourceMap"] = "index.js.map" + end + + @script + end + + if @source_map + get "/index.js.map" do |env| + env.response.content_type = "application/json" + + @script_map + end end get "/external-javascripts.js" do |env| diff --git a/src/skippable.cr b/src/skippable.cr index cf9868a6f..f6f145753 100644 --- a/src/skippable.cr +++ b/src/skippable.cr @@ -4,7 +4,7 @@ module Mint module Skippable @skip = [] of {String, String} - def replace_skipped(result) + def replace_skipped(result : String) @skip.reverse.reduce(result) do |memo, (digest, item)| memo.sub(digest, item) end diff --git a/src/style_builder.cr b/src/style_builder.cr index 2c8112bfe..458a5d2da 100644 --- a/src/style_builder.cr +++ b/src/style_builder.cr @@ -51,7 +51,7 @@ module Mint .try do |hash| items = hash - .each_with_object({} of String => String) do |(key, value), memo| + .each_with_object({} of String => Codegen::Node) do |(key, value), memo| memo["[`#{key}`]"] = compile value, quote_string: true end @@ -107,7 +107,7 @@ module Mint js.const("_", static), compiled_conditions, js.return("_"), - ]].flatten.reject!(&.empty?))) + ]].flatten.reject! { |item| Codegen.empty? item })) end end diff --git a/src/utils/codegen.cr b/src/utils/codegen.cr new file mode 100644 index 000000000..32a8dd794 --- /dev/null +++ b/src/utils/codegen.cr @@ -0,0 +1,235 @@ +module Mint + module Codegen + alias Node = String | + SourceMappable | + NodeContainer | + NodeIndent | + NodeNoIndent | + NodeStrip + + class NodeContainer + def initialize(@nodes : Array(Node)) + end + + def to_s(io) + io << (Codegen.build self)[:code] + raise "Please use Codegen.join to concatenate Strings with nodes." + end + end + + class SourceMappable + def initialize(@ast_node_from : Ast::Node?, @ast_node_to : Ast::Node, @node : Node, @is_symbol : Bool) + symbol.try do |s| + if s.includes?('\n') + raise "symbol is supposed to be just a name, e.g. for a variable or a function but it is: #{s}" + end + end + end + + def symbol + @is_symbol ? file.input[from, to - from] : nil + end + + def from + (@ast_node_from || @ast_node_to).from + end + + def to + @ast_node_to.to + end + + def file + @ast_node_to.input + end + + def to_s(io) + raise "Please use Codegen.join to concatenate Strings with nodes." + end + end + + class NodeIndent + def initialize(@child : Node, @spaces : Int32) + end + end + + class NodeNoIndent + def initialize(@child : Node) + end + end + + class NodeStrip + def initialize(@child : Node) + end + end + + def self.empty?(node : Node) + case node + in NodeContainer + node.@nodes.all? { |n| empty?(n) } + in String + node.empty? + in SourceMappable + false + in NodeIndent + empty? node.@child + in NodeNoIndent + empty? node.@child + in NodeStrip + empty? node.@child + end + end + + def self.join(nodes : Array(Node), separator : String? = nil) : Node + arr = [] of Node + i = 0 + + while i < nodes.size - 1 + node = nodes[i] + arr << node + + if !separator.nil? + arr << separator + end + i += 1 + end + if !nodes.empty? + arr << nodes.last + end + + NodeContainer.new arr + end + + def self.join(items, separator : String? = nil) + Codegen.join(items.map { |i| yield i }, separator) + end + + def self.source_mapped(ast_node : Ast::Node, node : Node) : Node + SourceMappable.new(nil, ast_node, node, false) + end + + def self.source_mapped(ast_node_from : Ast::Node, ast_node_to : Ast::Node, node : Node) : Node + SourceMappable.new(ast_node_from, ast_node_to, node, false) + end + + def self.symbol_mapped(ast_node : Ast::Node, node : Node) : Node + SourceMappable.new(nil, ast_node, node, true) + end + + def self.symbol_mapped(ast_node_from : Ast::Node, ast_node_to : Ast::Node, node : Node) : Node + SourceMappable.new(ast_node_from, ast_node_to, node, true) + end + + def self.indent(node : Node, spaces : Int32 = 2) : Node + NodeIndent.new(node, spaces) + end + + def self.indent(nodes : Array(Node), spaces : Int32 = 2) : Node + indent(join(nodes), spaces) + end + + def self.no_indent(node : Node) : Node + NodeNoIndent.new(node) + end + + def self.strip(node : Node) : Node + NodeStrip.new(node) + end + + def self.includes_endl?(node : Node) : Bool + traverse = uninitialized Node -> Bool + traverse = ->(cur : Node) { + case cur + in String + cur.includes? '\n' + in SourceMappable + traverse.call cur.@node + in NodeContainer + cur.@nodes.any? { |n| traverse.call n } + in NodeIndent + traverse.call cur.@child + in NodeNoIndent + traverse.call cur.@child + in NodeStrip + false + end + } + + traverse.call(node) + end + + def self.ends_with?(char : Char, node : Node) : Bool + last_str = "" + + traverse = uninitialized Node -> Nil + traverse = ->(cur : Node) { + case cur + in String + last_str = cur + in SourceMappable + traverse.call cur.@node + in NodeContainer + cur.@nodes.each { |n| traverse.call n } + in NodeIndent + traverse.call cur.@child + in NodeNoIndent + traverse.call cur.@child + in NodeStrip + if char == ' ' || char == '\n' + false + else + traverse.call cur.@child + end + end + } + + traverse.call(node) + last_str.ends_with? char + end + + def self.build(node : Node, source_map = false) : NamedTuple(code: String, source_map: SourceMap?) + builder = IndentedStringBuilder.new + map = source_map ? SourceMap.new : nil + + traverse = uninitialized Node -> Nil + traverse = ->(cur : Node) { + case cur + in String + builder << cur + in SourceMappable + pos = builder.get_position_for_next_input + dst_from_line = pos[:line] + dst_from_col = pos[:column] + + traverse.call cur.@node + + map.try do |map| + map.add_mapping(cur.file, cur.from, dst_from_line, dst_from_col, cur.symbol) + end + in NodeContainer + cur.@nodes.each { |n| traverse.call n } + in NodeIndent + builder.indent_size += cur.@spaces + traverse.call cur.@child + builder.indent_size -= cur.@spaces + in NodeNoIndent + old_indent = builder.indent_size + builder.indent_size = 0 + traverse.call cur.@child + builder.indent_size = old_indent + in NodeStrip + builder.strip_whitespace_around do + traverse.call cur.@child + end + end + } + + traverse.call(node) + + {code: builder.build, source_map: map} + end + + def self.build_with_mappings(node : Node) : Tuple(String, String) + build(node, true) + end + end +end diff --git a/src/utils/indented_string_builder.cr b/src/utils/indented_string_builder.cr new file mode 100644 index 000000000..2ced4816b --- /dev/null +++ b/src/utils/indented_string_builder.cr @@ -0,0 +1,136 @@ +module Mint + class IndentedStringBuilder + property indent_size : Int32 = 0 + + def initialize + @str_builder = String::Builder.new(16384) + @last_input_had_endl = false + @left_strip = false + @line = 0 + @column = 0 + end + + def size + @str_builder.bytesize + end + + def get_position_for_next_input + {line: @last_input_had_endl ? @line + 1 : @line, + column: @last_input_had_endl ? indent_size : (@str_builder.empty? ? indent_size : @column)} + end + + def <<(str : String) : self + if str.size == 0 + return self + end + + if @last_input_had_endl + @str_builder << '\n' + @line += 1 + @column = 0 + end + + if indent_size > 0 && !@left_strip + # let's not have lines filled with only spaces + if (@last_input_had_endl || @str_builder.empty?) && !str.blank? + @str_builder << " " * indent_size + @column += indent_size + end + end + + # add indentation to all but last endline + ch = ' ' + new_str = String.build do |s| + last_index = str.size - 1 + + i = 0 + while i <= last_index + ch = str[i] + next_ch = str[i + 1]? + + if ch != ' ' && ch != '\n' + @left_strip = false + end + + if i < last_index || ch != '\n' + if !@left_strip || ch != ' ' + s << ch + + if ch == '\n' + @line += 1 + @column = 0 + else + @column += 1 + end + end + end + + if ch == '\n' && next_ch != '\n' && i < last_index && !@left_strip + s << " " * indent_size + @column += indent_size + end + + i += 1 + end + end + @last_input_had_endl = ch == '\n' + @str_builder << new_str + + self + end + + def build + if @last_input_had_endl + @str_builder << '\n' + @line += 1 + @column = 0 + end + @str_builder.to_s + end + + def strip_whitespace_around + start_size = @str_builder.bytesize + @left_strip = true + yield + strip_end_whitespace start_size + end + + private def strip_end_whitespace(start_size) + @last_input_had_endl = false + + recalculate_column = false + i = @str_builder.bytesize + while i > start_size + ch = @str_builder.buffer[i - 1] + + if ch == ' '.ord + @column -= 1 + elsif ch == '\n'.ord + @line -= 1 + recalculate_column = true + else + break + end + + i -= 1 + end + + diff = @str_builder.bytesize - i + if diff > 0 + @str_builder.back diff + + if recalculate_column + n = @str_builder.bytesize + i = n - 1 + while i > 0 + ch = @str_builder.buffer[i] + break if ch == '\n' + i -= 1 + end + + @column = n - i + 1 + end + end + end + end +end diff --git a/src/utils/object_serializer.cr b/src/utils/object_serializer.cr index f3e917d58..1f53caf07 100644 --- a/src/utils/object_serializer.cr +++ b/src/utils/object_serializer.cr @@ -2,7 +2,7 @@ module Mint # This class handles the generation of a serializer of Mint types into # JavaScript Objects. class ObjectSerializer - @mappings = {} of TypeChecker::Record => String + @mappings = {} of TypeChecker::Record => Codegen::Node getter js : Js @@ -16,7 +16,7 @@ module Mint def generate_mappings(node : TypeChecker::Record) @mappings[node] ||= begin mappings = - node.fields.each_with_object({} of String => String) do |(key, value), memo| + node.fields.each_with_object({} of String => Codegen::Node) do |(key, value), memo| decoder = self.decoder value @@ -33,8 +33,8 @@ module Mint end end - def encoder(node : TypeChecker::Record) - "((_)=>#{js.class_of(node.name)}.encode(_))" + def encoder(node : TypeChecker::Record) : Codegen::Node + Codegen.join ["((_)=>", js.class_of(node.name), ".encode(_))"] end def encoder(node : TypeChecker::Variable) @@ -42,7 +42,7 @@ module Mint raise "Cannot generate an encoder for a type variable!" end - def encoder(node : TypeChecker::Type) + def encoder(node : TypeChecker::Type) : Codegen::Node? case node.name when "Time" "Encoder.time" @@ -63,7 +63,7 @@ module Mint def decoder(node : TypeChecker::Record) generate_mappings node - "((_)=>#{js.class_of(node.name)}.decode(_))" + Codegen.join ["((_)=>", js.class_of(node.name), ".decode(_))"] end def decoder(node : TypeChecker::Type) @@ -79,11 +79,11 @@ module Mint when "Time" "Decoder.time" when "Maybe" - "Decoder.maybe(#{decoder(node.parameters.first)})" + Codegen.join ["Decoder.maybe(", decoder(node.parameters.first), ")"] when "Array" - "Decoder.array(#{decoder(node.parameters.first)})" + Codegen.join ["Decoder.array(", decoder(node.parameters.first), ")"] when "Map" - "Decoder.map(#{decoder(node.parameters.last)})" + Codegen.join ["Decoder.map(", decoder(node.parameters.last), ")"] else # This should never happen because of the typechecker! raise "Cannot generate a decoder for #{node}!" diff --git a/src/utils/source_map.cr b/src/utils/source_map.cr new file mode 100644 index 000000000..890f83f58 --- /dev/null +++ b/src/utils/source_map.cr @@ -0,0 +1,206 @@ +require "json" + +module Mint + # Source Map rev. 3 + # https://docs.google.com/document/d/1U1RGAehQwRypUTovF1KRlpiOFze0b-_2gc6fAH0KY0k + class SourceMap + struct Mapping + def initialize(@source_file_index : Int32, + @src_line : Int32, + @src_col : Int32, + @dst_line : Int32, + @dst_col : Int32, + @source_symbol_index : Int32?) + end + end + + class SourceFile + def initialize(@file : Ast::Data, @index : Int32) + @lines = [0] of Int32 + @file.input.each_char_with_index do |ch, i| + if ch == '\n' + @lines.push i + 1 + end + end + end + + def path + @file.file + end + + def content + @file.input + end + + def get_line_column_position(file_position : Int32) + left = 0 + right = @lines.size - 1 + line_start_pos, line = begin + index = 0 + found = false + pos = 0 + while left <= right + middle = left + ((right - left) / 2).to_i + pos = @lines[middle] + if pos < file_position + left = middle + 1 + elsif pos > file_position + right = middle - 1 + else + index = middle + found = true + break + end + end + + if !found + index = left - 1 + pos = @lines[index] + end + + {pos, index} + end + + column = file_position - line_start_pos + {line, column} + end + end + + def initialize + @sources = {} of String => SourceFile + @sources_idx = [] of String + @source_symbols = [] of String + @source_symbols_to_idx = {} of String => Int32 + @mappings = {} of Int32 => Array(Mapping) + @max_output_line = 0 + end + + def add_mapping(src_file : Ast::Data, + src_from : Int32, + dst_from_line : Int32, + dst_from_col : Int32, + src_symbol : String? = nil) : Nil + file = @sources[src_file.file]? || + begin + index = @sources_idx.size + path = src_file.file + @sources_idx.push path + @sources[path] = SourceFile.new(src_file, index) + end + + symbol_index = src_symbol.try do |symbol| + @source_symbols_to_idx[symbol]? || + begin + index = @source_symbols.size + @source_symbols.push symbol + @source_symbols_to_idx[symbol] = index + index + end + end + + src_from_line, src_from_col = file.get_line_column_position src_from + + mapping = Mapping.new(file.@index, src_from_line, src_from_col, dst_from_line, dst_from_col, symbol_index) + line_mappings = @mappings[dst_from_line]? || (@mappings[dst_from_line] = [] of Mapping) + line_mappings.push mapping + @max_output_line = Math.max(dst_from_line, @max_output_line) + end + + def build_json(output_filename : String? = nil) + JSON.build do |j| + j.object do + j.field "version", 3 + output_filename.try { |f| j.field "file", f } + + j.field "sources" do + j.array do + @sources_idx.each do |path| + source = path + j.string source + end + end + end + + j.field "sourcesContent" do + j.array do + @sources_idx.each do |path| + content = @sources[path].content + j.string content + end + end + end + + j.field "names" do + j.array do + @source_symbols.each do |s| + j.string s + end + end + end + + j.field "mappings" do + mappings = String.build do |str| + is_first_input = true + last_source_file_index = -1 + last_src_line = -1 + last_src_col = -1 + last_source_symbol_index : Int32? = nil + + (0..@max_output_line).each do |line_index| + line_mappings = @mappings[line_index]? + + if line_mappings + is_first_in_line = true + last_dst_col = -1 + + line_mappings.each_with_index do |mapping, mapping_index| + if is_first_in_line + dst_col = mapping.@dst_col + is_first_in_line = false + else + dst_col = mapping.@dst_col - last_dst_col + end + last_dst_col = mapping.@dst_col + + source_file_index = mapping.@source_file_index + src_line = mapping.@src_line + src_col = mapping.@src_col + if is_first_input + is_first_input = false + else + source_file_index -= last_source_file_index + src_line -= last_src_line + src_col -= last_src_col + end + last_source_file_index = mapping.@source_file_index + last_src_line = mapping.@src_line + last_src_col = mapping.@src_col + + source_symbol_index = mapping.@source_symbol_index + if source_symbol_index + if last_source_symbol_index + source_symbol_index -= last_source_symbol_index + end + last_source_symbol_index = mapping.@source_symbol_index + end + + str << VLQ.encode(dst_col) + str << VLQ.encode(source_file_index) + str << VLQ.encode(src_line) + str << VLQ.encode(src_col) + str << VLQ.encode(source_symbol_index) if source_symbol_index + str << ',' if mapping_index < line_mappings.size - 1 + end + end + + str << ';' if line_index != @max_output_line + end + end + + j.string mappings + end + end + end + end + end +end diff --git a/src/utils/vlq.cr b/src/utils/vlq.cr new file mode 100644 index 000000000..901b98627 --- /dev/null +++ b/src/utils/vlq.cr @@ -0,0 +1,112 @@ +module Mint + # Support for encoding/decoding the variable length quantity format + # described in the spec at: + # + # https://docs.google.com/document/d/1U1RGAehQwRypUTovF1KRlpiOFze0b-_2gc6fAH0KY0k/edit + # + # This implementation is heavily based on https://github.com/mozilla/source-map + # Copyright 2009-2011, Mozilla Foundation and contributors, BSD + # + module VLQ + # A single base 64 digit can contain 6 bits of data. For the base 64 variable + # length quantities we use in the source map spec, the first bit is the sign, + # the next four bits are the actual value, and the 6th bit is the + # continuation bit. The continuation bit tells us whether there are more + # digits in this value following this digit. + # + # Continuation + # | Sign + # | | + # V V + # 101011 + VLQ_BASE_SHIFT = 5 + # binary: 100000 + VLQ_BASE = 1 << VLQ_BASE_SHIFT + # binary: 011111 + VLQ_BASE_MASK = VLQ_BASE - 1 + # binary: 100000 + VLQ_CONTINUATION_BIT = VLQ_BASE + BASE64_DIGITS = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".chars + BASE64_VALUES = (0..64).reduce({} of Char => Int32) { |h, i| h[BASE64_DIGITS[i]] = i } + + # Returns the base 64 VLQ encoded value. + def self.encode(int) + vlq = to_vlq_signed(int) + encoded = "" + + cond = true + while cond + digit = vlq & VLQ_BASE_MASK + vlq >>= VLQ_BASE_SHIFT + digit |= VLQ_CONTINUATION_BIT if vlq > 0 + encoded += base64_encode(digit) + cond = vlq > 0 + end + + encoded + end + + # Decodes the next base 64 VLQ value from the given string and returns the + # value and the rest of the string. + def self.decode(str) + vlq = 0 + shift = 0 + continue = true + chars = str.chars + + while continue + char = chars.shift or raise "Expected more digits in base 64 VLQ value." + digit = base64_decode(char) + continue = false if (digit & VLQ_CONTINUATION_BIT) == 0 + digit &= VLQ_BASE_MASK + vlq += digit << shift + shift += VLQ_BASE_SHIFT + end + + [from_vlq_signed(vlq), String.new(chars)] + end + + # Decode an array of variable length quantities from the given string + def self.decode_array(str) + output = [] of Tuple(Int32, String) + while !str.empty? + int, str = decode(str) + output << int + end + output + end + + private def self.base64_encode(int) + BASE64_DIGITS[int]? || raise ArgumentError.new "#{int} is not a valid base64 digit" + end + + private def self.base64_decode(char) + BASE64_VALUES[char]? || raise ArgumentError.new "#{char} is not a valid base64 digit" + end + + # Converts from a two's-complement integer to an integer where the + # sign bit is placed in the least significant bit. For example, as decimals: + # 1 becomes 2 (10 binary), -1 becomes 3 (11 binary) + # 2 becomes 4 (100 binary), -2 becomes 5 (101 binary) + private def self.to_vlq_signed(int) + if int < 0 + ((-int) << 1) + 1 + else + int << 1 + end + end + + # Converts to a two's-complement value from a value where the sign bit is + # placed in the least significant bit. For example, as decimals: + # + # 2 (10 binary) becomes 1, 3 (11 binary) becomes -1 + # 4 (100 binary) becomes 2, 5 (101 binary) becomes -2 + private def self.from_vlq_signed(vlq) + if vlq & 1 == 1 + -(vlq >> 1) + else + vlq >> 1 + end + end + end +end