Skip to content

Latest commit

 

History

History
167 lines (127 loc) · 4.13 KB

README.md

File metadata and controls

167 lines (127 loc) · 4.13 KB

JSrb

ruby.wasm の js.rb を Ruby っぽく使えるようにするためのラッパー

Usage

<!DOCTYPE html>
<html>
  <script src="https://cdn.jsdelivr.net/npm/@ruby/[email protected]/dist/browser.script.iife.js"></script>
  <script type="text/ruby" src="https://cdn.jsdelivr.net/gh/tmtm/[email protected]/jsrb.rb"></script>
  <script type="text/ruby">
    ...
  </script>
</html>

次のような機能がある。

値を Ruby で扱いやすいように変換する

JavaScript:

elements = document.querySelectorAll('div')
elements.length
elements[0].style.width

ruby.wasm JS:

elements = JS.global[:document].querySelectorAll('div')
elements[:length]           #=> 3 (JS::Object)
elements[0][:style][:width] #=> "100px" (JS::Object)

ruby.wasm JSrb:

elements = JSrb.document.querySelectorAll('div')
elements[:length]           #=> 3 (Integer)
elements[0][:style][:width] #=> "100px" (String)

プロパティを [] なしで参照できる

JavaScript:

elements = document.querySelectorAll('div')
elements.length
elements[0].style.width

ruby.wasm JS:

elements = JS.global[:document].querySelectorAll('div')
elements[:length]           #=> 3 (JS::Object)
elements[0][:style][:width] #=> "100px" (JS::Object)

ruby.wasm JSrb:

elements = JSrb.document.querySelectorAll('div')
elements.length             #=> 3 (Integer)
elements[0].style.width     #=> "100px" (String)

プロパティと同名の関数があったらそれが呼ばれてしまうので、その場合は [] で参照する必要がある。 undefined を返すプロパティをこの形式で呼ぶと NoMethodError になってしまうので、この場合も [] で参照する必要がある。

キャメルケースのプロパティやメソッドをスネークケースで呼べる

JavaScript:

div = document.querySelector('div')
div.innerText

ruby.wasm JS:

div = JS.global[:document].querySelector('div')
div[:innerText]

ruby.wasm JSrb:

div = JSrb.document.query_selector('div')
div.inner_text

length プロパティと item() メソッドがあるオブジェクトは Enumerable になる

ruby.wasm JS:

elements = JS.global[:document].querySelectorAll('div')
elements[:length].to_i.times do |i|
  elements[i][:style][:color] = 'red'
end

ruby.wasm JSrb:

elements = JSrb.document.query_selector_all('div')
elements.each do |element|
  element.style.color = 'red'
end

JSrb.window

JavaScript の window オブジェクトに対応

JSrb.global

JSrb.window と同じ

JSrb.document

JavaScript の document オブジェクトに対応

JSrb.convert

JS::Object を Ruby で扱いやすい形に変換する:

JSrb.convert(JS.eval('return 123'))        #=> 123 (Integer)
JSrb.convert(JS.eval('return 123.45'))     #=> 123.45 (Float)
JSrb.convert(JS.eval('return [1,2,3]'))    #=> [1, 2, 3] (Array)
JSrb.convert(JS.eval('return "abc"'))      #=> "abc" (String)
JSrb.convert(JS.eval('return null'))       #=> nil
JSrb.convert(JS.eval('return undefined'))  #=> nil
JSrb.convert(JS.eval('return new Date'))   #=> 2024-07-16 17:04:41.755 UTC (Time)
JSrb.convert(JS.eval('return {a:1,b:2}'))  #=> #<JSrb: [object Object]>

JSrb#to_h

JavaScript の Object を Hash に変換する:

JSrb.new(JS.eval('return {a:1,b:2}')).to_h #=> {:a=>1, :b=>2}

JSrb#timeout(sec) { ... }

sec 秒後にブロックを実行する

JS.global.setTimeout と異なり、ブロック内で await も使える。

<script src="https://cdn.jsdelivr.net/npm/@ruby/[email protected]/dist/browser.script.iife.js"></script>
<script type="text/ruby" src="jsrb.rb"></script>
<script type="text/ruby" data-eval="async">
  require 'js'
  def hoge = p JS.global.fetch("/").await
  JS.global.setTimeout(->{hoge}, 0)
    #=> Uncaught Error: /bundle/gems/js-2.6.2/lib/js.rb:86:in `await': JS::Object#await can be called only from RubyVM#evalAsync or RbValue#callAsync JS API
  JSrb.timeout(0){hoge}
    #=> OK!
</script>

JSrb#js_object

JSrb がラップしている JS::Object を返す

License

MIT