Upgrade to Pro — share decks privately, control downloads, hide ads and more …

ERB Hacks

ERB Hacks

Ruby World Conference 2023

seki at druby.org

November 08, 2023
Tweet

More Decks by seki at druby.org

Other Decks in Programming

Transcript

  1. ERB前史(1999年) HTMLのエンティティ表記を置換するライブラリを書い てたら、eRubyというものを作っていることを教えて もらった ruby-dev:5286 (1999-02-19) Re: htmlelem.rb (Re: HTML

    generator) 8 エンティティ表記ならWeb オーサリングツールの邪魔し ないだろう、と想像した (使ったことないけど)
  2. 退屈なputs 古来のCGIはだいたいこんな感じ 12 puts %Q{<!DOCTYPE html>} puts %Q{<html lang="ja">} puts

    %Q{<head>} puts %Q{</head>} puts %Q{<body>} puts %Q{<ul>} ENV.each do |row| puts %Q!<li>#{row[0]}</li>! end puts %Q{</ul>} puts %Q{</body>} puts %Q{</html>}
  3. here document使うとマシ putsの回数は減らせた 13 puts %Q{<!DOCTYPE html>} puts %Q{<html lang="ja">}

    puts %Q{<head>} puts %Q{</head>} puts %Q{<body>} puts %Q{<ul>} ENV.each do |row| puts %Q!<li>#{row[0]}</li>! end puts %Q{</ul>} puts %Q{</body>} puts %Q{</html>} puts <<EOS <!DOCTYPE html> <html lang="ja"> <head> </head> <body> <ul> EOS ENV.each do |row| puts %Q!<li>#{row[0]}</li>! end puts <<EOS </ul> </body> </html> EOS
  4. putsからeRubyへ Stringの式展開では制御構造が書きにくい(書ける?)のだがeRubyならかんたん 14 <!DOCTYPE html> <html lang="ja"> <head> </head> <body>

    <ul> <% ENV.each do |row| %> <li><%= row[0] %></li> <% end %> </ul> </body> </html> puts <<EOS <!DOCTYPE html> <html lang="ja"> <head> </head> <body> <ul> EOS ENV.each do |row| puts %Q!<li>#{row[0]}</li>! end puts <<EOS </ul> </body> </html> EOS
  5. 単純じゃなくなるぞ予言 ページの一部をメソッドで生成したくなると思う Helper? 16 <h2>report hoge</h2> <ul> <% result.each do

    |row| %> <li><%= format_date(row.date) %></li> <% end %> </ul> def format_date(date) date.strftime("%Y-%m-%d") end
  6. Procの利用 Procで処理の一部を使いまわせる 29 <% t = Proc.new do |arg| %>

    <h3><%= arg %>ճ౴ཝ</h3> <table id="table-<%= arg %>"> ... </table> <% end %> <h3>࣭໰1</h3> ࠷ॳͷ࣭໰จͩΑ <% t.call("q1") %> <h3>࣭໰2</h3> ೋͭ໨ͷ࣭໰ͩΑ <% t.call("q2") %> <h3>࣭໰1</h3> ࠷ॳͷ࣭໰จͩΑ <h3>q1ճ౴ཝ</h3> <table id="table-q1"> ... </table> <h3>࣭໰2</h3> ೋͭ໨ͷ࣭໰ͩΑ <h3>q2ճ౴ཝ</h3> <table id="table-q2"> ... </table> ここを再利用したい ここで呼び出す ここで呼び出す
  7. Procの利用 Procで処理の一部を使いまわせる 30 <% t = Proc.new do |arg| %>

    <h3><%= arg %>ճ౴ཝ</h3> <table id="table-<%= arg %>"> ... </table> <% end %> <h3>࣭໰1</h3> ࠷ॳͷ࣭໰จͩΑ <% t.call("q1") %> <h3>࣭໰2</h3> ೋͭ໨ͷ࣭໰ͩΑ <% t.call("q2") %> HTML片を組み立てる処理 そのものがProcになる パラメータ渡せる
  8. Procの利用 Procで処理の一部を使いまわせる あれ?なんかこれ見たことある... 31 <% t = Proc.new do |arg|

    %> <h3><%= arg %>ճ౴ཝ</h3> <table id="table-<%= arg %>"> ... </table> <% end %> <h3>࣭໰1</h3> ࠷ॳͷ࣭໰จͩΑ <% t.call("q1") %> <h3>࣭໰2</h3> ೋͭ໨ͷ࣭໰ͩΑ <% t.call("q2") %> <%= ... %>ではないので注意
  9. できませんでした Railsに正規表現でブロックの開始を検知して処理を切り替えるコードがあった 34 But gave up. ! Because he didn't

    like this part in ActionView + Erubis ! BLOCK_EXPR = /\s+(do|\{)(\s*\|[^|]*\|)?\s*\Z/ ! This code scans the template with the Regexp and detects the Ruby block, but this kind of code could be imperfect ! So this is not acceptable as an ERB spec, said Seki- san.
  10. <%= ... %>とブロック こうなっちゃう 37 <%= form_with do %> ...

    <% end %> #coding:UTF-8 _erbout = +''; _erbout.<<(( form_with do ).to_s); _erbout.<< "\n ... \n".freeze ; end ; _erbout.<< "\n".freeze ; _erbout SyntaxError
  11. なおしかた ((...).to_s)の括弧がじゃま 39 <%= form_with do %> ... <% end

    %> #coding:UTF-8 _erbout = +''; _erbout.<<(( form_with do ).to_s); _erbout.<< "\n ... \n".freeze ; end ; _erbout.<< "\n".freeze ; _erbout (( ... ).to_s) をなくせばよいかも? → 結果バッファクラスに移動させる
  12. なおしかた Rubyへの変換方法はカスタマイズできる 40 <%= form_with do %> ... <% end

    %> #coding:UTF-8 _erbout = +''; _erbout.<<(( form_with do ).to_s); _erbout.<< "\n ... \n".freeze ; end ; _erbout.<< "\n".freeze ; _erbout 結果の取得 一時変数の名前と初期化 リテラルの連結 式の連結 ((...).to_s)をなくせばOK
  13. Rubyむずかしい そんなに単純じゃなかった 41 _erbout << h("str") # OK _erbout <<

    h "str" # SyntaxError _erbout = h "str" # _erbout = form_with "arg" do |arg| # ... end 代入なら大丈夫なのに... <<と括弧なし のメソッド呼び 出し _erbout.concat h "str" # OK but .... _erbout.concat form_with "arg" do |arg| # NG ... # end # concatʹϒϩοΫ͕౉Δ <<でなくふつうのメソッド呼び出しにしても このブロックはconcatのブロック引数になる
  14. ERBOut 結果バッファクラスの追加 いろんなトリックがある +で自分自身を返して+=を だます captureのためのしかけ 43 class ERBOut Buffer

    = String # SafeBuffer if rails def initialize(s='') @str = Buffer.new(s) end def to_s @str end def <<(other) @str << other end def +(other) @str << other.to_s self end def capture(*arg, &block) save = @str @str = Buffer.new yield(*arg) return @str ensure @str = save end end 式の連結処理の一部 をここへ移動 自分自身を返す
  15. ERBOut 結果バッファクラスの追加 いろんなトリックがある +で自分自身を返して+=を だます captureのためのしかけ 44 class ERBOut Buffer

    = String # SafeBuffer if rails def initialize(s='') @str = Buffer.new(s) end def to_s @str end def <<(other) @str << other end def +(other) @str << other.to_s self end def capture(*arg, &block) save = @str @str = Buffer.new yield(*arg) return @str ensure @str = save end end もとのバッファを退避 一時バッファに差し替え 一時バッファをreturn もとのバッファを復元
  16. なんかできちゃった 中間結果のためのERBOutクラスも使ったらできた https://gist.github.com/seki/610a42932a85209aaa33547ae983bbdf 45 <%= form_with do %> ... <%

    end %> #coding:UTF-8 _erbout = ERB::ERBOut.new; _erbout += form_with do ; _erbout << "\n ...\n".freeze ; end ; _erbout << "\n".freeze ; _erbout.to_s 一時変数の名前と初期化 式の連結 += がミソ 結果の取得 ここはブロック の内側だよ!